diff options
Diffstat (limited to 'src/client/views/collections/TreeView.tsx')
| -rw-r--r-- | src/client/views/collections/TreeView.tsx | 81 | 
1 files changed, 55 insertions, 26 deletions
| diff --git a/src/client/views/collections/TreeView.tsx b/src/client/views/collections/TreeView.tsx index 206e48aac..a3da0e0e4 100644 --- a/src/client/views/collections/TreeView.tsx +++ b/src/client/views/collections/TreeView.tsx @@ -1,7 +1,7 @@  import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';  import { action, computed, IReactionDisposer, observable, reaction } from "mobx";  import { observer } from "mobx-react"; -import { DataSym, Doc, DocListCast, DocListCastOrNull, Field, HeightSym, Opt, WidthSym } from '../../../fields/Doc'; +import { DataSym, Doc, DocListCast, DocListCastOrNull, Field, HeightSym, Opt, WidthSym, StrListCast } from '../../../fields/Doc';  import { Id } from '../../../fields/FieldSymbols';  import { List } from '../../../fields/List';  import { RichTextField } from '../../../fields/RichTextField'; @@ -9,7 +9,7 @@ import { listSpec } from '../../../fields/Schema';  import { ComputedField, ScriptField } from '../../../fields/ScriptField';  import { BoolCast, Cast, NumCast, ScriptCast, StrCast } from '../../../fields/Types';  import { TraceMobx } from '../../../fields/util'; -import { emptyFunction, returnEmptyDoclist, returnEmptyFilter, returnEmptyString, returnFalse, returnTrue, simulateMouseClick, Utils } from '../../../Utils'; +import { emptyFunction, returnEmptyDoclist, returnEmptyFilter, returnEmptyString, returnFalse, returnTrue, simulateMouseClick, Utils, returnOne } from '../../../Utils';  import { Docs, DocUtils } from '../../documents/Documents';  import { DocumentType } from "../../documents/DocumentTypes";  import { CurrentUserUtils } from '../../util/CurrentUserUtils'; @@ -54,6 +54,7 @@ export interface TreeViewProps {      indentDocument?: (editTitle: boolean) => void;      outdentDocument?: (editTitle: boolean) => void;      ScreenToLocalTransform: () => Transform; +    contextMenuItems: { script: ScriptField, filter: ScriptField, icon: string, label: string }[];      dontRegisterView?: boolean;      styleProvider?: StyleProviderFunc | undefined;      treeViewHideHeaderFields: () => boolean; @@ -99,13 +100,14 @@ export class TreeView extends React.Component<TreeViewProps> {      @observable _dref: DocumentView | undefined | null;      get displayName() { return "TreeView(" + this.props.document.title + ")"; }  // this makes mobx trace() statements more descriptive      get defaultExpandedView() { -        return this.props.treeView.fileSysMode ? (this.doc.isFolder ? this.fieldKey : "aliases") : -            this.props.treeView.outlineMode || this.childDocs ? this.fieldKey : Doc.UserDoc().noviceMode ? "layout" : StrCast(this.props.treeView.doc.treeViewExpandedView, "fields"); +        return this.doc.viewType === CollectionViewType.Docking ? this.fieldKey : +            this.props.treeView.fileSysMode ? (this.doc.isFolder ? this.fieldKey : "layout") : +                this.props.treeView.outlineMode || this.childDocs ? this.fieldKey : Doc.UserDoc().noviceMode ? "layout" : StrCast(this.props.treeView.doc.treeViewExpandedView, "fields");      }      @computed get doc() { return this.props.document; }      @computed get treeViewOpen() { return (!this.treeViewOpenIsTransient && Doc.GetT(this.doc, "treeViewOpen", "boolean", true)) || this._transientOpenState; } -    @computed get treeViewExpandedView() { return StrCast(this.doc.treeViewExpandedView, this.defaultExpandedView); } +    @computed get treeViewExpandedView() { return this.validExpandViewTypes.includes(StrCast(this.doc.treeViewExpandedView)) ? StrCast(this.doc.treeViewExpandedView) : this.defaultExpandedView; }      @computed get MAX_EMBED_HEIGHT() { return NumCast(this.props.containerCollection.maxEmbedHeight, 200); }      @computed get dataDoc() { return this.doc[DataSym]; }      @computed get layoutDoc() { return Doc.Layout(this.doc); } @@ -152,8 +154,8 @@ export class TreeView extends React.Component<TreeViewProps> {          } else {              // choose an appropriate alias or make one. --- choose the first alias that (1) user owns,  (2) has no context field ... otherwise make a new alias              // this.props.addDocTab(CurrentUserUtils.ActiveDashboard.isShared ? Doc.MakeAlias(this.props.document) : this.props.document, "add:right"); -            // choose an appropriate alias or make one -- -- choose the first alias that (1) the user owns, (2) has no context field - if I own it and someone else does not have it open,, otherwise create an alias -            this.props.addDocTab(this.props.document, "add:right"); +            const bestAlias = DocListCast(this.props.document.aliases).find(doc => !doc.context && doc.author === Doc.CurrentUserEmail); +            this.props.addDocTab(bestAlias ?? Doc.MakeAlias(this.props.document), "add:right");          }      } @@ -238,10 +240,13 @@ export class TreeView extends React.Component<TreeViewProps> {      }      makeFolder = () => { -        const folder = Docs.Create.TreeDocument([], { title: "-folder-", _stayInCollection: true, isFolder: true }); +        const folder = Docs.Create.TreeDocument([], { title: "Untitled folder", _stayInCollection: true, isFolder: true });          TreeView._editTitleOnLoad = { id: folder[Id], parent: this.props.parentTreeView };          return this.props.addDocument(folder);      } +    deleteFolder = () => { +        return this.props.removeDoc?.(this.doc); +    }      preTreeDrop = (e: Event, de: DragManager.DropEvent, targetAction: dropActionType) => {          const dragData = de.complete.docDragData; @@ -294,7 +299,7 @@ export class TreeView extends React.Component<TreeViewProps> {          const aspect = Doc.NativeAspect(layoutDoc);          if (layoutDoc._fitWidth) return Math.min(this.props.panelWidth() - treeBulletWidth(), layoutDoc[WidthSym]());          if (aspect) return Math.min(layoutDoc[WidthSym](), Math.min(this.MAX_EMBED_HEIGHT * aspect, this.props.panelWidth() - treeBulletWidth())); -        return Math.min(this.props.panelWidth() - treeBulletWidth(), Doc.NativeWidth(layoutDoc) ? layoutDoc[WidthSym]() : this.layoutDoc[WidthSym]()); +        return Math.min((this.props.panelWidth() - treeBulletWidth()) / (this.props.treeView.props.scaling?.() || 1), Doc.NativeWidth(layoutDoc) ? layoutDoc[WidthSym]() : this.layoutDoc[WidthSym]());      }      docHeight = () => {          const layoutDoc = this.layoutDoc; @@ -336,7 +341,7 @@ export class TreeView extends React.Component<TreeViewProps> {                      this.props.dropAction, this.props.addDocTab, this.titleStyleProvider, this.props.ScreenToLocalTransform, this.props.isContentActive,                      this.props.panelWidth, this.props.renderDepth, this.props.treeViewHideHeaderFields,                      [...this.props.renderedIds, doc[Id]], this.props.onCheckedClick, this.props.onChildClick, this.props.skipFields, false, this.props.whenChildContentsActiveChanged, -                    this.props.dontRegisterView, emptyFunction, emptyFunction); +                    this.props.dontRegisterView, emptyFunction, emptyFunction, this.childContextMenuItems());              } else {                  contentElement = <EditableView key="editableView"                      contents={contents !== undefined ? Field.toString(contents as Field) : "null"} @@ -363,7 +368,7 @@ export class TreeView extends React.Component<TreeViewProps> {          return rows;      } -    rtfWidth = () => Math.min(this.layoutDoc?.[WidthSym](), this.props.panelWidth() - treeBulletWidth()); +    rtfWidth = () => Math.min(this.layoutDoc?.[WidthSym](), (this.props.panelWidth() - treeBulletWidth())) / (this.props.treeView.props.scaling?.() || 1);      rtfHeight = () => this.rtfWidth() <= this.layoutDoc?.[WidthSym]() ? Math.min(this.layoutDoc?.[HeightSym](), this.MAX_EMBED_HEIGHT) : this.MAX_EMBED_HEIGHT;      rtfOutlineHeight = () => Math.max(this.layoutDoc?.[HeightSym](), treeBulletWidth());      expandPanelHeight = () => { @@ -418,7 +423,7 @@ export class TreeView extends React.Component<TreeViewProps> {                          StrCast(this.doc.childDropAction, this.props.dropAction) as dropActionType, this.props.addDocTab, this.titleStyleProvider, this.props.ScreenToLocalTransform,                          this.props.isContentActive, this.props.panelWidth, this.props.renderDepth, this.props.treeViewHideHeaderFields,                          [...this.props.renderedIds, this.doc[Id]], this.props.onCheckedClick, this.props.onChildClick, this.props.skipFields, false, this.props.whenChildContentsActiveChanged, -                        this.props.dontRegisterView, emptyFunction, emptyFunction)} +                        this.props.dontRegisterView, emptyFunction, emptyFunction, this.childContextMenuItems())}              </ul >;          } else if (this.treeViewExpandedView === "fields") {              return <ul key={this.doc[Id] + this.doc.title}> @@ -475,16 +480,23 @@ export class TreeView extends React.Component<TreeViewProps> {          </div>;      } +    @computed get validExpandViewTypes() { +        if (this.props.treeView.dashboardMode && Doc.UserDoc().noviceMode) { +            return [this.doc.viewType === CollectionViewType.Docking ? this.fieldKey : "layout"]; +        } +        const annos = () => DocListCast(this.doc[this.fieldKey + "-annotations"]).length ? "annotations" : ""; +        const links = () => DocListCast(this.doc.links).length ? "links" : ""; +        const data = () => this.childDocs ? this.fieldKey : ""; +        const aliases = () => this.props.treeView.dashboardMode ? "" : "aliases"; +        const fields = () => Doc.UserDoc().noviceMode ? "" : "fields"; +        const layout = this.doc.viewType === CollectionViewType.Docking ? [] : ["layout"]; +        return [data(), ...layout, ...(this.props.treeView.fileSysMode ? [aliases(), links(), annos()] : []), fields()].filter(m => m); +    }      @action      expandNextviewType = () => {          if (this.treeViewOpen && !this.doc.isFolder && !this.props.treeView.outlineMode && !this.doc.treeViewExpandedViewLock) { -            const next = (modes: any[]) => modes[(modes.indexOf(StrCast(this.doc.treeViewExpandedView)) + 1) % modes.length]; -            const annos = () => DocListCast(this.doc[this.fieldKey + "-annotations"]).length ? "annotations" : ""; -            const links = () => DocListCast(this.doc.links).length ? "links" : ""; -            const children = () => this.childDocs ? this.fieldKey : ""; -            this.doc.treeViewExpandedView = next(this.props.treeView.fileSysMode ? -                (Doc.UserDoc().noviceMode ? ["layout", "aliases"] : ["layout", "aliases", "fields"]) : -                (Doc.UserDoc().noviceMode ? [children(), "layout"] : [children(), "fields", "layout", links(), annos()]).filter(mode => mode)); +            const next = (modes: any[]) => modes[(modes.indexOf(StrCast(this.treeViewExpandedView)) + 1) % modes.length]; +            this.doc.treeViewExpandedView = next(this.validExpandViewTypes);          }          this.treeViewOpen = true;      } @@ -506,12 +518,23 @@ export class TreeView extends React.Component<TreeViewProps> {          DocumentViewInternal.SelectAfterContextMenu = true;      }      contextMenuItems = () => { -        const makeFolder = { script: ScriptField.MakeFunction(`scriptContext.makeFolder()`, { scriptContext: "any" })!, label: "New Folder" }; -        return this.doc.isFolder ? [makeFolder] : +        const makeFolder = { script: ScriptField.MakeFunction(`scriptContext.makeFolder()`, { scriptContext: "any" })!, icon: "folder-plus", label: "New Folder" }; +        const deleteFolder = { script: ScriptField.MakeFunction(`scriptContext.deleteFolder()`, { scriptContext: "any" })!, icon: "folder-plus", label: "Delete Folder" }; +        const folderOp = this.childDocs?.length ? makeFolder : deleteFolder; +        const openAlias = { script: ScriptField.MakeFunction(`openOnRight(getAlias(self))`)!, icon: "copy", label: "Open Alias" }; +        const focusDoc = { script: ScriptField.MakeFunction(`DocFocusOrOpen(self)`)!, icon: "eye", label: "Focus or Open" }; +        return [...this.props.contextMenuItems.filter(mi => !mi.filter ? true : mi.filter.script.run({ doc: this.doc })?.result), ... (this.doc.isFolder ? [folderOp] :              Doc.IsSystem(this.doc) ? [] :                  this.props.treeView.fileSysMode && this.doc === Doc.GetProto(this.doc) ? -                    [{ script: ScriptField.MakeFunction(`openOnRight(getAlias(self))`)!, label: "Open Alias" }, makeFolder] : -                    [{ script: ScriptField.MakeFunction(`openOnRight(getAlias(self))`)!, label: "Open Alias" }, { script: ScriptField.MakeFunction(`DocFocusOrOpen(self)`)!, label: "Focus or Open" }]; +                    [openAlias, makeFolder] : +                    this.doc.viewType === CollectionViewType.Docking ? [] : +                        [openAlias, focusDoc])]; +    } +    childContextMenuItems = () => { +        const customScripts = Cast(this.doc.childContextMenuScripts, listSpec(ScriptField), []); +        const customFilters = Cast(this.doc.childContextMenuFilters, listSpec(ScriptField), []); +        const icons = StrListCast(this.doc.childContextMenuIcons); +        return StrListCast(this.doc.childContextMenuLabels).map((label, i) => ({ script: customScripts[i], filter: customFilters[i], icon: icons[i], label }));      }      onChildClick = () => this.props.onChildClick?.() ?? (this._editTitleScript?.() || ScriptCast(this.doc.treeChildClick)); @@ -530,7 +553,7 @@ export class TreeView extends React.Component<TreeViewProps> {          switch (property.split(":")[0]) {              case StyleProp.Opacity: return this.props.treeView.outlineMode ? undefined : 1;              case StyleProp.BackgroundColor: return this.selected ? "#7089bb" : StrCast(doc._backgroundColor, StrCast(doc.backgroundColor)); -            case StyleProp.DocContents: return testDocProps(props) && !props?.treeViewDoc ? (null) : +            case StyleProp.DocContents: return this.props.treeView.outlineMode ? (null) :                  <div className="treeView-label" style={{    // just render a title for a tree view label (identified by treeViewDoc being set in 'props')                      maxWidth: props?.PanelWidth() || undefined,                      background: props?.styleProvider?.(doc, props, StyleProp.BackgroundColor), @@ -632,6 +655,7 @@ export class TreeView extends React.Component<TreeViewProps> {                  searchFilterDocs={returnEmptyDoclist}                  ContainingCollectionView={undefined}                  ContainingCollectionDoc={this.props.treeView.props.Document} +                ContentScaling={returnOne}              />;          const buttons = this.props.styleProvider?.(this.doc, this.props.treeView.props, StyleProp.Decorations + (Doc.IsSystem(this.props.containerCollection) ? ":afterHeader" : "")); @@ -688,10 +712,12 @@ export class TreeView extends React.Component<TreeViewProps> {              hideDecorationTitle={this.props.treeView.outlineMode}              hideResizeHandles={this.props.treeView.outlineMode}              focus={this.refocus} +            ContentScaling={returnOne}              hideLinkButton={BoolCast(this.props.treeView.props.Document.childHideLinkButton)}              dontRegisterView={BoolCast(this.props.treeView.props.Document.childDontRegisterViews, this.props.dontRegisterView)}              ScreenToLocalTransform={this.docTransform}              renderDepth={this.props.renderDepth + 1} +            treeViewDoc={this.props.treeView?.props.Document}              rootSelected={returnTrue}              layerProvider={returnTrue}              docViewPath={this.props.treeView.props.docViewPath} @@ -817,7 +843,8 @@ export class TreeView extends React.Component<TreeViewProps> {          whenChildContentsActiveChanged: (isActive: boolean) => void,          dontRegisterView: boolean | undefined,          observerHeight: (ref: any) => void, -        unobserveHeight: (ref: any) => void +        unobserveHeight: (ref: any) => void, +        contextMenuItems: ({ script: ScriptField, filter: ScriptField, label: string, icon: string }[])      ) {          const viewSpecScript = Cast(conainerCollection.viewSpecScript, ScriptField);          if (viewSpecScript) { @@ -834,6 +861,7 @@ export class TreeView extends React.Component<TreeViewProps> {              }              const dentDoc = (editTitle: boolean, newParent: Doc, addAfter: Doc | undefined, parent: TreeView | CollectionTreeView | undefined) => { +                if (parent instanceof TreeView && parent.props.treeView.fileSysMode && !newParent.isFolder) return;                  const fieldKey = Doc.LayoutFieldKey(newParent);                  if (remove && fieldKey && Cast(newParent[fieldKey], listSpec(Doc)) !== undefined) {                      remove(child); @@ -850,7 +878,7 @@ export class TreeView extends React.Component<TreeViewProps> {              const childLayout = Doc.Layout(pair.layout);              const rowHeight = () => {                  const aspect = Doc.NativeAspect(childLayout); -                return aspect ? Math.min(childLayout[WidthSym](), rowWidth()) / aspect : childLayout[HeightSym](); +                return (aspect ? Math.min(childLayout[WidthSym](), rowWidth()) / aspect : childLayout[HeightSym]());              };              return <TreeView key={child[Id]} ref={r => treeViewRefs.set(child, r ? r : undefined)}                  document={pair.layout} @@ -882,6 +910,7 @@ export class TreeView extends React.Component<TreeViewProps> {                  parentTreeView={parentTreeView}                  observeHeight={observerHeight}                  unobserveHeight={unobserveHeight} +                contextMenuItems={contextMenuItems}              />;          });      } | 
