2021-05-12 14:34:49 +00:00
|
|
|
import {DataRowModel} from 'app/client/models/DataRowModel';
|
|
|
|
import {colors} from 'app/client/ui2018/cssVars';
|
|
|
|
import {ChoiceTextBox} from 'app/client/widgets/ChoiceTextBox';
|
2021-05-12 21:00:55 +00:00
|
|
|
import {CellValue} from 'app/common/DocActions';
|
2021-05-12 14:34:49 +00:00
|
|
|
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),
|
2021-05-12 21:00:55 +00:00
|
|
|
dom.domComputed((use) => {
|
|
|
|
return use(row._isAddRow) ? null : [use(value), use(this._choiceSet)] as [CellValue, Set<string>];
|
|
|
|
}, (input) => {
|
2021-05-12 14:34:49 +00:00
|
|
|
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),
|
2021-05-12 21:00:55 +00:00
|
|
|
cssInvalidToken.cls('-invalid', !choiceSet.has(token as string))
|
2021-05-12 14:34:49 +00:00
|
|
|
)
|
|
|
|
);
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
`);
|