mirror of
https://github.com/gristlabs/grist-core.git
synced 2024-10-27 20:44:07 +00:00
caf830db08
Summary: Adds a new Record Card view section to each non-summary table, which can be from opened from various parts of the Grist UI to view and edit records in a popup card view. Work is still ongoing, so the feature is locked away behind a flag; follow-up work is planned to finish up the implementation and add end-to-end tests. Test Plan: Python and server tests. Browser tests will be included in a follow-up. Reviewers: jarek, paulfitz Reviewed By: jarek Subscribers: paulfitz Differential Revision: https://phab.getgrist.com/D4114
176 lines
6.5 KiB
TypeScript
176 lines
6.5 KiB
TypeScript
import {makeT} from 'app/client/lib/localization';
|
|
import {DataRowModel} from 'app/client/models/DataRowModel';
|
|
import {TableRec} from 'app/client/models/DocModel';
|
|
import {ViewFieldRec} from 'app/client/models/entities/ViewFieldRec';
|
|
import {RECORD_CARDS} from 'app/client/models/features';
|
|
import {urlState} from 'app/client/models/gristUrlState';
|
|
import {cssLabel, cssRow} from 'app/client/ui/RightPanelStyles';
|
|
import {hideInPrintView, testId, theme} from 'app/client/ui2018/cssVars';
|
|
import {icon} from 'app/client/ui2018/icons';
|
|
import {IOptionFull, select} from 'app/client/ui2018/menus';
|
|
import {NTextBox} from 'app/client/widgets/NTextBox';
|
|
import {isFullReferencingType, isVersions} from 'app/common/gristTypes';
|
|
import {UIRowId} from 'app/plugin/GristAPI';
|
|
import {Computed, dom, styled} from 'grainjs';
|
|
|
|
|
|
const t = makeT('Reference');
|
|
|
|
/**
|
|
* Reference - The widget for displaying references to another table's records.
|
|
*/
|
|
export class Reference extends NTextBox {
|
|
private _refTable: Computed<TableRec | null>;
|
|
private _visibleColRef: Computed<number>;
|
|
private _validCols: Computed<Array<IOptionFull<number>>>;
|
|
|
|
constructor(field: ViewFieldRec) {
|
|
super(field);
|
|
|
|
this._visibleColRef = Computed.create(this, (use) => use(this.field.visibleColRef));
|
|
// Note that saveOnly is used here to prevent display value flickering on visible col change.
|
|
this._visibleColRef.onWrite((val) => this.field.visibleColRef.saveOnly(val));
|
|
|
|
this._refTable = Computed.create(this, (use) => use(use(this.field.column).refTable));
|
|
|
|
this._validCols = Computed.create(this, (use) => {
|
|
const refTable = use(this._refTable);
|
|
if (!refTable) { return []; }
|
|
return use(use(refTable.columns).getObservable())
|
|
.filter(col => !use(col.isHiddenCol))
|
|
.map<IOptionFull<number>>(col => ({
|
|
label: use(col.label),
|
|
value: col.getRowId(),
|
|
icon: 'FieldColumn',
|
|
disabled: isFullReferencingType(use(col.type)) || use(col.isTransforming)
|
|
}))
|
|
.concat([{label: t('Row ID'), value: 0, icon: 'FieldColumn'}]);
|
|
});
|
|
}
|
|
|
|
public buildConfigDom() {
|
|
return [
|
|
this.buildTransformConfigDom(),
|
|
cssLabel(t('CELL FORMAT')),
|
|
super.buildConfigDom()
|
|
];
|
|
}
|
|
|
|
public buildTransformConfigDom() {
|
|
const disabled = Computed.create(null, use => use(this.field.config.multiselect));
|
|
return [
|
|
cssLabel(t('SHOW COLUMN')),
|
|
cssRow(
|
|
dom.autoDispose(disabled),
|
|
select(this._visibleColRef, this._validCols, {
|
|
disabled
|
|
}),
|
|
testId('fbuilder-ref-col-select')
|
|
)
|
|
];
|
|
}
|
|
|
|
public buildDom(row: DataRowModel) {
|
|
// Note: we require 2 observables here because changes to the cell value (reference id)
|
|
// and the display value (display column) are not bundled. This can cause `formattedValue`
|
|
// to briefly display incorrect values (e.g. [Blank] when adding a reference to an empty cell)
|
|
// because the cell value changes before the display column has a chance to update.
|
|
//
|
|
// TODO: Look into a better solution (perhaps updating the display formula to return [Blank]).
|
|
const referenceId = Computed.create(null, (use) => {
|
|
const id = row.cells[use(this.field.colId)];
|
|
return id && use(id);
|
|
});
|
|
const formattedValue = Computed.create(null, (use) => {
|
|
let [value, hasBlankReference, hasRecordCard] = ['', false, false];
|
|
if (use(row._isAddRow) || this.isDisposed() || use(this.field.displayColModel).isDisposed()) {
|
|
// Work around JS errors during certain changes (noticed when visibleCol field gets removed
|
|
// for a column using per-field settings).
|
|
return {value, hasBlankReference, hasRecordCard};
|
|
}
|
|
|
|
const displayValueObs = row.cells[use(use(this.field.displayColModel).colId)];
|
|
if (!displayValueObs) {
|
|
return {value, hasBlankReference, hasRecordCard};
|
|
}
|
|
|
|
const displayValue = use(displayValueObs);
|
|
value = isVersions(displayValue) ?
|
|
// We can arrive here if the reference value is unchanged (viewed as a foreign key)
|
|
// but the content of its displayCol has changed. Postponing doing anything about
|
|
// this until we have three-way information for computed columns. For now,
|
|
// just showing one version of the cell. TODO: elaborate.
|
|
use(this.field.formatter).formatAny(displayValue[1].local || displayValue[1].parent) :
|
|
use(this.field.formatter).formatAny(displayValue);
|
|
|
|
hasBlankReference = referenceId.get() !== 0 && value.trim() === '';
|
|
const refTable = use(this._refTable);
|
|
if (refTable) {
|
|
hasRecordCard = !use(use(refTable.recordCardViewSection).disabled);
|
|
}
|
|
|
|
return {value, hasBlankReference, hasRecordCard};
|
|
});
|
|
|
|
return cssRef(
|
|
dom.autoDispose(formattedValue),
|
|
dom.autoDispose(referenceId),
|
|
cssRef.cls('-blank', use => use(formattedValue).hasBlankReference),
|
|
dom.style('text-align', this.alignment),
|
|
dom.cls('text_wrapping', this.wrapping),
|
|
cssRefIcon('FieldReference',
|
|
cssRefIcon.cls('-view-as-card', use =>
|
|
RECORD_CARDS() && use(referenceId) !== 0 && use(formattedValue).hasRecordCard),
|
|
dom.on('click', async () => {
|
|
if (!RECORD_CARDS()) { return; }
|
|
if (referenceId.get() === 0 || !formattedValue.get().hasRecordCard) { return; }
|
|
|
|
const rowId = referenceId.get() as UIRowId;
|
|
const sectionId = this._refTable.get()?.recordCardViewSectionRef();
|
|
if (sectionId === undefined) {
|
|
throw new Error('Unable to find Record Card section');
|
|
}
|
|
|
|
const anchorUrlState = {hash: {rowId, sectionId, recordCard: true}};
|
|
await urlState().pushUrl(anchorUrlState, {replace: true});
|
|
}),
|
|
dom.on('mousedown', (ev) => {
|
|
if (!RECORD_CARDS()) { return; }
|
|
|
|
ev.stopPropagation();
|
|
ev.preventDefault();
|
|
}),
|
|
hideInPrintView(),
|
|
testId('ref-link-icon'),
|
|
),
|
|
dom('span',
|
|
dom.text(use => {
|
|
if (use(referenceId) === 0) { return ''; }
|
|
if (use(formattedValue).hasBlankReference) { return '[Blank]'; }
|
|
return use(formattedValue).value;
|
|
}),
|
|
testId('ref-text'),
|
|
),
|
|
);
|
|
}
|
|
}
|
|
|
|
const cssRefIcon = styled(icon, `
|
|
float: left;
|
|
--icon-color: ${theme.lightText};
|
|
margin: -1px 2px 2px 0;
|
|
|
|
&-view-as-card {
|
|
cursor: pointer;
|
|
}
|
|
&-view-as-card:hover {
|
|
--icon-color: ${theme.controlFg};
|
|
}
|
|
`);
|
|
|
|
const cssRef = styled('div.field_clip', `
|
|
&-blank {
|
|
color: ${theme.lightText}
|
|
}
|
|
`);
|