diff options
Diffstat (limited to 'src/client/views')
5 files changed, 569 insertions, 16 deletions
| diff --git a/src/client/views/collections/CollectionSchemaCells.tsx b/src/client/views/collections/CollectionSchemaCells.tsx index 2b8110e27..4231a3941 100644 --- a/src/client/views/collections/CollectionSchemaCells.tsx +++ b/src/client/views/collections/CollectionSchemaCells.tsx @@ -1,21 +1,21 @@  import React = require("react"); -import { action, observable, trace } from "mobx"; +import { action, observable, trace, computed } from "mobx";  import { observer } from "mobx-react";  import { CellInfo } from "react-table";  import "react-table/react-table.css"; -import { emptyFunction, returnFalse, returnZero, returnOne, returnEmptyFilter } from "../../../Utils"; +import { emptyFunction, returnFalse, returnZero, returnOne, returnEmptyFilter, Utils, emptyPath } from "../../../Utils";  import { Doc, DocListCast, Field, Opt } from "../../../fields/Doc";  import { Id } from "../../../fields/FieldSymbols";  import { KeyCodes } from "../../util/KeyCodes";  import { SetupDrag, DragManager } from "../../util/DragManager";  import { CompileScript } from "../../util/Scripting";  import { Transform } from "../../util/Transform"; -import { MAX_ROW_HEIGHT } from '../globalCssVariables.scss'; +import { MAX_ROW_HEIGHT, COLLECTION_BORDER_WIDTH } from '../globalCssVariables.scss';  import '../DocumentDecorations.scss';  import { EditableView } from "../EditableView";  import { FieldView, FieldViewProps } from "../nodes/FieldView";  import "./CollectionSchemaView.scss"; -import { CollectionView } from "./CollectionView"; +import { CollectionView, Flyout } from "./CollectionView";  import { NumCast, StrCast, BoolCast, FieldValue, Cast } from "../../../fields/Types";  import { Docs } from "../../documents/Documents";  import { library } from '@fortawesome/fontawesome-svg-core'; @@ -24,6 +24,15 @@ import { SchemaHeaderField } from "../../../fields/SchemaHeaderField";  import { undoBatch } from "../../util/UndoManager";  import { SnappingManager } from "../../util/SnappingManager";  import { ComputedField } from "../../../fields/ScriptField"; +import { ImageField } from "../../../fields/URLField"; +import { List } from "../../../fields/List"; +import { LinkBox } from "../nodes/LinkBox"; +import { OverlayView } from "../OverlayView"; +import { DocumentIconContainer } from "../nodes/DocumentIcon"; +import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; +import { ContentFittingDocumentView } from "../nodes/ContentFittingDocumentView"; +import ReactDOM from "react-dom"; +const path = require('path');  library.add(faExpand); @@ -47,6 +56,7 @@ export interface CellProps {      setPreviewDoc: (doc: Doc) => void;      setComputed: (script: string, doc: Doc, field: string, row: number, col: number) => boolean;      getField: (row: number, col?: number) => void; +    showDoc: (doc: Doc | undefined, dataDoc?: any, screenX?: number, screenY?: number) => void;  }  @observer @@ -54,7 +64,7 @@ export class CollectionSchemaCell extends React.Component<CellProps> {      @observable protected _isEditing: boolean = false;      protected _focusRef = React.createRef<HTMLDivElement>();      protected _document = this.props.rowProps.original; -    private _dropDisposer?: DragManager.DragDropDisposer; +    protected _dropDisposer?: DragManager.DragDropDisposer;      componentDidMount() {          document.addEventListener("keydown", this.onKeyDown); @@ -129,7 +139,7 @@ export class CollectionSchemaCell extends React.Component<CellProps> {          }      } -    private dropRef = (ele: HTMLElement | null) => { +    protected dropRef = (ele: HTMLElement | null) => {          this._dropDisposer && this._dropDisposer();          if (ele) {              this._dropDisposer = DragManager.MakeDropTarget(ele, this.drop.bind(this)); @@ -206,6 +216,15 @@ export class CollectionSchemaCell extends React.Component<CellProps> {              const doc = FieldValue(Cast(field, Doc));              contents = typeof field === "object" ? doc ? StrCast(doc.title) === "" ? "--" : StrCast(doc.title) : `--${typeof field}--` : `--${typeof field}--`;          } +        if (type === "image") { +            const image = FieldValue(Cast(field, ImageField)); +            const doc = FieldValue(Cast(field, Doc)); +            contents = typeof field === "object" ? doc ? StrCast(doc.title) === "" ? "--" : StrCast(doc.title) : `--${typeof field}--` : `--${typeof field}--`; +        } +        if (type === "list") { +            contents = typeof field === "object" ? doc ? StrCast(field) === "" ? "--" : StrCast(field) : `--${typeof field}--` : `--${typeof field}--`; +        } +          let className = "collectionSchemaView-cellWrapper";          if (this._isEditing) className += " editing"; @@ -300,11 +319,414 @@ export class CollectionSchemaStringCell extends CollectionSchemaCell {  @observer  export class CollectionSchemaDocCell extends CollectionSchemaCell { + +    _overlayDisposer?: () => void; + +    private prop: FieldViewProps = { +        Document: this.props.rowProps.original, +        DataDoc: this.props.rowProps.original, +        LibraryPath: [], +        dropAction: "alias", +        bringToFront: emptyFunction, +        rootSelected: returnFalse, +        fieldKey: this.props.rowProps.column.id as string, +        ContainingCollectionView: this.props.CollectionView, +        ContainingCollectionDoc: this.props.CollectionView && this.props.CollectionView.props.Document, +        isSelected: returnFalse, +        select: emptyFunction, +        renderDepth: this.props.renderDepth + 1, +        ScreenToLocalTransform: Transform.Identity, +        focus: emptyFunction, +        active: returnFalse, +        whenActiveChanged: emptyFunction, +        PanelHeight: returnZero, +        PanelWidth: returnZero, +        NativeHeight: returnZero, +        NativeWidth: returnZero, +        addDocTab: this.props.addDocTab, +        pinToPres: this.props.pinToPres, +        ContentScaling: returnOne, +        docFilters: returnEmptyFilter +    }; +    @observable private _field = this.prop.Document[this.prop.fieldKey]; +    @observable private _doc = FieldValue(Cast(this._field, Doc)); +    @observable private _docTitle = this._doc?.title; +    @observable private _preview = false; +    @computed get previewWidth() { return () => NumCast(this.props.Document.schemaPreviewWidth); } +    @computed get borderWidth() { return Number(COLLECTION_BORDER_WIDTH); } +    @computed get tableWidth() { return this.prop.PanelWidth() - 2 * this.borderWidth - 4 - this.previewWidth(); } + +    @action +    onSetValue = (value: string) => { +        this._docTitle = value; +        //this.prop.Document[this.prop.fieldKey] = this._text; + +        const script = CompileScript(value, { +            addReturn: true, +            typecheck: false, +            transformer: DocumentIconContainer.getTransformer() +        }); + +        const results = script.compiled && script.run(); +        if (results && results.success) { + +            console.log(results.result); +            this._doc = results.result; +            this.prop.Document[this.prop.fieldKey] = results.result; +            this.prop.Document[this.prop.fieldKey] = results.result; +            this._docTitle = this._doc?.title; + +            return true; +        } +        return false; +    } + +    onFocus = () => { + +        console.log(this._field); +        this._overlayDisposer?.(); +        this._overlayDisposer = OverlayView.Instance.addElement(<DocumentIconContainer />, { x: 0, y: 0 }); +    } + +    @action +    onOpenClick = () => { +        this._preview = false; +        if (this._doc) { +            this.prop.addDocTab(this._doc, "onRight"); +            return true; +        } +        return false; +    } + +    @action +    showPreview = (bool: boolean, e: any) => { +        if (this._isEditing) { +            this._preview = false; +        } else { +            if (bool) { +                this.props.showDoc(this._doc, this.prop.DataDoc, e.screenX, e.screenY); +            } else { +                this.props.showDoc(undefined); +            } +        } +    } + +    render() { + +        const dragRef: React.RefObject<HTMLDivElement> = React.createRef(); +        const reference = React.createRef<HTMLDivElement>(); + +        if (typeof this._field === "object" && this._doc && this._docTitle) { + + +            return ( +                <div className="collectionSchemaView-cellWrapper" ref={this._focusRef} tabIndex={-1} onPointerDown={this.onPointerDown} +                    onPointerEnter={(e) => { this.showPreview(true, e); }} +                    onPointerLeave={(e) => { this.showPreview(false, e); }} +                > + +                    <div className="collectionSchemaView-cellContents-document" +                        style={{ padding: "5.9px" }} +                        onFocus={this.onFocus} onBlur={() => this._overlayDisposer?.()} +                        ref={this.dropRef} +                    > + +                        <EditableView +                            editing={this._isEditing} +                            isEditingCallback={this.isEditingCallback} +                            display={"inline"} +                            contents={this._docTitle} +                            height={"auto"} +                            maxHeight={Number(MAX_ROW_HEIGHT)} +                            GetValue={() => { +                                return StrCast(this._docTitle); +                            }} +                            SetValue={action((value: string) => { +                                this.onSetValue(value); +                                return true; +                            })} +                        /> +                    </div > +                    <div onClick={this.onOpenClick} className="collectionSchemaView-cellContents-docButton"> +                        <FontAwesomeIcon icon="external-link-alt" size="lg" ></FontAwesomeIcon> </div> +                </div> +            ); +        } else { +            return this.renderCellWithType("document"); +        } +    } +} + +@observer +export class CollectionSchemaImageCell extends CollectionSchemaCell { +    // render() { +    //     return this.renderCellWithType("image"); +    // } + +    choosePath(url: URL, dataDoc: any) { +        const lower = url.href.toLowerCase(); +        if (url.protocol === "data") { +            return url.href; +        } else if (url.href.indexOf(window.location.origin) === -1) { +            return Utils.CorsProxy(url.href); +        } else if (!/\.(png|jpg|jpeg|gif|webp)$/.test(lower)) { +            return url.href;//Why is this here +        } +        const ext = path.extname(url.href); +        const _curSuffix = "_o"; +        return url.href.replace(ext, _curSuffix + ext); +    } + +    render() { +        const props: FieldViewProps = { +            Document: this.props.rowProps.original, +            DataDoc: this.props.rowProps.original, +            LibraryPath: [], +            dropAction: "alias", +            bringToFront: emptyFunction, +            rootSelected: returnFalse, +            fieldKey: this.props.rowProps.column.id as string, +            ContainingCollectionView: this.props.CollectionView, +            ContainingCollectionDoc: this.props.CollectionView && this.props.CollectionView.props.Document, +            isSelected: returnFalse, +            select: emptyFunction, +            renderDepth: this.props.renderDepth + 1, +            ScreenToLocalTransform: Transform.Identity, +            focus: emptyFunction, +            active: returnFalse, +            whenActiveChanged: emptyFunction, +            PanelHeight: returnZero, +            PanelWidth: returnZero, +            NativeHeight: returnZero, +            NativeWidth: returnZero, +            addDocTab: this.props.addDocTab, +            pinToPres: this.props.pinToPres, +            ContentScaling: returnOne, +            docFilters: returnEmptyFilter +        }; + +        let image = true; +        let url = []; +        if (props.DataDoc) { +            const field = Cast(props.DataDoc[props.fieldKey], ImageField, null); // retrieve the primary image URL that is being rendered from the data doc +            const alts = DocListCast(props.DataDoc[props.fieldKey + "-alternates"]); // retrieve alternate documents that may be rendered as alternate images +            const altpaths = alts.map(doc => Cast(doc[Doc.LayoutFieldKey(doc)], ImageField, null)?.url).filter(url => url).map(url => this.choosePath(url, props.DataDoc)); // access the primary layout data of the alternate documents +            const paths = field ? [this.choosePath(field.url, props.DataDoc), ...altpaths] : altpaths; +            if (paths.length) { +                url = paths; +            } else { +                url = [Utils.CorsProxy("http://www.cs.brown.edu/~bcz/noImage.png")]; +                image = false; +            } +            //url = paths.length ? paths : [Utils.CorsProxy("http://www.cs.brown.edu/~bcz/noImage.png")]; +        } else { +            url = [Utils.CorsProxy("http://www.cs.brown.edu/~bcz/noImage.png")]; +            image = false; +        } + +        const heightToWidth = NumCast(props.DataDoc?._nativeHeight) / NumCast(props.DataDoc?._nativeWidth); +        const height = this.props.rowProps.width * heightToWidth; + +        if (props.fieldKey === "data") { +            if (url !== []) { +                const reference = React.createRef<HTMLDivElement>(); +                return ( +                    <div className="collectionSchemaView-cellWrapper" ref={this._focusRef} tabIndex={-1} onPointerDown={this.onPointerDown}> +                        <div className="collectionSchemaView-cellContents" key={this._document[Id]} ref={reference}> +                            <img src={url[0]} width={image ? this.props.rowProps.width : "30px"} +                                height={image ? height : "30px"} /> +                        </div > +                    </div> +                ); + +            } else { +                return this.renderCellWithType("image"); +            } +        } else { +            return this.renderCellWithType("image"); +        } +    } +} + + + + + +@observer +export class CollectionSchemaListCell extends CollectionSchemaCell { + +    _overlayDisposer?: () => void; + +    private prop: FieldViewProps = { +        Document: this.props.rowProps.original, +        DataDoc: this.props.rowProps.original, +        LibraryPath: [], +        dropAction: "alias", +        bringToFront: emptyFunction, +        rootSelected: returnFalse, +        fieldKey: this.props.rowProps.column.id as string, +        ContainingCollectionView: this.props.CollectionView, +        ContainingCollectionDoc: this.props.CollectionView && this.props.CollectionView.props.Document, +        isSelected: returnFalse, +        select: emptyFunction, +        renderDepth: this.props.renderDepth + 1, +        ScreenToLocalTransform: Transform.Identity, +        focus: emptyFunction, +        active: returnFalse, +        whenActiveChanged: emptyFunction, +        PanelHeight: returnZero, +        PanelWidth: returnZero, +        NativeHeight: returnZero, +        NativeWidth: returnZero, +        addDocTab: this.props.addDocTab, +        pinToPres: this.props.pinToPres, +        ContentScaling: returnOne, +        docFilters: returnEmptyFilter +    }; +    @observable private _field = this.prop.Document[this.prop.fieldKey]; +    @observable private _optionsList = this._field as List<any>; +    @observable private _opened = false; +    @observable private _text = "select an item"; +    @observable private _selectedNum = 0; + +    @action +    toggleOpened(open: boolean) { +        console.log("open: " + open); +        this._opened = open; +    } + +    // @action +    // onChange = (e: React.ChangeEvent<HTMLTextAreaElement>) => { +    //     this._text = e.target.value; + +    //     // change if its a document +    //     this._optionsList[this._selectedNum] = this._text; +    // } + +    @action +    onSetValue = (value: string) => { + + +        this._text = value; + +        // change if its a document +        this._optionsList[this._selectedNum] = this._text; + +        (this.prop.Document[this.prop.fieldKey] as List<any>).splice(this._selectedNum, 1, value); + +    } + +    @action +    onSelected = (element: string, index: number) => { +        this._text = element; +        this._selectedNum = index; +    } + +    onFocus = () => { +        this._overlayDisposer?.(); +        this._overlayDisposer = OverlayView.Instance.addElement(<DocumentIconContainer />, { x: 0, y: 0 }); +    } + +      render() { -        return this.renderCellWithType("document"); + +        const dragRef: React.RefObject<HTMLDivElement> = React.createRef(); + +        let type = "list"; + +        let link = false; +        let doc = false; +        const reference = React.createRef<HTMLDivElement>(); + +        if (typeof this._field === "object" && this._optionsList[0]) { + +            const options = this._optionsList.map((element, index) => { + +                if (element instanceof Doc) { +                    doc = true; +                    type = "document"; +                    if (this.prop.fieldKey.toLowerCase() === "links") { +                        link = true; +                        type = "link"; +                    } +                    const document = FieldValue(Cast(element, Doc)); +                    const title = element.title; +                    return <div +                        className="collectionSchemaView-dropdownOption" +                        onPointerDown={(e) => { this.onSelected(StrCast(element.title), index); }} +                        style={{ padding: "6px" }}> +                        {title} +                    </div>; + +                } else { +                    return <div +                        className="collectionSchemaView-dropdownOption" +                        onPointerDown={(e) => { this.onSelected(StrCast(element), index); }} +                        style={{ padding: "6px" }}>{element}</div>; +                } +            }); + +            const plainText = <div style={{ padding: "5.9px" }}>{this._text}</div>; +            // const textarea = <textarea onChange={this.onChange} value={this._text} +            //     onFocus={doc ? this.onFocus : undefined} +            //     onBlur={doc ? e => this._overlayDisposer?.() : undefined} +            //     style={{ resize: "none" }} +            //     placeholder={"select an item"}></textarea>; + +            const textarea = <div className="collectionSchemaView-cellContents" +                style={{ padding: "5.9px" }} +                ref={type === undefined || type === "document" ? this.dropRef : null} key={this.prop.Document[Id]}> +                <EditableView +                    editing={this._isEditing} +                    isEditingCallback={this.isEditingCallback} +                    display={"inline"} +                    contents={this._text} +                    height={"auto"} +                    maxHeight={Number(MAX_ROW_HEIGHT)} +                    GetValue={() => { +                        return this._text; +                    }} +                    SetValue={action((value: string) => { + +                        // add special for params  +                        this.onSetValue(value); +                        return true; +                    })} +                /> +            </div >; + +            //☰ + +            const dropdown = <div> +                {options} +            </div>; + +            return ( +                <div className="collectionSchemaView-cellWrapper" ref={this._focusRef} tabIndex={-1} onPointerDown={this.onPointerDown}> +                    <div className="collectionSchemaView-cellContents" key={this._document[Id]} ref={reference}> +                        <div className="collectionSchemaView-dropDownWrapper"> +                            <button type="button" className="collectionSchemaView-dropdownButton" onClick={(e) => { this.toggleOpened(!this._opened); }} +                                style={{ right: "length", position: "relative" }}> +                                {this._opened ? <FontAwesomeIcon icon="caret-up" size="lg" ></FontAwesomeIcon> +                                    : <FontAwesomeIcon icon="caret-down" size="lg" ></FontAwesomeIcon>} +                            </button> +                            <div className="collectionSchemaView-dropdownText"> {link ? plainText : textarea} </div> +                        </div> + +                        {this._opened ? dropdown : null} +                    </div > +                </div> +            ); +        } else { +            return this.renderCellWithType("list"); +        }      }  } + + + +  @observer  export class CollectionSchemaCheckboxCell extends CollectionSchemaCell {      @observable private _isChecked: boolean = typeof this.props.rowProps.original[this.props.rowProps.column.id as string] === "boolean" ? BoolCast(this.props.rowProps.original[this.props.rowProps.column.id as string]) : false; diff --git a/src/client/views/collections/CollectionSchemaHeaders.tsx b/src/client/views/collections/CollectionSchemaHeaders.tsx index dae0600b1..0d6cf4474 100644 --- a/src/client/views/collections/CollectionSchemaHeaders.tsx +++ b/src/client/views/collections/CollectionSchemaHeaders.tsx @@ -2,7 +2,7 @@ import React = require("react");  import { action, observable } from "mobx";  import { observer } from "mobx-react";  import "./CollectionSchemaView.scss"; -import { faPlus, faFont, faHashtag, faAlignJustify, faCheckSquare, faToggleOn, faSortAmountDown, faSortAmountUp, faTimes } from '@fortawesome/free-solid-svg-icons'; +import { faPlus, faFont, faHashtag, faAlignJustify, faCheckSquare, faToggleOn, faSortAmountDown, faSortAmountUp, faTimes, faImage, faListUl } from '@fortawesome/free-solid-svg-icons';  import { library, IconProp } from "@fortawesome/fontawesome-svg-core";  import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";  import { ColumnType } from "./CollectionSchemaView"; @@ -13,7 +13,7 @@ const higflyout = require("@hig/flyout");  export const { anchorPoints } = higflyout;  export const Flyout = higflyout.default; -library.add(faPlus, faFont, faHashtag, faAlignJustify, faCheckSquare, faToggleOn, faFile as any, faSortAmountDown, faSortAmountUp, faTimes); +library.add(faPlus, faFont, faHashtag, faAlignJustify, faCheckSquare, faToggleOn, faFile as any, faSortAmountDown, faSortAmountUp, faTimes, faImage, faListUl);  export interface HeaderProps {      keyValue: SchemaHeaderField; @@ -33,7 +33,8 @@ export interface HeaderProps {  export class CollectionSchemaHeader extends React.Component<HeaderProps> {      render() {          const icon: IconProp = this.props.keyType === ColumnType.Number ? "hashtag" : this.props.keyType === ColumnType.String ? "font" : -            this.props.keyType === ColumnType.Boolean ? "check-square" : this.props.keyType === ColumnType.Doc ? "file" : "align-justify"; +            this.props.keyType === ColumnType.Boolean ? "check-square" : this.props.keyType === ColumnType.Doc ? "file" : +                this.props.keyType === ColumnType.Image ? "image" : this.props.keyType === ColumnType.List ? "list-ul" : "align-justify";          return (              <div className="collectionSchemaView-header" style={{ background: this.props.keyValue.color }}>                  <CollectionSchemaColumnMenu @@ -160,10 +161,18 @@ export class CollectionSchemaColumnMenu extends React.Component<ColumnMenuProps>                          <FontAwesomeIcon icon={"check-square"} size="sm" />                          Checkbox                      </div> +                    <div className={"columnMenu-option" + (type === ColumnType.List ? " active" : "")} onClick={() => this.changeColumnType(ColumnType.List)}> +                        <FontAwesomeIcon icon={"list-ul"} size="sm" /> +                        List +                    </div>                      <div className={"columnMenu-option" + (type === ColumnType.Doc ? " active" : "")} onClick={() => this.changeColumnType(ColumnType.Doc)}>                          <FontAwesomeIcon icon={"file"} size="sm" />                          Document                      </div> +                    <div className={"columnMenu-option" + (type === ColumnType.Image ? " active" : "")} onClick={() => this.changeColumnType(ColumnType.Image)}> +                        <FontAwesomeIcon icon={"image"} size="sm" /> +                        Image +                    </div>                  </div>              </div >          ); @@ -258,7 +267,7 @@ export class CollectionSchemaColumnMenu extends React.Component<ColumnMenuProps>  } -interface KeysDropdownProps { +export interface KeysDropdownProps {      keyValue: string;      possibleKeys: string[];      existingKeys: string[]; @@ -268,7 +277,7 @@ interface KeysDropdownProps {      setIsEditing: (isEditing: boolean) => void;  }  @observer -class KeysDropdown extends React.Component<KeysDropdownProps> { +export class KeysDropdown extends React.Component<KeysDropdownProps> {      @observable private _key: string = this.props.keyValue;      @observable private _searchTerm: string = this.props.keyValue;      @observable private _isOpen: boolean = false; @@ -331,7 +340,9 @@ class KeysDropdown extends React.Component<KeysDropdownProps> {      renderOptions = (): JSX.Element[] | JSX.Element => {          if (!this._isOpen) return <></>; -        const keyOptions = this._searchTerm === "" ? this.props.possibleKeys : this.props.possibleKeys.filter(key => key.toUpperCase().indexOf(this._searchTerm.toUpperCase()) > -1); +        const searchTerm = this._searchTerm.trim() === "New field" ? "" : this._searchTerm; + +        const keyOptions = searchTerm === "" ? this.props.possibleKeys : this.props.possibleKeys.filter(key => key.toUpperCase().indexOf(this._searchTerm.toUpperCase()) > -1);          const exactFound = keyOptions.findIndex(key => key.toUpperCase() === this._searchTerm.toUpperCase()) > -1 ||              this.props.existingKeys.findIndex(key => key.toUpperCase() === this._searchTerm.toUpperCase()) > -1; diff --git a/src/client/views/collections/CollectionSchemaMovableTableHOC.tsx b/src/client/views/collections/CollectionSchemaMovableTableHOC.tsx index b206765e8..b77173b25 100644 --- a/src/client/views/collections/CollectionSchemaMovableTableHOC.tsx +++ b/src/client/views/collections/CollectionSchemaMovableTableHOC.tsx @@ -137,6 +137,7 @@ export interface MovableRowProps {      textWrapRow: (doc: Doc) => void;      rowWrapped: boolean;      dropAction: string; +    addDocTab: any;  }  export class MovableRow extends React.Component<MovableRowProps> { @@ -232,6 +233,7 @@ export class MovableRow extends React.Component<MovableRowProps> {                          <div className="row-dragger">                              <div className="row-option" onClick={undoBatch(() => this.props.removeDoc(this.props.rowInfo.original))}><FontAwesomeIcon icon="trash" size="sm" /></div>                              <div className="row-option" style={{ cursor: "grab" }} ref={reference} onPointerDown={onItemDown}><FontAwesomeIcon icon="grip-vertical" size="sm" /></div> +                            <div className="row-option" onClick={() => this.props.addDocTab(this.props.rowInfo.original, "onRight")}><FontAwesomeIcon icon="external-link-alt" size="sm" /></div>                          </div>                          {children}                      </ReactTableDefaults.TrComponent> diff --git a/src/client/views/collections/CollectionSchemaView.scss b/src/client/views/collections/CollectionSchemaView.scss index a24140b48..50971420d 100644 --- a/src/client/views/collections/CollectionSchemaView.scss +++ b/src/client/views/collections/CollectionSchemaView.scss @@ -412,6 +412,56 @@ button.add-column {      &:hover .collectionSchemaView-cellContents-docExpander {          display: block;      } + + +    .collectionSchemaView-cellContents-document { +        display: inline-block; +    } + +    .collectionSchemaView-cellContents-docButton { +        float: right; +        width: "15px"; +        height: "15px"; +    } + +    .collectionSchemaView-dropdownWrapper { + +        border: grey; +        border-style: solid; +        border-width: 1px; +        height: 100%; + +        .collectionSchemaView-dropdownButton { + +            //display: inline-block; +            float: left; +            height: 100%; + + +        } + +        .collectionSchemaView-dropdownText { +            display: inline-block; +            //float: right; +            height: 100%; +            display: "flex"; +            font-size: 13; +            justify-content: "center"; +            align-items: "center"; +        } + +    } + +    .collectionSchemaView-dropdownContainer { +        position: absolute; +        border: 1px solid rgba(0, 0, 0, 0.04); +        box-shadow: 0 16px 24px 2px rgba(0, 0, 0, 0.14); + +        .collectionSchemaView-dropdownOption:hover { +            background-color: rgba(0, 0, 0, 0.14); +            cursor: pointer; +        } +    }  }  .collectionSchemaView-cellContents-docExpander { @@ -422,6 +472,7 @@ button.add-column {      top: 0;      right: 0;      background-color: lightgray; +  }  .doc-drag-over { diff --git a/src/client/views/collections/CollectionSchemaView.tsx b/src/client/views/collections/CollectionSchemaView.tsx index 56a2a517c..196ebbdc6 100644 --- a/src/client/views/collections/CollectionSchemaView.tsx +++ b/src/client/views/collections/CollectionSchemaView.tsx @@ -20,15 +20,16 @@ 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 { CellProps, CollectionSchemaCell, CollectionSchemaCheckboxCell, CollectionSchemaDocCell, CollectionSchemaNumberCell, CollectionSchemaStringCell, CollectionSchemaImageCell, CollectionSchemaListCell } from "./CollectionSchemaCells";  import { CollectionSchemaAddColumnHeader, CollectionSchemaHeader } from "./CollectionSchemaHeaders";  import { MovableColumn, MovableRow } from "./CollectionSchemaMovableTableHOC";  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 { setupMoveUpEvents, emptyFunction, returnZero, returnOne, returnFalse, returnEmptyFilter, emptyPath } from "../../../Utils";  import { SnappingManager } from "../../util/SnappingManager"; +import ReactDOM from "react-dom";  library.add(faCog, faPlus, faSortUp, faSortDown);  library.add(faTable); @@ -40,6 +41,8 @@ export enum ColumnType {      String,      Boolean,      Doc, +    Image, +    List  }  // this map should be used for keys that should have a const type of value  const columnTypes: Map<string, ColumnType> = new Map([ @@ -226,6 +229,14 @@ export interface SchemaTableProps {      setPreviewDoc: (document: Doc) => void;  } + + + + + + + +  @observer  export class SchemaTable extends React.Component<SchemaTableProps> {      private DIVIDER_WIDTH = 4; @@ -235,6 +246,10 @@ export class SchemaTable extends React.Component<SchemaTableProps> {      @observable _focusedCell: { row: number, col: number } = { row: 0, col: 0 };      @observable _openCollections: Array<string> = []; +    @observable _showDoc: Doc | undefined; +    @observable _showDataDoc: any = ""; +    @observable _showDocPos: number[] = []; +      @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(); } @@ -347,6 +362,7 @@ export class SchemaTable extends React.Component<SchemaTableProps> {                          setPreviewDoc: this.props.setPreviewDoc,                          setComputed: this.setComputed,                          getField: this.getField, +                        showDoc: this.showDoc,                      };                      const colType = this.getColumnType(col); @@ -354,6 +370,8 @@ export class SchemaTable extends React.Component<SchemaTableProps> {                      if (colType === ColumnType.String) return <CollectionSchemaStringCell {...props} />;                      if (colType === ColumnType.Boolean) return <CollectionSchemaCheckboxCell {...props} />;                      if (colType === ColumnType.Doc) return <CollectionSchemaDocCell {...props} />; +                    if (colType === ColumnType.Image) return <CollectionSchemaImageCell {...props} />; +                    if (colType === ColumnType.List) return <CollectionSchemaListCell {...props} />;                      return <CollectionSchemaCell {...props} />;                  },                  minWidth: 200, @@ -403,7 +421,8 @@ export class SchemaTable extends React.Component<SchemaTableProps> {              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) +            dropAction: StrCast(this.props.Document.childDropAction), +            addDocTab: this.props.addDocTab          };      } @@ -751,8 +770,56 @@ export class SchemaTable extends React.Component<SchemaTableProps> {          return false;      } +    @action +    showDoc = (doc: Doc | undefined, dataDoc?: Doc, screenX?: number, screenY?: number) => { +        this._showDoc = doc; +        if (dataDoc && screenX && screenY) { +            this._showDocPos = this.props.ScreenToLocalTransform().transformPoint(screenX, screenY); +        } +    } + +    onOpenClick = () => { +        if (this._showDoc) { +            this.props.addDocTab(this._showDoc, "onRight"); +        } +    } + +    getPreviewTransform = (): Transform => { +        return this.props.ScreenToLocalTransform().translate(- this.borderWidth - 4 - this.tableWidth, - this.borderWidth); +    } +      render() { +        const preview = "";          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._showDoc ? (null) : +                <div onClick={() => { this.onOpenClick(); }} +                    style={{ position: "absolute", width: 400, height: 300, transform: `translate(${this._showDocPos[0]}px, ${this._showDocPos[1]}px)` }} +                    ref="overlay"><ContentFittingDocumentView +                        Document={this._showDoc} +                        DataDoc={this._showDataDoc} +                        NativeHeight={returnZero} +                        NativeWidth={returnZero} +                        fitToBox={true} +                        FreezeDimensions={true} +                        focus={emptyFunction} +                        LibraryPath={emptyPath} +                        renderDepth={this.props.renderDepth} +                        rootSelected={() => false} +                        PanelWidth={() => 150} +                        PanelHeight={() => 150} +                        ScreenToLocalTransform={this.getPreviewTransform} +                        docFilters={returnEmptyFilter} +                        ContainingCollectionDoc={this.props.CollectionView?.props.Document} +                        ContainingCollectionView={this.props.CollectionView} +                        moveDocument={this.props.moveDocument} +                        parentActive={this.props.active} +                        whenActiveChanged={emptyFunction} +                        addDocTab={this.props.addDocTab} +                        pinToPres={this.props.pinToPres} +                        bringToFront={returnFalse} +                        ContentScaling={returnOne}> +                    </ContentFittingDocumentView> +                </div>}              {this.reactTable}              <div className="collectionSchemaView-addRow" onClick={() => this.createRow()}>+ new</div>          </div>; | 
