diff options
Diffstat (limited to 'src/client/views/collections/CollectionSchemaView.tsx')
| -rw-r--r-- | src/client/views/collections/CollectionSchemaView.tsx | 992 | 
1 files changed, 436 insertions, 556 deletions
| diff --git a/src/client/views/collections/CollectionSchemaView.tsx b/src/client/views/collections/CollectionSchemaView.tsx index 56a2a517c..9722f8f26 100644 --- a/src/client/views/collections/CollectionSchemaView.tsx +++ b/src/client/views/collections/CollectionSchemaView.tsx @@ -4,31 +4,27 @@ import { faCog, faPlus, faSortDown, faSortUp, faTable } from '@fortawesome/free-  import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';  import { action, computed, observable, untracked } from "mobx";  import { observer } from "mobx-react"; -import ReactTable, { CellInfo, Column, ComponentPropsGetterR, Resize, SortingRule } from "react-table"; +import { Resize } from "react-table";  import "react-table/react-table.css"; -import { Doc, DocListCast, Field, Opt } from "../../../fields/Doc"; -import { Id } from "../../../fields/FieldSymbols"; +import { Doc } from "../../../fields/Doc";  import { List } from "../../../fields/List";  import { listSpec } from "../../../fields/Schema"; -import { SchemaHeaderField } from "../../../fields/SchemaHeaderField"; -import { ComputedField } from "../../../fields/ScriptField"; -import { Cast, FieldValue, NumCast, StrCast, BoolCast } from "../../../fields/Types"; +import { SchemaHeaderField, PastelSchemaPalette } from "../../../fields/SchemaHeaderField"; +import { Cast, NumCast, StrCast } from "../../../fields/Types";  import { Docs, DocumentOptions } from "../../documents/Documents"; -import { CompileScript, Transformer, ts } from "../../util/Scripting";  import { Transform } from "../../util/Transform";  import { undoBatch } from "../../util/UndoManager";  import { COLLECTION_BORDER_WIDTH } from '../../views/globalCssVariables.scss'; -import { ContextMenu } from "../ContextMenu";  import '../DocumentDecorations.scss'; -import { CellProps, CollectionSchemaCell, CollectionSchemaCheckboxCell, CollectionSchemaDocCell, CollectionSchemaNumberCell, CollectionSchemaStringCell } from "./CollectionSchemaCells"; -import { CollectionSchemaAddColumnHeader, CollectionSchemaHeader } from "./CollectionSchemaHeaders"; -import { MovableColumn, MovableRow } from "./CollectionSchemaMovableTableHOC"; +import { KeysDropdown } from "./CollectionSchemaHeaders";  import "./CollectionSchemaView.scss";  import { CollectionSubView } from "./CollectionSubView"; -import { CollectionView } from "./CollectionView";  import { ContentFittingDocumentView } from "../nodes/ContentFittingDocumentView";  import { setupMoveUpEvents, emptyFunction, returnZero, returnOne, returnFalse } from "../../../Utils";  import { SnappingManager } from "../../util/SnappingManager"; +import Measure from "react-measure"; +import { SchemaTable } from "./SchemaTable"; +import { TraceMobx } from "../../../fields/util";  library.add(faCog, faPlus, faSortUp, faSortDown);  library.add(faTable); @@ -40,6 +36,9 @@ export enum ColumnType {      String,      Boolean,      Doc, +    Image, +    List, +    Date  }  // this map should be used for keys that should have a const type of value  const columnTypes: Map<string, ColumnType> = new Map([ @@ -62,6 +61,359 @@ export class CollectionSchemaView extends CollectionSubView(doc => doc) {      @computed get tableWidth() { return this.props.PanelWidth() - 2 * this.borderWidth - this.DIVIDER_WIDTH - this.previewWidth(); }      @computed get borderWidth() { return Number(COLLECTION_BORDER_WIDTH); } +    @observable _menuWidth = 0; +    @observable _headerOpen = false; +    @observable _isOpen = false; +    @observable _node: HTMLDivElement | null = null; +    @observable _headerIsEditing = false; +    @observable _col: any = ""; +    @observable _menuHeight = 0; +    @observable _pointerX = 0; +    @observable _pointerY = 0; +    @observable _openTypes: boolean = false; +    @computed get menuCoordinates() { +        const x = Math.max(0, Math.min(document.body.clientWidth - this._menuWidth, this._pointerX)); +        const y = Math.max(0, Math.min(document.body.clientHeight - this._menuHeight, this._pointerY)); +        return this.props.ScreenToLocalTransform().transformPoint(x, y); +    } + +    @observable scale = this.props.ScreenToLocalTransform().Scale; + +    @computed get columns() { +        return Cast(this.props.Document._schemaHeaders, listSpec(SchemaHeaderField), []); +    } +    set columns(columns: SchemaHeaderField[]) { +        this.props.Document._schemaHeaders = new List<SchemaHeaderField>(columns); +    } + +    get documentKeys() { +        const docs = this.childDocs; +        const keys: { [key: string]: boolean } = {}; +        // bcz: ugh.  this is untracked since otherwise a large collection of documents will blast the server for all their fields. +        //  then as each document's fields come back, we update the documents _proxies.  Each time we do this, the whole schema will be +        //  invalidated and re-rendered.   This workaround will inquire all of the document fields before the options button is clicked. +        //  then by the time the options button is clicked, all of the fields should be in place.  If a new field is added while this menu +        //  is displayed (unlikely) it won't show up until something else changes. +        //TODO Types +        untracked(() => docs.map(doc => Doc.GetAllPrototypes(doc).map(proto => Object.keys(proto).forEach(key => keys[key] = false)))); + +        this.columns.forEach(key => keys[key.heading] = true); +        return Array.from(Object.keys(keys)); +    } +    @computed get possibleKeys() { return this.documentKeys.filter(key => this.columns.findIndex(existingKey => existingKey.heading.toUpperCase() === key.toUpperCase()) === -1); } + + +    componentDidMount() { +        document.addEventListener("pointerdown", this.detectClick); +    } + +    componentWillUnmount() { +        document.removeEventListener("pointerdown", this.detectClick); +    } + +    @action setHeaderIsEditing = (isEditing: boolean) => this._headerIsEditing = isEditing; + +    detectClick = (e: PointerEvent): void => { +        if (this._node && this._node.contains(e.target as Node)) { +        } else { +            this._isOpen = false; +            this.setHeaderIsEditing(false); +            this.closeHeader(); +        } +    } + +    @action +    toggleIsOpen = (): void => { +        this._isOpen = !this._isOpen; +        this.setHeaderIsEditing(this._isOpen); +    } + +    @action +    changeColumnType = (type: ColumnType, col: any): void => { +        this._openTypes = false; +        this.setColumnType(col, type); +    } + +    changeColumnSort = (desc: boolean | undefined, col: any): void => { +        this.setColumnSort(col, desc); +    } + +    changeColumnColor = (color: string, col: any): void => { +        this.setColumnColor(col, color); +    } + +    @undoBatch +    setColumnType = (columnField: SchemaHeaderField, type: ColumnType): void => { +        if (columnTypes.get(columnField.heading)) return; + +        const columns = this.columns; +        const index = columns.indexOf(columnField); +        if (index > -1) { +            columnField.setType(NumCast(type)); +            columns[index] = columnField; +            this.columns = columns; +        } +    } + +    @undoBatch +    setColumnColor = (columnField: SchemaHeaderField, color: string): void => { +        const columns = this.columns; +        const index = columns.indexOf(columnField); +        if (index > -1) { +            columnField.setColor(color); +            columns[index] = columnField; +            this.columns = columns; // need to set the columns to trigger rerender +        } +    } + +    @undoBatch +    @action +    setColumnSort = (columnField: SchemaHeaderField, descending: boolean | undefined) => { +        const columns = this.columns; +        const index = columns.findIndex(c => c.heading === columnField.heading); +        const column = columns[index]; +        column.setDesc(descending); +        columns[index] = column; +        this.columns = columns; +    } + +    @action +    setNode = (node: HTMLDivElement): void => { +        node && (this._node = node); +    } + +    @action +    typesDropdownChange = (bool: boolean) => { +        this._openTypes = bool; +    } + +    renderTypes = (col: any) => { +        if (columnTypes.get(col.heading)) return (null); + +        const type = col.type; + +        const anyType = <div className={"columnMenu-option" + (type === ColumnType.Any ? " active" : "")} onClick={() => this.changeColumnType(ColumnType.Any, col)}> +            <FontAwesomeIcon icon={"align-justify"} size="sm" /> +                Any +            </div>; + +        const numType = <div className={"columnMenu-option" + (type === ColumnType.Number ? " active" : "")} onClick={() => this.changeColumnType(ColumnType.Number, col)}> +            <FontAwesomeIcon icon={"hashtag"} size="sm" /> +                Number +            </div>; + +        const textType = <div className={"columnMenu-option" + (type === ColumnType.String ? " active" : "")} onClick={() => this.changeColumnType(ColumnType.String, col)}> +            <FontAwesomeIcon icon={"font"} size="sm" /> +            Text +            </div>; + +        const boolType = <div className={"columnMenu-option" + (type === ColumnType.Boolean ? " active" : "")} onClick={() => this.changeColumnType(ColumnType.Boolean, col)}> +            <FontAwesomeIcon icon={"check-square"} size="sm" /> +            Checkbox +            </div>; + +        const listType = <div className={"columnMenu-option" + (type === ColumnType.List ? " active" : "")} onClick={() => this.changeColumnType(ColumnType.List, col)}> +            <FontAwesomeIcon icon={"list-ul"} size="sm" /> +            List +            </div>; + +        const docType = <div className={"columnMenu-option" + (type === ColumnType.Doc ? " active" : "")} onClick={() => this.changeColumnType(ColumnType.Doc, col)}> +            <FontAwesomeIcon icon={"file"} size="sm" /> +            Document +            </div>; + +        const imageType = <div className={"columnMenu-option" + (type === ColumnType.Image ? " active" : "")} onClick={() => this.changeColumnType(ColumnType.Image, col)}> +            <FontAwesomeIcon icon={"image"} size="sm" /> +            Image +            </div>; + +        const dateType = <div className={"columnMenu-option" + (type === ColumnType.Date ? " active" : "")} onClick={() => this.changeColumnType(ColumnType.Date, col)}> +            <FontAwesomeIcon icon={"calendar"} size="sm" /> +                Date +                </div>; + + +        const allColumnTypes = <div className="columnMenu-types"> +            {anyType} +            {numType} +            {textType} +            {boolType} +            {listType} +            {docType} +            {imageType} +            {dateType} +        </div>; + +        const justColType = type === ColumnType.Any ? anyType : type === ColumnType.Number ? numType : +            type === ColumnType.String ? textType : type === ColumnType.Boolean ? boolType : +                type === ColumnType.List ? listType : type === ColumnType.Doc ? docType : +                    type === ColumnType.Date ? dateType : imageType; + +        return ( +            <div className="collectionSchema-headerMenu-group"> +                <div onClick={() => this.typesDropdownChange(!this._openTypes)}> +                    <label>Column type:</label> +                    <FontAwesomeIcon icon={"caret-down"} size="sm" style={{ float: "right" }} /> +                </div> +                {this._openTypes ? allColumnTypes : justColType} +            </div > +        ); +    } + +    renderSorting = (col: any) => { +        const sort = col.desc; +        return ( +            <div className="collectionSchema-headerMenu-group"> +                <label>Sort by:</label> +                <div className="columnMenu-sort"> +                    <div className={"columnMenu-option" + (sort === true ? " active" : "")} onClick={() => this.changeColumnSort(true, col)}> +                        <FontAwesomeIcon icon="sort-amount-down" size="sm" /> +                        Sort descending +                    </div> +                    <div className={"columnMenu-option" + (sort === false ? " active" : "")} onClick={() => this.changeColumnSort(false, col)}> +                        <FontAwesomeIcon icon="sort-amount-up" size="sm" /> +                        Sort ascending +                    </div> +                    <div className="columnMenu-option" onClick={() => this.changeColumnSort(undefined, col)}> +                        <FontAwesomeIcon icon="times" size="sm" /> +                        Clear sorting +                    </div> +                </div> +            </div> +        ); +    } + +    renderColors = (col: any) => { +        const selected = col.color; + +        const pink = PastelSchemaPalette.get("pink2"); +        const purple = PastelSchemaPalette.get("purple2"); +        const blue = PastelSchemaPalette.get("bluegreen1"); +        const yellow = PastelSchemaPalette.get("yellow4"); +        const red = PastelSchemaPalette.get("red2"); +        const gray = "#f1efeb"; + +        return ( +            <div className="collectionSchema-headerMenu-group"> +                <label>Color:</label> +                <div className="columnMenu-colors"> +                    <div className={"columnMenu-colorPicker" + (selected === pink ? " active" : "")} style={{ backgroundColor: pink }} onClick={() => this.changeColumnColor(pink!, col)}></div> +                    <div className={"columnMenu-colorPicker" + (selected === purple ? " active" : "")} style={{ backgroundColor: purple }} onClick={() => this.changeColumnColor(purple!, col)}></div> +                    <div className={"columnMenu-colorPicker" + (selected === blue ? " active" : "")} style={{ backgroundColor: blue }} onClick={() => this.changeColumnColor(blue!, col)}></div> +                    <div className={"columnMenu-colorPicker" + (selected === yellow ? " active" : "")} style={{ backgroundColor: yellow }} onClick={() => this.changeColumnColor(yellow!, col)}></div> +                    <div className={"columnMenu-colorPicker" + (selected === red ? " active" : "")} style={{ backgroundColor: red }} onClick={() => this.changeColumnColor(red!, col)}></div> +                    <div className={"columnMenu-colorPicker" + (selected === gray ? " active" : "")} style={{ backgroundColor: gray }} onClick={() => this.changeColumnColor(gray, col)}></div> +                </div> +            </div> +        ); +    } + +    @undoBatch +    @action +    changeColumns = (oldKey: string, newKey: string, addNew: boolean, filter?: string) => { +        console.log("COL"); +        const columns = this.columns; +        if (columns === undefined) { +            this.columns = new List<SchemaHeaderField>([new SchemaHeaderField(newKey, "f1efeb")]); +        } else { +            if (addNew) { +                columns.push(new SchemaHeaderField(newKey, "f1efeb")); +                this.columns = columns; +            } else { +                const index = columns.map(c => c.heading).indexOf(oldKey); +                if (index > -1) { +                    const column = columns[index]; +                    column.setHeading(newKey); +                    columns[index] = column; +                    this.columns = columns; +                    if (filter) { +                        console.log(newKey); +                        console.log(filter); +                        Doc.setDocFilter(this.props.Document, newKey, filter, "match"); +                    } +                    else { +                        this.props.Document._docFilters = undefined; +                    } +                } +            } +        } +    } + +    @action +    openHeader = (col: any, screenx: number, screeny: number) => { +        console.log("header opening"); +        this._col = col; +        this._headerOpen = !this._headerOpen; +        this._pointerX = screenx; +        this._pointerY = screeny; +    } + +    @action +    closeHeader = () => { this._headerOpen = false; } + +    renderKeysDropDown = (col: any) => { +        return <KeysDropdown +            keyValue={col.heading} +            possibleKeys={this.possibleKeys} +            existingKeys={this.columns.map(c => c.heading)} +            canAddNew={true} +            addNew={false} +            onSelect={this.changeColumns} +            setIsEditing={this.setHeaderIsEditing} +        />; +    } + +    @undoBatch +    @action +    deleteColumn = (key: string) => { +        const columns = this.columns; +        if (columns === undefined) { +            this.columns = new List<SchemaHeaderField>([]); +        } else { +            const index = columns.map(c => c.heading).indexOf(key); +            if (index > -1) { +                columns.splice(index, 1); +                this.columns = columns; +            } +        } +        this.closeHeader(); +    } + +    getPreviewTransform = (): Transform => { +        return this.props.ScreenToLocalTransform().translate(- this.borderWidth - NumCast(COLLECTION_BORDER_WIDTH) - this.tableWidth, - this.borderWidth); +    } + +    @action +    onHeaderClick = (e: React.PointerEvent) => { +        this.props.active(true); +        e.stopPropagation(); +    } + +    @action +    onWheel(e: React.WheelEvent) { +        const scale = this.props.ScreenToLocalTransform().Scale; +        this.props.active(true) && e.stopPropagation(); +        //this.menuCoordinates[0] -= e.screenX / scale; +        //this.menuCoordinates[1] -= e.screenY / scale; +    } + +    @computed get renderMenuContent() { +        TraceMobx(); +        return <div className="collectionSchema-header-menuOptions"> +            <div className="collectionSchema-headerMenu-group"> +                <label>Key:</label> +                {this.renderKeysDropDown(this._col)} +            </div> +            {this.renderTypes(this._col)} +            {this.renderSorting(this._col)} +            {this.renderColors(this._col)} +            <div className="collectionSchema-headerMenu-group"> +                <button onClick={() => { this.deleteColumn(this._col.heading); }} +                >Delete Column</button> +            </div> +        </div>; +    } +      private createTarget = (ele: HTMLDivElement) => {          this._previewCont = ele;          super.CreateDropTarget(ele); @@ -105,14 +457,12 @@ export class CollectionSchemaView extends CollectionSubView(doc => doc) {      @computed      get previewDocument(): Doc | undefined { return this.previewDoc; } -    getPreviewTransform = (): Transform => { -        return this.props.ScreenToLocalTransform().translate(- this.borderWidth - this.DIVIDER_WIDTH - this.tableWidth, - this.borderWidth); -    } -      @computed      get dividerDragger() {          return this.previewWidth() === 0 ? (null) : -            <div className="collectionSchemaView-dividerDragger" onPointerDown={this.onDividerDown} style={{ width: `${this.DIVIDER_WIDTH}px` }} />; +            <div className="collectionSchemaView-dividerDragger" +                onPointerDown={this.onDividerDown} +                style={{ width: `${this.DIVIDER_WIDTH}px` }} />;      }      @computed @@ -174,6 +524,17 @@ export class CollectionSchemaView extends CollectionSubView(doc => doc) {              deleteDocument={this.props.removeDocument}              addDocument={this.props.addDocument}              dataDoc={this.props.DataDoc} +            columns={this.columns} +            documentKeys={this.documentKeys} +            headerIsEditing={this._headerIsEditing} +            openHeader={this.openHeader} +            onPointerDown={this.onTablePointerDown} +            onResizedChange={this.onResizedChange} +            setColumns={this.setColumns} +            reorderColumns={this.reorderColumns} +            changeColumns={this.changeColumns} +            setHeaderIsEditing={this.setHeaderIsEditing} +            changeColumnSort={this.setColumnSort}          />;      } @@ -181,388 +542,33 @@ export class CollectionSchemaView extends CollectionSubView(doc => doc) {      public get schemaToolbar() {          return <div className="collectionSchemaView-toolbar">              <div className="collectionSchemaView-toolbar-item"> -                <div id="preview-schema-checkbox-div"><input type="checkbox" key={"Show Preview"} checked={this.previewWidth() !== 0} onChange={this.toggleExpander} />Show Preview</div> +                <div id="preview-schema-checkbox-div"> +                    <input type="checkbox" +                        key={"Show Preview"} checked={this.previewWidth() !== 0} +                        onChange={this.toggleExpander} />Show Preview</div>              </div>          </div>;      } -    render() { -        return <div className="collectionSchemaView-container" -            style={{ -                pointerEvents: !this.props.active() && !SnappingManager.GetIsDragging() ? "none" : undefined, -                width: this.props.PanelWidth() || "100%", height: this.props.PanelHeight() || "100%" -            }}  > -            <div className="collectionSchemaView-tableContainer" style={{ width: `calc(100% - ${this.previewWidth()}px)` }} onPointerDown={this.onPointerDown} onWheel={e => this.props.active(true) && e.stopPropagation()} onDrop={e => this.onExternalDrop(e, {})} ref={this.createTarget}> -                {this.schemaTable} -            </div> -            {this.dividerDragger} -            {!this.previewWidth() ? (null) : this.previewPanel} -        </div>; -    } -} - -export interface SchemaTableProps { -    Document: Doc; // child doc -    dataDoc?: Doc; -    PanelHeight: () => number; -    PanelWidth: () => number; -    childDocs?: Doc[]; -    CollectionView: Opt<CollectionView>; -    ContainingCollectionView: Opt<CollectionView>; -    ContainingCollectionDoc: Opt<Doc>; -    fieldKey: string; -    renderDepth: number; -    deleteDocument: (document: Doc | Doc[]) => boolean; -    addDocument: (document: Doc | Doc[]) => boolean; -    moveDocument: (document: Doc | Doc[], targetCollection: Doc | undefined, addDocument: (document: Doc | Doc[]) => boolean) => boolean; -    ScreenToLocalTransform: () => Transform; -    active: (outsideReaction: boolean) => boolean; -    onDrop: (e: React.DragEvent<Element>, options: DocumentOptions, completed?: (() => void) | undefined) => void; -    addDocTab: (document: Doc, where: string) => boolean; -    pinToPres: (document: Doc) => void; -    isSelected: (outsideReaction?: boolean) => boolean; -    isFocused: (document: Doc) => boolean; -    setFocused: (document: Doc) => void; -    setPreviewDoc: (document: Doc) => void; -} - -@observer -export class SchemaTable extends React.Component<SchemaTableProps> { -    private DIVIDER_WIDTH = 4; - -    @observable _headerIsEditing: boolean = false; -    @observable _cellIsEditing: boolean = false; -    @observable _focusedCell: { row: number, col: number } = { row: 0, col: 0 }; -    @observable _openCollections: Array<string> = []; - -    @computed get previewWidth() { return () => NumCast(this.props.Document.schemaPreviewWidth); } -    @computed get previewHeight() { return () => this.props.PanelHeight() - 2 * this.borderWidth; } -    @computed get tableWidth() { return this.props.PanelWidth() - 2 * this.borderWidth - this.DIVIDER_WIDTH - this.previewWidth(); } - -    @computed get columns() { -        return Cast(this.props.Document.schemaColumns, listSpec(SchemaHeaderField), []); -    } -    set columns(columns: SchemaHeaderField[]) { -        this.props.Document.schemaColumns = new List<SchemaHeaderField>(columns); -    } - -    @computed get childDocs() { -        if (this.props.childDocs) return this.props.childDocs; - -        const doc = this.props.dataDoc ? this.props.dataDoc : this.props.Document; -        return DocListCast(doc[this.props.fieldKey]); -    } -    set childDocs(docs: Doc[]) { -        const doc = this.props.dataDoc ? this.props.dataDoc : this.props.Document; -        doc[this.props.fieldKey] = new List<Doc>(docs); -    } - -    @computed get textWrappedRows() { -        return Cast(this.props.Document.textwrappedSchemaRows, listSpec("string"), []); -    } -    set textWrappedRows(textWrappedRows: string[]) { -        this.props.Document.textwrappedSchemaRows = new List<string>(textWrappedRows); -    } - -    @computed get resized(): { id: string, value: number }[] { -        return this.columns.reduce((resized, shf) => { -            (shf.width > -1) && resized.push({ id: shf.heading, value: shf.width }); -            return resized; -        }, [] as { id: string, value: number }[]); -    } -    @computed get sorted(): SortingRule[] { -        return this.columns.reduce((sorted, shf) => { -            shf.desc && sorted.push({ id: shf.heading, desc: shf.desc }); -            return sorted; -        }, [] as SortingRule[]); -    } - -    @computed get borderWidth() { return Number(COLLECTION_BORDER_WIDTH); } -    @computed get tableColumns(): Column<Doc>[] { -        const possibleKeys = this.documentKeys.filter(key => this.columns.findIndex(existingKey => existingKey.heading.toUpperCase() === key.toUpperCase()) === -1); -        const columns: Column<Doc>[] = []; -        const tableIsFocused = this.props.isFocused(this.props.Document); -        const focusedRow = this._focusedCell.row; -        const focusedCol = this._focusedCell.col; -        const isEditable = !this._headerIsEditing; - -        if (this.childDocs.reduce((found, doc) => found || doc.type === "collection", false)) { -            columns.push( -                { -                    expander: true, -                    Header: "", -                    width: 30, -                    Expander: (rowInfo) => { -                        if (rowInfo.original.type === "collection") { -                            if (rowInfo.isExpanded) return <div className="collectionSchemaView-expander" onClick={() => this.onCloseCollection(rowInfo.original)}><FontAwesomeIcon icon={"sort-up"} size="sm" /></div>; -                            if (!rowInfo.isExpanded) return <div className="collectionSchemaView-expander" onClick={() => this.onExpandCollection(rowInfo.original)}><FontAwesomeIcon icon={"sort-down"} size="sm" /></div>; -                        } else { -                            return null; -                        } -                    } -                } -            ); -        } - -        const cols = this.columns.map(col => { -            const header = <CollectionSchemaHeader -                keyValue={col} -                possibleKeys={possibleKeys} -                existingKeys={this.columns.map(c => c.heading)} -                keyType={this.getColumnType(col)} -                typeConst={columnTypes.get(col.heading) !== undefined} -                onSelect={this.changeColumns} -                setIsEditing={this.setHeaderIsEditing} -                deleteColumn={this.deleteColumn} -                setColumnType={this.setColumnType} -                setColumnSort={this.setColumnSort} -                setColumnColor={this.setColumnColor} -            />; - -            return { -                Header: <MovableColumn columnRenderer={header} columnValue={col} allColumns={this.columns} reorderColumns={this.reorderColumns} ScreenToLocalTransform={this.props.ScreenToLocalTransform} />, -                accessor: (doc: Doc) => doc ? doc[col.heading] : 0, -                id: col.heading, -                Cell: (rowProps: CellInfo) => { -                    const rowIndex = rowProps.index; -                    const columnIndex = this.columns.map(c => c.heading).indexOf(rowProps.column.id!); -                    const isFocused = focusedRow === rowIndex && focusedCol === columnIndex && tableIsFocused; - -                    const props: CellProps = { -                        row: rowIndex, -                        col: columnIndex, -                        rowProps: rowProps, -                        isFocused: isFocused, -                        changeFocusedCellByIndex: this.changeFocusedCellByIndex, -                        CollectionView: this.props.CollectionView, -                        ContainingCollection: this.props.ContainingCollectionView, -                        Document: this.props.Document, -                        fieldKey: this.props.fieldKey, -                        renderDepth: this.props.renderDepth, -                        addDocTab: this.props.addDocTab, -                        pinToPres: this.props.pinToPres, -                        moveDocument: this.props.moveDocument, -                        setIsEditing: this.setCellIsEditing, -                        isEditable: isEditable, -                        setPreviewDoc: this.props.setPreviewDoc, -                        setComputed: this.setComputed, -                        getField: this.getField, -                    }; - -                    const colType = this.getColumnType(col); -                    if (colType === ColumnType.Number) return <CollectionSchemaNumberCell {...props} />; -                    if (colType === ColumnType.String) return <CollectionSchemaStringCell {...props} />; -                    if (colType === ColumnType.Boolean) return <CollectionSchemaCheckboxCell {...props} />; -                    if (colType === ColumnType.Doc) return <CollectionSchemaDocCell {...props} />; -                    return <CollectionSchemaCell {...props} />; -                }, -                minWidth: 200, -            }; -        }); -        columns.push(...cols); - -        columns.push({ -            Header: <CollectionSchemaAddColumnHeader createColumn={this.createColumn} />, -            accessor: (doc: Doc) => 0, -            id: "add", -            Cell: (rowProps: CellInfo) => <></>, -            width: 28, -            resizable: false -        }); -        return columns; -    } - -    constructor(props: SchemaTableProps) { -        super(props); -        // convert old schema columns (list of strings) into new schema columns (list of schema header fields) -        const oldSchemaColumns = Cast(this.props.Document.schemaColumns, listSpec("string"), []); -        if (oldSchemaColumns && oldSchemaColumns.length && typeof oldSchemaColumns[0] !== "object") { -            const newSchemaColumns = oldSchemaColumns.map(i => typeof i === "string" ? new SchemaHeaderField(i, "#f1efeb") : i); -            this.props.Document.schemaColumns = new List<SchemaHeaderField>(newSchemaColumns); -        } -    } - -    componentDidMount() { -        document.addEventListener("keydown", this.onKeyDown); -    } - -    componentWillUnmount() { -        document.removeEventListener("keydown", this.onKeyDown); -    } - -    tableAddDoc = (doc: Doc, relativeTo?: Doc, before?: boolean) => { -        return Doc.AddDocToList(this.props.Document, this.props.fieldKey, doc, relativeTo, before); -    } - -    private getTrProps: ComponentPropsGetterR = (state, rowInfo) => { -        return !rowInfo ? {} : { -            ScreenToLocalTransform: this.props.ScreenToLocalTransform, -            addDoc: this.tableAddDoc, -            removeDoc: this.props.deleteDocument, -            rowInfo, -            rowFocused: !this._headerIsEditing && rowInfo.index === this._focusedCell.row && this.props.isFocused(this.props.Document), -            textWrapRow: this.toggleTextWrapRow, -            rowWrapped: this.textWrappedRows.findIndex(id => rowInfo.original[Id] === id) > -1, -            dropAction: StrCast(this.props.Document.childDropAction) -        }; -    } - -    private getTdProps: ComponentPropsGetterR = (state, rowInfo, column, instance) => { -        if (!rowInfo || column) return {}; - -        const row = rowInfo.index; -        //@ts-ignore -        const col = this.columns.map(c => c.heading).indexOf(column!.id); -        const isFocused = this._focusedCell.row === row && this._focusedCell.col === col && this.props.isFocused(this.props.Document); -        // TODO: editing border doesn't work :( -        return { -            style: { -                border: !this._headerIsEditing && isFocused ? "2px solid rgb(255, 160, 160)" : "1px solid #f1efeb" -            } -        }; -    } -      @action -    onCloseCollection = (collection: Doc): void => { -        const index = this._openCollections.findIndex(col => col === collection[Id]); -        if (index > -1) this._openCollections.splice(index, 1); -    } - -    @action onExpandCollection = (collection: Doc) => this._openCollections.push(collection[Id]); -    @action setCellIsEditing = (isEditing: boolean) => this._cellIsEditing = isEditing; -    @action setHeaderIsEditing = (isEditing: boolean) => this._headerIsEditing = isEditing; - -    onPointerDown = (e: React.PointerEvent): void => { -        this.props.setFocused(this.props.Document); +    onTablePointerDown = (e: React.PointerEvent): void => { +        this.setFocused(this.props.Document);          if (e.button === 0 && !e.altKey && !e.ctrlKey && !e.metaKey && this.props.isSelected(true)) {              e.stopPropagation();          } +        this._pointerY = e.screenY; +        this._pointerX = e.screenX;      } -    @action -    onKeyDown = (e: KeyboardEvent): void => { -        if (!this._cellIsEditing && !this._headerIsEditing && this.props.isFocused(this.props.Document)) {// && this.props.isSelected(true)) { -            const direction = e.key === "Tab" ? "tab" : e.which === 39 ? "right" : e.which === 37 ? "left" : e.which === 38 ? "up" : e.which === 40 ? "down" : ""; -            this._focusedCell = this.changeFocusedCellByDirection(direction, this._focusedCell.row, this._focusedCell.col); - -            const pdoc = FieldValue(this.childDocs[this._focusedCell.row]); -            pdoc && this.props.setPreviewDoc(pdoc); -        } -    } - -    changeFocusedCellByDirection = (direction: string, curRow: number, curCol: number) => { -        switch (direction) { -            case "tab": return { row: (curRow + 1 === this.childDocs.length ? 0 : curRow + 1), col: curCol + 1 === this.columns.length ? 0 : curCol + 1 }; -            case "right": return { row: curRow, col: curCol + 1 === this.columns.length ? curCol : curCol + 1 }; -            case "left": return { row: curRow, col: curCol === 0 ? curCol : curCol - 1 }; -            case "up": return { row: curRow === 0 ? curRow : curRow - 1, col: curCol }; -            case "down": return { row: curRow + 1 === this.childDocs.length ? curRow : curRow + 1, col: curCol }; -        } -        return this._focusedCell; -    } - -    @action -    changeFocusedCellByIndex = (row: number, col: number): void => { -        if (this._focusedCell.row !== row || this._focusedCell.col !== col) { -            this._focusedCell = { row: row, col: col }; -        } -        this.props.setFocused(this.props.Document); -    } - -    @undoBatch -    createRow = () => { -        this.props.addDocument(Docs.Create.TextDocument("", { title: "", _width: 100, _height: 30 })); -    } - -    @undoBatch -    @action -    createColumn = () => { -        let index = 0; -        let found = this.columns.findIndex(col => col.heading.toUpperCase() === "New field".toUpperCase()) > -1; -        while (found) { -            index++; -            found = this.columns.findIndex(col => col.heading.toUpperCase() === ("New field (" + index + ")").toUpperCase()) > -1; -        } -        this.columns.push(new SchemaHeaderField(`New field ${index ? "(" + index + ")" : ""}`, "#f1efeb")); -    } - -    @undoBatch -    @action -    deleteColumn = (key: string) => { -        const columns = this.columns; -        if (columns === undefined) { -            this.columns = new List<SchemaHeaderField>([]); -        } else { -            const index = columns.map(c => c.heading).indexOf(key); -            if (index > -1) { -                columns.splice(index, 1); -                this.columns = columns; -            } -        } -    } - -    @undoBatch -    @action -    changeColumns = (oldKey: string, newKey: string, addNew: boolean) => { -        const columns = this.columns; -        if (columns === undefined) { -            this.columns = new List<SchemaHeaderField>([new SchemaHeaderField(newKey, "f1efeb")]); -        } else { -            if (addNew) { -                columns.push(new SchemaHeaderField(newKey, "f1efeb")); -                this.columns = columns; -            } else { -                const index = columns.map(c => c.heading).indexOf(oldKey); -                if (index > -1) { -                    const column = columns[index]; -                    column.setHeading(newKey); -                    columns[index] = column; -                    this.columns = columns; -                } -            } -        } -    } - -    getColumnType = (column: SchemaHeaderField): ColumnType => { -        // added functionality to convert old column type stuff to new column type stuff -syip -        if (column.type && column.type !== 0) { -            return column.type; -        } -        if (columnTypes.get(column.heading)) { -            column.type = columnTypes.get(column.heading)!; -            return columnTypes.get(column.heading)!; -        } -        const typesDoc = FieldValue(Cast(this.props.Document.schemaColumnTypes, Doc)); -        if (!typesDoc) { -            column.type = ColumnType.Any; -            return ColumnType.Any; -        } -        column.type = NumCast(typesDoc[column.heading]); -        return NumCast(typesDoc[column.heading]); -    } - -    @undoBatch -    setColumnType = (columnField: SchemaHeaderField, type: ColumnType): void => { -        if (columnTypes.get(columnField.heading)) return; - -        const columns = this.columns; -        const index = columns.indexOf(columnField); -        if (index > -1) { -            columnField.setType(NumCast(type)); -            columns[index] = columnField; -            this.columns = columns; -        } -    } - -    @undoBatch -    setColumnColor = (columnField: SchemaHeaderField, color: string): void => { +    onResizedChange = (newResized: Resize[], event: any) => {          const columns = this.columns; -        const index = columns.indexOf(columnField); -        if (index > -1) { -            columnField.setColor(color); -            columns[index] = columnField; -            this.columns = columns; // need to set the columns to trigger rerender -        } +        newResized.forEach(resized => { +            const index = columns.findIndex(c => c.heading === resized.id); +            const column = columns[index]; +            column.setWidth(resized.value); +            columns[index] = column; +        }); +        this.columns = columns;      }      @action @@ -581,180 +587,54 @@ export class SchemaTable extends React.Component<SchemaTableProps> {          this.columns = columns;      } -    @undoBatch -    @action -    setColumnSort = (columnField: SchemaHeaderField, descending: boolean | undefined) => { -        const columns = this.columns; -        const index = columns.findIndex(c => c.heading === columnField.heading); -        const column = columns[index]; -        column.setDesc(descending); -        columns[index] = column; -        this.columns = columns; -    } - -    get documentKeys() { -        const docs = this.childDocs; -        const keys: { [key: string]: boolean } = {}; -        // bcz: ugh.  this is untracked since otherwise a large collection of documents will blast the server for all their fields. -        //  then as each document's fields come back, we update the documents _proxies.  Each time we do this, the whole schema will be -        //  invalidated and re-rendered.   This workaround will inquire all of the document fields before the options button is clicked. -        //  then by the time the options button is clicked, all of the fields should be in place.  If a new field is added while this menu -        //  is displayed (unlikely) it won't show up until something else changes. -        //TODO Types -        untracked(() => docs.map(doc => Doc.GetAllPrototypes(doc).map(proto => Object.keys(proto).forEach(key => keys[key] = false)))); - -        this.columns.forEach(key => keys[key.heading] = true); -        return Array.from(Object.keys(keys)); -    } - -    @undoBatch -    @action -    toggleTextwrap = async () => { -        const textwrappedRows = Cast(this.props.Document.textwrappedSchemaRows, listSpec("string"), []); -        if (textwrappedRows.length) { -            this.props.Document.textwrappedSchemaRows = new List<string>([]); +    onZoomMenu = (e: React.WheelEvent) => { +        this.props.active(true) && e.stopPropagation(); +        if (this.menuCoordinates[0] > e.screenX) { +            this.menuCoordinates[0] -= e.screenX; //* this.scale;          } else { -            const docs = DocListCast(this.props.Document[this.props.fieldKey]); -            const allRows = docs instanceof Doc ? [docs[Id]] : docs.map(doc => doc[Id]); -            this.props.Document.textwrappedSchemaRows = new List<string>(allRows); -        } -    } - -    @action -    toggleTextWrapRow = (doc: Doc): void => { -        const textWrapped = this.textWrappedRows; -        const index = textWrapped.findIndex(id => doc[Id] === id); - -        index > -1 ? textWrapped.splice(index, 1) : textWrapped.push(doc[Id]); - -        this.textWrappedRows = textWrapped; -    } - -    @computed -    get reactTable() { -        const children = this.childDocs; -        const hasCollectionChild = children.reduce((found, doc) => found || doc.type === "collection", false); -        const expandedRowsList = this._openCollections.map(col => children.findIndex(doc => doc[Id] === col).toString()); -        const expanded = {}; -        //@ts-ignore -        expandedRowsList.forEach(row => expanded[row] = true); -        const rerender = [...this.textWrappedRows]; // TODO: get component to rerender on text wrap change without needign to console.log :(((( - -        return <ReactTable -            style={{ position: "relative" }} -            data={children} -            page={0} -            pageSize={children.length} -            showPagination={false} -            columns={this.tableColumns} -            getTrProps={this.getTrProps} -            getTdProps={this.getTdProps} -            sortable={false} -            TrComponent={MovableRow} -            sorted={this.sorted} -            expanded={expanded} -            resized={this.resized} -            onResizedChange={this.onResizedChange} -            SubComponent={!hasCollectionChild ? undefined : row => (row.original.type !== "collection") ? (null) : -                <div className="reactTable-sub"><SchemaTable {...this.props} Document={row.original} dataDoc={undefined} childDocs={undefined} /></div>} - -        />; -    } - -    onResizedChange = (newResized: Resize[], event: any) => { -        const columns = this.columns; -        newResized.forEach(resized => { -            const index = columns.findIndex(c => c.heading === resized.id); -            const column = columns[index]; -            column.setWidth(resized.value); -            columns[index] = column; -        }); -        this.columns = columns; -    } - -    onContextMenu = (e: React.MouseEvent): void => { -        if (!e.isPropagationStopped() && this.props.Document[Id] !== "mainDoc") { // need to test this because GoldenLayout causes a parallel hierarchy in the React DOM for its children and the main document view7 -            // ContextMenu.Instance.addItem({ description: "Make DB", event: this.makeDB, icon: "table" }); -            ContextMenu.Instance.addItem({ description: "Toggle text wrapping", event: this.toggleTextwrap, icon: "table" }); -        } -    } - -    getField = (row: number, col?: number) => { -        const docs = this.childDocs; - -        row = row % docs.length; -        while (row < 0) row += docs.length; -        const columns = this.columns; -        const doc = docs[row]; -        if (col === undefined) { -            return doc; +            this.menuCoordinates[0] += e.screenX; //* this.scale;          } -        if (col >= 0 && col < columns.length) { -            const column = this.columns[col].heading; -            return doc[column]; -        } -        return undefined; -    } - -    createTransformer = (row: number, col: number): Transformer => { -        const self = this; -        const captures: { [name: string]: Field } = {}; - -        const transformer: ts.TransformerFactory<ts.SourceFile> = context => { -            return root => { -                function visit(node: ts.Node) { -                    node = ts.visitEachChild(node, visit, context); -                    if (ts.isIdentifier(node)) { -                        const isntPropAccess = !ts.isPropertyAccessExpression(node.parent) || node.parent.expression === node; -                        const isntPropAssign = !ts.isPropertyAssignment(node.parent) || node.parent.name !== node; -                        if (isntPropAccess && isntPropAssign) { -                            if (node.text === "$r") { -                                return ts.createNumericLiteral(row.toString()); -                            } else if (node.text === "$c") { -                                return ts.createNumericLiteral(col.toString()); -                            } else if (node.text === "$") { -                                if (ts.isCallExpression(node.parent)) { -                                    // captures.doc = self.props.Document; -                                    // captures.key = self.props.fieldKey; -                                } -                            } -                        } -                    } - -                    return node; -                } -                return ts.visitNode(root, visit); -            }; -        }; - -        // const getVars = () => { -        //     return { capturedVariables: captures }; -        // }; - -        return { transformer, /*getVars*/ }; -    } - -    setComputed = (script: string, doc: Doc, field: string, row: number, col: number): boolean => { -        script = -            `const $ = (row:number, col?:number) => { -                if(col === undefined) { -                    return (doc as any)[key][row + ${row}]; -                } -                return (doc as any)[key][row + ${row}][(doc as any).schemaColumns[col + ${col}].heading]; -            } -            return ${script}`; -        const compiled = CompileScript(script, { params: { this: Doc.name }, capturedVariables: { doc: this.props.Document, key: this.props.fieldKey }, typecheck: false, transformer: this.createTransformer(row, col) }); -        if (compiled.compiled) { -            doc[field] = new ComputedField(compiled); -            return true; +        if (this.menuCoordinates[1] > e.screenY) { +            this.menuCoordinates[1] -= e.screenY; //* this.scale; +        } else { +            this.menuCoordinates[1] += e.screenY; //* this.scale;          } -        return false;      }      render() { -        return <div className="collectionSchemaView-table" onPointerDown={this.onPointerDown} onWheel={e => this.props.active(true) && e.stopPropagation()} onDrop={e => this.props.onDrop(e, {})} onContextMenu={this.onContextMenu} > -            {this.reactTable} -            <div className="collectionSchemaView-addRow" onClick={() => this.createRow()}>+ new</div> +        TraceMobx(); +        const menuContent = this.renderMenuContent; +        const menu = <div className="collectionSchema-header-menu" ref={this.setNode} +            onWheel={e => this.onZoomMenu(e)} +            onPointerDown={e => this.onHeaderClick(e)} +            style={{ +                position: "fixed", background: "white", +                transform: `translate(${this.menuCoordinates[0] / this.scale}px, ${this.menuCoordinates[1] / this.scale}px)` +            }}> +            <Measure offset onResize={action((r: any) => { +                const dim = this.props.ScreenToLocalTransform().inverse().transformDirection(r.offset.width, r.offset.height); +                this._menuWidth = dim[0]; this._menuHeight = dim[1]; +            })}> +                {({ measureRef }) => <div ref={measureRef}> {menuContent} </div>} +            </Measure> +        </div>; + +        return <div className="collectionSchemaView-container" +            style={{ +                pointerEvents: !this.props.active() && !SnappingManager.GetIsDragging() ? "none" : undefined, +                width: this.props.PanelWidth() || "100%", height: this.props.PanelHeight() || "100%" +            }}  > +            <div className="collectionSchemaView-tableContainer" +                style={{ width: `calc(100% - ${this.previewWidth()}px)` }} +                onPointerDown={this.onPointerDown} +                onWheel={e => this.props.active(true) && e.stopPropagation()} +                onDrop={e => this.onExternalDrop(e, {})} +                ref={this.createTarget}> +                {this.schemaTable} +            </div> +            {this.dividerDragger} +            {!this.previewWidth() ? (null) : this.previewPanel} +            {this._headerOpen ? menu : null}          </div>;      }  }
\ No newline at end of file | 
