diff options
Diffstat (limited to 'src/client/views/nodes/DocumentView.tsx')
| -rw-r--r-- | src/client/views/nodes/DocumentView.tsx | 301 | 
1 files changed, 228 insertions, 73 deletions
| diff --git a/src/client/views/nodes/DocumentView.tsx b/src/client/views/nodes/DocumentView.tsx index fd012e7ea..7750b9173 100644 --- a/src/client/views/nodes/DocumentView.tsx +++ b/src/client/views/nodes/DocumentView.tsx @@ -1,9 +1,19 @@ -import { action, computed, runInAction } from "mobx"; +import { library } from '@fortawesome/fontawesome-svg-core'; +import { faAlignCenter, faCaretSquareRight, faCompressArrowsAlt, faExpandArrowsAlt, faLayerGroup, faSquare, faTrash, faConciergeBell, faFolder, faMapPin, faLink, faFingerprint, faCrosshairs, faDesktop } from '@fortawesome/free-solid-svg-icons'; +import { action, computed, IReactionDisposer, reaction } from "mobx";  import { observer } from "mobx-react"; +import { Doc, DocListCast, HeightSym, Opt, WidthSym, DocListCastAsync } from "../../../new_fields/Doc"; +import { List } from "../../../new_fields/List"; +import { ObjectField } from "../../../new_fields/ObjectField"; +import { createSchema, makeInterface } from "../../../new_fields/Schema"; +import { BoolCast, Cast, FieldValue, StrCast, NumCast } from "../../../new_fields/Types"; +import { CurrentUserUtils } from "../../../server/authentication/models/current_user_utils";  import { emptyFunction, Utils } from "../../../Utils"; -import { Docs } from "../../documents/Documents"; +import { DocServer } from "../../DocServer"; +import { Docs, DocUtils } from "../../documents/Documents";  import { DocumentManager } from "../../util/DocumentManager";  import { DragManager, dropActionType } from "../../util/DragManager"; +import { SearchUtil } from "../../util/SearchUtil";  import { SelectionManager } from "../../util/SelectionManager";  import { Transform } from "../../util/Transform";  import { undoBatch, UndoManager } from "../../util/UndoManager"; @@ -12,20 +22,30 @@ import { CollectionPDFView } from "../collections/CollectionPDFView";  import { CollectionVideoView } from "../collections/CollectionVideoView";  import { CollectionView } from "../collections/CollectionView";  import { ContextMenu } from "../ContextMenu"; -import { Template, Templates } from "./../Templates"; +import { DocComponent } from "../DocComponent"; +import { PresentationView } from "../PresentationView"; +import { Template } from "./../Templates";  import { DocumentContentsView } from "./DocumentContentsView";  import "./DocumentView.scss";  import React = require("react"); -import { Opt, Doc, WidthSym, HeightSym } from "../../../new_fields/Doc"; -import { DocComponent } from "../DocComponent"; -import { createSchema, makeInterface } from "../../../new_fields/Schema"; -import { FieldValue, StrCast, BoolCast } from "../../../new_fields/Types"; -import { List } from "../../../new_fields/List"; -import { CollectionFreeFormView } from "../collections/collectionFreeForm/CollectionFreeFormView"; -import { CurrentUserUtils } from "../../../server/authentication/models/current_user_utils"; -import { DocServer } from "../../DocServer"; -import { Id } from "../../../new_fields/RefField"; -import { PresentationView } from "../PresentationView"; +import { Id, Copy } from '../../../new_fields/FieldSymbols'; +import { ContextMenuProps } from '../ContextMenuItem'; +const JsxParser = require('react-jsx-parser').default; //TODO Why does this need to be imported like this? + +library.add(faTrash); +library.add(faExpandArrowsAlt); +library.add(faCompressArrowsAlt); +library.add(faLayerGroup); +library.add(faAlignCenter); +library.add(faCaretSquareRight); +library.add(faSquare); +library.add(faConciergeBell); +library.add(faFolder); +library.add(faMapPin); +library.add(faLink); +library.add(faFingerprint); +library.add(faCrosshairs); +library.add(faDesktop);  const linkSchema = createSchema({      title: "string", @@ -41,27 +61,28 @@ const LinkDoc = makeInterface(linkSchema);  export interface DocumentViewProps {      ContainingCollectionView: Opt<CollectionView | CollectionPDFView | CollectionVideoView>;      Document: Doc; -    addDocument?: (doc: Document, allowDuplicates?: boolean) => boolean; -    removeDocument?: (doc: Document) => boolean; -    moveDocument?: (doc: Document, targetCollection: Document, addDocument: (document: Document) => boolean) => boolean; +    addDocument?: (doc: Doc, allowDuplicates?: boolean) => boolean; +    removeDocument?: (doc: Doc) => boolean; +    moveDocument?: (doc: Doc, targetCollection: Doc, addDocument: (document: Doc) => boolean) => boolean;      ScreenToLocalTransform: () => Transform;      isTopMost: boolean;      ContentScaling: () => number;      PanelWidth: () => number;      PanelHeight: () => number; -    focus: (doc: Document) => void; +    focus: (doc: Doc) => void;      selectOnLoad: boolean;      parentActive: () => boolean;      whenActiveChanged: (isActive: boolean) => void; -    toggleMinimized: () => void;      bringToFront: (doc: Doc) => void; +    addDocTab: (doc: Doc, where: string) => void; +    collapseToPoint?: (scrpt: number[], expandedDocs: Doc[] | undefined) => void;  }  const schema = createSchema({      layout: "string",      nativeWidth: "number",      nativeHeight: "number", -    backgroundColor: "string" +    backgroundColor: "string",  });  export const positionSchema = createSchema({ @@ -83,6 +104,9 @@ const Document = makeInterface(schema);  export class DocumentView extends DocComponent<DocumentViewProps, Document>(Document) {      private _downX: number = 0;      private _downY: number = 0; +    private _lastTap: number = 0; +    private _doubleTap = false; +    private _hitExpander = false;      private _mainCont = React.createRef<HTMLDivElement>();      private _dropDisposer?: DragManager.DragDropDisposer; @@ -99,6 +123,7 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu      set templates(templates: List<string>) { this.props.Document.templates = templates; }      screenRect = (): ClientRect | DOMRect => this._mainCont.current ? this._mainCont.current.getBoundingClientRect() : new DOMRect(); +    _reactionDisposer?: IReactionDisposer;      @action      componentDidMount() {          if (this._mainCont.current) { @@ -106,6 +131,19 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu                  handlers: { drop: this.drop.bind(this) }              });          } +        // bcz: kind of ugly .. setup a reaction to update the title of a summary document's target (maximizedDocs) whenver the summary doc's title changes +        this._reactionDisposer = reaction(() => [DocListCast(this.props.Document.maximizedDocs).map(md => md.title), +        this.props.Document.summaryDoc, this.props.Document.summaryDoc instanceof Doc ? this.props.Document.summaryDoc.title : ""], +            () => { +                let maxDoc = DocListCast(this.props.Document.maximizedDocs); +                if (maxDoc.length === 1 && StrCast(this.props.Document.title).startsWith("-") && StrCast(this.props.Document.layout).indexOf("IconBox") !== -1) { +                    this.props.Document.proto!.title = "-" + maxDoc[0].title + ".icon"; +                } +                let sumDoc = Cast(this.props.Document.summaryDoc, Doc); +                if (sumDoc instanceof Doc && StrCast(this.props.Document.title).startsWith("-")) { +                    this.props.Document.proto!.title = "-" + sumDoc.title + ".expanded"; +                } +            }, { fireImmediately: true });          DocumentManager.Instance.DocumentViews.push(this);      }      @action @@ -121,9 +159,8 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu      }      @action      componentWillUnmount() { -        if (this._dropDisposer) { -            this._dropDisposer(); -        } +        if (this._reactionDisposer) this._reactionDisposer(); +        if (this._dropDisposer) this._dropDisposer();          DocumentManager.Instance.DocumentViews.splice(DocumentManager.Instance.DocumentViews.indexOf(this), 1);      } @@ -131,10 +168,11 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu          e.stopPropagation();      } -    startDragging(x: number, y: number, dropAction: dropActionType) { +    startDragging(x: number, y: number, dropAction: dropActionType, dragSubBullets: boolean) {          if (this._mainCont.current) { +            let allConnected = [this.props.Document, ...(dragSubBullets ? DocListCast(this.props.Document.subBulletDocs) : [])];              const [left, top] = this.props.ScreenToLocalTransform().scale(this.props.ContentScaling()).inverse().transformPoint(0, 0); -            let dragData = new DragManager.DocumentDragData([this.props.Document]); +            let dragData = new DragManager.DocumentDragData(allConnected);              const [xoff, yoff] = this.props.ScreenToLocalTransform().scale(this.props.ContentScaling()).transformDirection(x - left, y - top);              dragData.dropAction = dropAction;              dragData.xOffset = xoff; @@ -148,28 +186,95 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu              });          }      } +    toggleMinimized = async () => { +        let minimizedDoc = await Cast(this.props.Document.minimizedDoc, Doc); +        if (minimizedDoc) { +            let scrpt = this.props.ScreenToLocalTransform().inverse().transformPoint( +                NumCast(minimizedDoc.x) - NumCast(this.Document.x), NumCast(minimizedDoc.y) - NumCast(this.Document.y)); +            this.props.collapseToPoint && this.props.collapseToPoint(scrpt, await DocListCastAsync(minimizedDoc.maximizedDocs)); +        } +    } -    onClick = (e: React.MouseEvent): void => { -        if (CurrentUserUtils.MainDocId !== this.props.Document[Id] && +    onClick = async (e: React.MouseEvent) => { +        e.stopPropagation(); +        let altKey = e.altKey; +        let ctrlKey = e.ctrlKey; +        if (this._doubleTap && !this.props.isTopMost) { +            this.props.addDocTab(this.props.Document, "inTab"); +            SelectionManager.DeselectAll(); +            this.props.Document.libraryBrush = false; +        } +        else if (CurrentUserUtils.MainDocId !== this.props.Document[Id] &&              (Math.abs(e.clientX - this._downX) < Utils.DRAG_THRESHOLD &&                  Math.abs(e.clientY - this._downY) < Utils.DRAG_THRESHOLD)) {              SelectionManager.SelectDoc(this, e.ctrlKey); +            let isExpander = (e.target as any).id === "isExpander"; +            if (BoolCast(this.props.Document.isButton, false) || isExpander) { +                SelectionManager.DeselectAll(); +                let subBulletDocs = await DocListCastAsync(this.props.Document.subBulletDocs); +                let maximizedDocs = await DocListCastAsync(this.props.Document.maximizedDocs); +                let summarizedDocs = await DocListCastAsync(this.props.Document.summarizedDocs); +                let linkedToDocs = await DocListCastAsync(this.props.Document.linkedToDocs, []); +                let linkedFromDocs = await DocListCastAsync(this.props.Document.linkedFromDocs, []); +                let expandedDocs: Doc[] = []; +                expandedDocs = subBulletDocs ? [...subBulletDocs, ...expandedDocs] : expandedDocs; +                expandedDocs = maximizedDocs ? [...maximizedDocs, ...expandedDocs] : expandedDocs; +                expandedDocs = summarizedDocs ? [...summarizedDocs, ...expandedDocs] : expandedDocs; +                // let expandedDocs = [...(subBulletDocs ? subBulletDocs : []), ...(maximizedDocs ? maximizedDocs : []), ...(summarizedDocs ? summarizedDocs : []),]; +                if (expandedDocs.length) {   // bcz: need a better way to associate behaviors with click events on widget-documents +                    let expandedProtoDocs = expandedDocs.map(doc => Doc.GetProto(doc)); +                    let maxLocation = StrCast(this.props.Document.maximizeLocation, "inPlace"); +                    let getDispDoc = (target: Doc) => Object.getOwnPropertyNames(target).indexOf("isPrototype") === -1 ? target : Doc.MakeDelegate(target); +                    if (altKey) { +                        maxLocation = this.props.Document.maximizeLocation = (maxLocation === "inPlace" || !maxLocation ? "inTab" : "inPlace"); +                        if (!maxLocation || maxLocation === "inPlace") { +                            let hadView = expandedDocs.length === 1 && DocumentManager.Instance.getDocumentView(expandedProtoDocs[0], this.props.ContainingCollectionView); +                            let wasMinimized = !hadView && expandedDocs.reduce((min, d) => !min && !BoolCast(d.IsMinimized, false), false); +                            expandedDocs.forEach(maxDoc => Doc.GetProto(maxDoc).isMinimized = false); +                            let hasView = expandedDocs.length === 1 && DocumentManager.Instance.getDocumentView(expandedProtoDocs[0], this.props.ContainingCollectionView); +                            if (!hasView) { +                                this.props.addDocument && expandedDocs.forEach(async maxDoc => this.props.addDocument!(getDispDoc(maxDoc), false)); +                            } +                            expandedProtoDocs.forEach(maxDoc => maxDoc.isMinimized = wasMinimized); +                        } +                    } +                    if (maxLocation && maxLocation !== "inPlace" && CollectionDockingView.Instance) { +                        let dataDocs = DocListCast(CollectionDockingView.Instance.props.Document.data); +                        if (dataDocs) { +                            expandedDocs.forEach(maxDoc => +                                (!CollectionDockingView.Instance.CloseRightSplit(Doc.GetProto(maxDoc)) && +                                    this.props.addDocTab(getDispDoc(maxDoc), maxLocation))); +                        } +                    } else { +                        let scrpt = this.props.ScreenToLocalTransform().inverse().transformPoint(NumCast(this.Document.width) / 2, NumCast(this.Document.height) / 2); +                        this.props.collapseToPoint && this.props.collapseToPoint(scrpt, expandedProtoDocs); +                    } +                } +                else if (linkedToDocs.length || linkedFromDocs.length) { +                    let linkedFwdDocs = [ +                        linkedToDocs.length ? linkedToDocs[0].linkedTo as Doc : linkedFromDocs.length ? linkedFromDocs[0].linkedFrom as Doc : expandedDocs[0], +                        linkedFromDocs.length ? linkedFromDocs[0].linkedFrom as Doc : linkedToDocs.length ? linkedToDocs[0].linkedTo as Doc : expandedDocs[0]]; + +                    let linkedFwdPage = [ +                        linkedToDocs.length ? NumCast(linkedToDocs[0].linkedToPage, undefined) : linkedFromDocs.length ? NumCast(linkedFromDocs[0].linkedFromPage, undefined) : undefined, +                        linkedFromDocs.length ? NumCast(linkedFromDocs[0].linkedFromPage, undefined) : linkedToDocs.length ? NumCast(linkedToDocs[0].linkedToPage, undefined) : undefined]; +                    if (!linkedFwdDocs.some(l => l instanceof Promise)) { +                        let maxLocation = StrCast(linkedFwdDocs[altKey ? 1 : 0].maximizeLocation, "inTab"); +                        DocumentManager.Instance.jumpToDocument(linkedFwdDocs[altKey ? 1 : 0], ctrlKey, document => this.props.addDocTab(document, maxLocation), linkedFwdPage[altKey ? 1 : 0]); +                    } +                } +            }          }      }      onPointerDown = (e: React.PointerEvent): void => {          this._downX = e.clientX;          this._downY = e.clientY; -        if (CollectionFreeFormView.RIGHT_BTN_DRAG && (e.button === 2 || (e.button === 0 && e.altKey)) && !this.isSelected()) { -            return; -        } -        if (e.shiftKey && e.buttons === 1) { -            if (this.props.isTopMost) { -                this.startDragging(e.pageX, e.pageY, e.altKey || e.ctrlKey ? "alias" : undefined); -            } else { -                CollectionDockingView.Instance.StartOtherDrag([this.props.Document], e); -            } +        this._hitExpander = DocListCast(this.props.Document.subBulletDocs).length > 0; +        if (e.shiftKey && e.buttons === 1 && CollectionDockingView.Instance) { +            CollectionDockingView.Instance.StartOtherDrag([Doc.MakeAlias(this.props.Document)], e);              e.stopPropagation(); -        } else if (this.active) { +        } else { +            if (this.active) e.stopPropagation(); // events stop at the lowest document that is active.                document.removeEventListener("pointermove", this.onPointerMove);              document.addEventListener("pointermove", this.onPointerMove);              document.removeEventListener("pointerup", this.onPointerUp); @@ -177,12 +282,12 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu          }      }      onPointerMove = (e: PointerEvent): void => { -        if (!e.cancelBubble) { +        if (!e.cancelBubble && this.active) {              if (Math.abs(this._downX - e.clientX) > 3 || Math.abs(this._downY - e.clientY) > 3) {                  document.removeEventListener("pointermove", this.onPointerMove);                  document.removeEventListener("pointerup", this.onPointerUp); -                if (!e.altKey && !this.topMost && (!CollectionFreeFormView.RIGHT_BTN_DRAG && e.buttons === 1) || (CollectionFreeFormView.RIGHT_BTN_DRAG && e.buttons === 2)) { -                    this.startDragging(this._downX, this._downY, e.ctrlKey || e.altKey ? "alias" : undefined); +                if (!e.altKey && !this.topMost && e.buttons === 1) { +                    this.startDragging(this._downX, this._downY, e.ctrlKey || e.altKey ? "alias" : undefined, this._hitExpander);                  }              }              e.stopPropagation(); // doesn't actually stop propagation since all our listeners are listening to events on 'document'  however it does mark the event as cancelBubble=true which we test for in the move event handlers @@ -192,30 +297,29 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu      onPointerUp = (e: PointerEvent): void => {          document.removeEventListener("pointermove", this.onPointerMove);          document.removeEventListener("pointerup", this.onPointerUp); +        this._doubleTap = (Date.now() - this._lastTap < 300 && e.button === 0 && Math.abs(e.clientX - this._downX) < 2 && Math.abs(e.clientY - this._downY) < 2); +        this._lastTap = Date.now();      } -    deleteClicked = (): void => { -        this.props.removeDocument && this.props.removeDocument(this.props.Document); -    } -    fieldsClicked = (e: React.MouseEvent): void => { -        let kvp = Docs.KVPDocument(this.props.Document, { width: 300, height: 300 }); -        CollectionDockingView.Instance.AddRightSplit(kvp); -    } -    makeButton = (e: React.MouseEvent): void => { -        this.props.Document.isButton = !BoolCast(this.props.Document.isButton, false); -        if (this.props.Document.isButton && !this.props.Document.nativeWidth) { -            this.props.Document.nativeWidth = this.props.Document[WidthSym](); -            this.props.Document.nativeHeight = this.props.Document[HeightSym](); +    deleteClicked = (): void => { this.props.removeDocument && this.props.removeDocument(this.props.Document); } +    fieldsClicked = (): void => { this.props.addDocTab(Docs.KVPDocument(this.props.Document, { width: 300, height: 300 }), "onRight") }; +    makeBtnClicked = (): void => { +        let doc = Doc.GetProto(this.props.Document); +        doc.isButton = !BoolCast(doc.isButton, false); +        if (StrCast(doc.layout).indexOf("Formatted") !== -1) { // only need to freeze the dimensions of text boxes since they don't have a native width and height naturally +            if (doc.isButton && !doc.nativeWidth) { +                doc.nativeWidth = this.props.Document[WidthSym](); +                doc.nativeHeight = this.props.Document[HeightSym](); +            } else { +                doc.nativeWidth = doc.nativeHeight = undefined; +            }          }      } -    fullScreenClicked = (e: React.MouseEvent): void => { -        const doc = Doc.MakeDelegate(FieldValue(this.Document.proto)); -        if (doc) { -            CollectionDockingView.Instance.OpenFullScreen(doc); -        } -        ContextMenu.Instance.clearItems(); +    fullScreenClicked = (): void => { +        CollectionDockingView.Instance && CollectionDockingView.Instance.OpenFullScreen(Doc.MakeCopy(this.props.Document, false));          SelectionManager.DeselectAll();      } +      @undoBatch      @action      drop = async (e: Event, de: DragManager.DropEvent) => { @@ -223,9 +327,19 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu              let sourceDoc = de.data.linkSourceDocument;              let destDoc = this.props.Document; -            const protoDest = destDoc.proto; -            const protoSrc = sourceDoc.proto; -            Doc.MakeLink(protoSrc ? protoSrc : sourceDoc, protoDest ? protoDest : destDoc); +            if (de.mods === "AltKey") { +                const protoDest = destDoc.proto; +                const protoSrc = sourceDoc.proto; +                let src = protoSrc ? protoSrc : sourceDoc; +                let dst = protoDest ? protoDest : destDoc; +                dst.data = (src.data! as ObjectField)[Copy](); +                dst.nativeWidth = src.nativeWidth; +                dst.nativeHeight = src.nativeHeight; +            } +            else { +                DocUtils.MakeLink(sourceDoc, destDoc); +                de.data.droppedDocuments.push(destDoc); +            }              e.stopPropagation();          }      } @@ -242,7 +356,6 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu          }      } -      @action      addTemplate = (template: Template) => {          this.templates.push(template.Layout); @@ -260,6 +373,26 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu          this.templates = this.templates;      } +    freezeNativeDimensions = (e: React.MouseEvent): void => { +        if (NumCast(this.props.Document.nativeWidth)) { +            let proto = Doc.GetProto(this.props.Document); +            let nw = proto.nativeWidth; +            let nh = proto.nativeHeight; +            proto.nativeWidth = proto.nativeHeight = undefined; +            this.props.Document.width = this.props.Document.frozenWidth; +            this.props.Document.height = this.props.Document.frozenHeight; +        } +        else { +            let scale = this.props.ScreenToLocalTransform().Scale * NumCast(this.props.Document.zoomBasis, 1); +            let scr = this.screenRect(); +            let proto = Doc.GetProto(this.props.Document); +            this.props.Document.frozenWidth = this.props.Document.width; +            this.props.Document.frozenHeight = this.props.Document.height; +            this.props.Document.height = proto.nativeHeight = scr.height * scale; +            this.props.Document.width = proto.nativeWidth = scr.width * scale; +        } +    } +      @action      onContextMenu = (e: React.MouseEvent): void => {          e.stopPropagation(); @@ -270,16 +403,28 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu          }          e.preventDefault(); -        ContextMenu.Instance.addItem({ description: "Full Screen", event: this.fullScreenClicked }); -        ContextMenu.Instance.addItem({ description: this.props.Document.isButton ? "Remove Button" : "Make Button", event: this.makeButton }); -        ContextMenu.Instance.addItem({ description: "Fields", event: this.fieldsClicked }); -        ContextMenu.Instance.addItem({ description: "Center", event: () => this.props.focus(this.props.Document) }); -        ContextMenu.Instance.addItem({ description: "Open Right", event: () => CollectionDockingView.Instance.AddRightSplit(this.props.Document) }); -        ContextMenu.Instance.addItem({ description: "Copy URL", event: () => Utils.CopyText(DocServer.prepend("/doc/" + this.props.Document[Id])) }); -        ContextMenu.Instance.addItem({ description: "Copy ID", event: () => Utils.CopyText(this.props.Document[Id]) }); -        //ContextMenu.Instance.addItem({ description: "Docking", event: () => this.props.Document.SetNumber(KeyStore.ViewType, CollectionViewType.Docking) }) -        ContextMenu.Instance.addItem({ description: "Pin to Presentation", event: () => PresentationView.Instance.PinDoc(this.props.Document) }); -        ContextMenu.Instance.addItem({ description: "Delete", event: this.deleteClicked }); +        const cm = ContextMenu.Instance; +        let subitems: ContextMenuProps[] = []; +        subitems.push({ description: "Open Full Screen", event: this.fullScreenClicked, icon: "desktop" }); +        subitems.push({ description: "Open Tab", event: () => this.props.addDocTab && this.props.addDocTab(this.props.Document, "inTab"), icon: "folder" }); +        subitems.push({ description: "Open Tab Alias", event: () => this.props.addDocTab && this.props.addDocTab(Doc.MakeAlias(this.props.Document), "inTab"), icon: "folder" }); +        subitems.push({ description: "Open Right", event: () => this.props.addDocTab && this.props.addDocTab(this.props.Document, "onRight"), icon: "caret-square-right" }); +        subitems.push({ description: "Open Right Alias", event: () => this.props.addDocTab && this.props.addDocTab(Doc.MakeAlias(this.props.Document), "onRight"), icon: "caret-square-right" }); +        subitems.push({ description: "Open Fields", event: this.fieldsClicked, icon: "layer-group" }); +        cm.addItem({ description: "Open...", subitems: subitems }); +        cm.addItem({ description: NumCast(this.props.Document.nativeWidth) ? "Unfreeze" : "Freeze", event: this.freezeNativeDimensions, icon: "edit" }); +        cm.addItem({ description: "Pin to Pres", event: () => PresentationView.Instance.PinDoc(this.props.Document), icon: "map-pin" }); +        cm.addItem({ description: this.props.Document.isButton ? "Remove Button" : "Make Button", event: this.makeBtnClicked, icon: "concierge-bell" }); +        cm.addItem({ +            description: "Find aliases", event: async () => { +                const aliases = await SearchUtil.GetAliasesOfDocument(this.props.Document); +                this.props.addDocTab && this.props.addDocTab(Docs.SchemaDocument(["title"], aliases, {}), "onRight"); +            }, icon: "search" +        }); +        cm.addItem({ description: "Center View", event: () => this.props.focus(this.props.Document), icon: "crosshairs" }); +        cm.addItem({ description: "Copy URL", event: () => Utils.CopyText(DocServer.prepend("/doc/" + this.props.Document[Id])), icon: "link" }); +        cm.addItem({ description: "Copy ID", event: () => Utils.CopyText(this.props.Document[Id]), icon: "fingerprint" }); +        cm.addItem({ description: "Delete", event: this.deleteClicked, icon: "trash" });          if (!this.topMost) {              // DocumentViews should stop propagation of this event              e.stopPropagation(); @@ -290,6 +435,8 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu          }      } +    onPointerEnter = (e: React.PointerEvent): void => { this.props.Document.libraryBrush = true; }; +    onPointerLeave = (e: React.PointerEvent): void => { this.props.Document.libraryBrush = false; };      isSelected = () => SelectionManager.IsSelected(this);      select = (ctrlPressed: boolean) => SelectionManager.SelectDoc(this, ctrlPressed); @@ -300,13 +447,19 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu      render() {          var scaling = this.props.ContentScaling(); -        var nativeHeight = this.nativeHeight > 0 ? this.nativeHeight.toString() + "px" : "100%"; -        var nativeWidth = this.nativeWidth > 0 ? this.nativeWidth.toString() + "px" : "100%"; +        var nativeHeight = this.nativeHeight > 0 ? `${this.nativeHeight}px` : "100%"; +        var nativeWidth = this.nativeWidth > 0 ? `${this.nativeWidth}px` : "100%";          return (              <div className={`documentView-node${this.props.isTopMost ? "-topmost" : ""}`}                  ref={this._mainCont}                  style={{ +                    outlineColor: "maroon", +                    outlineStyle: "dashed", +                    outlineWidth: BoolCast(this.props.Document.libraryBrush, false) || +                        BoolCast(this.props.Document.protoBrush, false) ? +                        `${1 * this.props.ScreenToLocalTransform().Scale}px` +                        : "0px",                      borderRadius: "inherit",                      background: this.Document.backgroundColor || "",                      width: nativeWidth, @@ -314,6 +467,8 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu                      transform: `scale(${scaling}, ${scaling})`                  }}                  onDrop={this.onDrop} onContextMenu={this.onContextMenu} onPointerDown={this.onPointerDown} onClick={this.onClick} + +                onPointerEnter={this.onPointerEnter} onPointerLeave={this.onPointerLeave}              >                  {this.contents}              </div> | 
