mirror of
https://github.com/gristlabs/grist-core.git
synced 2024-10-19 16:52:29 +00:00
8d62a857e1
Summary: - Adds a new ChoiceList type, and widgets to view and edit it. - Store in SQLite as a JSON string - Support conversions between ChoiceList and other types Test Plan: Added browser tests, and a test for how these values are stored Reviewers: paulfitz Reviewed By: paulfitz Differential Revision: https://phab.getgrist.com/D2803
73 lines
2.0 KiB
TypeScript
73 lines
2.0 KiB
TypeScript
import {DataRowModel} from 'app/client/models/DataRowModel';
|
|
import {colors} from 'app/client/ui2018/cssVars';
|
|
import {ChoiceTextBox} from 'app/client/widgets/ChoiceTextBox';
|
|
import {decodeObject} from 'app/plugin/objtypes';
|
|
import {Computed, dom, styled} from 'grainjs';
|
|
|
|
/**
|
|
* ChoiceListCell - A cell that renders a list of choice tokens.
|
|
*/
|
|
export class ChoiceListCell extends ChoiceTextBox {
|
|
private _choiceSet = Computed.create(this, this.getChoiceValues(), (use, values) => new Set(values));
|
|
|
|
public buildDom(row: DataRowModel) {
|
|
const value = row.cells[this.field.colId.peek()];
|
|
|
|
return cssChoiceList(
|
|
dom.cls('field_clip'),
|
|
cssChoiceList.cls('-wrap', this.wrapping),
|
|
dom.style('justify-content', this.alignment),
|
|
dom.domComputed((use) => use(row._isAddRow) ? null : [use(value), use(this._choiceSet)], (input) => {
|
|
if (!input) { return null; }
|
|
const [rawValue, choiceSet] = input;
|
|
const val = decodeObject(rawValue);
|
|
if (!val) { return null; }
|
|
// Handle any unexpected values we might get (non-array, or array with non-strings).
|
|
const tokens: unknown[] = Array.isArray(val) ? val : [val];
|
|
return tokens.map(token =>
|
|
cssToken(
|
|
String(token),
|
|
cssInvalidToken.cls('-invalid', !choiceSet.has(token))
|
|
)
|
|
);
|
|
}),
|
|
);
|
|
}
|
|
}
|
|
|
|
const cssChoiceList = styled('div', `
|
|
display: flex;
|
|
align-items: start;
|
|
padding: 0 3px;
|
|
|
|
position: relative;
|
|
height: min-content;
|
|
min-height: 22px;
|
|
|
|
&-wrap {
|
|
flex-wrap: wrap;
|
|
}
|
|
`);
|
|
|
|
const cssToken = styled('div', `
|
|
flex: 0 1 auto;
|
|
min-width: 0px;
|
|
overflow: hidden;
|
|
border-radius: 3px;
|
|
background-color: ${colors.mediumGreyOpaque};
|
|
padding: 1px 4px;
|
|
margin: 2px;
|
|
line-height: 16px;
|
|
`);
|
|
|
|
export const cssInvalidToken = styled('div', `
|
|
&-invalid {
|
|
background-color: white !important;
|
|
box-shadow: inset 0 0 0 1px var(--grist-color-error);
|
|
color: ${colors.slate};
|
|
}
|
|
&-invalid.selected {
|
|
background-color: ${colors.lightGrey} !important;
|
|
}
|
|
`);
|