diff options
Diffstat (limited to 'src')
68 files changed, 3902 insertions, 1565 deletions
| diff --git a/src/Utils.ts b/src/Utils.ts index dba802f98..a01a94134 100644 --- a/src/Utils.ts +++ b/src/Utils.ts @@ -7,6 +7,22 @@ import { ColorState } from 'react-color';  export namespace Utils {      export let DRAG_THRESHOLD = 4; +    export function readUploadedFileAsText(inputFile: File) { +        const temporaryFileReader = new FileReader(); + +        return new Promise((resolve, reject) => { +            temporaryFileReader.onerror = () => { +                temporaryFileReader.abort(); +                reject(new DOMException("Problem parsing input file.")); +            }; + +            temporaryFileReader.onload = () => { +                resolve(temporaryFileReader.result); +            }; +            temporaryFileReader.readAsText(inputFile); +        }); +    } +      export function GenerateGuid(): string {          return v4();      } diff --git a/src/client/apis/google_docs/GooglePhotosClientUtils.ts b/src/client/apis/google_docs/GooglePhotosClientUtils.ts index a604c7de1..13bfb3a91 100644 --- a/src/client/apis/google_docs/GooglePhotosClientUtils.ts +++ b/src/client/apis/google_docs/GooglePhotosClientUtils.ts @@ -130,7 +130,6 @@ export namespace GooglePhotos {              const uploads = await Transactions.WriteMediaItemsToServer(response);              const children = uploads.map((upload: Transactions.UploadInformation) => {                  const document = Docs.Create.ImageDocument(Utils.fileUrl(upload.fileNames.clean)); -                document.fillColumn = true;                  document.contentSize = upload.contentSize;                  return document;              }); diff --git a/src/client/documents/DocumentTypes.ts b/src/client/documents/DocumentTypes.ts index 454fb2ad2..86d8c6db7 100644 --- a/src/client/documents/DocumentTypes.ts +++ b/src/client/documents/DocumentTypes.ts @@ -33,8 +33,10 @@ export enum DocumentType {      DOCHOLDER = "docholder",    // nested document (view of a document)      SEARCHITEM= "searchitem",      COMPARISON = "comparison",   // before/after view with slider (view of 2 images) +    GROUP = "group",            // group of users      LINKDB = "linkdb",          // database of links  ??? why do we have this      SCRIPTDB = "scriptdb",          // database of scripts      RECOMMENDATION = "recommendation", // view of a recommendation +    GROUPDB = "groupdb"         // database of groups  }
\ No newline at end of file diff --git a/src/client/documents/Documents.ts b/src/client/documents/Documents.ts index 56c384eca..5a2d746d6 100644 --- a/src/client/documents/Documents.ts +++ b/src/client/documents/Documents.ts @@ -1,5 +1,5 @@  import { runInAction } from "mobx"; -import { extname } from "path"; +import { extname, basename } from "path";  import { DateField } from "../../fields/DateField";  import { Doc, DocListCast, DocListCastAsync, Field, HeightSym, Opt, WidthSym } from "../../fields/Doc";  import { HtmlField } from "../../fields/HtmlField"; @@ -49,6 +49,9 @@ import { WebBox } from "../views/nodes/WebBox";  import { PresElementBox } from "../views/presentationview/PresElementBox";  import { RecommendationsBox } from "../views/RecommendationsBox";  import { DashWebRTCVideo } from "../views/webcam/DashWebRTCVideo"; +import { DocumentType } from "./DocumentTypes"; +import { Networking } from "../Network"; +import { Upload } from "../../server/SharedMediaTypes";  const path = require('path');  export interface DocumentOptions { @@ -63,7 +66,7 @@ export interface DocumentOptions {      _dimUnit?: string; // "px" or "*" (default = "*")      _fitWidth?: boolean;      _fitToBox?: boolean; // whether a freeformview should zoom/scale to create a shrinkwrapped view of its contents -    _LODdisable?: boolean; +    _freeformLOD?: boolean; // whether to use LOD to render a freeform document      _showTitleHover?: string; //       _showTitle?: string; // which field to display in the title area.  leave empty to have no title      _showCaption?: string; // which field to display in the caption area.  leave empty to have no caption @@ -100,7 +103,7 @@ export interface DocumentOptions {      childLayoutTemplate?: Doc; // template for collection to use to render its children (see PresBox or Buxton layout in tree view)      childLayoutString?: string; // template string for collection to use to render its children      hideFilterView?: boolean; // whether to hide the filter popout on collections -    hideHeadings?: boolean; // whether stacking view column headings should be hidden +    _columnsHideIfEmpty?: boolean; // whether stacking view column headings should be hidden      isTemplateForField?: string; // the field key for which the containing document is a rendering template      isTemplateDoc?: boolean;      targetScriptKey?: string; // where to write a template script (used by collections with click templates which need to target onClick, onDoubleClick, etc) @@ -120,7 +123,7 @@ export interface DocumentOptions {      defaultBackgroundColor?: string;      isBackground?: boolean;      isLinkButton?: boolean; -    columnWidth?: number; +    _columnWidth?: number;      _fontSize?: number;      _fontFamily?: string;      curPage?: number; @@ -138,7 +141,8 @@ export interface DocumentOptions {      "onCheckedClick-rawScript"?: string; // onChecked script in raw text form      "onCheckedClick-params"?: List<string>; // parameter list for onChecked treeview functions      _pivotField?: string; // field key used to determine headings for sections in stacking, masonry, pivot views -    schemaColumns?: List<SchemaHeaderField>; +    _columnHeaders?: List<SchemaHeaderField>; // headers for stacking views +    _schemaHeaders?: List<SchemaHeaderField>; // headers for schema view      dockingConfig?: string;      annotationOn?: Doc;      removeDropProperties?: List<string>; // list of properties that should be removed from a document when it is dropped.  e.g., a creator button may be forceActive to allow it be dragged, but the forceActive property can be removed from the dropped document @@ -316,6 +320,14 @@ export namespace Docs {              [DocumentType.COMPARISON, {                  layout: { view: ComparisonBox, dataField: defaultDataKey },              }], +            [DocumentType.GROUPDB, { +                data: new List<Doc>(), +                layout: { view: EmptyBox, dataField: defaultDataKey }, +                options: { childDropAction: "alias", title: "Global Group Database" } +            }], +            [DocumentType.GROUP, { +                layout: { view: EmptyBox, dataField: defaultDataKey } +            }]          ]);          // All document prototypes are initialized with at least these values @@ -379,6 +391,13 @@ export namespace Docs {          }          /** +         * A collection of all groups in the database +         */ +        export function MainGroupDocument() { +            return Prototypes.get(DocumentType.GROUPDB); +        } + +        /**           * This is a convenience method that is used to initialize           * prototype documents for the first time.           *  @@ -403,7 +422,7 @@ export namespace Docs {              // synthesize the default options, the type and title from computed values and              // whatever options pertain to this specific prototype              const options = { title, type, baseProto: true, ...defaultOptions, ...(template.options || {}) }; -            options.layout = layout.view.LayoutString(layout.dataField); +            options.layout = layout.view?.LayoutString(layout.dataField);              const doc = Doc.assign(new Doc(prototypeId, true), { layoutKey: "layout", ...options });              doc.layout_keyValue = KeyValueBox.LayoutString("");              return doc; @@ -433,8 +452,7 @@ export namespace Docs {              const parent = TreeDocument([loading], {                  title: "The Buxton Collection",                  _width: 400, -                _height: 400, -                _LODdisable: true +                _height: 400              });              const parentProto = Doc.GetProto(parent);              const { _socket } = DocServer; @@ -470,7 +488,7 @@ export namespace Docs {                          return imageDoc;                      });                      // the main document we create -                    const doc = StackingDocument(deviceImages, { title, _LODdisable: true, hero: new ImageField(constructed[0].url) }); +                    const doc = StackingDocument(deviceImages, { title, hero: new ImageField(constructed[0].url) });                      doc.nameAliases = new List<string>([title.toLowerCase()]);                      // add the parsed attributes to this main document                      Doc.Get.FromJson({ data: device, appendToExisting: { targetDoc: Doc.GetProto(doc) } }); @@ -703,15 +721,15 @@ export namespace Docs {          }          export function FreeformDocument(documents: Array<Doc>, options: DocumentOptions, id?: string) { -            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", schemaColumns: new List([new SchemaHeaderField("title", "#f1efeb")]), ...options, _viewType: CollectionViewType.Freeform }, id); +            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", ...options, _viewType: CollectionViewType.Freeform }, id);          }          export function PileDocument(documents: Array<Doc>, options: DocumentOptions, id?: string) { -            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", backgroundColor: "black", schemaColumns: new List([new SchemaHeaderField("title", "#f1efeb")]), ...options, _viewType: CollectionViewType.Pile }, id); +            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", backgroundColor: "black", ...options, _viewType: CollectionViewType.Pile }, id);          }          export function LinearDocument(documents: Array<Doc>, options: DocumentOptions, id?: string) { -            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", backgroundColor: "black", schemaColumns: new List([new SchemaHeaderField("title", "#f1efeb")]), ...options, _viewType: CollectionViewType.Linear }, id); +            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", backgroundColor: "black", ...options, _viewType: CollectionViewType.Linear }, id);          }          export function MapDocument(documents: Array<Doc>, options: DocumentOptions = {}) { @@ -719,35 +737,35 @@ export namespace Docs {          }          export function CarouselDocument(documents: Array<Doc>, options: DocumentOptions) { -            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", schemaColumns: new List([new SchemaHeaderField("title", "#f1efeb")]), ...options, _viewType: CollectionViewType.Carousel }); +            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", ...options, _viewType: CollectionViewType.Carousel });          }          export function Carousel3DDocument(documents: Array<Doc>, options: DocumentOptions) { -            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", schemaColumns: new List([new SchemaHeaderField("title", "#f1efeb")]), ...options, _viewType: CollectionViewType.Carousel3D }); +            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", ...options, _viewType: CollectionViewType.Carousel3D });          } -        export function SchemaDocument(schemaColumns: SchemaHeaderField[], documents: Array<Doc>, options: DocumentOptions) { -            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", schemaColumns: new List(schemaColumns.length ? schemaColumns : [new SchemaHeaderField("title", "#f1efeb")]), ...options, _viewType: CollectionViewType.Schema }); +        export function SchemaDocument(schemaHeaders: SchemaHeaderField[], documents: Array<Doc>, options: DocumentOptions) { +            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", _schemaHeaders: schemaHeaders.length ? new List(schemaHeaders) : undefined, ...options, _viewType: CollectionViewType.Schema });          }          export function TreeDocument(documents: Array<Doc>, options: DocumentOptions, id?: string) { -            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", schemaColumns: new List([new SchemaHeaderField("title", "#f1efeb")]), ...options, _viewType: CollectionViewType.Tree }, id); +            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", ...options, _viewType: CollectionViewType.Tree }, id);          }          export function StackingDocument(documents: Array<Doc>, options: DocumentOptions, id?: string) { -            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", schemaColumns: new List([new SchemaHeaderField("title", "#f1efeb")]), ...options, _viewType: CollectionViewType.Stacking }, id); +            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", ...options, _viewType: CollectionViewType.Stacking }, id);          }          export function MulticolumnDocument(documents: Array<Doc>, options: DocumentOptions) { -            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", schemaColumns: new List([new SchemaHeaderField("title", "#f1efeb")]), ...options, _viewType: CollectionViewType.Multicolumn }); +            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", ...options, _viewType: CollectionViewType.Multicolumn });          }          export function MultirowDocument(documents: Array<Doc>, options: DocumentOptions) { -            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", schemaColumns: new List([new SchemaHeaderField("title", "#f1efeb")]), ...options, _viewType: CollectionViewType.Multirow }); +            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", ...options, _viewType: CollectionViewType.Multirow });          }          export function MasonryDocument(documents: Array<Doc>, options: DocumentOptions) { -            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", schemaColumns: new List([new SchemaHeaderField("title", "#f1efeb")]), ...options, _viewType: CollectionViewType.Masonry }); +            return InstanceFromProto(Prototypes.get(DocumentType.COL), new List(documents), { _chromeStatus: "collapsed", ...options, _viewType: CollectionViewType.Masonry });          }          export function LabelDocument(options?: DocumentOptions) { @@ -971,7 +989,7 @@ export namespace DocUtils {          });          ContextMenu.Instance.addItem({              description: "Add Template Doc ...", -            subitems: DocListCast(Cast(Doc.UserDoc().dockedBtns, Doc, null)?.data).map(btnDoc => Cast(btnDoc?.dragFactory, Doc, null)).filter(doc => doc).map((dragDoc, i) => ({ +            subitems: DocListCast(Cast(Doc.UserDoc().myItemCreators, Doc, null)?.data).map(btnDoc => Cast(btnDoc?.dragFactory, Doc, null)).filter(doc => doc).map((dragDoc, i) => ({                  description: ":" + StrCast(dragDoc.title),                  event: (args: { x: number, y: number }) => {                      const newDoc = Doc.ApplyTemplate(dragDoc); @@ -1063,7 +1081,7 @@ export namespace DocUtils {              });          });          if (x !== undefined && y !== undefined) { -            const newCollection = Docs.Create.PileDocument(docList, { title: "pileup", x: x - 55, y: y - 55, _width: 110, _height: 100, _LODdisable: true }); +            const newCollection = Docs.Create.PileDocument(docList, { title: "pileup", x: x - 55, y: y - 55, _width: 110, _height: 100 });              newCollection.x = NumCast(newCollection.x) + NumCast(newCollection._width) / 2 - 55;              newCollection.y = NumCast(newCollection.y) + NumCast(newCollection._height) / 2 - 55;              newCollection._width = newCollection._height = 110; @@ -1098,6 +1116,32 @@ export namespace DocUtils {          });          return optionsCollection;      } + +    export async function uploadFilesToDocs(files: File[], options: DocumentOptions) { +        const generatedDocuments: Doc[] = []; +        for (const { source: { name, type }, result } of await Networking.UploadFilesToServer(files)) { +            if (result instanceof Error) { +                alert(`Upload failed: ${result.message}`); +                return []; +            } +            const full = { ...options, _width: 400, title: name }; +            const pathname = Utils.prepend(result.accessPaths.agnostic.client); +            const doc = await DocUtils.DocumentFromType(type, pathname, full); +            if (!doc) { +                continue; +            } +            const proto = Doc.GetProto(doc); +            proto.text = result.rawText; +            proto.fileUpload = basename(pathname).replace("upload_", "").replace(/\.[a-z0-9]*$/, ""); +            if (Upload.isImageInformation(result)) { +                proto["data-nativeWidth"] = (result.nativeWidth > result.nativeHeight) ? 400 * result.nativeWidth / result.nativeHeight : 400; +                proto["data-nativeHeight"] = (result.nativeWidth > result.nativeHeight) ? 400 : 400 / (result.nativeWidth / result.nativeHeight); +                proto.contentSize = result.contentSize; +            } +            generatedDocuments.push(doc); +        } +        return generatedDocuments; +    }  }  Scripting.addGlobal("Docs", Docs); diff --git a/src/client/util/CurrentUserUtils.ts b/src/client/util/CurrentUserUtils.ts index c5db002f4..276ad4c90 100644 --- a/src/client/util/CurrentUserUtils.ts +++ b/src/client/util/CurrentUserUtils.ts @@ -45,7 +45,7 @@ export class CurrentUserUtils {                      Docs.Create.SearchDocument({ _viewType: CollectionViewType.Stacking, ignoreClick: true, forceActive: true, lockedPosition: true, title: "query", _height: 200 }),                      Docs.Create.FreeformDocument([], { title: "data", _height: 100, _LODdisable: true })                  ], -                { _width: 400, _height: 300, title: "queryView", _chromeStatus: "disabled", _xMargin: 3,  _yMargin: 3, hideFilterView: true } +                { _width: 400, _height: 300, title: "queryView", _chromeStatus: "disabled", _xMargin: 3, _yMargin: 3, hideFilterView: true }              );              queryTemplate.isTemplateDoc = makeTemplate(queryTemplate);              doc["template-button-query"] = CurrentUserUtils.ficon({ @@ -136,9 +136,9 @@ export class CurrentUserUtils {          if (doc["template-button-switch"] === undefined) {              const { FreeformDocument, MulticolumnDocument, TextDocument } = Docs.Create; -            const yes = FreeformDocument([], { title: "yes", _height: 35, _width: 50, _LODdisable: true, _dimUnit: DimUnit.Pixel, _dimMagnitude: 40 }); +            const yes = FreeformDocument([], { title: "yes", _height: 35, _width: 50, _dimUnit: DimUnit.Pixel, _dimMagnitude: 40 });              const name = TextDocument("name", { title: "name", _height: 35, _width: 70, _dimMagnitude: 1 }); -            const no = FreeformDocument([], { title: "no", _height: 100, _width: 100, _LODdisable: true }); +            const no = FreeformDocument([], { title: "no", _height: 100, _width: 100 });              const labelTemplate = {                  doc: {                      type: "doc", content: [{ @@ -193,10 +193,10 @@ export class CurrentUserUtils {              const shared = { _chromeStatus: "disabled", _autoHeight: true, _xMargin: 0 };              const detailViewOpts = { title: "detailView", _width: 300, _fontFamily: "Arial", _fontSize: 12 }; -            const descriptionWrapperOpts = { title: "descriptions", _height: 300, columnWidth: -1, treeViewHideTitle: true, _pivotField: "title" }; +            const descriptionWrapperOpts = { title: "descriptions", _height: 300, _columnWidth: -1, treeViewHideTitle: true, _pivotField: "title" };              const descriptionWrapper = MasonryDocument([details, short, long], { ...shared, ...descriptionWrapperOpts }); -            descriptionWrapper.sectionHeaders = new List<SchemaHeaderField>([ +            descriptionWrapper._columnHeaders = new List<SchemaHeaderField>([                  new SchemaHeaderField("[A Short Description]", "dimGray", undefined, undefined, undefined, false),                  new SchemaHeaderField("[Long Description]", "dimGray", undefined, undefined, undefined, true),                  new SchemaHeaderField("[Details]", "dimGray", undefined, undefined, undefined, true), @@ -225,7 +225,7 @@ export class CurrentUserUtils {          if (doc["template-buttons"] === undefined) {              doc["template-buttons"] = new PrefetchProxy(Docs.Create.MasonryDocument(requiredTypes, {                  title: "Advanced Item Prototypes", _xMargin: 0, _showTitle: "title", -                _autoHeight: true, _width: 500, columnWidth: 35, ignoreClick: true, lockedPosition: true, _chromeStatus: "disabled", +                _autoHeight: true, _width: 500, _columnWidth: 35, ignoreClick: true, lockedPosition: true, _chromeStatus: "disabled",                  dropConverter: ScriptField.MakeScript("convertToButtons(dragData)", { dragData: DragManager.DocumentDragData.name }),              }));          } else { @@ -358,11 +358,17 @@ export class CurrentUserUtils {      }[] {          if (doc.emptyPresentation === undefined) {              doc.emptyPresentation = Docs.Create.PresDocument(new List<Doc>(), -                { title: "Presentation", _viewType: CollectionViewType.Stacking, targetDropAction: "alias", _LODdisable: true, _chromeStatus: "replaced", _showTitle: "title", boxShadow: "0 0" }); +                { title: "Presentation", _viewType: CollectionViewType.Stacking, targetDropAction: "alias", _chromeStatus: "replaced", _showTitle: "title", boxShadow: "0 0" });          }          if (doc.emptyCollection === undefined) {              doc.emptyCollection = Docs.Create.FreeformDocument([], -                { _nativeWidth: undefined, _nativeHeight: undefined, _LODdisable: true, _width: 150, _height: 100, title: "freeform" }); +                { _nativeWidth: undefined, _nativeHeight: undefined, _width: 150, _height: 100, title: "freeform" }); +        } +        if (doc.emptyComparison === undefined) { +            doc.emptyComparison = Docs.Create.ComparisonDocument({ title: "compare", _width: 300, _height: 300 }); +        } +        if (doc.emptyScript === undefined) { +            doc.emptyScript = Docs.Create.ScriptingDocument(undefined, { _width: 200, _height: 250, title: "script" });          }          if (doc.emptyDocHolder === undefined) {              doc.emptyDocHolder = Docs.Create.DocumentDocument( @@ -370,10 +376,10 @@ export class CurrentUserUtils {                  { _width: 250, _height: 250, title: "container" });          }          if (doc.emptyWebpage === undefined) { -            doc.emptyWebpage = Docs.Create.WebDocument("", { title: "New Webpage", _nativeWidth: 850, _nativeHeight: 962, _width: 600, UseCors: true }); +            doc.emptyWebpage = Docs.Create.WebDocument("", { title: "webpage", _nativeWidth: 850, _nativeHeight: 962, _width: 600, UseCors: true });          }          return [ -            { title: "Drag a comparison box", label: "Comp", icon: "columns", ignoreClick: true, drag: 'Docs.Create.ComparisonDocument()' }, +            { title: "Drag a comparison box", label: "Comp", icon: "columns", click: 'openOnRight(getCopy(this.dragFactory, true))', drag: 'getCopy(this.dragFactory, true)', dragFactory: doc.emptyComparison as Doc },              { title: "Drag a collection", label: "Col", icon: "folder", click: 'openOnRight(getCopy(this.dragFactory, true))', drag: 'getCopy(this.dragFactory, true)', dragFactory: doc.emptyCollection as Doc },              { title: "Drag a web page", label: "Web", icon: "globe-asia", click: 'openOnRight(getCopy(this.dragFactory, true))', drag: 'getCopy(this.dragFactory, true)', dragFactory: doc.emptyWebpage as Doc },              { title: "Drag a cat image", label: "Img", icon: "cat", ignoreClick: true, drag: 'Docs.Create.ImageDocument("https://upload.wikimedia.org/wikipedia/commons/thumb/3/3a/Cat03.jpg/1200px-Cat03.jpg", { _width: 250, _nativeWidth:250, title: "an image of a cat" })' }, @@ -430,7 +436,7 @@ export class CurrentUserUtils {          if (dragCreatorSet === undefined) {              doc.myItemCreators = new PrefetchProxy(Docs.Create.MasonryDocument(creatorBtns, {                  title: "Basic Item Creators", _showTitle: "title", _xMargin: 0, -                _autoHeight: true, _width: 500, columnWidth: 35, ignoreClick: true, lockedPosition: true, _chromeStatus: "disabled", +                _autoHeight: true, _width: 500, _columnWidth: 35, ignoreClick: true, lockedPosition: true, _chromeStatus: "disabled",                  dropConverter: ScriptField.MakeScript("convertToButtons(dragData)", { dragData: DragManager.DocumentDragData.name }),              }));          } else { @@ -493,7 +499,7 @@ export class CurrentUserUtils {      static setupMobileDoc(userDoc: Doc) {          return userDoc.activeMoble ?? Docs.Create.MasonryDocument(CurrentUserUtils.setupMobileButtons(userDoc), { -            columnWidth: 100, ignoreClick: true, lockedPosition: true, _chromeStatus: "disabled", title: "buttons", _autoHeight: true, _yMargin: 5 +            _columnWidth: 100, ignoreClick: true, lockedPosition: true, _chromeStatus: "disabled", title: "buttons", _autoHeight: true, _yMargin: 5          });      } @@ -625,7 +631,7 @@ export class CurrentUserUtils {              doc["tabs-button-search"] = new PrefetchProxy(Docs.Create.ButtonDocument({                  _width: 50, _height: 25, title: "Search", _fontSize: 10,                  letterSpacing: "0px", textTransform: "unset", borderRounding: "5px 5px 0px 0px", boxShadow: "3px 3px 0px rgb(34, 34, 34)", -                sourcePanel: new PrefetchProxy(Docs.Create.SearchDocument({ignoreClick: true, childDropAction: "alias", lockedPosition: true, _viewType: CollectionViewType.Stacking, title: "sidebar search stack", })) as any as Doc, +                sourcePanel: new PrefetchProxy(Docs.Create.SearchDocument({ ignoreClick: true, childDropAction: "alias", lockedPosition: true, _viewType: CollectionViewType.Stacking, title: "sidebar search stack", })) as any as Doc,                  searchFileTypes: new List<string>([DocumentType.RTF, DocumentType.IMG, DocumentType.PDF, DocumentType.VID, DocumentType.WEB, DocumentType.SCRIPTING]),                  targetContainer: new PrefetchProxy(sidebarContainer) as any as Doc,                  lockedPosition: true, @@ -654,8 +660,8 @@ export class CurrentUserUtils {          // Finally, setup the list of buttons to display in the sidebar          if (doc["tabs-buttons"] === undefined) { -            doc["tabs-buttons"] = new PrefetchProxy(Docs.Create.StackingDocument([searchBtn, libraryBtn, toolsBtn], { -                _width: 500, _height: 80, boxShadow: "0 0", _pivotField: "title", hideHeadings: true, ignoreClick: true, _chromeStatus: "view-mode", +            doc["tabs-buttons"] = new PrefetchProxy(Docs.Create.StackingDocument([libraryBtn, searchBtn, toolsBtn], { +                _width: 500, _height: 80, boxShadow: "0 0", _pivotField: "title", _columnsHideIfEmpty: true, ignoreClick: true, _chromeStatus: "view-mode",                  title: "sidebar btn row stack", backgroundColor: "dimGray",              }));              (toolsBtn.onClick as ScriptField).script.run({ this: toolsBtn }); @@ -709,7 +715,7 @@ export class CurrentUserUtils {          if (doc.activePresentation === undefined) {              doc.activePresentation = Docs.Create.PresDocument(new List<Doc>(), {                  title: "Presentation", _viewType: CollectionViewType.Stacking, targetDropAction: "alias", -                _LODdisable: true, _chromeStatus: "replaced", _showTitle: "title", boxShadow: "0 0" +                _chromeStatus: "replaced", _showTitle: "title", boxShadow: "0 0"              });          }      } @@ -782,6 +788,7 @@ export class CurrentUserUtils {          await this.setupSidebarButtons(doc); // the pop-out left sidebar of tools/panels          doc.globalLinkDatabase = Docs.Prototypes.MainLinkDocument();          doc.globalScriptDatabase = Docs.Prototypes.MainScriptDocument(); +        doc.globalGroupDatabase = Docs.Prototypes.MainGroupDocument();          // setup reactions to change the highlights on the undo/redo buttons -- would be better to encode this in the undo/redo buttons, but the undo/redo stacks are not wired up that way yet          doc["dockedBtn-undo"] && reaction(() => UndoManager.undoStack.slice(), () => Doc.GetProto(doc["dockedBtn-undo"] as Doc).opacity = UndoManager.CanUndo() ? 1 : 0.4, { fireImmediately: true }); diff --git a/src/client/util/DocumentManager.ts b/src/client/util/DocumentManager.ts index fb5d1717e..1fa5faeb3 100644 --- a/src/client/util/DocumentManager.ts +++ b/src/client/util/DocumentManager.ts @@ -146,7 +146,7 @@ export class DocumentManager {          };          const docView = getFirstDocView(targetDoc, originatingDoc);          let annotatedDoc = await Cast(targetDoc.annotationOn, Doc); -        if (annotatedDoc) { +        if (annotatedDoc && !linkDoc?.isPushpin) {              const first = getFirstDocView(annotatedDoc);              if (first) {                  annotatedDoc = first.props.Document; @@ -156,7 +156,11 @@ export class DocumentManager {              }          }          if (docView) {  // we have a docView already and aren't forced to create a new one ... just focus on the document.  TODO move into view if necessary otherwise just highlight? -            docView.props.focus(docView.props.Document, willZoom, undefined, focusAndFinish); +            if (linkDoc?.isPushpin) docView.props.Document.hidden = !docView.props.Document.hidden; +            else { +                docView.props.Document.hidden && (docView.props.Document.hidden = undefined); +                docView.props.focus(docView.props.Document, willZoom, undefined, focusAndFinish); +            }              highlight();          } else {              const contextDocs = docContext ? await DocListCastAsync(docContext.data) : undefined; diff --git a/src/client/util/DragManager.ts b/src/client/util/DragManager.ts index 0db3963b2..2ceafff30 100644 --- a/src/client/util/DragManager.ts +++ b/src/client/util/DragManager.ts @@ -202,7 +202,6 @@ export namespace DragManager {              dropDoc instanceof Doc && DocUtils.MakeLinkToActiveAudio(dropDoc);              return dropDoc;          }; -        const batch = UndoManager.StartBatch("dragging");          const finishDrag = (e: DragCompleteEvent) => {              const docDragData = e.docDragData;              if (docDragData && !docDragData.droppedDocuments.length) { @@ -216,7 +215,6 @@ export namespace DragManager {                      const remProps = (dragData?.removeDropProperties || []).concat(Array.from(dragProps));                      remProps.map(prop => drop[prop] = undefined);                  }); -                batch.end();              }              return e;          }; @@ -315,6 +313,7 @@ export namespace DragManager {      export let docsBeingDragged: Doc[] = [];      export let CanEmbed = false;      export function StartDrag(eles: HTMLElement[], dragData: { [id: string]: any }, downX: number, downY: number, options?: DragOptions, finishDrag?: (dropData: DragCompleteEvent) => void) { +        const batch = UndoManager.StartBatch("dragging");          eles = eles.filter(e => e);          CanEmbed = false;          if (!dragDiv) { @@ -449,6 +448,7 @@ export namespace DragManager {              document.removeEventListener("pointermove", moveHandler, true);              document.removeEventListener("pointerup", upHandler);              SnappingManager.clearSnapLines(); +            batch.end();          });          AbortDrag = () => { diff --git a/src/client/util/GroupManager.scss b/src/client/util/GroupManager.scss new file mode 100644 index 000000000..544a79e98 --- /dev/null +++ b/src/client/util/GroupManager.scss @@ -0,0 +1,136 @@ +@import "../views/globalCssVariables"; + +.group-interface { +    background-color: whitesmoke !important; +    color: grey; +    width: 450px; +    height: 300px; + +    .dialogue-box { +        width: 450; +        height: 300; +    } + +    button { +        background: $lighter-alt-accent; +        outline: none; +        border-radius: 5px; +        border: 0px; +        color: #fcfbf7; +        text-transform: uppercase; +        letter-spacing: 2px; +        font-size: 75%; +        padding: 10px; +        margin: 10px; +        transition: transform 0.2s; +        margin: 2px; +    } +} + +.group-interface { +    display: flex; +    flex-direction: column; + +    .overlay { +        transform: translate(-20px, -20px); +        border-radius: 10px; +    } + +    button { +        width: 100%; +        align-self: center; +        background: $darker-alt-accent; +    } + +    .delete-button { +        background: rgb(227, 86, 86); +    } + +    .close-button { +        position: absolute; +        right: 1em; +        top: 1em; +        cursor: pointer; +        z-index: 999; +    } + +    .group-heading { +        letter-spacing: .5em; +    } + + +    .group-body { +        display: flex; +        justify-content: space-between; +        max-height: 80%; + +        .group-create { +            display: flex; +            flex-direction: column; +            flex-basis: 30%; +            margin-left: 5px; + +            input { +                border-radius: 5px; +                border: none; +                padding: 4px; +                min-width: 100%; +                margin: 4px 0 4px 0; +            } + +        } + +        .group-content { +            padding-left: 1em; +            padding-right: 1em; +            justify-content: space-around; +            text-align: left; + +            overflow-y: auto; +            width: 100%; + +            .group-row { +                display: flex; +                position: relative; +                margin-bottom: 5px; +                min-height: 40px; +                border: 1px solid; +                border-radius: 10px; +                align-items: center; + +                .group-name { +                    position: relative; +                    max-width: 65%; +                    left: 10; +                } + +                button { +                    position: absolute; +                    width: 30%; +                    right: 2; +                    margin-top: 0; +                } +            } + +            ::placeholder { +                color: $intermediate-color; +            } + +            input { +                border-radius: 5px; +                border: none; +                padding: 4px; +                min-width: 100%; +                margin: 2px 0; +            } + +        } +    } + +    h1 { +        color: $dark-color; +        text-transform: uppercase; +        letter-spacing: 2px; +        font-size: 120%; +    } +}
\ No newline at end of file diff --git a/src/client/util/GroupManager.tsx b/src/client/util/GroupManager.tsx new file mode 100644 index 000000000..7c68fc2a0 --- /dev/null +++ b/src/client/util/GroupManager.tsx @@ -0,0 +1,360 @@ +import * as React from "react"; +import { observable, action, runInAction, computed } from "mobx"; +import { SelectionManager } from "./SelectionManager"; +import MainViewModal from "../views/MainViewModal"; +import { observer } from "mobx-react"; +import { Doc, DocListCast, Opt } from "../../fields/Doc"; +import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; +import * as fa from '@fortawesome/free-solid-svg-icons'; +import { library } from "@fortawesome/fontawesome-svg-core"; +import SharingManager, { User } from "./SharingManager"; +import { Utils } from "../../Utils"; +import * as RequestPromise from "request-promise"; +import Select from 'react-select'; +import "./GroupManager.scss"; +import { StrCast } from "../../fields/Types"; +import GroupMemberView from "./GroupMemberView"; + +library.add(fa.faWindowClose); + +export interface UserOptions { +    label: string; +    value: string; +} + +@observer +export default class GroupManager extends React.Component<{}> { + +    static Instance: GroupManager; +    @observable isOpen: boolean = false; // whether the GroupManager is to be displayed or not. +    @observable private dialogueBoxOpacity: number = 1; // opacity of the dialogue box div of the MainViewModal. +    @observable private overlayOpacity: number = 0.4; // opacity of the overlay div of the MainViewModal. +    @observable private users: string[] = []; // list of users populated from the database. +    @observable private selectedUsers: UserOptions[] | null = null; // list of users selected in the "Select users" dropdown. +    @observable currentGroup: Opt<Doc>; // the currently selected group. +    private inputRef: React.RefObject<HTMLInputElement> = React.createRef(); // the ref for the input box. + +    constructor(props: Readonly<{}>) { +        super(props); +        GroupManager.Instance = this; +    } + +    // sets up the list of users +    componentDidMount() { +        this.populateUsers().then(resolved => runInAction(() => this.users = resolved)); +    } + +    /** +     * Fetches the list of users stored on the database and @returns a list of the emails. +     */ +    populateUsers = async () => { +        const userList: User[] = JSON.parse(await RequestPromise.get(Utils.prepend("/getUsers"))); +        const currentUserIndex = userList.findIndex(user => user.email === Doc.CurrentUserEmail); +        currentUserIndex !== -1 && userList.splice(currentUserIndex, 1); +        return userList.map(user => user.email); +    } + +    /** +     * @returns the options to be rendered in the dropdown menu to add users and create a group. +     */ +    @computed get options() { +        return this.users.map(user => ({ label: user, value: user })); +    } + +    /** +     * Makes the GroupManager visible. +     */ +    @action +    open = () => { +        SelectionManager.DeselectAll(); +        this.isOpen = true; +    } + +    /** +     * Hides the GroupManager. +    */ +    @action +    close = () => { +        this.isOpen = false; +        this.currentGroup = undefined; +    } + +    /** +     * @returns the database of groups. +     */ +    get GroupManagerDoc(): Doc | undefined { +        return Doc.UserDoc().globalGroupDatabase as Doc; +    } + +    /** +     * @returns a list of all group documents. +     */ +    private getAllGroups(): Doc[] { +        const groupDoc = this.GroupManagerDoc; +        return groupDoc ? DocListCast(groupDoc.data) : []; +    } + +    /** +     * @returns a group document based on the group name. +     * @param groupName  +     */ +    private getGroup(groupName: string): Doc | undefined { +        const groupDoc = this.getAllGroups().find(group => group.groupName === groupName); +        return groupDoc; +    } + +    /** +     * @returns a readonly copy of a single group document +     */ +    getGroupCopy(groupName: string): Doc | undefined { +        const groupDoc = this.getGroup(groupName); +        if (groupDoc) { +            const { members, owners } = groupDoc; +            return Doc.assign(new Doc, { groupName, members: StrCast(members), owners: StrCast(owners) }); +        } +        return undefined; +    } +    /** +     * @returns a readonly copy of the list of group documents +     */ +    getAllGroupsCopy(): Doc[] { +        return this.getAllGroups().map(({ groupName, owners, members }) => +            Doc.assign(new Doc, { groupName: (StrCast(groupName)), owners: (StrCast(owners)), members: (StrCast(members)) }) +        ); +    } + +    /** +     * @returns the members of the admin group. +     */ +    get adminGroupMembers(): string[] { +        return this.getGroup("admin") ? JSON.parse(StrCast(this.getGroup("admin")!.members)) : ""; +    } + +    /** +     * @returns a boolean indicating whether the current user has access to edit group documents. +     * @param groupDoc  +     */ +    hasEditAccess(groupDoc: Doc): boolean { +        if (!groupDoc) return false; +        const accessList: string[] = JSON.parse(StrCast(groupDoc.owners)); +        return accessList.includes(Doc.CurrentUserEmail) || this.adminGroupMembers?.includes(Doc.CurrentUserEmail); +    } + +    /** +     * Helper method that sets up the group document. +     * @param groupName  +     * @param memberEmails  +     */ +    createGroupDoc(groupName: string, memberEmails: string[] = []) { +        const groupDoc = new Doc; +        groupDoc.groupName = groupName; +        groupDoc.owners = JSON.stringify([Doc.CurrentUserEmail]); +        groupDoc.members = JSON.stringify(memberEmails); +        this.addGroup(groupDoc); +    } + +    /** +     * Helper method that adds a group document to the database of group documents and @returns whether it was successfully added or not. +     * @param groupDoc  +     */ +    addGroup(groupDoc: Doc): boolean { +        if (this.GroupManagerDoc) { +            Doc.AddDocToList(this.GroupManagerDoc, "data", groupDoc); +            return true; +        } +        return false; +    } + +    /** +     * Deletes a group from the database of group documents and @returns whether the group was deleted or not. +     * @param group  +     */ +    deleteGroup(group: Doc): boolean { +        if (group) { +            if (this.GroupManagerDoc && this.hasEditAccess(group)) { +                Doc.RemoveDocFromList(this.GroupManagerDoc, "data", group); +                SharingManager.Instance.setInternalGroupSharing(group, "Not Shared"); +                if (group === this.currentGroup) { +                    runInAction(() => this.currentGroup = undefined); +                } +                return true; +            } +        } +        return false; +    } + +    /** +     * Adds a member to a group. +     * @param groupDoc  +     * @param email  +     */ +    addMemberToGroup(groupDoc: Doc, email: string) { +        if (this.hasEditAccess(groupDoc)) { +            const memberList: string[] = JSON.parse(StrCast(groupDoc.members)); +            !memberList.includes(email) && memberList.push(email); +            groupDoc.members = JSON.stringify(memberList); +        } +    } + +    /** +     * Removes a member from the group. +     * @param groupDoc  +     * @param email  +     */ +    removeMemberFromGroup(groupDoc: Doc, email: string) { +        if (this.hasEditAccess(groupDoc)) { +            const memberList: string[] = JSON.parse(StrCast(groupDoc.members)); +            const index = memberList.indexOf(email); +            index !== -1 && memberList.splice(index, 1); +            groupDoc.members = JSON.stringify(memberList); +        } +    } + +    /** +     * Handles changes in the users selected in the "Select users" dropdown. +     * @param selectedOptions  +     */ +    @action +    handleChange = (selectedOptions: any) => { +        this.selectedUsers = selectedOptions as UserOptions[]; +    } + +    /** +     * Creates the group when the enter key has been pressed (when in the input). +     * @param e  +     */ +    handleKeyDown = (e: React.KeyboardEvent) => { +        e.key === "Enter" && this.createGroup(); +    } + +    /** +     * Handles the input of required fields in the setup of a group and resets the relevant variables. +     */ +    @action +    createGroup = () => { +        if (!this.inputRef.current?.value) { +            alert("Please enter a group name"); +            return; +        } +        if (this.getAllGroups().find(group => group.groupName === this.inputRef.current!.value)) { // why do I need a null check here? +            alert("Please select a unique group name"); +            return; +        } +        this.createGroupDoc(this.inputRef.current.value, this.selectedUsers?.map(user => user.value)); +        this.selectedUsers = null; +        this.inputRef.current.value = ""; +    } + +    /** +     * A getter that @returns the interface rendered to view an individual group. +     */ +    private get editingInterface() { +        const members: string[] = this.currentGroup ? JSON.parse(StrCast(this.currentGroup.members)) : []; +        const options: UserOptions[] = this.currentGroup ? this.options.filter(option => !(JSON.parse(StrCast(this.currentGroup!.members)) as string[]).includes(option.value)) : []; +        return (!this.currentGroup ? null : +            <div className="editing-interface"> +                <div className="editing-header"> +                    <b>{this.currentGroup.groupName}</b> +                    <div className={"close-button"} onClick={action(() => this.currentGroup = undefined)}> +                        <FontAwesomeIcon icon={fa.faWindowClose} size={"lg"} /> +                    </div> + +                    {this.hasEditAccess(this.currentGroup) ? +                        <div className="group-buttons"> +                            <div className="add-member-dropdown"> +                                <Select +                                    // isMulti={true} +                                    isSearchable={true} +                                    options={options} +                                    onChange={selectedOption => this.addMemberToGroup(this.currentGroup!, (selectedOption as UserOptions).value)} +                                    placeholder={"Add members"} +                                    value={null} +                                    closeMenuOnSelect={true} +                                /> +                            </div> +                            <button onClick={() => this.deleteGroup(this.currentGroup!)}>Delete group</button> +                        </div> : +                        null} +                </div> +                <div className="editing-contents"> +                    {members.map(member => ( +                        <div className="editing-row"> +                            <div className="user-email"> +                                {member} +                            </div> +                            {this.hasEditAccess(this.currentGroup!) ? <button onClick={() => this.removeMemberFromGroup(this.currentGroup!, member)}> Remove </button> : null} +                        </div> +                    ))} +                </div> +            </div> +        ); + +    } + +    /** +     * A getter that @returns the main interface for the GroupManager. +     */ +    private get groupInterface() { +        return ( +            <div className="group-interface"> +                {/* <MainViewModal +                    contents={this.editingInterface} +                    isDisplayed={this.currentGroup ? true : false} +                    interactive={true} +                    dialogueBoxDisplayedOpacity={this.dialogueBoxOpacity} +                    overlayDisplayedOpacity={this.overlayOpacity} +                /> */} +                {this.currentGroup ? +                    <GroupMemberView +                        group={this.currentGroup} +                        onCloseButtonClick={() => this.currentGroup = undefined} +                    /> +                    : null} +                <div className="group-heading"> +                    <h1>Groups</h1> +                    <div className={"close-button"} onClick={this.close}> +                        <FontAwesomeIcon icon={fa.faWindowClose} size={"lg"} /> +                    </div> +                </div> +                <div className="group-body"> +                    <div className="group-create"> +                        <button onClick={this.createGroup}>Create group</button> +                        <input ref={this.inputRef} onKeyDown={this.handleKeyDown} type="text" placeholder="Group name" /> +                        <Select +                            isMulti={true} +                            isSearchable={true} +                            options={this.options} +                            onChange={this.handleChange} +                            placeholder={"Select users"} +                            value={this.selectedUsers} +                            closeMenuOnSelect={false} +                        /> +                    </div> +                    <div className="group-content"> +                        {this.getAllGroups().map(group => +                            <div className="group-row"> +                                <div className="group-name">{group.groupName}</div> +                                <button onClick={action(() => this.currentGroup = group)}> +                                    {this.hasEditAccess(group) ? "Edit" : "View"} +                                </button> +                            </div> +                        )} +                    </div> +                </div> +            </div> +        ); +    } + +    render() { +        return ( +            <MainViewModal +                contents={this.groupInterface} +                isDisplayed={this.isOpen} +                interactive={true} +                dialogueBoxDisplayedOpacity={this.dialogueBoxOpacity} +                overlayDisplayedOpacity={this.overlayOpacity} +            /> +        ); +    } + +}
\ No newline at end of file diff --git a/src/client/util/GroupMemberView.scss b/src/client/util/GroupMemberView.scss new file mode 100644 index 000000000..7833c485f --- /dev/null +++ b/src/client/util/GroupMemberView.scss @@ -0,0 +1,68 @@ +@import "../views/globalCssVariables"; + +.editing-interface { +    background-color: whitesmoke !important; +    color: grey; +    width: 100%; +    height: 100%; + +    button { +        background: $darker-alt-accent; +        outline: none; +        border-radius: 5px; +        border: 0px; +        color: #fcfbf7; +        text-transform: uppercase; +        letter-spacing: 2px; +        font-size: 75%; +        padding: 10px; +        margin: 10px; +        transition: transform 0.2s; +        margin: 2px; +    } + +    .memberView-closeButton { +        position: absolute; +        right: 1em; +        top: 1em; +        cursor: pointer; +        z-index: 1000; +    } + +    .editing-header { +        margin-bottom: 5; + +        .group-buttons { +            display: flex; +            margin-top: 5; + +            .add-member-dropdown { +                width: 100%; +                margin: 0 5; +            } +        } +    } + +    .editing-contents { +        overflow-y: auto; +        // max-height: 67%; +        height: 67%; +        width: 100%; + +        .editing-row { +            display: flex; +            align-items: center; +            // border: 1px solid; +            // border-radius: 10px; + +            .user-email { +                // position: relative; +                min-width: 65%; +                word-break: break-all; +                padding: 0 5; +            } +        } +    } + + +}
\ No newline at end of file diff --git a/src/client/util/GroupMemberView.tsx b/src/client/util/GroupMemberView.tsx new file mode 100644 index 000000000..b2d75158e --- /dev/null +++ b/src/client/util/GroupMemberView.tsx @@ -0,0 +1,75 @@ +import * as React from "react"; +import MainViewModal from "../views/MainViewModal"; +import { observer } from "mobx-react"; +import GroupManager, { UserOptions } from "./GroupManager"; +import { library } from "@fortawesome/fontawesome-svg-core"; +import { StrCast } from "../../fields/Types"; +import { action } from "mobx"; +import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; +import * as fa from '@fortawesome/free-solid-svg-icons'; +import Select from "react-select"; +import { Doc, Opt } from "../../fields/Doc"; +import "./GroupMemberView.scss"; + +library.add(fa.faWindowClose); + +interface GroupMemberViewProps { +    group: Doc; +    onCloseButtonClick: () => void; +} + +@observer +export default class GroupMemberView extends React.Component<GroupMemberViewProps> { + +    private get editingInterface() { +        const members: string[] = this.props.group ? JSON.parse(StrCast(this.props.group.members)) : []; +        const options: UserOptions[] = this.props.group ? GroupManager.Instance.options.filter(option => !(JSON.parse(StrCast(this.props.group.members)) as string[]).includes(option.value)) : []; +        return (!this.props.group ? null : +            <div className="editing-interface"> +                <div className="editing-header"> +                    <b>{this.props.group.groupName}</b> +                    <div className={"memberView-closeButton"} onClick={action(this.props.onCloseButtonClick)}> +                        <FontAwesomeIcon icon={fa.faWindowClose} size={"lg"} /> +                    </div> + +                    {GroupManager.Instance.hasEditAccess(this.props.group) ? +                        <div className="group-buttons"> +                            <div className="add-member-dropdown"> +                                <Select +                                    isSearchable={true} +                                    options={options} +                                    onChange={selectedOption => GroupManager.Instance.addMemberToGroup(this.props.group, (selectedOption as UserOptions).value)} +                                    placeholder={"Add members"} +                                    value={null} +                                    closeMenuOnSelect={true} +                                /> +                            </div> +                            <button onClick={() => GroupManager.Instance.deleteGroup(this.props.group)}>Delete group</button> +                        </div> : +                        null} +                </div> +                <div className="editing-contents"> +                    {members.map(member => ( +                        <div className="editing-row"> +                            <div className="user-email"> +                                {member} +                            </div> +                            {GroupManager.Instance.hasEditAccess(this.props.group) ? <button onClick={() => GroupManager.Instance.removeMemberFromGroup(this.props.group, member)}> Remove </button> : null} +                        </div> +                    ))} +                </div> +            </div> +        ); + +    } + +    render() { +        return <MainViewModal +            isDisplayed={true} +            interactive={true} +            contents={this.editingInterface} +        />; +    } + + +}
\ No newline at end of file diff --git a/src/client/util/Import & Export/DirectoryImportBox.tsx b/src/client/util/Import & Export/DirectoryImportBox.tsx index af6c57e68..77f13e9f4 100644 --- a/src/client/util/Import & Export/DirectoryImportBox.tsx +++ b/src/client/util/Import & Export/DirectoryImportBox.tsx @@ -161,7 +161,7 @@ export class DirectoryImportBox extends React.Component<FieldViewProps> {                  importContainer = Docs.Create.SchemaDocument(headers, docs, options);              }              runInAction(() => this.phase = 'External: uploading files to Google Photos...'); -            importContainer.singleColumn = false; +            importContainer._columnsStack = false;              await GooglePhotos.Export.CollectionToAlbum({ collection: importContainer });              Doc.AddDocToList(Doc.GetProto(parent.props.Document), "data", importContainer);              !this.persistent && this.props.removeDocument && this.props.removeDocument(doc); diff --git a/src/client/util/LinkManager.ts b/src/client/util/LinkManager.ts index 0aec81ab0..749fabfcc 100644 --- a/src/client/util/LinkManager.ts +++ b/src/client/util/LinkManager.ts @@ -41,24 +41,17 @@ export class LinkManager {      }      public addLink(linkDoc: Doc): boolean { -        const linkList = LinkManager.Instance.getAllLinks(); -        linkList.push(linkDoc);          if (LinkManager.Instance.LinkManagerDoc) { -            LinkManager.Instance.LinkManagerDoc.data = new List<Doc>(linkList); +            Doc.AddDocToList(LinkManager.Instance.LinkManagerDoc, "data", linkDoc);              return true;          }          return false;      }      public deleteLink(linkDoc: Doc): boolean { -        const linkList = LinkManager.Instance.getAllLinks(); -        const index = LinkManager.Instance.getAllLinks().indexOf(linkDoc); -        if (index > -1) { -            linkList.splice(index, 1); -            if (LinkManager.Instance.LinkManagerDoc) { -                LinkManager.Instance.LinkManagerDoc.data = new List<Doc>(linkList); -                return true; -            } +        if (LinkManager.Instance.LinkManagerDoc && linkDoc instanceof Doc) { +            Doc.RemoveDocFromList(LinkManager.Instance.LinkManagerDoc, "data", linkDoc); +            return true;          }          return false;      } diff --git a/src/client/util/SearchUtil.ts b/src/client/util/SearchUtil.ts index e4c4f5fb7..911340ab1 100644 --- a/src/client/util/SearchUtil.ts +++ b/src/client/util/SearchUtil.ts @@ -77,7 +77,7 @@ export namespace SearchUtil {          const docs = ids.map((id: string) => docMap[id]).map(doc => doc as Doc);          for (let i = 0; i < ids.length; i++) {              const testDoc = docs[i]; -            if (testDoc instanceof Doc && testDoc.type !== DocumentType.KVP && (options.allowAliases || theDocs.findIndex(d => Doc.AreProtosEqual(d, testDoc)) === -1)) { +            if (testDoc instanceof Doc && testDoc.type !== DocumentType.KVP && (options.allowAliases || testDoc.proto === undefined || theDocs.findIndex(d => Doc.AreProtosEqual(d, testDoc)) === -1)) {                  theDocs.push(testDoc);                  theLines.push([]);              } diff --git a/src/client/util/SettingsManager.scss b/src/client/util/SettingsManager.scss index 6513cb223..fa2609ca2 100644 --- a/src/client/util/SettingsManager.scss +++ b/src/client/util/SettingsManager.scss @@ -41,6 +41,7 @@          position: absolute;          right: 1em;          top: 1em; +        cursor: pointer;      }      .settings-heading { diff --git a/src/client/util/SharingManager.scss b/src/client/util/SharingManager.scss index dec9f751a..fcbc05f8a 100644 --- a/src/client/util/SharingManager.scss +++ b/src/client/util/SharingManager.scss @@ -1,13 +1,75 @@ +@import "../views/globalCssVariables"; +  .sharing-interface {      display: flex;      flex-direction: column; +    width: 730px; + +    .dialogue-box { +        width: 450; +        height: 300; +    } + +    .overlay { +        transform: translate(-20px, -20px); +    } + +    .sharing-contents { +        display: flex; + +        button { +            background: $darker-alt-accent; +            outline: none; +            border-radius: 5px; +            border: 0px; +            color: #fcfbf7; +            text-transform: uppercase; +            letter-spacing: 2px; +            font-size: 75%; +            padding: 0 10; +            margin: 0 5; +            transition: transform 0.2s; +            height: 25; +        } + +        .individual-container, +        .group-container { +            width: 50%; + +            .share-groups, +            .share-individual { +                margin-top: 20px; +                margin-bottom: 20px; +            } + +            .groups-list, +            .users-list { +                font-style: italic; +                background: white; +                border: 1px solid black; +                padding-left: 10px; +                padding-right: 10px; +                overflow-y: scroll; +                overflow-x: hidden; +                text-align: left; +                display: flex; +                align-content: center; +                align-items: center; +                text-align: center; +                justify-content: center; +                color: red; +                height: 150px; +                margin: 0 2; +            } +        } +    }      .focus-span {          text-decoration: underline;      }      p { -        font-size: 20px; +        font-size: 15px;          text-align: left;          font-style: italic;          padding: 0; @@ -36,33 +98,10 @@          }      } -    .share-individual { -        margin-top: 20px; -        margin-bottom: 20px; -    } - -    .users-list { -        font-style: italic; -        background: white; -        border: 1px solid black; -        padding-left: 10px; -        padding-right: 10px; -        max-height: 200px; -        overflow: scroll; -        height: -webkit-fill-available; -        text-align: left; -        display: flex; -        align-content: center; -        align-items: center; -        text-align: center; -        justify-content: center; -        color: red; -    } -      .container { -        display: block; +        display: flex;          position: relative; -        margin-top: 10px; +        margin-top: 5px;          margin-bottom: 10px;          font-size: 22px;          -webkit-user-select: none; @@ -74,18 +113,27 @@          max-width: 700px;          text-align: left;          font-style: normal; -        font-size: 15; +        font-size: 14;          font-weight: normal;          padding: 0; +        align-items: baseline;          .padding { -            padding: 0 0 0 20px; +            padding: 0 10px 0 0;              color: black;          }          .permissions-dropdown {              outline: none; +            height: 25;          } + +        .edit-actions { +            display: flex; +            position: absolute; +            right: 51.5%; +        } +      }      .no-users { diff --git a/src/client/util/SharingManager.tsx b/src/client/util/SharingManager.tsx index dc67145fc..127ee33ce 100644 --- a/src/client/util/SharingManager.tsx +++ b/src/client/util/SharingManager.tsx @@ -17,6 +17,8 @@ import { SelectionManager } from "./SelectionManager";  import { DocumentManager } from "./DocumentManager";  import { CollectionView } from "../views/collections/CollectionView";  import { DictationOverlay } from "../views/DictationOverlay"; +import GroupManager from "./GroupManager"; +import GroupMemberView from "./GroupMemberView";  library.add(fa.faCopy); @@ -28,17 +30,30 @@ export interface User {  export enum SharingPermissions {      None = "Not Shared",      View = "Can View", -    Comment = "Can Comment", +    Add = "Can Add",      Edit = "Can Edit"  }  const ColorMapping = new Map<string, string>([      [SharingPermissions.None, "red"],      [SharingPermissions.View, "maroon"], -    [SharingPermissions.Comment, "blue"], +    [SharingPermissions.Add, "blue"],      [SharingPermissions.Edit, "green"]  ]); +const HierarchyMapping = new Map<string, string>([ +    [SharingPermissions.None, "0"], +    [SharingPermissions.View, "1"], +    [SharingPermissions.Add, "2"], +    [SharingPermissions.Edit, "3"], + +    ["0", SharingPermissions.None], +    ["1", SharingPermissions.View], +    ["2", SharingPermissions.Add], +    ["3", SharingPermissions.Edit] + +]); +  const SharingKey = "sharingPermissions";  const PublicKey = "publicLinkPermissions";  const DefaultColor = "black"; @@ -55,11 +70,13 @@ export default class SharingManager extends React.Component<{}> {      public static Instance: SharingManager;      @observable private isOpen = false;      @observable private users: ValidatedUser[] = []; +    @observable private groups: Doc[] = [];      @observable private targetDoc: Doc | undefined;      @observable private targetDocView: DocumentView | undefined;      @observable private copied = false;      @observable private dialogueBoxOpacity = 1;      @observable private overlayOpacity = 0.4; +    @observable private groupToView: Opt<Doc>;      private get linkVisible() {          return this.sharingDoc ? this.sharingDoc[PublicKey] !== SharingPermissions.None : false; @@ -76,6 +93,8 @@ export default class SharingManager extends React.Component<{}> {                  this.sharingDoc = new Doc;              }          })); + +        runInAction(() => this.groups = GroupManager.Instance.getAllGroupsCopy());      }      public close = action(() => { @@ -121,26 +140,71 @@ export default class SharingManager extends React.Component<{}> {          return Promise.all(evaluating);      } -    setInternalSharing = async (recipient: ValidatedUser, state: string) => { +    setInternalGroupSharing = (group: Doc, permission: string) => { +        const members: string[] = JSON.parse(StrCast(group.members)); +        const users: ValidatedUser[] = this.users.filter(user => members.includes(user.user.email)); + +        const sharingDoc = this.sharingDoc!; +        if (permission === SharingPermissions.None) { +            const metadata = sharingDoc[StrCast(group.groupName)]; +            if (metadata) sharingDoc[StrCast(group.groupName)] = undefined; +        } +        else { +            sharingDoc[StrCast(group.groupName)] = permission; +        } + +        users.forEach(user => { +            this.setInternalSharing(user, permission, group); +        }); +    } + +    setInternalSharing = async (recipient: ValidatedUser, state: string, group: Opt<Doc>) => {          const { user, notificationDoc } = recipient;          const target = this.targetDoc!;          const manager = this.sharingDoc!;          const key = user.userDocumentId; -        if (state === SharingPermissions.None) { -            const metadata = (await DocCastAsync(manager[key])); -            if (metadata) { -                const sharedAlias = (await DocCastAsync(metadata.sharedAlias))!; -                Doc.RemoveDocFromList(notificationDoc, storage, sharedAlias); -                manager[key] = undefined; -            } -        } else { -            const sharedAlias = Doc.MakeAlias(target); -            Doc.AddDocToList(notificationDoc, storage, sharedAlias); -            const metadata = new Doc; -            metadata.permissions = state; -            metadata.sharedAlias = sharedAlias; -            manager[key] = metadata; + +        let metadata = await DocCastAsync(manager[key]); +        const permissions: { [key: string]: number } = metadata?.permissions ? JSON.parse(StrCast(metadata.permissions)) : {}; +        permissions[StrCast(group ? group.groupName : Doc.CurrentUserEmail)] = parseInt(HierarchyMapping.get(state)!); +        const max = Math.max(...Object.values(permissions)); + +        // let max = 0; +        // const keys: string[] = []; +        // for (const [key, value] of Object.entries(permissions)) { +        //     if (value === max && max !== 0) { +        //         keys.push(key); +        //     } +        //     else if (value > max) { +        //         keys.splice(0, keys.length); +        //         keys.push(key); +        //         max = value; +        //     } +        // } + +        switch (max) { +            case 0: +                if (metadata) { +                    const sharedAlias = (await DocCastAsync(metadata.sharedAlias))!; +                    Doc.RemoveDocFromList(notificationDoc, storage, sharedAlias); +                    manager[key] = undefined; +                } +                break; + +            case 1: case 2: case 3: +                if (!metadata) { +                    metadata = new Doc; +                    const sharedAlias = Doc.MakeAlias(target); +                    Doc.AddDocToList(notificationDoc, storage, sharedAlias); +                    metadata.sharedAlias = sharedAlias; +                    manager[key] = metadata; +                } +                metadata.permissions = JSON.stringify(permissions); +                // metadata.usersShared = JSON.stringify(keys); +                break;          } + +        if (metadata) metadata.maxPermission = HierarchyMapping.get(`${max}`);      }      private setExternalSharing = (state: string) => { @@ -211,17 +275,27 @@ export default class SharingManager extends React.Component<{}> {          if (!sharingDoc) {              return SharingPermissions.None;          } -        const metadata = sharingDoc[userKey] as Doc; +        const metadata = sharingDoc[userKey] as Doc | string;          if (!metadata) {              return SharingPermissions.None;          } -        return StrCast(metadata.permissions, SharingPermissions.None); +        return StrCast(metadata instanceof Doc ? metadata.maxPermission : metadata, SharingPermissions.None);      }      private get sharingInterface() {          const existOtherUsers = this.users.length > 0; +        const existGroups = this.groups.length > 0; + +        // const manager = this.sharingDoc!; +          return (              <div className={"sharing-interface"}> +                {this.groupToView ? +                    <GroupMemberView +                        group={this.groupToView} +                        onCloseButtonClick={action(() => this.groupToView = undefined)} +                    /> : +                    null}                  <p className={"share-link"}>Manage the public link to {this.focusOn("this document...")}</p>                  {!this.linkVisible ? (null) :                      <div className={"link-container"}> @@ -252,31 +326,77 @@ export default class SharingManager extends React.Component<{}> {                      </select>                  </div>                  <div className={"hr-substitute"} /> -                <p className={"share-individual"}>Privately share {this.focusOn("this document")} with an individual...</p> -                <div className={"users-list"} style={{ display: existOtherUsers ? "block" : "flex", minHeight: existOtherUsers ? undefined : 200 }}> -                    {!existOtherUsers ? "There are no other users in your database." : -                        this.users.map(({ user, notificationDoc }) => { -                            const userKey = user.userDocumentId; -                            const permissions = this.computePermissions(userKey); -                            const color = ColorMapping.get(permissions); -                            return ( -                                <div -                                    key={userKey} -                                    className={"container"} -                                > -                                    <select -                                        className={"permissions-dropdown"} -                                        value={permissions} -                                        style={{ color, borderColor: color }} -                                        onChange={e => this.setInternalSharing({ user, notificationDoc }, e.currentTarget.value)} -                                    > -                                        {this.sharingOptions} -                                    </select> -                                    <span className={"padding"}>{user.email}</span> -                                </div> -                            ); -                        }) -                    } +                <div className="sharing-contents"> +                    <div className={"individual-container"}> +                        <p className={"share-individual"}>Privately share {this.focusOn("this document")} with an individual...</p> +                        <div className={"users-list"} style={{ display: existOtherUsers ? "block" : "flex", minHeight: existOtherUsers ? undefined : 150 }}>{/*200*/} +                            {!existOtherUsers ? "There are no other users in your database." : +                                this.users.map(({ user, notificationDoc }) => { // can't use async here +                                    const userKey = user.userDocumentId; +                                    const permissions = this.computePermissions(userKey); +                                    const color = ColorMapping.get(permissions); + +                                    // console.log(manager); +                                    // const metadata = manager[userKey] as Doc; +                                    // const usersShared = StrCast(metadata?.usersShared, ""); +                                    // console.log(usersShared) + + +                                    return ( +                                        <div +                                            key={userKey} +                                            className={"container"} +                                        > +                                            <span className={"padding"}>{user.email}</span> +                                            {/* <div className={"shared-by"}>{usersShared}</div> */} +                                            <div className="edit-actions"> +                                                <select +                                                    className={"permissions-dropdown"} +                                                    value={permissions} +                                                    style={{ color, borderColor: color }} +                                                    onChange={e => this.setInternalSharing({ user, notificationDoc }, e.currentTarget.value, undefined)} +                                                > +                                                    {this.sharingOptions} +                                                </select> +                                            </div> +                                        </div> +                                    ); +                                }) +                            } +                        </div> +                    </div> +                    <div className={"group-container"}> +                        <p className={"share-groups"}>Privately share {this.focusOn("this document")} with a group...</p> +                        <div className={"groups-list"} style={{ display: existGroups ? "block" : "flex", minHeight: existOtherUsers ? undefined : 150 }}>{/*200*/} +                            {!existGroups ? "There are no groups in your database." : +                                this.groups.map(group => { +                                    const permissions = this.computePermissions(StrCast(group.groupName)); +                                    const color = ColorMapping.get(permissions); +                                    return ( +                                        <div +                                            key={StrCast(group.groupName)} +                                            className={"container"} +                                        > +                                            <span className={"padding"}>{group.groupName}</span> +                                            <div className="edit-actions"> +                                                <select +                                                    className={"permissions-dropdown"} +                                                    value={permissions} +                                                    style={{ color, borderColor: color }} +                                                    onChange={e => this.setInternalGroupSharing(group, e.currentTarget.value)} +                                                > +                                                    {this.sharingOptions} +                                                </select> +                                                <button onClick={action(() => this.groupToView = group)}>Edit</button> +                                            </div> +                                        </div> +                                    ); +                                }) + +                            } + +                        </div> +                    </div>                  </div>                  <div className={"close-button"} onClick={this.close}>Done</div>              </div> @@ -284,6 +404,7 @@ export default class SharingManager extends React.Component<{}> {      }      render() { +        // console.log(this.sharingDoc);          return (              <MainViewModal                  contents={this.sharingInterface} diff --git a/src/client/views/AntimodeMenu.tsx b/src/client/views/AntimodeMenu.tsx index 3e4d20fea..a4634103c 100644 --- a/src/client/views/AntimodeMenu.tsx +++ b/src/client/views/AntimodeMenu.tsx @@ -51,18 +51,15 @@ export default abstract class AntimodeMenu extends React.Component {              if (this._opacity === 0.2) {                  this._transitionProperty = "opacity";                  this._transitionDuration = "0.1s"; -                this._transitionDelay = ""; -                this._opacity = 0; -                this._left = this._top = -300;              }              if (forceOut) {                  this._transitionProperty = "";                  this._transitionDuration = ""; -                this._transitionDelay = ""; -                this._opacity = 0; -                this._left = this._top = -300;              } +            this._transitionDelay = ""; +            this._opacity = 0; +            this._left = this._top = -300;          }      } diff --git a/src/client/views/ContextMenu.tsx b/src/client/views/ContextMenu.tsx index 941d7b44a..07f7b8e6d 100644 --- a/src/client/views/ContextMenu.tsx +++ b/src/client/views/ContextMenu.tsx @@ -155,9 +155,11 @@ export class ContextMenu extends React.Component {      @action      closeMenu = () => { +        const wasOpen = this._display;          this.clearItems();          this._display = false;          this._shouldDisplay = false; +        return wasOpen;      }      @computed get filteredItems(): (OriginalMenuProps | string[])[] { diff --git a/src/client/views/DocumentDecorations.tsx b/src/client/views/DocumentDecorations.tsx index 4fda10926..a45ef8862 100644 --- a/src/client/views/DocumentDecorations.tsx +++ b/src/client/views/DocumentDecorations.tsx @@ -600,52 +600,54 @@ export class DocumentDecorations extends React.Component<{}, { value: string }>                  zIndex: SelectionManager.SelectedDocuments().length > 1 ? 900 : 0,              }} onPointerDown={this.onBackgroundDown} onContextMenu={e => { e.preventDefault(); e.stopPropagation(); }} >              </div> -            <div className="documentDecorations-container" ref={this.setTextBar} style={{ -                width: (bounds.r - bounds.x + this._resizeBorderWidth) + "px", -                height: (bounds.b - bounds.y + this._resizeBorderWidth + this._titleHeight) + "px", -                left: bounds.x - this._resizeBorderWidth / 2, -                top: bounds.y - this._resizeBorderWidth / 2 - this._titleHeight, -            }}> -                {maximizeIcon} -                {titleArea} -                {SelectionManager.SelectedDocuments().length !== 1 || seldoc.Document.type === DocumentType.INK ? (null) : -                    <div className="documentDecorations-iconifyButton" title={`${seldoc.finalLayoutKey.includes("icon") ? "De" : ""}Iconify Document`} onPointerDown={this.onIconifyDown}> -                        {"_"} -                    </div>} -                <div className="documentDecorations-closeButton" title="Open Document in Tab" onPointerDown={this.onMaximizeDown}> -                    {SelectionManager.SelectedDocuments().length === 1 ? DocumentDecorations.DocumentIcon(StrCast(seldoc.props.Document.layout, "...")) : "..."} +            {bounds.r - bounds.x < 15 && bounds.b - bounds.y < 15 ? (null) : <> +                <div className="documentDecorations-container" key="container" ref={this.setTextBar} style={{ +                    width: (bounds.r - bounds.x + this._resizeBorderWidth) + "px", +                    height: (bounds.b - bounds.y + this._resizeBorderWidth + this._titleHeight) + "px", +                    left: bounds.x - this._resizeBorderWidth / 2, +                    top: bounds.y - this._resizeBorderWidth / 2 - this._titleHeight, +                }}> +                    {maximizeIcon} +                    {titleArea} +                    {SelectionManager.SelectedDocuments().length !== 1 || seldoc.Document.type === DocumentType.INK ? (null) : +                        <div className="documentDecorations-iconifyButton" title={`${seldoc.finalLayoutKey.includes("icon") ? "De" : ""}Iconify Document`} onPointerDown={this.onIconifyDown}> +                            {"_"} +                        </div>} +                    <div className="documentDecorations-closeButton" title="Open Document in Tab" onPointerDown={this.onMaximizeDown}> +                        {SelectionManager.SelectedDocuments().length === 1 ? DocumentDecorations.DocumentIcon(StrCast(seldoc.props.Document.layout, "...")) : "..."} +                    </div> +                    <div id="documentDecorations-rotation" className="documentDecorations-rotation" +                        onPointerDown={this.onRotateDown}> ⟲ </div> +                    <div id="documentDecorations-topLeftResizer" className="documentDecorations-resizer" +                        onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> +                    <div id="documentDecorations-topResizer" className="documentDecorations-resizer" +                        onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> +                    <div id="documentDecorations-topRightResizer" className="documentDecorations-resizer" +                        onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> +                    <div id="documentDecorations-leftResizer" className="documentDecorations-resizer" +                        onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> +                    <div id="documentDecorations-centerCont"></div> +                    <div id="documentDecorations-rightResizer" className="documentDecorations-resizer" +                        onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> +                    <div id="documentDecorations-bottomLeftResizer" className="documentDecorations-resizer" +                        onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> +                    <div id="documentDecorations-bottomResizer" className="documentDecorations-resizer" +                        onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> +                    <div id="documentDecorations-bottomRightResizer" className="documentDecorations-resizer" +                        onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> +                    {seldoc.props.renderDepth <= 1 || !seldoc.props.ContainingCollectionView ? (null) : +                        <div id="documentDecorations-levelSelector" className="documentDecorations-selector" +                            title="tap to select containing document" onPointerDown={this.onSelectorUp} onContextMenu={e => e.preventDefault()}> +                            <FontAwesomeIcon className="documentdecorations-times" icon={faArrowAltCircleUp} size="lg" /> +                        </div>} +                    <div id="documentDecorations-borderRadius" className="documentDecorations-radius" +                        onPointerDown={this.onRadiusDown} onContextMenu={(e) => e.preventDefault()}></div> + +                </div > +                <div className="link-button-container" key="links" style={{ left: bounds.x - this._resizeBorderWidth / 2 + 10, top: bounds.b + this._resizeBorderWidth / 2 }}> +                    <DocumentButtonBar views={SelectionManager.SelectedDocuments} />                  </div> -                <div id="documentDecorations-rotation" className="documentDecorations-rotation" -                    onPointerDown={this.onRotateDown}> ⟲ </div> -                <div id="documentDecorations-topLeftResizer" className="documentDecorations-resizer" -                    onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> -                <div id="documentDecorations-topResizer" className="documentDecorations-resizer" -                    onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> -                <div id="documentDecorations-topRightResizer" className="documentDecorations-resizer" -                    onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> -                <div id="documentDecorations-leftResizer" className="documentDecorations-resizer" -                    onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> -                <div id="documentDecorations-centerCont"></div> -                <div id="documentDecorations-rightResizer" className="documentDecorations-resizer" -                    onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> -                <div id="documentDecorations-bottomLeftResizer" className="documentDecorations-resizer" -                    onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> -                <div id="documentDecorations-bottomResizer" className="documentDecorations-resizer" -                    onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> -                <div id="documentDecorations-bottomRightResizer" className="documentDecorations-resizer" -                    onPointerDown={this.onPointerDown} onContextMenu={(e) => e.preventDefault()}></div> -                {seldoc.props.renderDepth <= 1 || !seldoc.props.ContainingCollectionView ? (null) : -                    <div id="documentDecorations-levelSelector" className="documentDecorations-selector" -                        title="tap to select containing document" onPointerDown={this.onSelectorUp} onContextMenu={e => e.preventDefault()}> -                        <FontAwesomeIcon className="documentdecorations-times" icon={faArrowAltCircleUp} size="lg" /> -                    </div>} -                <div id="documentDecorations-borderRadius" className="documentDecorations-radius" -                    onPointerDown={this.onRadiusDown} onContextMenu={(e) => e.preventDefault()}></div> - -            </div > -            <div className="link-button-container" style={{ left: bounds.x - this._resizeBorderWidth / 2 + 10, top: bounds.b + this._resizeBorderWidth / 2 }}> -                <DocumentButtonBar views={SelectionManager.SelectedDocuments} /> -            </div> +            </>}          </div >          );      } diff --git a/src/client/views/EditableView.tsx b/src/client/views/EditableView.tsx index bab3a1634..628db366f 100644 --- a/src/client/views/EditableView.tsx +++ b/src/client/views/EditableView.tsx @@ -194,7 +194,6 @@ export class EditableView extends React.Component<EditableProps> {                      ref={this._ref}                      style={{ display: this.props.display, minHeight: "20px", height: `${this.props.height ? this.props.height : "auto"}`, maxHeight: `${this.props.maxHeight}` }}                      onClick={this.onClick} placeholder={this.props.placeholder}> -                      <span style={{ fontStyle: this.props.fontStyle, fontSize: this.props.fontSize, color: this.props.contents ? "black" : "grey" }}>{this.props.contents ? this.props.contents?.valueOf() : this.props.placeholder?.valueOf()}</span>                  </div>              ); diff --git a/src/client/views/GlobalKeyHandler.ts b/src/client/views/GlobalKeyHandler.ts index e3546dece..c85849adb 100644 --- a/src/client/views/GlobalKeyHandler.ts +++ b/src/client/views/GlobalKeyHandler.ts @@ -20,6 +20,7 @@ import { MainView } from "./MainView";  import { DocumentView } from "./nodes/DocumentView";  import { DocumentLinksButton } from "./nodes/DocumentLinksButton";  import PDFMenu from "./pdf/PDFMenu"; +import { ContextMenu } from "./ContextMenu";  const modifiers = ["control", "meta", "shift", "alt"];  type KeyHandler = (keycode: string, e: KeyboardEvent) => KeyControlInfo | Promise<KeyControlInfo>; @@ -81,16 +82,17 @@ export default class KeyManager {                  DocumentLinksButton.StartLink = undefined;                  const main = MainView.Instance;                  Doc.SetSelectedTool(InkTool.None); +                var doDeselect = true;                  if (main.isPointerDown) {                      DragManager.AbortDrag();                  } else {                      if (CollectionDockingView.Instance.HasFullScreen()) {                          CollectionDockingView.Instance.CloseFullScreen();                      } else { -                        SelectionManager.DeselectAll(); +                        doDeselect = !ContextMenu.Instance.closeMenu();                      }                  } -                SelectionManager.DeselectAll(); +                doDeselect && SelectionManager.DeselectAll();                  DictationManager.Controls.stop();                  // RecommendationsBox.Instance.closeMenu();                  GoogleAuthenticationManager.Instance.cancel(); diff --git a/src/client/views/MainView.scss b/src/client/views/MainView.scss index e84969565..5b142ffda 100644 --- a/src/client/views/MainView.scss +++ b/src/client/views/MainView.scss @@ -28,10 +28,11 @@      left: 0;      width: 100%;      height: 100%; -    pointer-events:none; +    pointer-events: none;  } -.mainView-container, .mainView-container-dark { +.mainView-container, +.mainView-container-dark {      width: 100%;      height: 100%;      position: absolute; @@ -40,40 +41,50 @@      left: 0;      z-index: 1;      touch-action: none; +      .searchBox-container {          background: lightgray;      }  }  .mainView-container { -    color:dimgray; +    color: dimgray; +      .lm_title {          background: #cacaca; -        color:black; +        color: black;      }  }  .mainView-container-dark {      color: lightgray; +      .lm_goldenlayout {          background: dimgray;      } +      .lm_title {          background: black; -        color:unset; +        color: unset;      } +      .marquee {          border-color: white;      } +      #search-input {          background: lightgray;      } -    .searchBox-container  { -        background: rgb(45,45,45); + +    .searchBox-container { +        background: rgb(45, 45, 45);      } -    .contextMenu-cont, .contextMenu-item { + +    .contextMenu-cont, +    .contextMenu-item {          background: dimGray;      } +      .contextMenu-item:hover {          background: gray;      } @@ -108,20 +119,27 @@      overflow: hidden;  } +.buttonContainer { -.mainView-settings {      position: absolute; -    left: 0;      bottom: 0; -    border-radius: 25%; -    margin-left: -5px; -    background: darkblue; -} -.mainView-settings:hover { -    transform: none !important; +    .mainView-settings { +        // position: absolute; +        // left: 0; +        // bottom: 0; +        border-radius: 25%; +        margin-left: -5px; +        background: darkblue; +    } + +    .mainView-settings:hover { +        transform: none !important; +    }  } + +  .mainView-logout {      position: absolute;      right: 0; diff --git a/src/client/views/MainView.tsx b/src/client/views/MainView.tsx index 68b81ab4f..f6db1af66 100644 --- a/src/client/views/MainView.tsx +++ b/src/client/views/MainView.tsx @@ -5,7 +5,7 @@ import {      faQuestionCircle, faArrowLeft, faArrowRight, faArrowDown, faArrowUp, faBolt, faBullseye, faCaretUp, faCat, faCheck, faChevronRight, faClipboard, faClone, faCloudUploadAlt,      faCommentAlt, faCompressArrowsAlt, faCut, faEllipsisV, faEraser, faExclamation, faFileAlt, faFileAudio, faFilePdf, faFilm, faFilter, faFont, faGlobeAsia, faHighlighter,      faLongArrowAltRight, faMicrophone, faMousePointer, faMusic, faObjectGroup, faPause, faPen, faPenNib, faPhone, faPlay, faPortrait, faRedoAlt, faStamp, faStickyNote, -    faThumbtack, faTree, faTv, faUndoAlt, faVideo, faAsterisk, faBrain, faImage, faPaintBrush, faTimes, faEye, faArrowsAlt, faQuoteLeft, faSortAmountDown +    faThumbtack, faTree, faTv, faUndoAlt, faVideo, faAsterisk, faBrain, faImage, faPaintBrush, faTimes, faEye, faArrowsAlt, faQuoteLeft, faSortAmountDown, faAlignLeft, faAlignCenter, faAlignRight  } from '@fortawesome/free-solid-svg-icons';  import { ANTIMODEMENU_HEIGHT } from './globalCssVariables.scss';  import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'; @@ -30,6 +30,7 @@ import { HistoryUtil } from '../util/History';  import RichTextMenu from './nodes/formattedText/RichTextMenu';  import { Scripting } from '../util/Scripting';  import SettingsManager from '../util/SettingsManager'; +import GroupManager from '../util/GroupManager';  import SharingManager from '../util/SharingManager';  import { Transform } from '../util/Transform';  import { CollectionDockingView } from './collections/CollectionDockingView'; @@ -141,7 +142,7 @@ export class MainView extends React.Component {              faQuestionCircle, faArrowLeft, faArrowRight, faArrowDown, faArrowUp, faBolt, faBullseye, faCaretUp, faCat, faCheck, faChevronRight, faClipboard, faClone, faCloudUploadAlt,              faCommentAlt, faCompressArrowsAlt, faCut, faEllipsisV, faEraser, faExclamation, faFileAlt, faFileAudio, faFilePdf, faFilm, faFilter, faFont, faGlobeAsia, faHighlighter,              faLongArrowAltRight, faMicrophone, faMousePointer, faMusic, faObjectGroup, faPause, faPen, faPenNib, faPhone, faPlay, faPortrait, faRedoAlt, faStamp, faStickyNote, faTrashAlt, faAngleRight, faBell, -            faThumbtack, faTree, faTv, faUndoAlt, faVideo, faAsterisk, faBrain, faImage, faPaintBrush, faTimes, faEye, faArrowsAlt, faQuoteLeft, faSortAmountDown); +            faThumbtack, faTree, faTv, faUndoAlt, faVideo, faAsterisk, faBrain, faImage, faPaintBrush, faTimes, faEye, faArrowsAlt, faQuoteLeft, faSortAmountDown, faAlignLeft, faAlignCenter, faAlignRight);          this.initEventListeners();          this.initAuthenticationRouters();      } @@ -206,7 +207,6 @@ export class MainView extends React.Component {              _width: this._panelWidth * .7,              _height: this._panelHeight,              title: "Collection " + workspaceCount, -            _LODdisable: true          };          const freeformDoc = CurrentUserUtils.GuestTarget || Docs.Create.FreeformDocument([], freeformOptions);          const workspaceDoc = Docs.Create.StandardCollectionDockingDocument([{ doc: freeformDoc, initialWidth: 600, path: [Doc.UserDoc().myCatalog as Doc] }], { title: `Workspace ${workspaceCount}` }, id, "row"); @@ -355,15 +355,6 @@ export class MainView extends React.Component {      }      @action -    pointerOverDragger = () => { -        // if (this.flyoutWidth === 0) { -        //     this.flyoutWidth = 250; -        //     this.sidebarButtonsDoc.columnWidth = this.flyoutWidth / 3 - 30; -        //     this._flyoutTranslate = false; -        // } -    } - -    @action      pointerLeaveDragger = () => {          if (!this._flyoutTranslate) {              this.flyoutWidth = 0; @@ -375,13 +366,13 @@ export class MainView extends React.Component {      onPointerMove = (e: PointerEvent) => {          this.flyoutWidth = Math.max(e.clientX, 0);          Math.abs(this.flyoutWidth - this._flyoutSizeOnDown) > 6 && (this._canClick = false); -        this.sidebarButtonsDoc.columnWidth = this.flyoutWidth / 3 - 30; +        this.sidebarButtonsDoc._columnWidth = this.flyoutWidth / 3 - 30;      }      @action      onPointerUp = (e: PointerEvent) => {          if (Math.abs(e.clientX - this._flyoutSizeOnDown) < 4 && this._canClick) {              this.flyoutWidth = this.flyoutWidth < 15 ? 250 : 0; -            this.flyoutWidth && (this.sidebarButtonsDoc.columnWidth = this.flyoutWidth / 3 - 30); +            this.flyoutWidth && (this.sidebarButtonsDoc._columnWidth = this.flyoutWidth / 3 - 30);          }          document.removeEventListener("pointermove", this.onPointerMove);          document.removeEventListener("pointerup", this.onPointerUp); @@ -392,7 +383,8 @@ export class MainView extends React.Component {              doc.dockingConfig ? this.openWorkspace(doc) :                  CollectionDockingView.AddRightSplit(doc, libraryPath);      } -    mainContainerXf = () => new Transform(0, -this._buttonBarHeight, 1); +    sidebarScreenToLocal = () => new Transform(0, RichTextMenu.Instance.Pinned ? -35 : 0, 1); +    mainContainerXf = () => this.sidebarScreenToLocal().translate(0, -this._buttonBarHeight);      @computed get flyout() {          const sidebarContent = this.userDoc?.["tabs-panelContainer"]; @@ -411,7 +403,7 @@ export class MainView extends React.Component {                      pinToPres={emptyFunction}                      removeDocument={undefined}                      onClick={undefined} -                    ScreenToLocalTransform={Transform.Identity} +                    ScreenToLocalTransform={this.sidebarScreenToLocal}                      ContentScaling={returnOne}                      NativeHeight={returnZero}                      NativeWidth={returnZero} @@ -453,9 +445,14 @@ export class MainView extends React.Component {                      docFilters={returnEmptyFilter}                      ContainingCollectionView={undefined}                      ContainingCollectionDoc={undefined} /> -                <button className="mainView-settings" key="settings" onClick={() => SettingsManager.Instance.open()}> -                    <FontAwesomeIcon icon="cog" size="lg" /> -                </button> +                <div className="buttonContainer" > +                    <button className="mainView-settings" key="settings" onClick={() => SettingsManager.Instance.open()}> +                        <FontAwesomeIcon icon="cog" size="lg" /> +                    </button> +                    <button className="mainView-settings" key="groups" onClick={() => GroupManager.Instance.open()}> +                        <FontAwesomeIcon icon="columns" size="lg" /> +                    </button> +                </div>              </div>              {this.docButtons}          </div>; @@ -470,7 +467,7 @@ export class MainView extends React.Component {              }} >                  <div style={{ display: "contents", flexDirection: "row", position: "relative" }}>                      <div className="mainView-flyoutContainer" onPointerLeave={this.pointerLeaveDragger} style={{ width: this.flyoutWidth }}> -                        <div className="mainView-libraryHandle" onPointerDown={this.onPointerDown} onPointerOver={this.pointerOverDragger} +                        <div className="mainView-libraryHandle" onPointerDown={this.onPointerDown}                              style={{ backgroundColor: this.defaultBackgroundColors(sidebar) }}>                              <span title="library View Dragger" style={{                                  width: (this.flyoutWidth !== 0 && this._flyoutTranslate) ? "100%" : "3vw", @@ -497,7 +494,7 @@ export class MainView extends React.Component {      public static expandFlyout = action(() => {          MainView.Instance._flyoutTranslate = true;          MainView.Instance.flyoutWidth = (MainView.Instance.flyoutWidth || 250); -        MainView.Instance.sidebarButtonsDoc.columnWidth = MainView.Instance.flyoutWidth / 3 - 30; +        MainView.Instance.sidebarButtonsDoc._columnWidth = MainView.Instance.flyoutWidth / 3 - 30;      });      @computed get expandButton() { @@ -601,6 +598,7 @@ export class MainView extends React.Component {              <DictationOverlay />              <SharingManager />              <SettingsManager /> +            <GroupManager />              <GoogleAuthenticationManager />              <DocumentDecorations />              <GestureOverlay> diff --git a/src/client/views/PreviewCursor.tsx b/src/client/views/PreviewCursor.tsx index 1ab99881d..6583589f3 100644 --- a/src/client/views/PreviewCursor.tsx +++ b/src/client/views/PreviewCursor.tsx @@ -3,13 +3,18 @@ import { observer } from 'mobx-react';  import "normalize.css";  import * as React from 'react';  import "./PreviewCursor.scss"; -import { Docs } from '../documents/Documents'; +import { Docs, DocUtils } from '../documents/Documents';  import { Doc } from '../../fields/Doc';  import { Transform } from "../util/Transform";  import { DocServer } from '../DocServer'; -import { undoBatch } from '../util/UndoManager'; +import { undoBatch, UndoManager } from '../util/UndoManager';  import { NumCast } from '../../fields/Types';  import { FormattedTextBox } from './nodes/formattedText/FormattedTextBox'; +import * as rp from 'request-promise'; +import { Utils } from '../../Utils'; +import { Networking } from '../Network'; +import { Upload } from '../../server/SharedMediaTypes'; +import { basename } from 'path';  @observer  export class PreviewCursor extends React.Component<{}> { @@ -26,7 +31,7 @@ export class PreviewCursor extends React.Component<{}> {          document.addEventListener("paste", this.paste);      } -    paste = (e: ClipboardEvent) => { +    paste = async (e: ClipboardEvent) => {          if (PreviewCursor.Visible && e.clipboardData) {              const newPoint = PreviewCursor._getTransform().transformPoint(PreviewCursor._clickPoint[0], PreviewCursor._clickPoint[1]);              runInAction(() => PreviewCursor.Visible = false); @@ -104,6 +109,16 @@ export class PreviewCursor extends React.Component<{}> {                          x: newPoint[0],                          y: newPoint[1],                      })))(); +                } else if (e.clipboardData.items.length) { +                    const batch = UndoManager.StartBatch("collection view drop"); +                    const files: File[] = []; +                    for (let i = 0; i < e.clipboardData.items.length; i++) { +                        const file = e.clipboardData.items[i].getAsFile(); +                        file && files.push(file); +                    } +                    const generatedDocuments = await DocUtils.uploadFilesToDocs(files, { x: newPoint[0], y: newPoint[1] }); +                    generatedDocuments.forEach(PreviewCursor._addDocument); +                    batch.end();                  }          }      } diff --git a/src/client/views/collections/CollectionMasonryViewFieldRow.tsx b/src/client/views/collections/CollectionMasonryViewFieldRow.tsx index e0b53e762..627b22417 100644 --- a/src/client/views/collections/CollectionMasonryViewFieldRow.tsx +++ b/src/client/views/collections/CollectionMasonryViewFieldRow.tsx @@ -111,8 +111,8 @@ export class CollectionMasonryViewFieldRow extends React.Component<CMVFieldRowPr          const key = StrCast(this.props.parent.props.Document._pivotField);          const castedValue = this.getValue(value);          if (castedValue) { -            if (this.props.parent.sectionHeaders) { -                if (this.props.parent.sectionHeaders.map(i => i.heading).indexOf(castedValue.toString()) > -1) { +            if (this.props.parent.columnHeaders) { +                if (this.props.parent.columnHeaders.map(i => i.heading).indexOf(castedValue.toString()) > -1) {                      return false;                  }              } @@ -151,9 +151,9 @@ export class CollectionMasonryViewFieldRow extends React.Component<CMVFieldRowPr          this._createAliasSelected = false;          const key = StrCast(this.props.parent.props.Document._pivotField);          this.props.docList.forEach(d => d[key] = undefined); -        if (this.props.parent.sectionHeaders && this.props.headingObject) { -            const index = this.props.parent.sectionHeaders.indexOf(this.props.headingObject); -            this.props.parent.sectionHeaders.splice(index, 1); +        if (this.props.parent.columnHeaders && this.props.headingObject) { +            const index = this.props.parent.columnHeaders.indexOf(this.props.headingObject); +            this.props.parent.columnHeaders.splice(index, 1);          }      })); diff --git a/src/client/views/collections/CollectionSchemaCells.tsx b/src/client/views/collections/CollectionSchemaCells.tsx index 2b8110e27..d76b6d204 100644 --- a/src/client/views/collections/CollectionSchemaCells.tsx +++ b/src/client/views/collections/CollectionSchemaCells.tsx @@ -1,22 +1,22 @@  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 { NumCast, StrCast, BoolCast, FieldValue, Cast } from "../../../fields/Types"; +import { CollectionView, Flyout } from "./CollectionView"; +import { NumCast, StrCast, BoolCast, FieldValue, Cast, DateCast } from "../../../fields/Types";  import { Docs } from "../../documents/Documents";  import { library } from '@fortawesome/fontawesome-svg-core';  import { faExpand } from '@fortawesome/free-solid-svg-icons'; @@ -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 { OverlayView } from "../OverlayView"; +import { DocumentIconContainer } from "../nodes/DocumentIcon"; +import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; +import DatePicker from "react-datepicker"; +import "react-datepicker/dist/react-datepicker.css"; +import { DateField } from "../../../fields/DateField"; +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); @@ -84,6 +94,7 @@ export class CollectionSchemaCell extends React.Component<CellProps> {      @action      onPointerDown = async (e: React.PointerEvent): Promise<void> => { +          this.props.changeFocusedCellByIndex(this.props.row, this.props.col);          this.props.setPreviewDoc(this.props.rowProps.original); @@ -129,7 +140,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 +217,18 @@ 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}--`; +        } +        if (type === "date") { +            contents = typeof field === "object" ? doc ? StrCast(field) === "" ? "--" : StrCast(field) : `--${typeof field}--` : `--${typeof field}--`; +        } +          let className = "collectionSchemaView-cellWrapper";          if (this._isEditing) className += " editing"; @@ -220,40 +243,60 @@ export class CollectionSchemaCell extends React.Component<CellProps> {          // );             trace(); + +          return (              <div className="collectionSchemaView-cellContainer" style={{ cursor: fieldIsDoc ? "grab" : "auto" }} ref={dragRef} onPointerDown={this.onPointerDown} onPointerEnter={onPointerEnter} onPointerLeave={onPointerLeave}>                  <div className={className} ref={this._focusRef} onPointerDown={onItemDown} tabIndex={-1}>                      <div className="collectionSchemaView-cellContents" ref={type === undefined || type === "document" ? this.dropRef : null} key={props.Document[Id]}> + +                          <EditableView                              editing={this._isEditing}                              isEditingCallback={this.isEditingCallback}                              display={"inline"} -                            contents={contents} +                            contents={contents ? contents : type === "number" ? "0" : "undefined"} +                            //contents={StrCast(contents)}                              height={"auto"}                              maxHeight={Number(MAX_ROW_HEIGHT)} +                            placeholder={"enter value"}                              GetValue={() => { -                                const cfield = ComputedField.WithoutComputed(() => FieldValue(props.Document[props.fieldKey])); -                                const cscript = cfield instanceof ComputedField ? cfield.script.originalScript : undefined; -                                const cfinalScript = cscript?.split("return")[cscript.split("return").length - 1]; -                                const val = cscript !== undefined ? `:=${cfinalScript?.substring(0, cfinalScript.length - 2)}` : -                                    Field.IsField(cfield) ? Field.toScriptString(cfield) : ""; -                                return val; +                                if (type === "number" && (contents === 0 || contents === "0")) { +                                    return "0"; +                                } else { +                                    const cfield = ComputedField.WithoutComputed(() => FieldValue(props.Document[props.fieldKey])); +                                    console.log(cfield); +                                    if (type === "number") { +                                        return StrCast(cfield); +                                    } +                                    const cscript = cfield instanceof ComputedField ? cfield.script.originalScript : undefined; +                                    const cfinalScript = cscript?.split("return")[cscript.split("return").length - 1]; +                                    const val = cscript !== undefined ? (cfinalScript?.endsWith(";") ? `:=${cfinalScript?.substring(0, cfinalScript.length - 2)}` : cfinalScript) : +                                        Field.IsField(cfield) ? Field.toScriptString(cfield) : ""; +                                    return val; +                                } +                              }}                              SetValue={action((value: string) => {                                  let retVal = false; +                                  if (value.startsWith(":=")) {                                      retVal = this.props.setComputed(value.substring(2), props.Document, this.props.rowProps.column.id!, this.props.row, this.props.col);                                  } else {                                      const script = CompileScript(value, { requiredType: type, typecheck: false, editable: true, addReturn: true, params: { this: Doc.name, $r: "number", $c: "number", $: "any" } });                                      if (script.compiled) {                                          retVal = this.applyToDoc(props.Document, this.props.row, this.props.col, script.run); +                                        console.log("compiled");                                      } +                                  }                                  if (retVal) {                                      this._isEditing = false; // need to set this here. otherwise, the assignment of the field will invalidate & cause render() to be called with the wrong value for 'editing'                                      this.props.setIsEditing(false);                                  }                                  return retVal; + +                                //return true;                              })}                              OnFillDown={async (value: string) => {                                  const script = CompileScript(value, { requiredType: type, typecheck: false, editable: true, addReturn: true, params: { this: Doc.name, $r: "number", $c: "number", $: "any" } }); @@ -265,6 +308,8 @@ export class CollectionSchemaCell extends React.Component<CellProps> {                                  }                              }}                          /> + +                      </div >                      {/* {fieldIsDoc ? docExpander : null} */}                  </div> @@ -299,12 +344,473 @@ export class CollectionSchemaStringCell extends CollectionSchemaCell {  }  @observer +export class CollectionSchemaDateCell extends CollectionSchemaCell { +    @observable private _date: Date = this.props.rowProps.original[this.props.rowProps.column.id as string] instanceof DateField ? DateCast(this.props.rowProps.original[this.props.rowProps.column.id as string]).date : +        this.props.rowProps.original[this.props.rowProps.column.id as string] instanceof Date ? this.props.rowProps.original[this.props.rowProps.column.id as string] : new Date(); + +    @action +    handleChange = (date: any) => { +        console.log(date); +        this._date = date; +        // const script = CompileScript(date.toString(), { requiredType: "Date", addReturn: true, params: { this: Doc.name } }); +        // if (script.compiled) { +        //     console.log("scripting"); +        //     this.applyToDoc(this._document, this.props.row, this.props.col, script.run); +        // } else { +        console.log(DateCast(date)); +        // ^ DateCast is always undefined for some reason, but that is what the field should be set to +        this._document[this.props.rowProps.column.id as string] = date as Date; +        console.log(this._document[this.props.rowProps.column.id as string]); +        //} +    } + +    render() { +        return <DatePicker +            selected={this._date} +            onSelect={date => this.handleChange(date)} +            onChange={date => this.handleChange(date)} +        />; +    } +} + +@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._document[this.prop.fieldKey] = results.result; +            this._docTitle = this._doc?.title; + +            return true; +        } +        return false; +    } + +    onFocus = () => { +        this._overlayDisposer?.(); +        this._overlayDisposer = OverlayView.Instance.addElement(<DocumentIconContainer />, { x: 0, y: 0 }); +    } + +    @action +    onOpenClick = () => { +        this._preview = false; +        if (this._doc) { +            this.props.addDocTab(this._doc, "onRight"); +            return true; +        } +        return false; +    } + +    @action +    showPreview = (bool: boolean, e: any) => { +        if (this._isEditing) { +            this._preview = false; +        } else { +            if (bool) { +                console.log("show doc"); +                this.props.showDoc(this._doc, this.prop.DataDoc, e.clientX, e.clientY); +            } else { +                console.log("no doc"); +                this.props.showDoc(undefined); +            } +        } +    } + +    @action +    isEditingCalling = (isEditing: boolean): void => { +        this.showPreview(false, ""); +        document.removeEventListener("keydown", this.onKeyDown); +        isEditing && document.addEventListener("keydown", this.onKeyDown); +        this._isEditing = isEditing; +        this.props.setIsEditing(isEditing); +        this.props.changeFocusedCellByIndex(this.props.row, this.props.col); +    } + +    onDown = (e: any) => { +        this.props.changeFocusedCellByIndex(this.props.row, this.props.col); +        this.props.setPreviewDoc(this.props.rowProps.original); + +        let url: string; +        if (url = StrCast(this.props.rowProps.row.href)) { +            try { +                new URL(url); +                const temp = window.open(url)!; +                temp.blur(); +                window.focus(); +            } catch { } +        } + +        const field = this.props.rowProps.original[this.props.rowProps.column.id!]; +        const doc = FieldValue(Cast(field, Doc)); +        if (typeof field === "object" && doc) this.props.setPreviewDoc(doc); + +        this.showPreview(true, e); + +    } +      render() { -        return this.renderCellWithType("document"); +        if (typeof this._field === "object" && this._doc && this._docTitle) { +            return ( +                <div className="collectionSchemaView-cellWrapper" ref={this._focusRef} tabIndex={-1} +                    onPointerDown={(e) => { this.onDown(e); }} +                    onPointerEnter={(e) => { this.showPreview(true, e); }} +                    onPointerLeave={(e) => { this.showPreview(false, e); }} +                > + +                    <div className="collectionSchemaView-cellContents-document" +                        style={{ padding: "5.9px" }} +                        ref={this.dropRef} +                        onFocus={this.onFocus} +                        onBlur={() => this._overlayDisposer?.()} +                    > + +                        <EditableView +                            editing={this._isEditing} +                            isEditingCallback={this.isEditingCalling} +                            display={"inline"} +                            contents={this._docTitle} +                            height={"auto"} +                            maxHeight={Number(MAX_ROW_HEIGHT)} +                            GetValue={() => { +                                return StrCast(this._docTitle); +                            }} +                            SetValue={action((value: string) => { +                                this.onSetValue(value); +                                this.showPreview(false, ""); +                                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() { + +        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..213a72a85 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, faCalendar } 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, faCalendar);  export interface HeaderProps {      keyValue: SchemaHeaderField; @@ -33,7 +33,9 @@ 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" : this.props.keyType === ColumnType.Date ? "calendar" : +                    "align-justify";          return (              <div className="collectionSchemaView-header" style={{ background: this.props.keyValue.color }}>                  <CollectionSchemaColumnMenu @@ -72,6 +74,16 @@ export class CollectionSchemaAddColumnHeader extends React.Component<AddColumnHe      }  } + + + + + + + + + +  export interface ColumnMenuProps {      columnField: SchemaHeaderField;      // keyValue: string; @@ -160,10 +172,22 @@ 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 className={"columnMenu-option" + (type === ColumnType.Date ? " active" : "")} onClick={() => this.changeColumnType(ColumnType.Date)}> +                        <FontAwesomeIcon icon={"calendar"} size="sm" /> +                        Date +                    </div>                  </div>              </div >          ); @@ -258,17 +282,18 @@ export class CollectionSchemaColumnMenu extends React.Component<ColumnMenuProps>  } -interface KeysDropdownProps { +export interface KeysDropdownProps {      keyValue: string;      possibleKeys: string[];      existingKeys: string[];      canAddNew: boolean;      addNew: boolean; -    onSelect: (oldKey: string, newKey: string, addnew: boolean) => void; +    onSelect: (oldKey: string, newKey: string, addnew: boolean, filter?: string) => void;      setIsEditing: (isEditing: boolean) => void; +    width?: string;  }  @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; @@ -281,14 +306,23 @@ class KeysDropdown extends React.Component<KeysDropdownProps> {      @action      onSelect = (key: string): void => { -        this.props.onSelect(this._key, key, this.props.addNew); -        this.setKey(key); -        this._isOpen = false; -        this.props.setIsEditing(false); +        if (key.slice(0, this._key.length) === this._key && this._key !== key) { +            let filter = key.slice(this._key.length - key.length); +            this.props.onSelect(this._key, this._key, this.props.addNew, filter); +            console.log("YEE"); +        } +        else { +            this.props.onSelect(this._key, key, this.props.addNew); +            this.setKey(key); +            this._isOpen = false; +            this.props.setIsEditing(false); +        }      }      @undoBatch      onKeyDown = (e: React.KeyboardEvent): void => { +        //if (this._key !==) +          if (e.key === "Enter") {              const keyOptions = this._searchTerm === "" ? this.props.possibleKeys : this.props.possibleKeys.filter(key => key.toUpperCase().indexOf(this._searchTerm.toUpperCase()) > -1);              if (keyOptions.length) { @@ -331,19 +365,30 @@ 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;          const options = keyOptions.map(key => { -            return <div key={key} className="key-option" onPointerDown={e => e.stopPropagation()} onClick={() => { this.onSelect(key); this.setSearchTerm(""); }}>{key}</div>; +            return <div key={key} className="key-option" style={{ +                border: "1px solid lightgray", +                width: this.props.width, maxWidth: this.props.width, overflowX: "hidden" +            }} +                onPointerDown={e => e.stopPropagation()} onClick={() => { this.onSelect(key); this.setSearchTerm(""); }}>{key}</div>;          });          // if search term does not already exist as a group type, give option to create new group type -        if (!exactFound && this._searchTerm !== "" && this.props.canAddNew) { -            options.push(<div key={""} className="key-option" -                onClick={() => { this.onSelect(this._searchTerm); this.setSearchTerm(""); }}> -                Create "{this._searchTerm}" key</div>); +        if (this._key !== this._searchTerm.slice(0, this._key.length)) { +            if (!exactFound && this._searchTerm !== "" && this.props.canAddNew) { +                options.push(<div key={""} className="key-option" style={{ +                    border: "1px solid lightgray", +                    width: this.props.width, maxWidth: this.props.width, overflowX: "hidden" +                }} +                    onClick={() => { this.onSelect(this._searchTerm); this.setSearchTerm(""); }}> +                    Create "{this._searchTerm}" key</div>); +            }          }          return options; @@ -351,10 +396,24 @@ class KeysDropdown extends React.Component<KeysDropdownProps> {      render() {          return ( -            <div className="keys-dropdown"> -                <input className="keys-search" ref={this._inputRef} type="text" value={this._searchTerm} placeholder="Column key" onKeyDown={this.onKeyDown} -                    onChange={e => this.onChange(e.target.value)} onFocus={this.onFocus} onBlur={this.onBlur}></input> -                <div className="keys-options-wrapper" onPointerEnter={this.onPointerEnter} onPointerLeave={this.onPointerOut}> +            <div className="keys-dropdown" style={{ zIndex: 10, width: this.props.width, maxWidth: this.props.width, overflowX: "hidden" }}> +                {this._key === this._searchTerm.slice(0, this._key.length) ? +                    <div style={{ position: "absolute", marginLeft: "4px", marginTop: "3", color: "grey", pointerEvents: "none", lineHeight: 1.15 }}> +                        {this._key} +                    </div> +                    : undefined} +                <input className="keys-search" style={{ width: "100%" }} +                    ref={this._inputRef} type="text" value={this._searchTerm} placeholder="Column key" onKeyDown={this.onKeyDown} +                    onChange={e => this.onChange(e.target.value)} +                    onClick={(e) => { +                        //this._inputRef.current!.select(); +                        e.stopPropagation(); +                    }} onFocus={this.onFocus} onBlur={this.onBlur}></input> +                <div className="keys-options-wrapper" style={{ +                    backgroundColor: "white", +                    width: this.props.width, maxWidth: this.props.width, overflowX: "hidden" +                }} +                    onPointerEnter={this.onPointerEnter} onPointerLeave={this.onPointerOut}>                      {this.renderOptions()}                  </div>              </div > 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..5226a60f1 100644 --- a/src/client/views/collections/CollectionSchemaView.scss +++ b/src/client/views/collections/CollectionSchemaView.scss @@ -62,10 +62,15 @@          width: calc(100% - 52px);          margin-left: 50px; +        z-index: 100; +        overflow-y: visible; +          &.-header {              font-size: 12px;              height: 30px;              box-shadow: none; +            z-index: 100; +            overflow-y: visible;          }          .rt-resizable-header-content { @@ -172,27 +177,36 @@  } -.collectionSchemaView-header { +.collectionSchema-header-menu {      height: 100%; -    color: gray; +    z-index: 100; +    position: absolute; +    background:white; -    .collectionSchema-header-menu { +    .collectionSchema-header-toggler { +        z-index: 100; +        width: 100%;          height: 100%; +        padding: 4px; +        letter-spacing: 2px; +        text-transform: uppercase; -        .collectionSchema-header-toggler { -            width: 100%; -            height: 100%; -            padding: 4px; -            letter-spacing: 2px; -            text-transform: uppercase; - -            svg { -                margin-right: 4px; -            } +        svg { +            margin-right: 4px;          }      }  } +.collectionSchemaView-header { +    height: 100%; +    color: gray; +    z-index: 100; +    overflow-y: visible; +    display: flex; +    justify-content: space-between; +    flex-wrap: wrap; +} +  button.add-column {      width: 28px;  } @@ -253,13 +267,16 @@ button.add-column {      .keys-dropdown {          position: relative; -        width: 100%; +        //width: 100%; +        background-color: white;          input {              border: 2px solid $light-color-secondary;              padding: 3px;              height: 28px;              font-weight: bold; +            letter-spacing: "2px"; +            text-transform: "uppercase";              &:focus {                  font-weight: normal; @@ -273,11 +290,14 @@ button.add-column {              position: absolute;              top: 28px;              box-shadow: 0 10px 16px rgba(0, 0, 0, 0.1); +            background-color: white;              .key-option { -                background-color: $light-color; +                //background-color: $light-color; +                background-color: white;                  border: 1px solid lightgray;                  padding: 2px 3px; +                overflow-x: hidden;                  &:not(:first-child) {                      border-top: 0; @@ -412,6 +432,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 +492,7 @@ button.add-column {      top: 0;      right: 0;      background-color: lightgray; +  }  .doc-drag-over { @@ -429,6 +500,10 @@ button.add-column {  }  .collectionSchemaView-toolbar { +    z-index: 100; +} + +.collectionSchemaView-toolbar {      height: 30px;      display: flex;      justify-content: flex-end; 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 diff --git a/src/client/views/collections/CollectionStackingView.tsx b/src/client/views/collections/CollectionStackingView.tsx index 9f6643ee0..460f1e486 100644 --- a/src/client/views/collections/CollectionStackingView.tsx +++ b/src/client/views/collections/CollectionStackingView.tsx @@ -43,8 +43,8 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)      @observable _heightMap = new Map<string, number>();      @observable _cursor: CursorProperty = "grab";      @observable _scroll = 0; // used to force the document decoration to update when scrolling -    @computed get sectionHeaders() { return Cast(this.props.Document.sectionHeaders, listSpec(SchemaHeaderField)); } -    @computed get pivotField() { return StrCast(this.props.Document._pivotField); } +    @computed get columnHeaders() { return Cast(this.layoutDoc._columnHeaders, listSpec(SchemaHeaderField)); } +    @computed get pivotField() { return StrCast(this.layoutDoc._pivotField); }      @computed get filteredChildren() { return this.childLayoutPairs.filter(pair => pair.layout instanceof Doc).map(pair => pair.layout); }      @computed get xMargin() { return NumCast(this.props.Document._xMargin, 2 * Math.min(this.gridGap, .05 * this.props.PanelWidth())); }      @computed get yMargin() { return Math.max(this.props.Document._showTitle && !this.props.Document._showTitleHover ? 30 : 0, NumCast(this.props.Document._yMargin, 0)); } // 2 * this.gridGap)); } @@ -53,19 +53,18 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)      @computed get isStackingView() { return BoolCast(this.props.Document.singleColumn, true); }      @computed get numGroupColumns() { return this.isStackingView ? Math.max(1, this.Sections.size + (this.showAddAGroup ? 1 : 0)) : 1; } -    @computed get showAddAGroup() { return (this.pivotField && (this.props.Document._chromeStatus !== 'view-mode' && this.props.Document._chromeStatus !== 'disabled')); } +    @computed get showAddAGroup() { return (this.pivotField && (this.layoutDoc._chromeStatus !== 'view-mode' && this.layoutDoc._chromeStatus !== 'disabled')); }      @computed get columnWidth() { -        TraceMobx(); -        return Math.min(this.props.PanelWidth() / (this.props as any).ContentScaling() - 2 * this.xMargin, -            this.isStackingView ? Number.MAX_VALUE : this.props.Document.columnWidth === -1 ? this.props.PanelWidth() - 2 * this.xMargin : NumCast(this.props.Document.columnWidth, 250)); +        return Math.min(this.props.PanelWidth() / this.props.ContentScaling() - 2 * this.xMargin, +            this.isStackingView ? Number.MAX_VALUE : this.layoutDoc._columnWidth === -1 ? this.props.PanelWidth() - 2 * this.xMargin : NumCast(this.layoutDoc._columnWidth, 250));      }      @computed get NodeWidth() { return this.props.PanelWidth() - this.gridGap; }      constructor(props: any) {          super(props); -        if (this.sectionHeaders === undefined) { -            this.props.Document.sectionHeaders = new List<SchemaHeaderField>(); +        if (this.columnHeaders === undefined) { +            this.layoutDoc._columnHeaders = new List<SchemaHeaderField>();          }      } @@ -79,8 +78,8 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)              const dxf = () => this.getDocTransform(d, dref.current!);              this._docXfs.push({ dxf, width, height });              const rowSpan = Math.ceil((height() + this.gridGap) / this.gridGap); -             -            const style = this.isStackingView ? { width: width(), marginTop: i || this.searchDoc? this.gridGap : 0, marginBottom: this.searchDoc? 10:0, height: height() } : { gridRowEnd: `span ${rowSpan}` }; + +            const style = this.isStackingView ? { width: width(), marginTop: i || this.searchDoc ? this.gridGap : 0, marginBottom: this.searchDoc ? 10 : 0, height: height() } : { gridRowEnd: `span ${rowSpan}` };              return <div className={`collectionStackingView-${this.isStackingView ? "columnDoc" : "masonryDoc"}`} key={d[Id]} ref={dref} style={style} >                  {this.getDisplayDoc(d, (!d.isTemplateDoc && !d.isTemplateForField && !d.PARAMS) ? undefined : this.props.DataDoc, dxf, width)}              </div>; @@ -92,14 +91,14 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)      }      get Sections() { -        if (!this.pivotField || this.sectionHeaders instanceof Promise) return new Map<SchemaHeaderField, Doc[]>(); +        if (!this.pivotField || this.columnHeaders instanceof Promise) return new Map<SchemaHeaderField, Doc[]>(); -        if (this.sectionHeaders === undefined) { -            setTimeout(() => this.props.Document.sectionHeaders = new List<SchemaHeaderField>(), 0); +        if (this.columnHeaders === undefined) { +            setTimeout(() => this.layoutDoc._columnHeaders = new List<SchemaHeaderField>(), 0);              return new Map<SchemaHeaderField, Doc[]>();          } -        const sectionHeaders = Array.from(this.sectionHeaders); -        const fields = new Map<SchemaHeaderField, Doc[]>(sectionHeaders.map(sh => [sh, []] as [SchemaHeaderField, []])); +        const columnHeaders = Array.from(this.columnHeaders); +        const fields = new Map<SchemaHeaderField, Doc[]>(columnHeaders.map(sh => [sh, []] as [SchemaHeaderField, []]));          let changed = false;          this.filteredChildren.map(d => {              const sectionValue = (d[this.pivotField] ? d[this.pivotField] : `NO ${this.pivotField.toUpperCase()} VALUE`) as object; @@ -108,26 +107,26 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)              const castedSectionValue = !isNaN(parsed) ? parsed : sectionValue;              // look for if header exists already -            const existingHeader = sectionHeaders.find(sh => sh.heading === (castedSectionValue ? castedSectionValue.toString() : `NO ${this.pivotField.toUpperCase()} VALUE`)); +            const existingHeader = columnHeaders.find(sh => sh.heading === (castedSectionValue ? castedSectionValue.toString() : `NO ${this.pivotField.toUpperCase()} VALUE`));              if (existingHeader) {                  fields.get(existingHeader)!.push(d);              }              else {                  const newSchemaHeader = new SchemaHeaderField(castedSectionValue ? castedSectionValue.toString() : `NO ${this.pivotField.toUpperCase()} VALUE`);                  fields.set(newSchemaHeader, [d]); -                sectionHeaders.push(newSchemaHeader); +                columnHeaders.push(newSchemaHeader);                  changed = true;              }          });          // remove all empty columns if hideHeadings is set -        if (this.props.Document.hideHeadings) { +        if (this.layoutDoc._columnsHideIfEmpty) {              Array.from(fields.keys()).filter(key => !fields.get(key)!.length).map(header => {                  fields.delete(header); -                sectionHeaders.splice(sectionHeaders.indexOf(header), 1); +                columnHeaders.splice(columnHeaders.indexOf(header), 1);                  changed = true;              });          } -        changed && setTimeout(action(() => { if (this.sectionHeaders) { this.sectionHeaders.length = 0; this.sectionHeaders.push(...sectionHeaders); } }), 0); +        changed && setTimeout(action(() => { if (this.columnHeaders) { this.columnHeaders.length = 0; this.columnHeaders.push(...columnHeaders); } }), 0);          return fields;      } @@ -139,7 +138,7 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)          let wid = this.columnWidth / (this.isStackingView ? this.numGroupColumns : 1);          if (!layoutDoc._fitWidth && nw && nh) {              const aspect = nw && nh ? nh / nw : 1; -            if (!(this.props.Document.fillColumn)) wid = Math.min(layoutDoc[WidthSym](), wid); +            if (!(this.layoutDoc._columnsFill)) wid = Math.min(layoutDoc[WidthSym](), wid);              return wid * aspect;          }          return layoutDoc._fitWidth ? wid * NumCast(layoutDoc.scrollHeight, nh) / (nw || 1) : layoutDoc[HeightSym](); @@ -150,7 +149,7 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)          // reset section headers when a new filter is inputted          this._pivotFieldDisposer = reaction(              () => this.pivotField, -            () => this.props.Document.sectionHeaders = new List() +            () => this.layoutDoc._columnHeaders = new List()          );      }      componentWillUnmount() { @@ -214,7 +213,7 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)              fitToBox={false}              dontRegisterView={this.props.dontRegisterView}              rootSelected={this.rootSelected} -            dropAction={StrCast(this.props.Document.childDropAction) as dropActionType} +            dropAction={StrCast(this.layoutDoc.childDropAction) as dropActionType}              onClick={this.onChildClickHandler}              onDoubleClick={this.onChildDoubleClickHandler}              ScreenToLocalTransform={dxf} @@ -239,7 +238,7 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)          if (!d) return 0;          const layoutDoc = Doc.Layout(d, this.props.ChildLayoutTemplate?.());          const nw = NumCast(layoutDoc._nativeWidth); -        return Math.min(nw && !this.props.Document.fillColumn ? d[WidthSym]() : Number.MAX_VALUE, this.columnWidth / this.numGroupColumns); +        return Math.min(nw && !this.layoutDoc._columnsFill ? d[WidthSym]() : Number.MAX_VALUE, this.columnWidth / this.numGroupColumns);      }      getDocHeight(d?: Doc) {          if (!d) return 0; @@ -249,7 +248,7 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)          let wid = this.columnWidth / (this.isStackingView ? this.numGroupColumns : 1);          if (!layoutDoc._fitWidth && nw && nh) {              const aspect = nw && nh ? nh / nw : 1; -            if (!(this.props.Document.fillColumn)) wid = Math.min(layoutDoc[WidthSym](), wid); +            if (!(this.layoutDoc._columnsFill)) wid = Math.min(layoutDoc[WidthSym](), wid);              return wid * aspect;          }          return layoutDoc._fitWidth ? !nh ? this.props.PanelHeight() - 2 * this.yMargin : @@ -262,7 +261,7 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)      }      @action      onDividerMove = (e: PointerEvent, down: number[], delta: number[]) => { -        this.layoutDoc.columnWidth = Math.max(10, this.columnWidth + delta[0]); +        this.layoutDoc._columnWidth = Math.max(10, this.columnWidth + delta[0]);          return false;      } @@ -344,7 +343,7 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)                      this.refList.push(ref);                      const doc = this.props.DataDoc && this.props.DataDoc.layout === this.layoutDoc ? this.props.DataDoc : this.layoutDoc;                      this.observer = new _global.ResizeObserver(action((entries: any) => { -                        if (this.props.Document._autoHeight && ref && this.refList.length && !SnappingManager.GetIsDragging()) { +                        if (this.layoutDoc._autoHeight && ref && this.refList.length && !SnappingManager.GetIsDragging()) {                              Doc.Layout(doc)._height = Math.min(1200, Math.max(...this.refList.map(r => Number(getComputedStyle(r).height.replace("px", "")))));                          }                      })); @@ -391,7 +390,7 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)                      this.refList.push(ref);                      const doc = this.props.DataDoc && this.props.DataDoc.layout === this.layoutDoc ? this.props.DataDoc : this.layoutDoc;                      this.observer = new _global.ResizeObserver(action((entries: any) => { -                        if (this.props.Document._autoHeight && ref && this.refList.length && !SnappingManager.GetIsDragging()) { +                        if (this.layoutDoc._autoHeight && ref && this.refList.length && !SnappingManager.GetIsDragging()) {                              Doc.Layout(doc)._height = this.refList.reduce((p, r) => p + Number(getComputedStyle(r).height.replace("px", "")), 0);                          }                      })); @@ -414,9 +413,9 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)      @action      addGroup = (value: string) => { -        if (value && this.sectionHeaders) { +        if (value && this.columnHeaders) {              const schemaHdrField = new SchemaHeaderField(value); -            this.sectionHeaders.push(schemaHdrField); +            this.columnHeaders.push(schemaHdrField);              DocUtils.addFieldEnumerations(undefined, this.pivotField, [{ title: value, _backgroundColor: schemaHdrField.color }]);              return true;          } @@ -424,22 +423,22 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)      }      sortFunc = (a: [SchemaHeaderField, Doc[]], b: [SchemaHeaderField, Doc[]]): 1 | -1 => { -        const descending = BoolCast(this.props.Document.stackingHeadersSortDescending); +        const descending = StrCast(this.layoutDoc._columnsSort) === "descending";          const firstEntry = descending ? b : a;          const secondEntry = descending ? a : b;          return firstEntry[0].heading > secondEntry[0].heading ? 1 : -1;      }      onToggle = (checked: Boolean) => { -        this.props.Document._chromeStatus = checked ? "collapsed" : "view-mode"; +        this.layoutDoc._chromeStatus = checked ? "collapsed" : "view-mode";      }      onContextMenu = (e: React.MouseEvent): void => {          // need to test if propagation has stopped because GoldenLayout forces a parallel react hierarchy to be created for its top-level layout          if (!e.isPropagationStopped()) {              const subItems: ContextMenuProps[] = []; -            subItems.push({ description: `${this.props.Document.fillColumn ? "Variable Size" : "Autosize"} Column`, event: () => this.props.Document.fillColumn = !this.props.Document.fillColumn, icon: "plus" }); -            subItems.push({ description: `${this.Document._autoHeight ? "Variable Height" : "Auto Height"}`, event: () => this.layoutDoc._autoHeight = !this.layoutDoc._autoHeight, icon: "plus" }); +            subItems.push({ description: `${this.layoutDoc._columnsFill ? "Variable Size" : "Autosize"} Column`, event: () => this.layoutDoc._columnsFill = !this.layoutDoc._columnsFill, icon: "plus" }); +            subItems.push({ description: `${this.layoutDoc._autoHeight ? "Variable Height" : "Auto Height"}`, event: () => this.layoutDoc._autoHeight = !this.layoutDoc._autoHeight, icon: "plus" });              ContextMenu.Instance.addItem({ description: "Options...", subitems: subItems, icon: "eye" });          }      } @@ -449,7 +448,7 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)          let sections = [[undefined, this.filteredChildren] as [SchemaHeaderField | undefined, Doc[]]];          if (this.pivotField) {              const entries = Array.from(this.Sections.entries()); -            sections = entries.sort(this.sortFunc); +            sections = this.layoutDoc._columnsSort ? entries.sort(this.sortFunc) : entries;          }          return sections.map((section, i) => this.isStackingView ? this.sectionStacking(section[0], section[1]) : this.sectionMasonry(section[0], section[1], i === 0));      } @@ -492,10 +491,10 @@ export class CollectionStackingView extends CollectionSubView(StackingDocument)                              style={{ width: !this.isStackingView ? "100%" : this.columnWidth / this.numGroupColumns - 10, marginTop: 10 }}>                              <EditableView {...editableViewProps} />                          </div>} -                    {this.props.Document._chromeStatus !== 'disabled' && this.props.isSelected() ? <Switch +                    {this.layoutDoc._chromeStatus !== 'disabled' && this.props.isSelected() ? <Switch                          onChange={this.onToggle}                          onClick={this.onToggle} -                        defaultChecked={this.props.Document._chromeStatus !== 'view-mode'} +                        defaultChecked={this.layoutDoc._chromeStatus !== 'view-mode'}                          checkedChildren="edit"                          unCheckedChildren="view"                      /> : null} diff --git a/src/client/views/collections/CollectionStackingViewFieldColumn.tsx b/src/client/views/collections/CollectionStackingViewFieldColumn.tsx index b60ed853b..2f4a25bfe 100644 --- a/src/client/views/collections/CollectionStackingViewFieldColumn.tsx +++ b/src/client/views/collections/CollectionStackingViewFieldColumn.tsx @@ -96,8 +96,8 @@ export class CollectionStackingViewFieldColumn extends React.Component<CSVFieldC          const key = StrCast(this.props.parent.props.Document._pivotField);          const castedValue = this.getValue(value);          if (castedValue) { -            if (this.props.parent.sectionHeaders) { -                if (this.props.parent.sectionHeaders.map(i => i.heading).indexOf(castedValue.toString()) > -1) { +            if (this.props.parent.columnHeaders) { +                if (this.props.parent.columnHeaders.map(i => i.heading).indexOf(castedValue.toString()) > -1) {                      return false;                  }              } @@ -148,9 +148,9 @@ export class CollectionStackingViewFieldColumn extends React.Component<CSVFieldC      deleteColumn = () => {          const key = StrCast(this.props.parent.props.Document._pivotField);          this.props.docList.forEach(d => d[key] = undefined); -        if (this.props.parent.sectionHeaders && this.props.headingObject) { -            const index = this.props.parent.sectionHeaders.indexOf(this.props.headingObject); -            this.props.parent.sectionHeaders.splice(index, 1); +        if (this.props.parent.columnHeaders && this.props.headingObject) { +            const index = this.props.parent.columnHeaders.indexOf(this.props.headingObject); +            this.props.parent.columnHeaders.splice(index, 1);          }      } @@ -168,7 +168,7 @@ export class CollectionStackingViewFieldColumn extends React.Component<CSVFieldC      startDrag = (e: PointerEvent, down: number[], delta: number[]) => {          const alias = Doc.MakeAlias(this.props.parent.props.Document); -        alias._width = this.props.parent.props.PanelWidth() / (Cast(this.props.parent.props.Document.sectionHeaders, listSpec(SchemaHeaderField))?.length || 1); +        alias._width = this.props.parent.props.PanelWidth() / (Cast(this.props.parent.columnHeaders, listSpec(SchemaHeaderField))?.length || 1);          alias._pivotField = undefined;          const key = StrCast(this.props.parent.props.Document._pivotField);          let value = this.getValue(this._heading); @@ -259,8 +259,8 @@ export class CollectionStackingViewFieldColumn extends React.Component<CSVFieldC                      }                  }, icon: "compress-arrows-alt"              })); -        layoutItems.push({ description: ":freeform", event: () => this.props.parent.props.addDocument(Docs.Create.FreeformDocument([], { _width: 200, _height: 200, _LODdisable: true })), icon: "compress-arrows-alt" }); -        layoutItems.push({ description: ":carousel", event: () => this.props.parent.props.addDocument(Docs.Create.CarouselDocument([], { _width: 400, _height: 200, _LODdisable: true })), icon: "compress-arrows-alt" }); +        layoutItems.push({ description: ":freeform", event: () => this.props.parent.props.addDocument(Docs.Create.FreeformDocument([], { _width: 200, _height: 200 })), icon: "compress-arrows-alt" }); +        layoutItems.push({ description: ":carousel", event: () => this.props.parent.props.addDocument(Docs.Create.CarouselDocument([], { _width: 400, _height: 200 })), icon: "compress-arrows-alt" });          layoutItems.push({ description: ":columns", event: () => this.props.parent.props.addDocument(Docs.Create.MulticolumnDocument([], { _width: 200, _height: 200 })), icon: "compress-arrows-alt" });          layoutItems.push({ description: ":image", event: () => this.props.parent.props.addDocument(Docs.Create.ImageDocument("http://www.cs.brown.edu/~bcz/face.gif", { _width: 200, _height: 200 })), icon: "compress-arrows-alt" }); @@ -359,10 +359,10 @@ export class CollectionStackingViewFieldColumn extends React.Component<CSVFieldC                      background: this._background                  }}                  ref={this.createColumnDropRef} onPointerEnter={this.pointerEntered} onPointerLeave={this.pointerLeave}> -                {this.props.parent.Document.hideHeadings ? (null) : headingView} +                {this.props.parent.Document._columnsHideIfEmpty ? (null) : headingView}                  {                      this.collapsed ? (null) : -                        <div> +                        <div style={{ marginTop: 5 }}>                              <div key={`${heading}-stack`} className={`collectionStackingView-masonry${singleColumn ? "Single" : "Grid"}`}                                  style={{                                      padding: singleColumn ? `${columnYMargin}px ${0}px ${style.yMargin}px ${0}px` : `${columnYMargin}px ${0}px`, diff --git a/src/client/views/collections/CollectionSubView.tsx b/src/client/views/collections/CollectionSubView.tsx index 3a13ac822..ed8535ecb 100644 --- a/src/client/views/collections/CollectionSubView.tsx +++ b/src/client/views/collections/CollectionSubView.tsx @@ -1,15 +1,14 @@  import { action, computed, IReactionDisposer, reaction } from "mobx";  import { basename } from 'path';  import CursorField from "../../../fields/CursorField"; -import { Doc, Opt } from "../../../fields/Doc"; +import { Doc, Opt, Field } from "../../../fields/Doc";  import { Id } from "../../../fields/FieldSymbols";  import { List } from "../../../fields/List";  import { listSpec } from "../../../fields/Schema";  import { ScriptField } from "../../../fields/ScriptField";  import { WebField } from "../../../fields/URLField"; -import { Cast, ScriptCast, NumCast } from "../../../fields/Types"; +import { Cast, ScriptCast, NumCast, StrCast } from "../../../fields/Types";  import { GestureUtils } from "../../../pen-gestures/GestureUtils"; -import { Upload } from "../../../server/SharedMediaTypes";  import { Utils, returnFalse, returnEmptyFilter } from "../../../Utils";  import { DocServer } from "../../DocServer";  import { Networking } from "../../Network"; @@ -136,8 +135,12 @@ export function CollectionSubView<T, X>(schemaCtor: (doc: Doc) => T, moreProps?:              const filteredDocs = docFilters.length && !this.props.dontRegisterView ? childDocs.filter(d => {                  for (const facetKey of Object.keys(filterFacets)) {                      const facet = filterFacets[facetKey]; -                    const satisfiesFacet = Object.keys(facet).some(value => -                        (facet[value] === "x") !== Doc.matchFieldValue(d, facetKey, value)); +                    const satisfiesFacet = Object.keys(facet).some(value => { +                        if (facet[value] === "match") { +                            return d[facetKey] === undefined || Field.toString(d[facetKey] as Field).includes(value); +                        } +                        return (facet[value] === "x") !== Doc.matchFieldValue(d, facetKey, value); +                    });                      if (!satisfiesFacet) {                          return false;                      } @@ -208,7 +211,6 @@ export function CollectionSubView<T, X>(schemaCtor: (doc: Doc) => T, moreProps?:          addDocument = (doc: Doc | Doc[]) => this.props.addDocument(doc); -        @undoBatch          @action          protected onInternalDrop(e: Event, de: DragManager.DropEvent): boolean {              const docDragData = de.complete.docDragData; @@ -222,7 +224,7 @@ export function CollectionSubView<T, X>(schemaCtor: (doc: Doc) => T, moreProps?:                      const movedDocs = docDragData.droppedDocuments.filter((d, i) => docDragData.draggedDocuments[i] === d);                      const addedDocs = docDragData.droppedDocuments.filter((d, i) => docDragData.draggedDocuments[i] !== d);                      const res = addedDocs.length ? this.addDocument(addedDocs) : true; -                    added = movedDocs.length ? docDragData.moveDocument(movedDocs, this.props.Document, de.embedKey || !this.props.isAnnotationOverlay ? this.addDocument : returnFalse) : res; +                    added = movedDocs.length ? docDragData.moveDocument(movedDocs, this.props.Document, Doc.AreProtosEqual(Cast(movedDocs[0].annotationOn, Doc, null), this.props.Document) || de.embedKey || !this.props.isAnnotationOverlay ? this.addDocument : returnFalse) : res;                  } else {                      added = this.addDocument(docDragData.droppedDocuments);                  } @@ -235,21 +237,7 @@ export function CollectionSubView<T, X>(schemaCtor: (doc: Doc) => T, moreProps?:              }              return false;          } -        readUploadedFileAsText = (inputFile: File) => { -            const temporaryFileReader = new FileReader(); - -            return new Promise((resolve, reject) => { -                temporaryFileReader.onerror = () => { -                    temporaryFileReader.abort(); -                    reject(new DOMException("Problem parsing input file.")); -                }; -                temporaryFileReader.onload = () => { -                    resolve(temporaryFileReader.result); -                }; -                temporaryFileReader.readAsText(inputFile); -            }); -        }          @undoBatch          @action          protected async onExternalDrop(e: React.DragEvent, options: DocumentOptions, completed?: () => void) { @@ -268,8 +256,7 @@ export function CollectionSubView<T, X>(schemaCtor: (doc: Doc) => T, moreProps?:              e.stopPropagation();              e.preventDefault(); -            const { addDocument } = this; -            if (!addDocument) { +            if (!this.addDocument) {                  alert("this.props.addDocument does not exist. Aborting drop operation.");                  return;              } @@ -283,14 +270,14 @@ export function CollectionSubView<T, X>(schemaCtor: (doc: Doc) => T, moreProps?:                              DocServer.GetRefField(docid).then(f => {                                  if (f instanceof Doc) {                                      if (options.x || options.y) { f.x = options.x; f.y = options.y; } // should be in CollectionFreeFormView -                                    (f instanceof Doc) && addDocument(f); +                                    (f instanceof Doc) && this.addDocument(f);                                  }                              });                          } else { -                            addDocument(Docs.Create.WebDocument(href, { ...options, title: href })); +                            this.addDocument(Docs.Create.WebDocument(href, { ...options, title: href }));                          }                      } else if (text) { -                        addDocument(Docs.Create.TextDocument(text, { ...options, _width: 100, _height: 25 })); +                        this.addDocument(Docs.Create.TextDocument(text, { ...options, _width: 100, _height: 25 }));                      }                      return;                  } @@ -310,7 +297,7 @@ export function CollectionSubView<T, X>(schemaCtor: (doc: Doc) => T, moreProps?:                          if (source.startsWith("http")) {                              const doc = Docs.Create.ImageDocument(source, { ...options, _width: 300 });                              ImageUtils.ExtractExif(doc); -                            addDocument(doc); +                            this.addDocument(doc);                          }                          return;                      } else { @@ -341,7 +328,7 @@ export function CollectionSubView<T, X>(schemaCtor: (doc: Doc) => T, moreProps?:                                      const rect = "getBoundingClientRect" in focusNode ? focusNode.getBoundingClientRect() : focusNode?.parentElement.getBoundingClientRect();                                      const x = (rect?.x || 0);                                      const y = NumCast(srcWeb._scrollTop) + (rect?.y || 0); -                                    const anchor = Docs.Create.FreeformDocument([], { _LODdisable: true, _backgroundColor: "transparent", _width: 25, _height: 25, x, y, annotationOn: srcWeb }); +                                    const anchor = Docs.Create.FreeformDocument([], { _backgroundColor: "transparent", _width: 25, _height: 25, x, y, annotationOn: srcWeb });                                      anchor.context = srcWeb;                                      const key = Doc.LayoutFieldKey(srcWeb);                                      Doc.AddDocToList(srcWeb, key + "-annotations", anchor); @@ -355,9 +342,9 @@ export function CollectionSubView<T, X>(schemaCtor: (doc: Doc) => T, moreProps?:              }              if (text) { -                if (text.includes("www.youtube.com/watch")) { +                if (text.includes("www.youtube.com/watch") || text.includes("www.youtube.com/embed")) {                      const url = text.replace("youtube.com/watch?v=", "youtube.com/embed/").split("&")[0]; -                    addDocument(Docs.Create.VideoDocument(url, { +                    this.addDocument(Docs.Create.VideoDocument(url, {                          ...options,                          title: url,                          _width: 400, @@ -410,10 +397,10 @@ export function CollectionSubView<T, X>(schemaCtor: (doc: Doc) => T, moreProps?:                      const file = item.getAsFile();                      file?.type && files.push(file); -                    file?.type === "application/json" && this.readUploadedFileAsText(file).then(result => { +                    file?.type === "application/json" && Utils.readUploadedFileAsText(file).then(result => {                          console.log(result);                          const json = JSON.parse(result as string); -                        addDocument(Docs.Create.TreeDocument( +                        this.addDocument(Docs.Create.TreeDocument(                              json["rectangular-puzzle"].crossword.clues[0].clue.map((c: any) => {                                  const label = Docs.Create.LabelDocument({ title: c["#text"], _width: 120, _height: 20 });                                  const proto = Doc.GetProto(label); @@ -425,38 +412,18 @@ export function CollectionSubView<T, X>(schemaCtor: (doc: Doc) => T, moreProps?:                      });                  }              } -            for (const { source: { name, type }, result } of await Networking.UploadFilesToServer(files)) { -                if (result instanceof Error) { -                    alert(`Upload failed: ${result.message}`); -                    return; -                } -                const full = { ...options, _width: 400, title: name }; -                const pathname = Utils.prepend(result.accessPaths.agnostic.client); -                const doc = await DocUtils.DocumentFromType(type, pathname, full); -                if (!doc) { -                    continue; -                } -                const proto = Doc.GetProto(doc); -                proto.text = result.rawText; -                proto.fileUpload = basename(pathname).replace("upload_", "").replace(/\.[a-z0-9]*$/, ""); -                if (Upload.isImageInformation(result)) { -                    proto["data-nativeWidth"] = (result.nativeWidth > result.nativeHeight) ? 400 * result.nativeWidth / result.nativeHeight : 400; -                    proto["data-nativeHeight"] = (result.nativeWidth > result.nativeHeight) ? 400 : 400 / (result.nativeWidth / result.nativeHeight); -                    proto.contentSize = result.contentSize; -                } -                generatedDocuments.push(doc); -            } +            generatedDocuments.push(...await DocUtils.uploadFilesToDocs(files, options));              if (generatedDocuments.length) {                  const set = generatedDocuments.length > 1 && generatedDocuments.map(d => DocUtils.iconify(d));                  if (set) { -                    addDocument(DocUtils.pileup(generatedDocuments, options.x!, options.y!)!); +                    this.addDocument(DocUtils.pileup(generatedDocuments, options.x!, options.y!)!);                  } else { -                    generatedDocuments.forEach(addDocument); +                    generatedDocuments.forEach(this.addDocument);                  }                  completed?.();              } else {                  if (text && !text.includes("https://")) { -                    addDocument(Docs.Create.TextDocument(text, { ...options, _width: 400, _height: 315 })); +                    this.addDocument(Docs.Create.TextDocument(text, { ...options, _width: 400, _height: 315 }));                  }              }              batch.end(); @@ -473,3 +440,4 @@ import { DocumentType } from "../../documents/DocumentTypes";  import { FormattedTextBox, GoogleRef } from "../nodes/formattedText/FormattedTextBox";  import { CollectionView } from "./CollectionView";  import { SelectionManager } from "../../util/SelectionManager"; + diff --git a/src/client/views/collections/CollectionTreeView.tsx b/src/client/views/collections/CollectionTreeView.tsx index d54f4d6e6..620b977fa 100644 --- a/src/client/views/collections/CollectionTreeView.tsx +++ b/src/client/views/collections/CollectionTreeView.tsx @@ -176,7 +176,7 @@ class TreeView extends React.Component<TreeViewProps> {          })}          OnFillDown={undoBatch((value: string) => {              Doc.SetInPlace(this.doc, key, value, false); -            const doc = Docs.Create.FreeformDocument([], { title: "-", x: 0, y: 0, _width: 100, _height: 25, _LODdisable: true, templates: new List<string>([Templates.Title.Layout]) }); +            const doc = Docs.Create.FreeformDocument([], { title: "-", x: 0, y: 0, _width: 100, _height: 25, templates: new List<string>([Templates.Title.Layout]) });              Doc.SetInPlace(this.doc, "editTitle", undefined, false);              Doc.SetInPlace(doc, "editTitle", "*", false);              return this.props.addDocument(doc); @@ -304,7 +304,7 @@ class TreeView extends React.Component<TreeViewProps> {      }      rtfWidth = () => Math.min(this.layoutDoc?.[WidthSym](), this.props.panelWidth() - 20); -    rtfHeight = () => this.rtfWidth() < this.layoutDoc?.[WidthSym]() ? Math.min(this.layoutDoc?.[HeightSym](), this.MAX_EMBED_HEIGHT) : this.MAX_EMBED_HEIGHT; +    rtfHeight = () => this.rtfWidth() <= this.layoutDoc?.[WidthSym]() ? Math.min(this.layoutDoc?.[HeightSym](), this.MAX_EMBED_HEIGHT) : this.MAX_EMBED_HEIGHT;      @computed get renderContent() {          TraceMobx(); @@ -332,8 +332,8 @@ class TreeView extends React.Component<TreeViewProps> {              </div></ul>;          } else {              const layoutDoc = this.layoutDoc; -            const panelHeight = layoutDoc.type === DocumentType.RTF ? this.rtfHeight : this.docHeight; -            const panelWidth = layoutDoc.type === DocumentType.RTF ? this.rtfWidth : this.docWidth; +            const panelHeight = StrCast(Doc.LayoutField(layoutDoc)).includes("FormattedTextBox") ? this.rtfHeight : this.docHeight; +            const panelWidth = StrCast(Doc.LayoutField(layoutDoc)).includes("FormattedTextBox") ? this.rtfWidth : this.docWidth;              return <div ref={this._dref} style={{ display: "inline-block", height: panelHeight() }} key={this.doc[Id]}>                  <ContentFittingDocumentView                      Document={layoutDoc} @@ -386,8 +386,8 @@ class TreeView extends React.Component<TreeViewProps> {          e.stopPropagation();      } -    @computed -    get renderBullet() { +    @computed get renderBullet() { +        TraceMobx();          const checked = this.doc.type === DocumentType.COL ? undefined : this.onCheckedClick ? (this.doc.treeViewChecked ?? "unchecked") : undefined;          return <div className="bullet"              title={this.childDocs?.length ? `click to see ${this.childDocs?.length} items` : "view fields"} @@ -828,7 +828,7 @@ export class CollectionTreeView extends CollectionSubView<Document, Partial<coll                          SetValue={undoBatch((value: string) => Doc.SetInPlace(this.dataDoc, "title", value, false) || true)}                          OnFillDown={undoBatch((value: string) => {                              Doc.SetInPlace(this.dataDoc, "title", value, false); -                            const doc = Docs.Create.FreeformDocument([], { title: "", x: 0, y: 0, _width: 100, _height: 25, _LODdisable: true, templates: new List<string>([Templates.Title.Layout]) }); +                            const doc = Docs.Create.FreeformDocument([], { title: "", x: 0, y: 0, _width: 100, _height: 25, templates: new List<string>([Templates.Title.Layout]) });                              Doc.SetInPlace(doc, "editTitle", "*", false);                              this.addDoc(doc, childDocs.length ? childDocs[0] : undefined, true);                          })} />} diff --git a/src/client/views/collections/CollectionView.tsx b/src/client/views/collections/CollectionView.tsx index 8c953a23f..43bfbc913 100644 --- a/src/client/views/collections/CollectionView.tsx +++ b/src/client/views/collections/CollectionView.tsx @@ -18,7 +18,7 @@ import { BoolCast, Cast, NumCast, ScriptCast, StrCast } from '../../../fields/Ty  import { ImageField } from '../../../fields/URLField';  import { TraceMobx } from '../../../fields/util';  import { emptyFunction, emptyPath, returnFalse, returnOne, returnZero, setupMoveUpEvents, Utils, returnEmptyFilter } from '../../../Utils'; -import { Docs } from '../../documents/Documents'; +import { Docs, DocUtils } from '../../documents/Documents';  import { DocumentType } from '../../documents/DocumentTypes';  import { CurrentUserUtils } from '../../util/CurrentUserUtils';  import { ImageUtils } from '../../util/Import & Export/ImageUtils'; @@ -47,6 +47,8 @@ import { CollectionGridView } from './collectionGrid/CollectionGridView';  import './CollectionView.scss';  import { CollectionViewBaseChrome } from './CollectionViewChromes';  import { UndoManager } from '../../util/UndoManager'; +import { RichTextField } from '../../../fields/RichTextField'; +import { TextField } from '../../util/ProsemirrorCopy/prompt';  const higflyout = require("@hig/flyout");  export const { anchorPoints } = higflyout;  export const Flyout = higflyout.default; @@ -137,7 +139,23 @@ export class CollectionView extends Touchable<FieldViewProps & CollectionViewCus              } else if (this.dataDoc[AclSym] === AclAddonly) {                  added.map(doc => Doc.AddDocToList(targetDataDoc, this.props.fieldKey, doc));              } else { -                added.map(doc => doc.context = this.props.Document); +                added.map(doc => { +                    const context = Cast(doc.context, Doc, null); +                    if (context && (context.type === DocumentType.VID || context.type === DocumentType.WEB || context.type === DocumentType.PDF || context.type === DocumentType.IMG) && +                        !DocListCast(doc.links).some(d => d.isPushpin)) { +                        const pushpin = Docs.Create.FontIconDocument({ +                            icon: "map-pin", x: Cast(doc.x, "number", null), y: Cast(doc.y, "number", null), _backgroundColor: "#0000003d", color: "#ACCEF7", +                            _width: 15, _height: 15, _xPadding: 0, isLinkButton: true, displayTimecode: Cast(doc.displayTimecode, "number", null) +                        }); +                        Doc.AddDocToList(context, Doc.LayoutFieldKey(context) + "-annotations", pushpin); +                        const pushpinLink = DocUtils.MakeLink({ doc: pushpin }, { doc: doc }, "pushpin"); +                        const first = DocListCast(pushpin.links).find(d => d instanceof Doc); +                        first && (first.hidden = true); +                        pushpinLink && (Doc.GetProto(pushpinLink).isPushpin = true); +                        doc.displayTimecode = undefined; +                    } +                    doc.context = this.props.Document; +                });                  added.map(add => Doc.AddDocToList(Cast(Doc.UserDoc().myCatalog, Doc, null), "data", add));                  targetDataDoc[this.props.fieldKey] = new List<Doc>([...docList, ...added]);                  targetDataDoc[this.props.fieldKey + "-lastModified"] = new DateField(new Date(Date.now())); @@ -207,8 +225,8 @@ export class CollectionView extends Touchable<FieldViewProps & CollectionViewCus              case CollectionViewType.Pile: { return (<CollectionPileView key="collview" {...props} />); }              case CollectionViewType.Carousel: { return (<CollectionCarouselView key="collview" {...props} />); }              case CollectionViewType.Carousel3D: { return (<CollectionCarousel3DView key="collview" {...props} />); } -            case CollectionViewType.Stacking: { this.props.Document.singleColumn = true; return (<CollectionStackingView key="collview" {...props} />); } -            case CollectionViewType.Masonry: { this.props.Document.singleColumn = false; return (<CollectionStackingView key="collview" {...props} />); } +            case CollectionViewType.Stacking: { this.props.Document._columnsStack = true; return (<CollectionStackingView key="collview" {...props} />); } +            case CollectionViewType.Masonry: { this.props.Document._columnsStack = false; return (<CollectionStackingView key="collview" {...props} />); }              case CollectionViewType.Time: { return (<CollectionTimeView key="collview" {...props} />); }              case CollectionViewType.Map: return (<CollectionMapView key="collview" {...props} />);              case CollectionViewType.Grid: return (<CollectionGridView key="gridview" {...props} />); @@ -357,10 +375,11 @@ export class CollectionView extends Touchable<FieldViewProps & CollectionViewCus          return viewSpecScript ? docs.filter(d => viewSpecScript.script.run({ doc: d }, console.log).result) : docs;      }      @computed get _allFacets() { -        const facets = new Set<string>(); -        this.childDocs.filter(child => child).forEach(child => Object.keys(Doc.GetProto(child)).forEach(key => facets.add(key))); +        TraceMobx(); +        const facets = new Set<string>(["type", "text", "data", "author", "ACL"]); +        this.childDocs.filter(child => child).forEach(child => child && Object.keys(Doc.GetProto(child)).forEach(key => facets.add(key)));          Doc.AreProtosEqual(this.dataDoc, this.props.Document) && this.childDocs.filter(child => child).forEach(child => Object.keys(child).forEach(key => facets.add(key))); -        return Array.from(facets); +        return Array.from(facets).filter(f => !f.startsWith("_") && !["proto", "zIndex", "isPrototype", "context", "text-noTemplate"].includes(f)).sort();      }      /** @@ -387,8 +406,13 @@ export class CollectionView extends Touchable<FieldViewProps & CollectionViewCus              }          } else {              const allCollectionDocs = DocListCast(this.dataDoc[this.props.fieldKey]); -            const facetValues = Array.from(allCollectionDocs.reduce((set, child) => -                set.add(Field.toString(child[facetHeader] as Field)), new Set<string>())); +            var rtfields = 0; +            const facetValues = Array.from(allCollectionDocs.reduce((set, child) => { +                const field = child[facetHeader] as Field; +                const fieldStr = Field.toString(field); +                if (field instanceof RichTextField || (typeof (field) === "string" && fieldStr.split(" ").length > 2)) rtfields++; +                return set.add(fieldStr); +            }, new Set<string>()));              let nonNumbers = 0;              let minVal = Number.MAX_VALUE, maxVal = -Number.MAX_VALUE; @@ -402,13 +426,18 @@ export class CollectionView extends Touchable<FieldViewProps & CollectionViewCus                  }              });              let newFacet: Opt<Doc>; -            if (nonNumbers / allCollectionDocs.length < .1) { -                newFacet = Docs.Create.SliderDocument({ title: facetHeader }); +            if (facetHeader === "text" || rtfields / allCollectionDocs.length > 0.1) { +                newFacet = Docs.Create.TextDocument("", { _width: 100, _height: 25, treeViewExpandedView: "layout", title: facetHeader, treeViewOpen: true, forceActive: true, ignoreClick: true }); +                Doc.GetProto(newFacet).type = DocumentType.COL; // forces item to show an open/close button instead ofa checkbox +                newFacet.target = this.props.Document; +                newFacet._textBoxPadding = 4; +                const scriptText = `setDocFilter(this.target, "${facetHeader}", text, "match")`; +                newFacet.onTextChanged = ScriptField.MakeScript(scriptText, { this: Doc.name, text: "string" }); +            } else if (nonNumbers / facetValues.length < .1) { +                newFacet = Docs.Create.SliderDocument({ title: facetHeader, treeViewExpandedView: "layout", treeViewOpen: true });                  const newFacetField = Doc.LayoutFieldKey(newFacet);                  const ranged = Doc.readDocRangeFilter(this.props.Document, facetHeader);                  Doc.GetProto(newFacet).type = DocumentType.COL; // forces item to show an open/close button instead ofa checkbox -                newFacet.treeViewExpandedView = "layout"; -                newFacet.treeViewOpen = true;                  const extendedMinVal = minVal - Math.min(1, Math.abs(maxVal - minVal) * .05);                  const extendedMaxVal = maxVal + Math.min(1, Math.abs(maxVal - minVal) * .05);                  newFacet[newFacetField + "-min"] = ranged === undefined ? extendedMinVal : ranged[0]; @@ -418,7 +447,6 @@ export class CollectionView extends Touchable<FieldViewProps & CollectionViewCus                  newFacet.target = this.props.Document;                  const scriptText = `setDocFilterRange(this.target, "${facetHeader}", range)`;                  newFacet.onThumbChanged = ScriptField.MakeScript(scriptText, { this: Doc.name, range: "number" }); -                  Doc.AddDocToList(facetCollection, this.props.fieldKey + "-filter", newFacet);              } else {                  newFacet = new Doc(); @@ -445,6 +473,7 @@ export class CollectionView extends Touchable<FieldViewProps & CollectionViewCus          return ScriptField.MakeScript(scriptText, { this: Doc.name, heading: "string", checked: "string", containingTreeView: Doc.name });      }      @computed get filterView() { +        TraceMobx();          const facetCollection = this.props.Document;          const flyout = (              <div className="collectionTimeView-flyout" style={{ width: `${this.facetWidth()}`, height: this.props.PanelHeight() - 30 }} onWheel={e => e.stopPropagation()}> @@ -534,7 +563,7 @@ export class CollectionView extends Touchable<FieldViewProps & CollectionViewCus                          Utils.CorsProxy(Cast(d.data, ImageField)!.url.href) : Cast(d.data, ImageField)!.url.href                      :                      ""))} -            {!this.props.isSelected() || this.props.PanelHeight() < 100 || this.props.Document.hideFilterView ? (null) : +            {(!this.props.isSelected() || this.props.Document.hideFilterView) && !this.props.Document.forceActive ? (null) :                  <div className="collectionView-filterDragger" title="library View Dragger" onPointerDown={this.onPointerDown}                      style={{ right: this.facetWidth() - 1, top: this.props.Document._viewType === CollectionViewType.Docking ? "25%" : "55%" }} />              } diff --git a/src/client/views/collections/CollectionViewChromes.scss b/src/client/views/collections/CollectionViewChromes.scss index 2885ac763..822e15aed 100644 --- a/src/client/views/collections/CollectionViewChromes.scss +++ b/src/client/views/collections/CollectionViewChromes.scss @@ -16,6 +16,7 @@          height: 32px;          border-bottom: .5px solid rgb(180, 180, 180);          overflow: visible; +        z-index: 9001;          .collectionViewBaseChrome {              display: flex; diff --git a/src/client/views/collections/CollectionViewChromes.tsx b/src/client/views/collections/CollectionViewChromes.tsx index 276bccede..4e91a2928 100644 --- a/src/client/views/collections/CollectionViewChromes.tsx +++ b/src/client/views/collections/CollectionViewChromes.tsx @@ -406,7 +406,7 @@ export class CollectionStackingViewChrome extends React.Component<CollectionView      @observable private _currentKey: string = "";      @observable private suggestions: string[] = []; -    @computed private get descending() { return BoolCast(this.props.CollectionView.props.Document.stackingHeadersSortDescending); } +    @computed private get descending() { return StrCast(this.props.CollectionView.props.Document._columnsSort) === "descending"; }      @computed get pivotField() { return StrCast(this.props.CollectionView.props.Document._pivotField); }      getKeySuggestions = async (value: string): Promise<string[]> => { @@ -450,7 +450,11 @@ export class CollectionStackingViewChrome extends React.Component<CollectionView          return true;      } -    @action toggleSort = () => { this.props.CollectionView.props.Document.stackingHeadersSortDescending = !this.props.CollectionView.props.Document.stackingHeadersSortDescending; }; +    @action toggleSort = () => { +        this.props.CollectionView.props.Document._columnsSort = +            this.props.CollectionView.props.Document._columnsSort === "descending" ? "ascending" : +                this.props.CollectionView.props.Document._columnsSort === "ascending" ? undefined : "descending"; +    }      @action resetValue = () => { this._currentKey = this.pivotField; };      render() { diff --git a/src/client/views/collections/SchemaTable.tsx b/src/client/views/collections/SchemaTable.tsx new file mode 100644 index 000000000..695965cb4 --- /dev/null +++ b/src/client/views/collections/SchemaTable.tsx @@ -0,0 +1,615 @@ +import React = require("react"); +import { IconProp } from '@fortawesome/fontawesome-svg-core'; +import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'; +import { action, computed, observable } from "mobx"; +import { observer } from "mobx-react"; +import ReactTable, { CellInfo, Column, ComponentPropsGetterR, Resize, SortingRule } from "react-table"; +import "react-table/react-table.css"; +import { Doc, DocListCast, Field, Opt } from "../../../fields/Doc"; +import { Id } from "../../../fields/FieldSymbols"; +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 { 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, CollectionSchemaImageCell, CollectionSchemaListCell, CollectionSchemaDateCell } from "./CollectionSchemaCells"; +import { CollectionSchemaAddColumnHeader, KeysDropdown } from "./CollectionSchemaHeaders"; +import { MovableColumn, MovableRow } from "./CollectionSchemaMovableTableHOC"; +import "./CollectionSchemaView.scss"; +import { CollectionView } from "./CollectionView"; +import { ContentFittingDocumentView } from "../nodes/ContentFittingDocumentView"; +import { emptyFunction, returnZero, returnOne, returnFalse, returnEmptyFilter, emptyPath } from "../../../Utils"; +import { TouchScrollableMenuItem } from "../TouchScrollableMenu"; + + +enum ColumnType { +    Any, +    Number, +    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([ +    ["title", ColumnType.String], +    ["x", ColumnType.Number], ["y", ColumnType.Number], ["_width", ColumnType.Number], ["_height", ColumnType.Number], +    ["_nativeWidth", ColumnType.Number], ["_nativeHeight", ColumnType.Number], ["isPrototype", ColumnType.Boolean], +    ["page", ColumnType.Number], ["curPage", ColumnType.Number], ["currentTimecode", ColumnType.Number], ["zIndex", ColumnType.Number] +]); + +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; +    columns: SchemaHeaderField[]; +    documentKeys: any[]; +    headerIsEditing: boolean; +    openHeader: (column: any, screenx: number, screeny: number) => void; +    onPointerDown: (e: React.PointerEvent) => void; +    onResizedChange: (newResized: Resize[], event: any) => void; +    setColumns: (columns: SchemaHeaderField[]) => void; +    reorderColumns: (toMove: SchemaHeaderField, relativeTo: SchemaHeaderField, before: boolean, columnsValues: SchemaHeaderField[]) => void; +    changeColumns: (oldKey: string, newKey: string, addNew: boolean) => void; +    setHeaderIsEditing: (isEditing: boolean) => void; +    changeColumnSort: (columnField: SchemaHeaderField, descending: boolean | undefined) => void; +} + +@observer +export class SchemaTable extends React.Component<SchemaTableProps> { +    private DIVIDER_WIDTH = 4; + +    @observable _cellIsEditing: boolean = false; +    @observable _focusedCell: { row: number, col: number } = { row: 0, col: 0 }; +    @observable _openCollections: Array<string> = []; + +    @observable _showDoc: Doc | undefined; +    @observable _showDataDoc: any = ""; +    @observable _showDocPos: number[] = []; + +    @observable _showTitleDropdown: boolean = false; + +    @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 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.props.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.props.columns.reduce((sorted, shf) => { +            shf.desc && sorted.push({ id: shf.heading, desc: shf.desc }); +            return sorted; +        }, [] as SortingRule[]); +    } + +    @action +    changeSorting = (col: any) => { +        console.log(col.heading); +        if (col.desc === undefined) { +            // no sorting +            this.props.changeColumnSort(col, true); +        } else if (col.desc === true) { +            // descending sort +            this.props.changeColumnSort(col, false); +        } else if (col.desc === false) { +            // ascending sort +            this.props.changeColumnSort(col, undefined); +        } +    } + +    @action +    changeTitleMode = () => { console.log("header clicked"); this._showTitleDropdown = !this._showTitleDropdown; } + +    @computed get borderWidth() { return Number(COLLECTION_BORDER_WIDTH); } +    @computed get tableColumns(): Column<Doc>[] { + +        const possibleKeys = this.props.documentKeys.filter(key => this.props.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.props.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.props.columns.map(col => { + +            const keysDropdown = <KeysDropdown +                keyValue={col.heading} +                possibleKeys={possibleKeys} +                existingKeys={this.props.columns.map(c => c.heading)} +                canAddNew={true} +                addNew={false} +                onSelect={this.props.changeColumns} +                setIsEditing={this.props.setHeaderIsEditing} + +                // try commenting this out +                width={"100%"} +            />; + +            const icon: IconProp = this.getColumnType(col) === ColumnType.Number ? "hashtag" : this.getColumnType(col) === ColumnType.String ? "font" : +                this.getColumnType(col) === ColumnType.Boolean ? "check-square" : this.getColumnType(col) === ColumnType.Doc ? "file" : +                    this.getColumnType(col) === ColumnType.Image ? "image" : this.getColumnType(col) === ColumnType.List ? "list-ul" : +                        this.getColumnType(col) === ColumnType.Date ? "calendar" : "align-justify"; + +            const headerText = this._showTitleDropdown ? keysDropdown : <div +                onClick={this.changeTitleMode} +                style={{ +                    background: col.color, padding: "2px", +                    letterSpacing: "2px", +                    textTransform: "uppercase", +                    display: "flex" +                }}> +                {col.heading}</div>; + +            const sortIcon = col.desc === undefined ? "circle" : col.desc === true ? "caret-down" : "caret-up"; + +            const header = +                <div //className="collectionSchemaView-header" +                    //onClick={e => this.props.openHeader(col, menuContent, e.clientX, e.clientY)} +                    className="collectionSchemaView-menuOptions-wrapper" +                    style={{ +                        background: col.color, padding: "2px", +                        display: "flex" +                    }}> +                    <FontAwesomeIcon icon={icon} size="lg" style={{ display: "inline", paddingLeft: "7px" }} /> +                    {/* <div className="keys-dropdown" +                        style={{ display: "inline", zIndex: 1000 }}> */} +                    {keysDropdown} +                    {/* </div> */} +                    <div onClick={e => this.changeSorting(col)} +                        style={{ paddingRight: "6px", display: "inline", zIndex: 1, background: "inherit" }}> +                        <FontAwesomeIcon icon={sortIcon} size="sm" /> +                    </div> +                    <div onClick={e => this.props.openHeader(col, e.clientX, e.clientY)} +                        style={{ float: "right", paddingRight: "6px", zIndex: 1, background: "inherit" }}> +                        <FontAwesomeIcon icon={"compass"} size="sm" /> +                    </div> +                </div>; + +            return { +                Header: <MovableColumn columnRenderer={header} columnValue={col} allColumns={this.props.columns} reorderColumns={this.props.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.props.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, +                        showDoc: this.showDoc, +                    }; + +                    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} />; +                    if (colType === ColumnType.Image) return <CollectionSchemaImageCell {...props} />; +                    if (colType === ColumnType.List) return <CollectionSchemaListCell {...props} />; +                    if (colType === ColumnType.Date) return <CollectionSchemaDateCell {...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 oldSchemaHeaders = Cast(this.props.Document._schemaHeaders, listSpec("string"), []); +        if (oldSchemaHeaders?.length && typeof oldSchemaHeaders[0] !== "object") { +            const newSchemaHeaders = oldSchemaHeaders.map(i => typeof i === "string" ? new SchemaHeaderField(i, "#f1efeb") : i); +            this.props.Document._schemaHeaders = new List<SchemaHeaderField>(newSchemaHeaders); +        } else if (this.props.Document._schemaHeaders === undefined) { +            this.props.Document._schemaHeaders = new List<SchemaHeaderField>([new SchemaHeaderField("title", "#f1efeb")]); +        } +    } + +    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.props.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), +            addDocTab: this.props.addDocTab +        }; +    } + +    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.props.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 +    onKeyDown = (e: KeyboardEvent): void => { +        if (!this._cellIsEditing && !this.props.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.props.columns.length ? 0 : curCol + 1 }; +            case "right": return { row: curRow, col: curCol + 1 === this.props.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.props.columns.findIndex(col => col.heading.toUpperCase() === "New field".toUpperCase()) > -1; +        while (found) { +            index++; +            found = this.props.columns.findIndex(col => col.heading.toUpperCase() === ("New field (" + index + ")").toUpperCase()) > -1; +        } +        this.props.columns.push(new SchemaHeaderField(`New field ${index ? "(" + index + ")" : ""}`, "#f1efeb")); +    } + +    @action +    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 +    @action +    toggleTextwrap = async () => { +        const textwrappedRows = Cast(this.props.Document.textwrappedSchemaRows, listSpec("string"), []); +        if (textwrappedRows.length) { +            this.props.Document.textwrappedSchemaRows = new List<string>([]); +        } 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} +            NoDataComponent={() => null} +            onResizedChange={this.props.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>} + +        />; +    } + +    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.props.columns; +        const doc = docs[row]; +        if (col === undefined) { +            return doc; +        } +        if (col >= 0 && col < columns.length) { +            const column = this.props.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)._schemaHeaders[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; +        } +        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.props.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> +            {!this._showDoc ? (null) : +                <div className="collectionSchemaView-documentPreview" //onClick={() => { this.onOpenClick(); }} +                    style={{ +                        position: "absolute", width: 150, height: 150, +                        background: "dimGray", display: "block", top: 0, left: 0, +                        transform: `translate(${this._showDocPos[0]}px, ${this._showDocPos[1] - 180}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>} +        </div>; +    } +}
\ No newline at end of file diff --git a/src/client/views/collections/collectionFreeForm/CollectionFreeFormView.tsx b/src/client/views/collections/collectionFreeForm/CollectionFreeFormView.tsx index 5135c4ae4..546a4307c 100644 --- a/src/client/views/collections/collectionFreeForm/CollectionFreeFormView.tsx +++ b/src/client/views/collections/collectionFreeForm/CollectionFreeFormView.tsx @@ -194,7 +194,6 @@ export class CollectionFreeFormView extends CollectionSubView<PanZoomDocument, P          return (pt => super.onExternalDrop(e, { x: pt[0], y: pt[1] }))(this.getTransform().transformPoint(e.pageX, e.pageY));      } -    @undoBatch      @action      internalDocDrop(e: Event, de: DragManager.DropEvent, docDragData: DragManager.DocumentDragData, xp: number, yp: number) {          if (!super.onInternalDrop(e, de)) return false; @@ -1226,7 +1225,7 @@ export class CollectionFreeFormView extends CollectionSubView<PanZoomDocument, P          optionItems.push({ description: "Arrange contents in grid", event: this.layoutDocsInGrid, icon: "table" });          if (!Doc.UserDoc().noviceMode) {              optionItems.push({ description: (!this.layoutDoc._nativeWidth || !this.layoutDoc._nativeHeight ? "Freeze" : "Unfreeze") + " Aspect", event: this.toggleNativeDimensions, icon: "snowflake" }); -            optionItems.push({ description: `${this.Document._LODdisable ? "Enable LOD" : "Disable LOD"}`, event: () => this.Document._LODdisable = !this.Document._LODdisable, icon: "table" }); +            optionItems.push({ description: `${this.Document._freeformLOD ? "Enable LOD" : "Disable LOD"}`, event: () => this.Document._freeformLOD = !this.Document._freeformLOD, icon: "table" });              optionItems.push({                  description: "Import document", icon: "upload", event: ({ x, y }) => {                      const input = document.createElement("input"); @@ -1390,7 +1389,7 @@ export class CollectionFreeFormView extends CollectionSubView<PanZoomDocument, P                  width: this.contentScaling ? `${100 / this.contentScaling}%` : "",                  height: this.contentScaling ? `${100 / this.contentScaling}%` : this.isAnnotationOverlay ? (this.props.Document.scrollHeight ? this.Document.scrollHeight : "100%") : this.props.PanelHeight()              }}> -            {!this.Document._LODdisable && !this.props.active() && !this.props.isAnnotationOverlay && !this.props.annotationsKey && this.props.renderDepth > 0 ? +            {this.Document._freeformLOD && !this.props.active() && !this.props.isAnnotationOverlay && !this.props.annotationsKey && this.props.renderDepth > 0 ?                  this.placeholder : this.marqueeView}              <CollectionFreeFormOverlayView elements={this.elementFunc} /> diff --git a/src/client/views/collections/collectionFreeForm/MarqueeView.tsx b/src/client/views/collections/collectionFreeForm/MarqueeView.tsx index 099859109..97ed74c10 100644 --- a/src/client/views/collections/collectionFreeForm/MarqueeView.tsx +++ b/src/client/views/collections/collectionFreeForm/MarqueeView.tsx @@ -352,7 +352,6 @@ export class MarqueeView extends React.Component<SubCollectionViewProps & Marque              backgroundColor: this.props.isAnnotationOverlay ? "#00000015" : isBackground ? "cyan" : undefined,              _width: bounds.width,              _height: bounds.height, -            _LODdisable: true,              title: "a nested collection",          });          selected.forEach(d => d.context = newCollection); diff --git a/src/client/views/linking/LinkMenu.tsx b/src/client/views/linking/LinkMenu.tsx index 0fcc0f0b9..de1d60a09 100644 --- a/src/client/views/linking/LinkMenu.tsx +++ b/src/client/views/linking/LinkMenu.tsx @@ -28,7 +28,7 @@ export class LinkMenu extends React.Component<Props> {      @action      onClick = (e: PointerEvent) => { -        if (!Array.from(this._linkMenuRef?.getElementsByTagName((e.target as HTMLElement).tagName) || []).includes(e.target as any)) { +        if (this._linkMenuRef?.contains(e.target as any)) {              DocumentLinksButton.EditLink = undefined;          }      } diff --git a/src/client/views/nodes/DocumentView.tsx b/src/client/views/nodes/DocumentView.tsx index 3a3bef2e0..09eeaee36 100644 --- a/src/client/views/nodes/DocumentView.tsx +++ b/src/client/views/nodes/DocumentView.tsx @@ -689,7 +689,7 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu      @undoBatch      @action -    setAcl = (acl: "readOnly" | "addOnly" | "ownerOnly") => { +    setAcl = (acl: "readOnly" | "addOnly" | "ownerOnly" | "write") => {          this.dataDoc.ACL = this.props.Document.ACL = acl;          DocListCast(this.dataDoc[Doc.LayoutFieldKey(this.dataDoc)]).map(d => {              if (d.author === Doc.CurrentUserEmail) d.ACL = acl; @@ -699,7 +699,7 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu      }      @undoBatch      @action -    testAcl = (acl: "readOnly" | "addOnly" | "ownerOnly") => { +    testAcl = (acl: "readOnly" | "addOnly" | "ownerOnly" | "write") => {          this.dataDoc.author = this.props.Document.author = "ADMIN";          this.dataDoc.ACL = this.props.Document.ACL = acl;          DocListCast(this.dataDoc[Doc.LayoutFieldKey(this.dataDoc)]).map(d => { @@ -811,6 +811,7 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu          aclItems.push({ description: "Make Add Only", event: () => this.setAcl("addOnly"), icon: "concierge-bell" });          aclItems.push({ description: "Make Read Only", event: () => this.setAcl("readOnly"), icon: "concierge-bell" });          aclItems.push({ description: "Make Private", event: () => this.setAcl("ownerOnly"), icon: "concierge-bell" }); +        aclItems.push({ description: "Make Editable", event: () => this.setAcl("write"), icon: "concierge-bell" });          aclItems.push({ description: "Test Private", event: () => this.testAcl("ownerOnly"), icon: "concierge-bell" });          aclItems.push({ description: "Test Readonly", event: () => this.testAcl("readOnly"), icon: "concierge-bell" });          !existingAcls && cm.addItem({ description: "Privacy...", subitems: aclItems, icon: "question" }); @@ -1168,8 +1169,9 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu      }      render() { -        if (this.props.Document[AclSym] && this.props.Document[AclSym] === AclPrivate) return (null);          if (!(this.props.Document instanceof Doc)) return (null); +        if (this.props.Document[AclSym] && this.props.Document[AclSym] === AclPrivate) return (null); +        if (this.props.Document.hidden) return (null);          const backgroundColor = Doc.UserDoc().renderStyle === "comic" ? undefined : this.props.forcedBackgroundColor?.(this.Document) || StrCast(this.layoutDoc._backgroundColor) || StrCast(this.layoutDoc.backgroundColor) || StrCast(this.Document.backgroundColor) || this.props.backgroundColor?.(this.Document);          const opacity = Cast(this.layoutDoc._opacity, "number", Cast(this.layoutDoc.opacity, "number", Cast(this.Document.opacity, "number", null)));          const finalOpacity = this.props.opacity ? this.props.opacity() : opacity; diff --git a/src/client/views/nodes/FontIconBox.tsx b/src/client/views/nodes/FontIconBox.tsx index cf0b16c7c..5e8dd2497 100644 --- a/src/client/views/nodes/FontIconBox.tsx +++ b/src/client/views/nodes/FontIconBox.tsx @@ -5,7 +5,7 @@ import { createSchema, makeInterface } from '../../../fields/Schema';  import { DocComponent } from '../DocComponent';  import './FontIconBox.scss';  import { FieldView, FieldViewProps } from './FieldView'; -import { StrCast, Cast } from '../../../fields/Types'; +import { StrCast, Cast, NumCast } from '../../../fields/Types';  import { Utils } from "../../../Utils";  import { runInAction, observable, reaction, IReactionDisposer } from 'mobx';  import { Doc } from '../../../fields/Doc'; @@ -59,13 +59,14 @@ export class FontIconBox extends DocComponent<FieldViewProps, FontIconDocument>(      render() {          const referenceDoc = (this.layoutDoc.dragFactory instanceof Doc ? this.layoutDoc.dragFactory : this.layoutDoc); -        const referenceLayout = Doc.Layout(referenceDoc); +        const refLayout = Doc.Layout(referenceDoc);          return <button className="fontIconBox-outerDiv" title={StrCast(this.layoutDoc.title)} ref={this._ref} onContextMenu={this.specificContextMenu}              style={{ -                background: StrCast(referenceLayout.backgroundColor), +                padding: Cast(this.layoutDoc._xPadding, "number", null), +                background: StrCast(refLayout._backgroundColor, StrCast(refLayout.backgroundColor)),                  boxShadow: this.layoutDoc.ischecked ? `4px 4px 12px black` : undefined              }}> -            <FontAwesomeIcon className="fontIconBox-icon" icon={this.dataDoc.icon as any} color={this._foregroundColor} size="sm" /> +            <FontAwesomeIcon className="fontIconBox-icon" icon={this.dataDoc.icon as any} color={StrCast(this.layoutDoc.color, this._foregroundColor)} size="sm" />              {!this.rootDoc.label ? (null) : <div className="fontIconBox-label"> {StrCast(this.rootDoc.label).substring(0, 5)} </div>}          </button>;      } diff --git a/src/client/views/nodes/KeyValueBox.tsx b/src/client/views/nodes/KeyValueBox.tsx index d375466c9..b732f5f83 100644 --- a/src/client/views/nodes/KeyValueBox.tsx +++ b/src/client/views/nodes/KeyValueBox.tsx @@ -169,8 +169,8 @@ export class KeyValueBox extends React.Component<FieldViewProps> {      getTemplate = async () => {          const parent = Docs.Create.StackingDocument([], { _width: 800, _height: 800, title: "Template" }); -        parent.singleColumn = false; -        parent.columnWidth = 100; +        parent._columnsStack = false; +        parent._columnWidth = 100;          for (const row of this.rows.filter(row => row.isChecked)) {              await this.createTemplateField(parent, row);              row.uncheck(); diff --git a/src/client/views/nodes/PDFBox.tsx b/src/client/views/nodes/PDFBox.tsx index 6b1c9fcde..eb2a85eeb 100644 --- a/src/client/views/nodes/PDFBox.tsx +++ b/src/client/views/nodes/PDFBox.tsx @@ -55,25 +55,28 @@ export class PDFBox extends ViewBoxAnnotatableComponent<FieldViewProps, PdfDocum          const backup = "oldPath";          const { Document } = this.props; -        const { url: { href } } = Cast(this.dataDoc[this.props.fieldKey], PdfField)!; -        const pathCorrectionTest = /upload\_[a-z0-9]{32}.(.*)/g; -        const matches = pathCorrectionTest.exec(href); -        console.log("\nHere's the { url } being fed into the outer regex:"); -        console.log(href); -        console.log("And here's the 'properPath' build from the captured filename:\n"); -        if (matches !== null && href.startsWith(window.location.origin)) { -            const properPath = Utils.prepend(`/files/pdfs/${matches[0]}`); -            console.log(properPath); -            if (!properPath.includes(href)) { -                console.log(`The two (url and proper path) were not equal`); -                const proto = Doc.GetProto(Document); -                proto[this.props.fieldKey] = new PdfField(properPath); -                proto[backup] = href; +        const pdf = Cast(this.dataDoc[this.props.fieldKey], PdfField); +        const href = pdf?.url?.href; +        if (href) { +            const pathCorrectionTest = /upload\_[a-z0-9]{32}.(.*)/g; +            const matches = pathCorrectionTest.exec(href); +            console.log("\nHere's the { url } being fed into the outer regex:"); +            console.log(href); +            console.log("And here's the 'properPath' build from the captured filename:\n"); +            if (matches !== null && href.startsWith(window.location.origin)) { +                const properPath = Utils.prepend(`/files/pdfs/${matches[0]}`); +                console.log(properPath); +                if (!properPath.includes(href)) { +                    console.log(`The two (url and proper path) were not equal`); +                    const proto = Doc.GetProto(Document); +                    proto[this.props.fieldKey] = new PdfField(properPath); +                    proto[backup] = href; +                } else { +                    console.log(`The two (url and proper path) were equal`); +                }              } else { -                console.log(`The two (url and proper path) were equal`); +                console.log("Outer matches was null!");              } -        } else { -            console.log("Outer matches was null!");          }      } diff --git a/src/client/views/nodes/VideoBox.tsx b/src/client/views/nodes/VideoBox.tsx index 71556bfd3..a5c6c4a48 100644 --- a/src/client/views/nodes/VideoBox.tsx +++ b/src/client/views/nodes/VideoBox.tsx @@ -19,6 +19,7 @@ import { FieldView, FieldViewProps } from './FieldView';  import "./VideoBox.scss";  import { documentSchema } from "../../../fields/documentSchemas";  import { Networking } from "../../Network"; +import { SnappingManager } from "../../util/SnappingManager";  const path = require('path');  export const timeSchema = createSchema({ @@ -58,21 +59,21 @@ export class VideoBox extends ViewBoxAnnotatableComponent<FieldViewProps, VideoD      @action public Play = (update: boolean = true) => {          this._playing = true; -        update && this.player && this.player.play(); -        update && this._youtubePlayer && this._youtubePlayer.playVideo(); +        update && this.player?.play(); +        update && this._youtubePlayer?.playVideo();          this._youtubePlayer && !this._playTimer && (this._playTimer = setInterval(this.updateTimecode, 5));          this.updateTimecode();      }      @action public Seek(time: number) { -        this._youtubePlayer && this._youtubePlayer.seekTo(Math.round(time), true); +        this._youtubePlayer?.seekTo(Math.round(time), true);          this.player && (this.player.currentTime = time);      }      @action public Pause = (update: boolean = true) => {          this._playing = false; -        update && this.player && this.player.pause(); -        update && this._youtubePlayer && this._youtubePlayer.pauseVideo && this._youtubePlayer.pauseVideo(); +        update && this.player?.pause(); +        update && this._youtubePlayer?.pauseVideo();          this._youtubePlayer && this._playTimer && clearInterval(this._playTimer);          this._playTimer = undefined;          this.updateTimecode(); @@ -261,21 +262,20 @@ export class VideoBox extends ViewBoxAnnotatableComponent<FieldViewProps, VideoD          const onYoutubePlayerStateChange = (event: any) => runInAction(() => {              if (started && event.data === YT.PlayerState.PLAYING) {                  started = false; -                this._youtubePlayer && this._youtubePlayer.unMute(); -                this.Pause(); +                this._youtubePlayer?.unMute(); +                //this.Pause();                  return;              }              if (event.data === YT.PlayerState.PLAYING && !this._playing) this.Play(false);              if (event.data === YT.PlayerState.PAUSED && this._playing) this.Pause(false);          });          const onYoutubePlayerReady = (event: any) => { -            this._reactionDisposer && this._reactionDisposer(); -            this._youtubeReactionDisposer && this._youtubeReactionDisposer(); +            this._reactionDisposer?.(); +            this._youtubeReactionDisposer?.();              this._reactionDisposer = reaction(() => this.layoutDoc.currentTimecode, () => !this._playing && this.Seek((this.layoutDoc.currentTimecode || 0))); -            this._youtubeReactionDisposer = reaction(() => [this.props.isSelected(), DocumentDecorations.Instance.Interacting, Doc.GetSelectedTool()], () => { -                const interactive = Doc.GetSelectedTool() === InkTool.None && this.props.isSelected(true) && !DocumentDecorations.Instance.Interacting; -                iframe.style.pointerEvents = interactive ? "all" : "none"; -            }, { fireImmediately: true }); +            this._youtubeReactionDisposer = reaction( +                () => Doc.GetSelectedTool() === InkTool.None && this.props.isSelected(true) && !SnappingManager.GetIsDragging() && !DocumentDecorations.Instance.Interacting, +                (interactive) => iframe.style.pointerEvents = interactive ? "all" : "none", { fireImmediately: true });          };          this._youtubePlayer = new YT.Player(`${this.youtubeVideoId + this._youtubeIframeId}-player`, {              events: { @@ -346,7 +346,7 @@ export class VideoBox extends ViewBoxAnnotatableComponent<FieldViewProps, VideoD          const start = untracked(() => Math.round((this.layoutDoc.currentTimecode || 0)));          return <iframe key={this._youtubeIframeId} id={`${this.youtubeVideoId + this._youtubeIframeId}-player`}              onLoad={this.youtubeIframeLoaded} className={`${style}`} width={(this.layoutDoc._nativeWidth || 640)} height={(this.layoutDoc._nativeHeight || 390)} -            src={`https://www.youtube.com/embed/${this.youtubeVideoId}?enablejsapi=1&rel=0&showinfo=1&autoplay=1&mute=1&start=${start}&modestbranding=1&controls=${VideoBox._showControls ? 1 : 0}`} />; +            src={`https://www.youtube.com/embed/${this.youtubeVideoId}?enablejsapi=1&rel=0&showinfo=1&autoplay=0&mute=1&start=${start}&modestbranding=1&controls=${VideoBox._showControls ? 1 : 0}`} />;      }      @action.bound diff --git a/src/client/views/nodes/formattedText/DashFieldView.tsx b/src/client/views/nodes/formattedText/DashFieldView.tsx index 8c16f4a1a..8718bf329 100644 --- a/src/client/views/nodes/formattedText/DashFieldView.tsx +++ b/src/client/views/nodes/formattedText/DashFieldView.tsx @@ -184,9 +184,9 @@ export class DashFieldViewInternal extends React.Component<IDashFieldViewInterna          if (container) {              const alias = Doc.MakeAlias(container.props.Document);              alias.viewType = CollectionViewType.Time; -            let list = Cast(alias.schemaColumns, listSpec(SchemaHeaderField)); +            let list = Cast(alias._columnHeaders, listSpec(SchemaHeaderField));              if (!list) { -                alias.schemaColumns = list = new List<SchemaHeaderField>(); +                alias._columnHeaders = list = new List<SchemaHeaderField>();              }              list.map(c => c.heading).indexOf(this._fieldKey) === -1 && list.push(new SchemaHeaderField(this._fieldKey, "#f1efeb"));              list.map(c => c.heading).indexOf("text") === -1 && list.push(new SchemaHeaderField("text", "#f1efeb")); diff --git a/src/client/views/nodes/formattedText/FormattedTextBox.tsx b/src/client/views/nodes/formattedText/FormattedTextBox.tsx index d26954dbc..11f25a208 100644 --- a/src/client/views/nodes/formattedText/FormattedTextBox.tsx +++ b/src/client/views/nodes/formattedText/FormattedTextBox.tsx @@ -8,7 +8,7 @@ import { baseKeymap, selectAll } from "prosemirror-commands";  import { history } from "prosemirror-history";  import { inputRules } from 'prosemirror-inputrules';  import { keymap } from "prosemirror-keymap"; -import { Fragment, Mark, Node, Slice } from "prosemirror-model"; +import { Fragment, Mark, Node, Slice, Schema } from "prosemirror-model";  import { EditorState, NodeSelection, Plugin, TextSelection, Transaction } from "prosemirror-state";  import { ReplaceStep } from 'prosemirror-transform';  import { EditorView } from "prosemirror-view"; @@ -16,13 +16,14 @@ import { DateField } from '../../../../fields/DateField';  import { DataSym, Doc, DocListCast, DocListCastAsync, Field, HeightSym, Opt, WidthSym, AclSym } from "../../../../fields/Doc";  import { documentSchema } from '../../../../fields/documentSchemas';  import applyDevTools = require("prosemirror-dev-tools"); +import { removeMarkWithAttrs } from "./prosemirrorPatches";  import { Id } from '../../../../fields/FieldSymbols';  import { InkTool } from '../../../../fields/InkField';  import { PrefetchProxy } from '../../../../fields/Proxy';  import { RichTextField } from "../../../../fields/RichTextField";  import { RichTextUtils } from '../../../../fields/RichTextUtils';  import { createSchema, makeInterface } from "../../../../fields/Schema"; -import { Cast, DateCast, NumCast, StrCast } from "../../../../fields/Types"; +import { Cast, DateCast, NumCast, StrCast, ScriptCast } from "../../../../fields/Types";  import { TraceMobx, OVERRIDE_ACL } from '../../../../fields/util';  import { addStyleSheet, addStyleSheetRule, clearStyleSheetRules, emptyFunction, numberRange, returnOne, returnZero, Utils, setupMoveUpEvents } from '../../../../Utils';  import { GoogleApiClientUtils, Pulls, Pushes } from '../../../apis/google_docs/GoogleApiClientUtils'; @@ -32,7 +33,7 @@ import { DocumentType } from '../../../documents/DocumentTypes';  import { DictationManager } from '../../../util/DictationManager';  import { DragManager } from "../../../util/DragManager";  import { makeTemplate } from '../../../util/DropConverter'; -import buildKeymap from "./ProsemirrorExampleTransfer"; +import buildKeymap, { updateBullets } from "./ProsemirrorExampleTransfer";  import RichTextMenu from './RichTextMenu';  import { RichTextRules } from "./RichTextRules"; @@ -56,7 +57,7 @@ import { DocumentButtonBar } from '../../DocumentButtonBar';  import { AudioBox } from '../AudioBox';  import { FieldView, FieldViewProps } from "../FieldView";  import "./FormattedTextBox.scss"; -import { FormattedTextBoxComment, formattedTextBoxCommentPlugin } from './FormattedTextBoxComment'; +import { FormattedTextBoxComment, formattedTextBoxCommentPlugin, findLinkMark } from './FormattedTextBoxComment';  import React = require("react");  library.add(faEdit); @@ -68,15 +69,10 @@ export interface FormattedTextBoxProps {      xMargin?: number;   // used to override document's settings for xMargin --- see CollectionCarouselView      yMargin?: number;  } - -const richTextSchema = createSchema({ -    documentText: "string", -}); -  export const GoogleRef = "googleDocId"; -type RichTextDocument = makeInterface<[typeof richTextSchema, typeof documentSchema]>; -const RichTextDocument = makeInterface(richTextSchema, documentSchema); +type RichTextDocument = makeInterface<[typeof documentSchema]>; +const RichTextDocument = makeInterface(documentSchema);  type PullHandler = (exportState: Opt<GoogleApiClientUtils.Docs.ImportResult>, dataDoc: Doc) => void; @@ -86,14 +82,16 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp      public static blankState = () => EditorState.create(FormattedTextBox.Instance.config);      public static Instance: FormattedTextBox;      public ProseRef?: HTMLDivElement; +    public get EditorView() { return this._editorView; }      private _ref: React.RefObject<HTMLDivElement> = React.createRef();      private _scrollRef: React.RefObject<HTMLDivElement> = React.createRef();      private _editorView: Opt<EditorView>;      private _applyingChange: boolean = false;      private _searchIndex = 0; +    private _cachedLinks: Doc[] = [];      private _undoTyping?: UndoManager.Batch;      private _disposers: { [name: string]: IReactionDisposer } = {}; -    private dropDisposer?: DragManager.DragDropDisposer; +    private _dropDisposer?: DragManager.DragDropDisposer;      @computed get _recording() { return this.dataDoc.audioState === "recording"; }      set _recording(value) { this.dataDoc.audioState = value ? "recording" : undefined; } @@ -145,6 +143,33 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp      public get CurrentDiv(): HTMLDivElement { return this._ref.current!; } +    // removes all hyperlink anchors for the removed linkDoc +    // TODO: bcz: Argh... if a section of text has multiple anchors, this should just remove the intended one.  +    // but since removing one anchor from the list of attr anchors isn't implemented, this will end up removing nothing. +    public RemoveLinkFromDoc(linkDoc?: Doc) { +        const state = this._editorView?.state; +        if (state && linkDoc && this._editorView) { +            var allLinks: any[] = []; +            state.doc.nodesBetween(0, state.doc.nodeSize - 2, (node: any, pos: number, parent: any) => { +                const foundMark = findLinkMark(node.marks); +                const newHrefs = foundMark?.attrs.allLinks.filter((a: any) => a.href.includes(linkDoc[Id])) || []; +                allLinks = newHrefs.length ? newHrefs : allLinks; +                return true; +            }); +            if (allLinks.length) { +                this._editorView.dispatch(removeMarkWithAttrs(state.tr, 0, state.doc.nodeSize - 2, state.schema.marks.linkAnchor, { allLinks })); +            } +        } +    } +    // removes all the specified link referneces from the selection.  +    // NOTE: as above, this won't work correctly if there are marks with overlapping but not exact sets of link references. +    public RemoveLinkFromSelection(allLinks: { href: string, title: string, linkId: string, targetId: string }[]) { +        const state = this._editorView?.state; +        if (state && this._editorView) { +            this._editorView.dispatch(removeMarkWithAttrs(state.tr, state.selection.from, state.selection.to, state.schema.marks.link, { allLinks })); +        } +    } +      linkOnDeselect: Map<string, string> = new Map();      doLinkOnDeselect() { @@ -181,8 +206,8 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp                      this.linkOnDeselect.set(key, value);                      const id = Utils.GenerateDeterministicGuid(this.dataDoc[Id] + key); -                    const allHrefs = [{ href: Utils.prepend("/doc/" + id), title: value, targetId: id }]; -                    const link = this._editorView.state.schema.marks.link.create({ allHrefs, location: "onRight", title: value }); +                    const allLinks = [{ href: Utils.prepend("/doc/" + id), title: value, targetId: id }]; +                    const link = this._editorView.state.schema.marks.linkAnchor.create({ allLinks, location: "onRight", title: value });                      const mval = this._editorView.state.schema.marks.metadataVal.create();                      const offset = (tx.selection.to === range!.end - 1 ? -1 : 0);                      tx = tx.addMark(textEndSelection - value.length + offset, textEndSelection, link).addMark(textEndSelection - value.length + offset, textEndSelection, mval); @@ -203,13 +228,14 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp              if (!this.dataDoc[AclSym]) {                  if (!this._applyingChange && json.replace(/"selection":.*/, "") !== curProto?.Data.replace(/"selection":.*/, "")) {                      this._applyingChange = true; -                    this.dataDoc[this.props.fieldKey + "-lastModified"] = new DateField(new Date(Date.now())); +                    (curText !== Cast(this.dataDoc[this.fieldKey], RichTextField)?.Text) && (this.dataDoc[this.props.fieldKey + "-lastModified"] = new DateField(new Date(Date.now())));                      if ((!curTemp && !curProto) || curText || curLayout?.Data.includes("dash")) { // if no template, or there's text that didn't come from the layout template, write it to the document. (if this is driven by a template, then this overwrites the template text which is intended)                          if (json !== curLayout?.Data) {                              !curText && tx.storedMarks?.map(m => m.type.name === "pFontSize" && (Doc.UserDoc().fontSize = this.layoutDoc._fontSize = m.attrs.fontSize));                              !curText && tx.storedMarks?.map(m => m.type.name === "pFontFamily" && (Doc.UserDoc().fontFamily = this.layoutDoc._fontFamily = m.attrs.fontFamily));                              this.dataDoc[this.props.fieldKey] = new RichTextField(json, curText);                              this.dataDoc[this.props.fieldKey + "-noTemplate"] = (curTemp?.Text || "") !== curText; // mark the data field as being split from the template if it has been edited +                            ScriptCast(this.layoutDoc.onTextChanged, null)?.script.run({ this: this.layoutDoc, self: this.rootDoc, text: curText });                          }                      } else { // if we've deleted all the text in a note driven by a template, then restore the template data                          this.dataDoc[this.props.fieldKey] = undefined; @@ -249,8 +275,8 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp              const lastSel = Math.min(flattened.length - 1, this._searchIndex);              this._searchIndex = ++this._searchIndex > flattened.length - 1 ? 0 : this._searchIndex;              const alink = DocUtils.MakeLink({ doc: this.rootDoc }, { doc: target }, "automatic")!; -            const allHrefs = [{ href: Utils.prepend("/doc/" + alink[Id]), title: "a link", targetId: target[Id], linkId: alink[Id] }]; -            const link = this._editorView.state.schema.marks.link.create({ allHrefs, title: "a link", location }); +            const allLinks = [{ href: Utils.prepend("/doc/" + alink[Id]), title: "a link", targetId: target[Id], linkId: alink[Id] }]; +            const link = this._editorView.state.schema.marks.linkAnchor.create({ allLinks, title: "a link", location });              this._editorView.dispatch(tr.addMark(flattened[lastSel].from, flattened[lastSel].to, link));          }      } @@ -331,8 +357,8 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp      }      protected createDropTarget = (ele: HTMLDivElement) => {          this.ProseRef = ele; -        this.dropDisposer?.(); -        ele && (this.dropDisposer = DragManager.MakeDropTarget(ele, this.drop.bind(this), this.layoutDoc)); +        this._dropDisposer?.(); +        ele && (this._dropDisposer = DragManager.MakeDropTarget(ele, this.drop.bind(this), this.layoutDoc));      }      @undoBatch @@ -658,9 +684,9 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp              let tr = state.tr.addMark(sel.from, sel.to, splitter);              sel.from !== sel.to && tr.doc.nodesBetween(sel.from, sel.to, (node: any, pos: number, parent: any) => {                  if (node.firstChild === null && node.marks.find((m: Mark) => m.type.name === schema.marks.splitter.name)) { -                    const allHrefs = [{ href, title, targetId, linkId }]; -                    allHrefs.push(...(node.marks.find((m: Mark) => m.type.name === schema.marks.link.name)?.attrs.allHrefs ?? [])); -                    const link = state.schema.marks.link.create({ allHrefs, title, location, linkId }); +                    const allLinks = [{ href, title, targetId, linkId }]; +                    allLinks.push(...(node.marks.find((m: Mark) => m.type.name === schema.marks.linkAnchor.name)?.attrs.allLinks ?? [])); +                    const link = state.schema.marks.linkAnchor.create({ allLinks, title, location, linkId });                      tr = tr.addMark(pos, pos + node.nodeSize, link);                  }              }); @@ -670,6 +696,12 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp          }      }      componentDidMount() { +        this._cachedLinks = DocListCast(this.Document.links); +        this._disposers.links = reaction(() => DocListCast(this.Document.links), // if a link is deleted, then remove all hyperlinks that reference it from the text's marks +            newLinks => { +                this._cachedLinks.forEach(l => !newLinks.includes(l) && this.RemoveLinkFromDoc(l)); +                this._cachedLinks = newLinks; +            });          this._disposers.buttonBar = reaction(              () => DocumentButtonBar.Instance,              instance => { @@ -700,8 +732,10 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp              incomingValue => {                  if (incomingValue !== undefined && this._editorView && !this._applyingChange) {                      const updatedState = JSON.parse(incomingValue); -                    this._editorView.updateState(EditorState.fromJSON(this.config, updatedState)); -                    this.tryUpdateHeight(); +                    if (JSON.stringify(this._editorView.state.toJSON()) !== JSON.stringify(updatedState)) { +                        this._editorView.updateState(EditorState.fromJSON(this.config, updatedState)); +                        this.tryUpdateHeight(); +                    }                  }              }          ); @@ -776,8 +810,8 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp                          return node.copy(content.frag);                      }                      const marks = [...node.marks]; -                    const linkIndex = marks.findIndex(mark => mark.type === editor.state.schema.marks.link); -                    return linkIndex !== -1 && marks[linkIndex].attrs.allHrefs.find((item: { href: string }) => scrollToLinkID === item.href.replace(/.*\/doc\//, "")) ? node : undefined; +                    const linkIndex = marks.findIndex(mark => mark.type === editor.state.schema.marks.linkAnchor); +                    return linkIndex !== -1 && marks[linkIndex].attrs.allLinks.find((item: { href: string }) => scrollToLinkID === item.href.replace(/.*\/doc\//, "")) ? node : undefined;                  };                  let start = 0; @@ -959,8 +993,8 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp              }              const marks = [...node.marks];              const linkIndex = marks.findIndex(mark => mark.type.name === "link"); -            const allHrefs = [{ href: Utils.prepend(`/doc/${linkId}`), title, linkId }]; -            const link = view.state.schema.mark(view.state.schema.marks.link, { allHrefs, location: "onRight", title, docref: true }); +            const allLinks = [{ href: Utils.prepend(`/doc/${linkId}`), title, linkId }]; +            const link = view.state.schema.mark(view.state.schema.marks.linkAnchor, { allLinks, location: "onRight", title, docref: true });              marks.splice(linkIndex === -1 ? 0 : linkIndex, 1, link);              return node.mark(marks);          } @@ -1014,7 +1048,9 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp          }          (selectOnLoad /* || !rtfField?.Text*/) && this._editorView!.focus();          // add user mark for any first character that was typed since the user mark that gets set in KeyPress won't have been called yet. -        this._editorView!.state.storedMarks = [...(this._editorView!.state.storedMarks ? this._editorView!.state.storedMarks : []), schema.marks.user_mark.create({ userid: Doc.CurrentUserEmail, modified: Math.floor(Date.now() / 1000) })]; +        if (!this._editorView!.state.storedMarks || !this._editorView!.state.storedMarks.some(mark => mark.type === schema.marks.user_mark)) { +            this._editorView!.state.storedMarks = [...(this._editorView!.state.storedMarks ? this._editorView!.state.storedMarks : []), schema.marks.user_mark.create({ userid: Doc.CurrentUserEmail, modified: Math.floor(Date.now() / 1000) })]; +        }      }      getFont(font: string) {          switch (font) { @@ -1204,18 +1240,27 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp          });      } -    public static HadSelection: boolean = false; -    onBlur = (e: any) => { -        FormattedTextBox.HadSelection = window.getSelection()?.toString() !== ""; -        //DictationManager.Controls.stop(false); +    public startUndoTypingBatch() { +        this._undoTyping = UndoManager.StartBatch("undoTyping"); +    } + +    public endUndoTypingBatch() { +        const wasUndoing = this._undoTyping;          if (this._undoTyping) {              this._undoTyping.end();              this._undoTyping = undefined;          } +        return wasUndoing; +    } +    public static HadSelection: boolean = false; +    onBlur = (e: any) => { +        FormattedTextBox.HadSelection = window.getSelection()?.toString() !== ""; +        //DictationManager.Controls.stop(false); +        this.endUndoTypingBatch();          this.doLinkOnDeselect();          // move the richtextmenu offscreen -        if (!RichTextMenu.Instance.Pinned && !RichTextMenu.Instance.overMenu) RichTextMenu.Instance.jumpTo(-300, -300); +        if (!RichTextMenu.Instance.Pinned) RichTextMenu.Instance.delayHide();      }      _lastTimedMark: Mark | undefined = undefined; @@ -1249,11 +1294,12 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp          // this._editorView!.dispatch(this._editorView!.state.tr.removeStoredMark(schema.marks.user_mark.create({})).addStoredMark(mark));          if (!this._undoTyping) { -            this._undoTyping = UndoManager.StartBatch("undoTyping"); +            this.startUndoTypingBatch();          }      }      ondrop = (eve: React.DragEvent) => { +        this._editorView!.dispatch(updateBullets(this._editorView!.state.tr, this._editorView!.state.schema));          eve.stopPropagation(); // drag n drop of text within text note will generate a new note if not caughst, as will dragging in from outside of Dash.      }      onscrolled = (ev: React.UIEvent) => { @@ -1321,7 +1367,8 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp                          color: this.props.color ? this.props.color : StrCast(this.layoutDoc[this.props.fieldKey + "-color"], this.props.hideOnLeave ? "white" : "inherit"),                          pointerEvents: interactive ? undefined : "none",                          fontSize: Cast(this.layoutDoc._fontSize, "number", null), -                        fontFamily: StrCast(this.layoutDoc._fontFamily, "inherit") +                        fontFamily: StrCast(this.layoutDoc._fontFamily, "inherit"), +                        transition: "opacity 1s"                      }}                      onContextMenu={this.specificContextMenu}                      onKeyDown={this.onKeyPress} @@ -1349,7 +1396,7 @@ export class FormattedTextBox extends ViewBoxAnnotatableComponent<(FieldViewProp                          onScroll={this.onscrolled} onDrop={this.ondrop} >                          <div className={`formattedTextBox-inner${rounded}${selclass}`} ref={this.createDropTarget}                              style={{ -                                padding: `${Math.max(0, NumCast(this.layoutDoc._yMargin, this.props.yMargin || 0) + selPad)}px  ${NumCast(this.layoutDoc._xMargin, this.props.xMargin || 0) + selPad}px`, +                                padding: this.layoutDoc._textBoxPadding ? StrCast(this.layoutDoc._textBoxPadding) : `${Math.max(0, NumCast(this.layoutDoc._yMargin, this.props.yMargin || 0) + selPad)}px  ${NumCast(this.layoutDoc._xMargin, this.props.xMargin || 0) + selPad}px`,                                  pointerEvents: !this.props.isSelected() ? ((this.layoutDoc.isLinkButton || this.props.onClick) ? "none" : "all") : undefined                              }}                          /> diff --git a/src/client/views/nodes/formattedText/FormattedTextBoxComment.tsx b/src/client/views/nodes/formattedText/FormattedTextBoxComment.tsx index 90f2c0aa6..4c90b6afd 100644 --- a/src/client/views/nodes/formattedText/FormattedTextBoxComment.tsx +++ b/src/client/views/nodes/formattedText/FormattedTextBoxComment.tsx @@ -27,7 +27,7 @@ export function findUserMark(marks: Mark[]): Mark | undefined {      return marks.find(m => m.attrs.userid);  }  export function findLinkMark(marks: Mark[]): Mark | undefined { -    return marks.find(m => m.type === schema.marks.link); +    return marks.find(m => m.type === schema.marks.linkAnchor);  }  export function findStartOfMark(rpos: ResolvedPos, view: EditorView, finder: (marks: Mark[]) => Mark | undefined) {      let before = 0; @@ -182,7 +182,7 @@ export class FormattedTextBoxComment {              state.doc.nodesBetween(state.selection.from, state.selection.to, (node: any, pos: number, parent: any) => !child && node.marks.length && (child = node));              child = child || (nbef && state.selection.$from.nodeBefore);              const mark = child ? findLinkMark(child.marks) : undefined; -            const href = (!mark?.attrs.docref || naft === nbef) && mark?.attrs.allHrefs.find((item: { href: string }) => item.href)?.href || forceUrl; +            const href = (!mark?.attrs.docref || naft === nbef) && mark?.attrs.allLinks.find((item: { href: string }) => item.href)?.href || forceUrl;              if (forceUrl || (href && child && nbef && naft && mark?.attrs.showPreview)) {                  FormattedTextBoxComment.tooltipText.textContent = "external => " + href;                  (FormattedTextBoxComment.tooltipText as any).href = href; diff --git a/src/client/views/nodes/formattedText/ProsemirrorExampleTransfer.ts b/src/client/views/nodes/formattedText/ProsemirrorExampleTransfer.ts index 1bbcb9fa8..9d69f4be7 100644 --- a/src/client/views/nodes/formattedText/ProsemirrorExampleTransfer.ts +++ b/src/client/views/nodes/formattedText/ProsemirrorExampleTransfer.ts @@ -1,7 +1,6 @@  import { chainCommands, exitCode, joinDown, joinUp, lift, deleteSelection, joinBackward, selectNodeBackward, setBlockType, splitBlockKeepMarks, toggleMark, wrapIn, newlineInCode } from "prosemirror-commands";  import { liftTarget } from "prosemirror-transform";  import { redo, undo } from "prosemirror-history"; -import { undoInputRule } from "prosemirror-inputrules";  import { Schema } from "prosemirror-model";  import { liftListItem, sinkListItem } from "./prosemirrorPatches.js";  import { splitListItem, wrapInList, } from "prosemirror-schema-list"; @@ -12,7 +11,6 @@ import { Doc, DataSym } from "../../../../fields/Doc";  import { FormattedTextBox } from "./FormattedTextBox";  import { Id } from "../../../../fields/FieldSymbols";  import { Docs } from "../../../documents/Documents"; -import { update } from "lodash";  const mac = typeof navigator !== "undefined" ? /Mac/.test(navigator.platform) : false; @@ -215,10 +213,13 @@ export default function buildKeymap<S extends Schema<any>>(schema: S, props: any                  marks && tx3.setStoredMarks([...marks]);                  dispatch(tx3);              })) { +                const fromattrs = state.selection.$from.node().attrs;                  if (!splitBlockKeepMarks(state, (tx3: Transaction) => { -                    splitMetadata(marks, tx3); -                    if (!liftListItem(schema.nodes.list_item)(tx3, dispatch as ((tx: Transaction<Schema<any, any>>) => void))) { -                        dispatch(tx3); +                    const tonode = tx3.selection.$to.node(); +                    const tx4 = tx3.setNodeMarkup(tx3.selection.to - 1, tonode.type, fromattrs, tonode.marks); +                    splitMetadata(marks, tx4); +                    if (!liftListItem(schema.nodes.list_item)(tx4, dispatch as ((tx: Transaction<Schema<any, any>>) => void))) { +                        dispatch(tx4);                      }                  })) {                      return false; @@ -281,19 +282,6 @@ export default function buildKeymap<S extends Schema<any>>(schema: S, props: any          return false;      }); -    // bind("^", (state: EditorState<S>, dispatch: (tx: Transaction<S>) => void) => { -    //     let newNode = schema.nodes.footnote.create({}); -    //     if (dispatch && state.selection.from === state.selection.to) { -    //         let tr = state.tr; -    //         tr.replaceSelectionWith(newNode); // replace insertion with a footnote. -    //         dispatch(tr.setSelection(new NodeSelection( // select the footnote node to open its display -    //             tr.doc.resolve(  // get the location of the footnote node by subtracting the nodesize of the footnote from the current insertion point anchor (which will be immediately after the footnote node) -    //                 tr.selection.anchor - tr.selection.$anchor.nodeBefore!.nodeSize)))); -    //         return true; -    //     } -    //     return false; -    // }); -      return keys;  } diff --git a/src/client/views/nodes/formattedText/RichTextMenu.scss b/src/client/views/nodes/formattedText/RichTextMenu.scss index 7a0718c16..fbc468292 100644 --- a/src/client/views/nodes/formattedText/RichTextMenu.scss +++ b/src/client/views/nodes/formattedText/RichTextMenu.scss @@ -77,6 +77,12 @@              color: white;          }      } +    .richTextMenu-divider { +        margin: auto; +        border-left: solid #ffffff70 0.5px; +        height: 20px; +        width: 1px; +    }  }  .link-menu { diff --git a/src/client/views/nodes/formattedText/RichTextMenu.tsx b/src/client/views/nodes/formattedText/RichTextMenu.tsx index 839943aac..95d6c9fac 100644 --- a/src/client/views/nodes/formattedText/RichTextMenu.tsx +++ b/src/client/views/nodes/formattedText/RichTextMenu.tsx @@ -1,8 +1,8 @@  import React = require("react");  import { IconProp, library } from '@fortawesome/fontawesome-svg-core'; -import { faBold, faCaretDown, faChevronLeft, faEyeDropper, faHighlighter, faIndent, faItalic, faLink, faPaintRoller, faPalette, faStrikethrough, faSubscript, faSuperscript, faUnderline } from "@fortawesome/free-solid-svg-icons"; +import { faBold, faCaretDown, faChevronLeft, faEyeDropper, faHighlighter, faOutdent, faIndent, faHandPointLeft, faHandPointRight, faItalic, faLink, faPaintRoller, faPalette, faStrikethrough, faSubscript, faSuperscript, faUnderline } from "@fortawesome/free-solid-svg-icons";  import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { action, observable } from "mobx"; +import { action, observable, IReactionDisposer, reaction } from "mobx";  import { observer } from "mobx-react";  import { lift, wrapIn } from "prosemirror-commands";  import { Mark, MarkType, Node as ProsNode, NodeType, ResolvedPos } from "prosemirror-model"; @@ -23,9 +23,10 @@ import { updateBullets } from "./ProsemirrorExampleTransfer";  import "./RichTextMenu.scss";  import { schema } from "./schema_rts";  import { TraceMobx } from "../../../../fields/util"; +import { UndoManager } from "../../../util/UndoManager";  const { toggleMark } = require("prosemirror-commands"); -library.add(faBold, faItalic, faChevronLeft, faUnderline, faStrikethrough, faSuperscript, faSubscript, faIndent, faEyeDropper, faCaretDown, faPalette, faHighlighter, faLink, faPaintRoller); +library.add(faBold, faItalic, faChevronLeft, faUnderline, faStrikethrough, faSuperscript, faSubscript, faOutdent, faIndent, faHandPointLeft, faHandPointRight, faEyeDropper, faCaretDown, faPalette, faHighlighter, faLink, faPaintRoller);  @observer @@ -68,6 +69,8 @@ export default class RichTextMenu extends AntimodeMenu {      @observable private currentLink: string | undefined = "";      @observable private showLinkDropdown: boolean = false; +    _reaction: IReactionDisposer | undefined; +    _delayHide = false;      constructor(props: Readonly<{}>) {          super(props);          RichTextMenu.Instance = this; @@ -138,6 +141,16 @@ export default class RichTextMenu extends AntimodeMenu {          ];      } +    componentDidMount() { +        this._reaction = reaction(() => SelectionManager.SelectedDocuments(), +            () => this._delayHide && !(this._delayHide = false) && this.fadeOut(true)); +    } +    componentWillUnmount() { +        this._reaction?.(); +    } + +    public delayHide = () => this._delayHide = true; +      @action      changeView(view: EditorView) {          this.view = view; @@ -147,16 +160,6 @@ export default class RichTextMenu extends AntimodeMenu {          this.updateFromDash(view, lastState, this.editorProps);      } -    public MakeLinkToSelection = (linkDocId: string, title: string, location: string, targetDocId: string): string => { -        if (this.view) { -            const link = this.view.state.schema.marks.link.create({ href: Utils.prepend("/doc/" + linkDocId), title: title, location: location, linkId: linkDocId, targetId: targetDocId }); -            this.view.dispatch(this.view.state.tr.removeMark(this.view.state.selection.from, this.view.state.selection.to, this.view.state.schema.marks.link). -                addMark(this.view.state.selection.from, this.view.state.selection.to, link)); -            return this.view.state.selection.$from.nodeAfter?.text || ""; -        } -        return ""; -    } -      @action      public async updateFromDash(view: EditorView, lastState: EditorState | undefined, props: any) {          if (!view) { @@ -310,8 +313,11 @@ export default class RichTextMenu extends AntimodeMenu {          function onClick(e: React.PointerEvent) {              e.preventDefault();              e.stopPropagation(); -            self.view && command && command(self.view.state, self.view.dispatch, self.view); -            self.view && onclick && onclick(self.view.state, self.view.dispatch, self.view); +            self.TextView.endUndoTypingBatch(); +            UndoManager.RunInBatch(() => { +                self.view && command && command(self.view.state, self.view.dispatch, self.view); +                self.view && onclick && onclick(self.view.state, self.view.dispatch, self.view); +            }, "rich text menu command");              self.setActiveMarkButtons(self.getActiveMarksOnSelection());          } @@ -338,9 +344,10 @@ export default class RichTextMenu extends AntimodeMenu {          function onChange(e: React.ChangeEvent<HTMLSelectElement>) {              e.stopPropagation();              e.preventDefault(); +            self.TextView.endUndoTypingBatch();              options.forEach(({ label, mark, command }) => {                  if (e.target.value === label) { -                    self.view && mark && command(mark, self.view); +                    UndoManager.RunInBatch(() => self.view && mark && command(mark, self.view), "text mark dropdown");                  }              });          } @@ -361,9 +368,10 @@ export default class RichTextMenu extends AntimodeMenu {          const self = this;          function onChange(val: string) { +            self.TextView.endUndoTypingBatch();              options.forEach(({ label, node, command }) => {                  if (val === label) { -                    self.view && node && command(node); +                    UndoManager.RunInBatch(() => self.view && node && command(node), "nodes dropdown");                  }              });          } @@ -412,6 +420,85 @@ export default class RichTextMenu extends AntimodeMenu {          dispatch?.(tr.replaceSelectionWith(newNode).removeMark(tr.selection.from - 1, tr.selection.from, mark));          return true;      } +    alignCenter = (state: EditorState<any>, dispatch: any) => { +        return this.alignParagraphs(state, "center", dispatch); +    } +    alignLeft = (state: EditorState<any>, dispatch: any) => { +        return this.alignParagraphs(state, "left", dispatch); +    } +    alignRight = (state: EditorState<any>, dispatch: any) => { +        return this.alignParagraphs(state, "right", dispatch); +    } + +    alignParagraphs(state: EditorState<any>, align: "left" | "right" | "center", dispatch: any) { +        var tr = state.tr; +        state.doc.nodesBetween(state.selection.from, state.selection.to, (node, pos, parent, index) => { +            if (node.type === schema.nodes.paragraph) { +                tr = tr.setNodeMarkup(pos, node.type, { ...node.attrs, align }, node.marks); +                return false; +            } +            return true; +        }); +        dispatch?.(tr); +        return true; +    } + +    insetParagraph(state: EditorState<any>, dispatch: any) { +        var tr = state.tr; +        state.doc.nodesBetween(state.selection.from, state.selection.to, (node, pos, parent, index) => { +            if (node.type === schema.nodes.paragraph) { +                const inset = (node.attrs.inset ? Number(node.attrs.inset) : 0) + 10; +                tr = tr.setNodeMarkup(pos, node.type, { ...node.attrs, inset }, node.marks); +                return false; +            } +            return true; +        }); +        dispatch?.(tr); +        return true; +    } +    outsetParagraph(state: EditorState<any>, dispatch: any) { +        var tr = state.tr; +        state.doc.nodesBetween(state.selection.from, state.selection.to, (node, pos, parent, index) => { +            if (node.type === schema.nodes.paragraph) { +                const inset = Math.max(0, (node.attrs.inset ? Number(node.attrs.inset) : 0) - 10); +                tr = tr.setNodeMarkup(pos, node.type, { ...node.attrs, inset }, node.marks); +                return false; +            } +            return true; +        }); +        dispatch?.(tr); +        return true; +    } + +    indentParagraph(state: EditorState<any>, dispatch: any) { +        var tr = state.tr; +        state.doc.nodesBetween(state.selection.from, state.selection.to, (node, pos, parent, index) => { +            if (node.type === schema.nodes.paragraph) { +                const nodeval = node.attrs.indent ? Number(node.attrs.indent) : undefined; +                const indent = !nodeval ? 25 : nodeval < 0 ? 0 : nodeval + 25; +                tr = tr.setNodeMarkup(pos, node.type, { ...node.attrs, indent }, node.marks); +                return false; +            } +            return true; +        }); +        dispatch?.(tr); +        return true; +    } + +    hangingIndentParagraph(state: EditorState<any>, dispatch: any) { +        var tr = state.tr; +        state.doc.nodesBetween(state.selection.from, state.selection.to, (node, pos, parent, index) => { +            if (node.type === schema.nodes.paragraph) { +                const nodeval = node.attrs.indent ? Number(node.attrs.indent) : undefined; +                const indent = !nodeval ? -25 : nodeval > 0 ? 0 : nodeval - 10; +                tr = tr.setNodeMarkup(pos, node.type, { ...node.attrs, indent }, node.marks); +                return false; +            } +            return true; +        }); +        dispatch?.(tr); +        return true; +    }      insertBlockquote(state: EditorState<any>, dispatch: any) {          const path = (state.selection.$from as any).path; @@ -423,6 +510,11 @@ export default class RichTextMenu extends AntimodeMenu {          return true;      } +    insertHorizontalRule(state: EditorState<any>, dispatch: any) { +        dispatch(state.tr.replaceSelectionWith(state.schema.nodes.horizontal_rule.create()).scrollIntoView()); +        return true; +    } +      @action toggleBrushDropdown() { this.showBrushDropdown = !this.showBrushDropdown; }      // todo: add brushes to brushMap to save with a style name @@ -439,7 +531,8 @@ export default class RichTextMenu extends AntimodeMenu {          function onBrushClick(e: React.PointerEvent) {              e.preventDefault();              e.stopPropagation(); -            self.view && self.fillBrush(self.view.state, self.view.dispatch); +            self.TextView.endUndoTypingBatch(); +            UndoManager.RunInBatch(() => self.view && self.fillBrush(self.view.state, self.view.dispatch), "rt brush");          }          let label = "Stored marks: "; @@ -506,19 +599,24 @@ export default class RichTextMenu extends AntimodeMenu {      @action toggleColorDropdown() { this.showColorDropdown = !this.showColorDropdown; }      @action setActiveColor(color: string) { this.activeFontColor = color; } +    get TextView() { return (this.view as any).TextView as FormattedTextBox; }      createColorButton() {          const self = this;          function onColorClick(e: React.PointerEvent) {              e.preventDefault();              e.stopPropagation(); -            self.view && self.insertColor(self.activeFontColor, self.view.state, self.view.dispatch); +            self.TextView.endUndoTypingBatch(); +            UndoManager.RunInBatch(() => self.view && self.insertColor(self.activeFontColor, self.view.state, self.view.dispatch), "rt menu color"); +            self.TextView.EditorView!.focus();          }          function changeColor(e: React.PointerEvent, color: string) {              e.preventDefault();              e.stopPropagation();              self.setActiveColor(color); -            self.view && self.insertColor(self.activeFontColor, self.view.state, self.view.dispatch); +            self.TextView.endUndoTypingBatch(); +            UndoManager.RunInBatch(() => self.view && self.insertColor(self.activeFontColor, self.view.state, self.view.dispatch), "rt menu color"); +            self.TextView.EditorView!.focus();          }          const button = @@ -563,13 +661,15 @@ export default class RichTextMenu extends AntimodeMenu {          function onHighlightClick(e: React.PointerEvent) {              e.preventDefault();              e.stopPropagation(); -            self.view && self.insertHighlight(self.activeHighlightColor, self.view.state, self.view.dispatch); +            self.TextView.endUndoTypingBatch(); +            UndoManager.RunInBatch(() => self.view && self.insertHighlight(self.activeHighlightColor, self.view.state, self.view.dispatch), "rt highligher");          }          function changeHighlight(e: React.PointerEvent, color: string) {              e.preventDefault();              e.stopPropagation();              self.setActiveHighlight(color); -            self.view && self.insertHighlight(self.activeHighlightColor, self.view.state, self.view.dispatch); +            self.TextView.endUndoTypingBatch(); +            UndoManager.RunInBatch(() => self.view && self.insertHighlight(self.activeHighlightColor, self.view.state, self.view.dispatch), "rt highlighter");          }          const button = @@ -609,7 +709,8 @@ export default class RichTextMenu extends AntimodeMenu {          const self = this;          function onLinkChange(e: React.ChangeEvent<HTMLInputElement>) { -            self.setCurrentLink(e.target.value); +            self.TextView.endUndoTypingBatch(); +            UndoManager.RunInBatch(() => self.setCurrentLink(e.target.value), "link change");          }          const link = this.currentLink ? this.currentLink : ""; @@ -636,7 +737,7 @@ export default class RichTextMenu extends AntimodeMenu {          const node = this.view.state.selection.$from.nodeAfter;          const link = node && node.marks.find(m => m.type.name === "link");          if (link) { -            const href = link.attrs.allHrefs.length > 0 ? link.attrs.allHrefs[0].href : undefined; +            const href = link.attrs.allLinks.length > 0 ? link.attrs.allLinks[0].href : undefined;              if (href) {                  if (href.indexOf(Utils.prepend("/doc/")) === 0) {                      const linkclicked = href.replace(Utils.prepend("/doc/"), "").split("?")[0]; @@ -671,40 +772,28 @@ export default class RichTextMenu extends AntimodeMenu {      }      deleteLink = () => { -        if (!this.view) return; - -        const node = this.view.state.selection.$from.nodeAfter; -        const link = node && node.marks.find(m => m.type === this.view!.state.schema.marks.link); -        const href = link!.attrs.allHrefs.length > 0 ? link!.attrs.allHrefs[0].href : undefined; -        if (href) { -            if (href.indexOf(Utils.prepend("/doc/")) === 0) { -                const linkclicked = href.replace(Utils.prepend("/doc/"), "").split("?")[0]; -                if (linkclicked) { -                    DocServer.GetRefField(linkclicked).then(async linkDoc => { -                        if (linkDoc instanceof Doc) { -                            LinkManager.Instance.deleteLink(linkDoc); -                            this.view!.dispatch(this.view!.state.tr.removeMark(this.view!.state.selection.from, this.view!.state.selection.to, this.view!.state.schema.marks.link)); -                        } -                    }); -                } -            } else { -                if (node) { -                    const { tr, schema, selection } = this.view.state; -                    const extension = this.linkExtend(selection.$anchor, href); -                    this.view.dispatch(tr.removeMark(extension.from, extension.to, schema.marks.link)); -                } +        if (this.view) { +            const link = this.view.state.selection.$from.nodeAfter?.marks.find(m => m.type === this.view!.state.schema.marks.linkAnchor); +            if (link) { +                const allLinks = link.attrs.allLinks.slice(); +                this.TextView.RemoveLinkFromSelection(link.attrs.allLinks); +                // bcz: Argh ... this will remove the link from the document even it's anchored somewhere else in the text which happens if only part of the anchor text was selected. +                allLinks.filter((aref: any) => aref?.href.indexOf(Utils.prepend("/doc/")) === 0).forEach((aref: any) => { +                    const linkId = aref.href.replace(Utils.prepend("/doc/"), "").split("?")[0]; +                    linkId && DocServer.GetRefField(linkId).then(linkDoc => LinkManager.Instance.deleteLink(linkDoc as Doc)); +                });              }          }      }      linkExtend($start: ResolvedPos, href: string) { -        const mark = this.view!.state.schema.marks.link; +        const mark = this.view!.state.schema.marks.linkAnchor;          let startIndex = $start.index();          let endIndex = $start.indexAfter(); -        while (startIndex > 0 && $start.parent.child(startIndex - 1).marks.filter(m => m.type === mark && m.attrs.allHrefs.find((item: { href: string }) => item.href === href)).length) startIndex--; -        while (endIndex < $start.parent.childCount && $start.parent.child(endIndex).marks.filter(m => m.type === mark && m.attrs.allHrefs.find((item: { href: string }) => item.href === href)).length) endIndex++; +        while (startIndex > 0 && $start.parent.child(startIndex - 1).marks.filter(m => m.type === mark && m.attrs.allLinks.find((item: { href: string }) => item.href === href)).length) startIndex--; +        while (endIndex < $start.parent.childCount && $start.parent.child(endIndex).marks.filter(m => m.type === mark && m.attrs.allLinks.find((item: { href: string }) => item.href === href)).length) endIndex++;          let startPos = $start.start();          let endPos = startPos; @@ -744,7 +833,7 @@ export default class RichTextMenu extends AntimodeMenu {          return ref_node;      } -    @action onPointerEnter(e: React.PointerEvent) { RichTextMenu.Instance.overMenu = true; } +    @action onPointerEnter(e: React.PointerEvent) { RichTextMenu.Instance.overMenu = false; }      @action onPointerLeave(e: React.PointerEvent) { RichTextMenu.Instance.overMenu = false; }      @action @@ -768,26 +857,41 @@ export default class RichTextMenu extends AntimodeMenu {          TraceMobx();          const row1 = <div className="antimodeMenu-row" key="row1" style={{ display: this.collapsed ? "none" : undefined }}>{[              !this.collapsed ? this.getDragger() : (null), -            this.createButton("bold", "Bold", this.boldActive, toggleMark(schema.marks.strong)), -            this.createButton("italic", "Italic", this.italicsActive, toggleMark(schema.marks.em)), -            this.createButton("underline", "Underline", this.underlineActive, toggleMark(schema.marks.underline)), -            this.createButton("strikethrough", "Strikethrough", this.strikethroughActive, toggleMark(schema.marks.strikethrough)), -            this.createButton("superscript", "Superscript", this.superscriptActive, toggleMark(schema.marks.superscript)), -            this.createButton("subscript", "Subscript", this.subscriptActive, toggleMark(schema.marks.subscript)), +            !this.Pinned ? (null) : <> {[ +                this.createButton("bold", "Bold", this.boldActive, toggleMark(schema.marks.strong)), +                this.createButton("italic", "Italic", this.italicsActive, toggleMark(schema.marks.em)), +                this.createButton("underline", "Underline", this.underlineActive, toggleMark(schema.marks.underline)), +                this.createButton("strikethrough", "Strikethrough", this.strikethroughActive, toggleMark(schema.marks.strikethrough)), +                this.createButton("superscript", "Superscript", this.superscriptActive, toggleMark(schema.marks.superscript)), +                this.createButton("subscript", "Subscript", this.subscriptActive, toggleMark(schema.marks.subscript)), +                <div className="richTextMenu-divider" /> +            ]}</>,              this.createColorButton(),              this.createHighlighterButton(),              this.createLinkButton(),              this.createBrushButton(), -            this.createButton("sort-amount-down", "Summarize", undefined, this.insertSummarizer), -            this.createButton("quote-left", "Blockquote", undefined, this.insertBlockquote), +            <div className="richTextMenu-divider" />, +            this.createButton("align-left", "Align Left", undefined, this.alignLeft), +            this.createButton("align-center", "Align Center", undefined, this.alignCenter), +            this.createButton("align-right", "Align Right", undefined, this.alignRight), +            this.createButton("indent", "Inset More", undefined, this.insetParagraph), +            this.createButton("outdent", "Inset Less", undefined, this.outsetParagraph), +            this.createButton("hand-point-left", "Hanging Indent", undefined, this.hangingIndentParagraph), +            this.createButton("hand-point-right", "Indent", undefined, this.indentParagraph),          ]}</div>;          const row2 = <div className="antimodeMenu-row row-2" key="antimodemenu row2">              {this.collapsed ? this.getDragger() : (null)}              <div key="row" style={{ display: this.collapsed ? "none" : undefined }}> +                <div className="richTextMenu-divider" />,                  {[this.createMarksDropdown(this.activeFontSize, this.fontSizeOptions, "font size"),                  this.createMarksDropdown(this.activeFontFamily, this.fontFamilyOptions, "font family"), -                this.createNodesDropdown(this.activeListType, this.listTypeOptions, "nodes")]} +                <div className="richTextMenu-divider" />, +                this.createNodesDropdown(this.activeListType, this.listTypeOptions, "nodes"), +                this.createButton("sort-amount-down", "Summarize", undefined, this.insertSummarizer), +                this.createButton("quote-left", "Blockquote", undefined, this.insertBlockquote), +                this.createButton("minus", "Horizontal Rule", undefined, this.insertHorizontalRule), +                <div className="richTextMenu-divider" />,]}              </div>              <div key="button">                  {/* <div key="collapser"> @@ -817,7 +921,7 @@ interface ButtonDropdownProps {  }  @observer -class ButtonDropdown extends React.Component<ButtonDropdownProps> { +export class ButtonDropdown extends React.Component<ButtonDropdownProps> {      @observable private showDropdown: boolean = false;      private ref: HTMLDivElement | null = null; diff --git a/src/client/views/nodes/formattedText/RichTextRules.ts b/src/client/views/nodes/formattedText/RichTextRules.ts index ba3230801..ca30dde9d 100644 --- a/src/client/views/nodes/formattedText/RichTextRules.ts +++ b/src/client/views/nodes/formattedText/RichTextRules.ts @@ -275,11 +275,11 @@ export class RichTextRules {                      if (!fieldKey) {                          if (docid) {                              DocServer.GetRefField(docid).then(docx => { -                                const target = ((docx instanceof Doc) && docx) || Docs.Create.FreeformDocument([], { title: docid, _width: 500, _height: 500, _LODdisable: true, }, docid); +                                const target = ((docx instanceof Doc) && docx) || Docs.Create.FreeformDocument([], { title: docid, _width: 500, _height: 500, }, docid);                                  DocUtils.Publish(target, docid, returnFalse, returnFalse);                                  DocUtils.MakeLink({ doc: this.Document }, { doc: target }, "portal to");                              }); -                            const link = state.schema.marks.link.create({ href: Utils.prepend("/doc/" + docid), location: "onRight", title: docid, targetId: docid }); +                            const link = state.schema.marks.linkAnchor.create({ href: Utils.prepend("/doc/" + docid), location: "onRight", title: docid, targetId: docid });                              return state.tr.deleteRange(end - 1, end).deleteRange(start, start + 2).addMark(start, end - 3, link);                          }                          return state.tr; @@ -305,7 +305,7 @@ export class RichTextRules {                      if (!fieldKey && !docid) return state.tr;                      docid && DocServer.GetRefField(docid).then(docx => {                          if (!(docx instanceof Doc && docx)) { -                            const docx = Docs.Create.FreeformDocument([], { title: docid, _width: 500, _height: 500, _LODdisable: true }, docid); +                            const docx = Docs.Create.FreeformDocument([], { title: docid, _width: 500, _height: 500 }, docid);                              DocUtils.Publish(docx, docid, returnFalse, returnFalse);                          }                      }); @@ -315,8 +315,6 @@ export class RichTextRules {                      return node ? state.tr.replaceRangeWith(start, end, dashDoc).setStoredMarks([...node.marks, ...(sm ? sm : [])]) : state.tr;                  }), - -              // create an inline view of a tag stored under the '#' field              new InputRule(                  new RegExp(/#([a-zA-Z_\-]+[a-zA-Z_;\-0-9]*)\s$/), @@ -374,7 +372,6 @@ export class RichTextRules {              new InputRule(                  new RegExp(/%\)/),                  (state, match, start, end) => { -                      return state.tr.deleteRange(start, end).removeStoredMark(state.schema.marks.summarizeInclusive.create());                  }), diff --git a/src/client/views/nodes/formattedText/marks_rts.ts b/src/client/views/nodes/formattedText/marks_rts.ts index b09ac0678..3d7d71b14 100644 --- a/src/client/views/nodes/formattedText/marks_rts.ts +++ b/src/client/views/nodes/formattedText/marks_rts.ts @@ -17,12 +17,12 @@ export const marks: { [index: string]: MarkSpec } = {              return ["div", { className: "dummy" }, 0];          }      }, -    // :: MarkSpec A link. Has `href` and `title` attributes. `title` +    // :: MarkSpec A linkAnchor. The anchor can have multiple links, where each link has an href URL and a title for use in menus and hover (Dash links have linkIDs & targetIDs). `title`      // defaults to the empty string. Rendered and parsed as an `<a>`      // element. -    link: { +    linkAnchor: {          attrs: { -            allHrefs: { default: [] as { href: string, title: string, linkId: string, targetId: string }[] }, +            allLinks: { default: [] as { href: string, title: string, linkId: string, targetId: string }[] },              showPreview: { default: true },              location: { default: null },              title: { default: null }, @@ -31,22 +31,22 @@ export const marks: { [index: string]: MarkSpec } = {          inclusive: false,          parseDOM: [{              tag: "a[href]", getAttrs(dom: any) { -                return { allHrefs: [{ href: dom.getAttribute("href"), title: dom.getAttribute("title"), linkId: dom.getAttribute("linkids"), targetId: dom.getAttribute("targetids") }], location: dom.getAttribute("location"), }; +                return { allLinks: [{ href: dom.getAttribute("href"), title: dom.getAttribute("title"), linkId: dom.getAttribute("linkids"), targetId: dom.getAttribute("targetids") }], location: dom.getAttribute("location"), };              }          }],          toDOM(node: any) { -            const targetids = node.attrs.allHrefs.reduce((p: string, item: { href: string, title: string, targetId: string, linkId: string }) => p + " " + item.targetId, ""); -            const linkids = node.attrs.allHrefs.reduce((p: string, item: { href: string, title: string, targetId: string, linkId: string }) => p + " " + item.linkId, ""); +            const targetids = node.attrs.allLinks.reduce((p: string, item: { href: string, title: string, targetId: string, linkId: string }) => p + " " + item.targetId, ""); +            const linkids = node.attrs.allLinks.reduce((p: string, item: { href: string, title: string, targetId: string, linkId: string }) => p + " " + item.linkId, "");              return node.attrs.docref && node.attrs.title ? -                ["div", ["span", `"`], ["span", 0], ["span", `"`], ["br"], ["a", { ...node.attrs, href: node.attrs.allHrefs[0].href, class: "prosemirror-attribution" }, node.attrs.title], ["br"]] : -                node.attrs.allHrefs.length === 1 ? -                    ["a", { ...node.attrs, class: linkids, targetids, title: `${node.attrs.title}`, href: node.attrs.allHrefs[0].href }, 0] : +                ["div", ["span", `"`], ["span", 0], ["span", `"`], ["br"], ["a", { ...node.attrs, href: node.attrs.allLinks[0].href, class: "prosemirror-attribution" }, node.attrs.title], ["br"]] : +                node.attrs.allLinks.length === 1 ? +                    ["a", { ...node.attrs, class: linkids, targetids, title: `${node.attrs.title}`, href: node.attrs.allLinks[0].href }, 0] :                      ["div", { class: "prosemirror-anchor" },                          ["span", { class: "prosemirror-linkBtn" },                              ["a", { ...node.attrs, class: linkids, targetids, title: `${node.attrs.title}` }, 0],                              ["input", { class: "prosemirror-hrefoptions" }],                          ], -                        ["div", { class: "prosemirror-links" }, ...node.attrs.allHrefs.map((item: { href: string, title: string }) => +                        ["div", { class: "prosemirror-links" }, ...node.attrs.allLinks.map((item: { href: string, title: string }) =>                              ["a", { class: "prosemirror-dropdownlink", href: item.href }, item.title]                          )]                      ]; @@ -270,6 +270,7 @@ export const marks: { [index: string]: MarkSpec } = {              userid: { default: "" },              modified: { default: "when?" }, // 1 second intervals since 1970          }, +        excludes: "user_mark",          group: "inline",          toDOM(node: any) {              const uid = node.attrs.userid.replace(".", "").replace("@", ""); diff --git a/src/client/views/nodes/formattedText/nodes_rts.ts b/src/client/views/nodes/formattedText/nodes_rts.ts index afb1f57b7..f83cff9b9 100644 --- a/src/client/views/nodes/formattedText/nodes_rts.ts +++ b/src/client/views/nodes/formattedText/nodes_rts.ts @@ -312,7 +312,7 @@ export const nodes: { [index: string]: NodeSpec } = {              const map = node.attrs.bulletStyle ? node.attrs.mapStyle + node.attrs.bulletStyle : "";              return node.attrs.visibility ?                  ["li", { class: `${map}`, "data-mapStyle": node.attrs.mapStyle, "data-bulletStyle": node.attrs.bulletStyle }, 0] : -                ["li", { class: `${map}`, "data-mapStyle": node.attrs.mapStyle, "data-bulletStyle": node.attrs.bulletStyle }, "..."]; +                ["li", { class: `${map}`, "data-mapStyle": node.attrs.mapStyle, "data-bulletStyle": node.attrs.bulletStyle }, `${node.firstChild?.textContent}...`];          }      },  };
\ No newline at end of file diff --git a/src/client/views/nodes/formattedText/prosemirrorPatches.js b/src/client/views/nodes/formattedText/prosemirrorPatches.js index 763961958..0969ea4ef 100644 --- a/src/client/views/nodes/formattedText/prosemirrorPatches.js +++ b/src/client/views/nodes/formattedText/prosemirrorPatches.js @@ -9,6 +9,7 @@ var prosemirrorModel = require('prosemirror-model');  exports.liftListItem = liftListItem;  exports.sinkListItem = sinkListItem;  exports.wrappingInputRule = wrappingInputRule; +exports.removeMarkWithAttrs = removeMarkWithAttrs;  // :: (NodeType) → (state: EditorState, dispatch: ?(tr: Transaction)) → bool  // Create a command to lift the list item around the selection up into  // a wrapping list. @@ -139,3 +140,57 @@ function wrappingInputRule(regexp, nodeType, getAttrs, joinPredicate, customWith  } +// :: ([Mark]) → ?Mark +// Tests whether there is a mark of this type in the given set. +function isInSetWithAttrs(mark, set, attrs) { +    for (var i = 0; i < set.length; i++) { +        if (set[i].type == mark) { +            if (Array.from(Object.keys(attrs)).reduce((p, akey) => { +                return p && JSON.stringify(set[i].attrs[akey]) === JSON.stringify(attrs[akey]); +            }, true)) { +                return set[i]; +            } +        } +    } +}; + +// :: (number, number, ?union<Mark, MarkType>) → this +// Remove marks from inline nodes between `from` and `to`. When `mark` +// is a single mark, remove precisely that mark. When it is a mark type, +// remove all marks of that type. When it is null, remove all marks of +// any type. +function removeMarkWithAttrs(tr, from, to, mark, attrs) { +    if (mark === void 0) mark = null; + +    var matched = [], step = 0; +    tr.doc.nodesBetween(from, to, function (node, pos) { +        if (!node.isInline) { return } +        step++; +        var toRemove = null; +        if (mark) { +            if (isInSetWithAttrs(mark, node.marks, attrs)) { toRemove = [mark]; } +        } else { +            toRemove = node.marks; +        } +        if (toRemove && toRemove.length) { +            var end = Math.min(pos + node.nodeSize, to); +            for (var i = 0; i < toRemove.length; i++) { +                var style = toRemove[i], found$1 = (void 0); +                for (var j = 0; j < matched.length; j++) { +                    var m = matched[j]; +                    if (m.step == step - 1 && style.eq(matched[j].style)) { found$1 = m; } +                } +                if (found$1) { +                    found$1.to = end; +                    found$1.step = step; +                } else { +                    matched.push({ style: style, from: Math.max(pos, from), to: end, step: step }); +                } +            } +        } +    }); +    matched.forEach(function (m) { return tr.step(new prosemirrorTransform.RemoveMarkStep(m.from, m.to, m.style)); }); +    return tr +}; + + diff --git a/src/client/views/pdf/PDFMenu.scss b/src/client/views/pdf/PDFMenu.scss index 3c08ba80d..fa43a99b2 100644 --- a/src/client/views/pdf/PDFMenu.scss +++ b/src/client/views/pdf/PDFMenu.scss @@ -3,4 +3,23 @@      width: 200px;      padding: 5px;      grid-template-columns: 90px 20px 90px; +} + +.color-wrapper { +    display: flex; +    flex-wrap: wrap; +    justify-content: space-between; + +    button.color-button { +        width: 20px; +        height: 20px; +        border-radius: 15px !important; +        margin: 3px; +        border: 2px solid transparent !important; +        padding: 3px; + +        &.active { +            border: 2px solid white; +        } +    }  }
\ No newline at end of file diff --git a/src/client/views/pdf/PDFMenu.tsx b/src/client/views/pdf/PDFMenu.tsx index 6dcf5cce6..00c56d73e 100644 --- a/src/client/views/pdf/PDFMenu.tsx +++ b/src/client/views/pdf/PDFMenu.tsx @@ -1,22 +1,43 @@  import React = require("react");  import "./PDFMenu.scss"; -import { observable, action, } from "mobx"; +import { observable, action, computed, } from "mobx";  import { observer } from "mobx-react";  import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { unimplementedFunction, returnFalse } from "../../../Utils"; +import { unimplementedFunction, returnFalse, Utils } from "../../../Utils";  import AntimodeMenu from "../AntimodeMenu";  import { Doc, Opt } from "../../../fields/Doc"; +import { ColorState } from "react-color"; +import { ButtonDropdown } from "../nodes/formattedText/RichTextMenu"; +  @observer  export default class PDFMenu extends AntimodeMenu {      static Instance: PDFMenu;      private _commentCont = React.createRef<HTMLButtonElement>(); +    private _palette = [ +        "rgba(208, 2, 27, 0.8)", +        "rgba(238, 0, 0, 0.8)", +        "rgba(245, 166, 35, 0.8)", +        "rgba(248, 231, 28, 0.8)", +        "rgba(245, 230, 95, 0.616)", +        "rgba(139, 87, 42, 0.8)", +        "rgba(126, 211, 33, 0.8)", +        "rgba(65, 117, 5, 0.8)", +        "rgba(144, 19, 254, 0.8)", +        "rgba(238, 169, 184, 0.8)", +        "rgba(224, 187, 228, 0.8)", +        "rgba(225, 223, 211, 0.8)", +        "rgba(255, 255, 255, 0.8)", +        "rgba(155, 155, 155, 0.8)", +        "rgba(0, 0, 0, 0.8)"];      @observable private _keyValue: string = "";      @observable private _valueValue: string = "";      @observable private _added: boolean = false; +    @observable private highlightColor: string = "rgba(245, 230, 95, 0.616)"; +    @observable public _colorBtn = false;      @observable public Highlighting: boolean = false;      @observable public Status: "pdf" | "annotation" | "" = ""; @@ -70,11 +91,47 @@ export default class PDFMenu extends AntimodeMenu {      @action      highlightClicked = (e: React.MouseEvent) => { -        if (!this.Highlight("rgba(245, 230, 95, 0.616)") && this.Pinned) { // yellowish highlight color for a marker type highlight +        if (!this.Highlight(this.highlightColor) && this.Pinned) {              this.Highlighting = !this.Highlighting;          }      } +    @computed get highlighter() { +        const button = +            <button className="antimodeMenu-button color-preview-button" title="" key="highilghter-button" onPointerDown={this.highlightClicked}> +                <FontAwesomeIcon icon="highlighter" size="lg" style={{ transition: "transform 0.1s", transform: this.Highlighting ? "" : "rotate(-45deg)" }} /> +                <div className="color-preview" style={{ backgroundColor: this.highlightColor }}></div> +            </button>; + +        const dropdownContent = +            <div className="dropdown"> +                <p>Change highlighter color:</p> +                <div className="color-wrapper"> +                    {this._palette.map(color => { +                        if (color) { +                            return this.highlightColor === color ? +                                <button className="color-button active" key={`active ${color}`} style={{ backgroundColor: color }} onPointerDown={e => this.changeHighlightColor(color, e)}></button> : +                                <button className="color-button" key={`inactive ${color}`} style={{ backgroundColor: color }} onPointerDown={e => this.changeHighlightColor(color, e)}></button>; +                        } +                    })} +                </div> +            </div>; +        return ( +            <ButtonDropdown key={"highlighter"} button={button} dropdownContent={dropdownContent} /> +        ); +    } + +    @action +    changeHighlightColor = (color: string, e: React.PointerEvent) => { +        const col: ColorState = { +            hex: color, hsl: { a: 0, h: 0, s: 0, l: 0, source: "" }, hsv: { a: 0, h: 0, s: 0, v: 0, source: "" }, +            rgb: { a: 0, r: 0, b: 0, g: 0, source: "" }, oldHue: 0, source: "", +        }; +        e.preventDefault(); +        e.stopPropagation(); +        this.highlightColor = Utils.colorString(col); +    } +      deleteClicked = (e: React.PointerEvent) => {          this.Delete();      } @@ -101,12 +158,11 @@ export default class PDFMenu extends AntimodeMenu {      render() {          const buttons = this.Status === "pdf" ?              [ -                <button key="1" className="antimodeMenu-button" title="Click to Highlight" onClick={this.highlightClicked} style={this.Highlighting ? { backgroundColor: "#121212" } : {}}> -                    <FontAwesomeIcon icon="highlighter" size="lg" style={{ transition: "transform 0.1s", transform: this.Highlighting ? "" : "rotate(-45deg)" }} /></button>, +                this.highlighter,                  <button key="2" className="antimodeMenu-button" title="Drag to Annotate" ref={this._commentCont} onPointerDown={this.pointerDown}>                      <FontAwesomeIcon icon="comment-alt" size="lg" /></button>,                  <button key="4" className="antimodeMenu-button" title="Pin Menu" onClick={this.togglePin} style={this.Pinned ? { backgroundColor: "#121212" } : {}}> -                    <FontAwesomeIcon icon="thumbtack" size="lg" style={{ transition: "transform 0.1s", transform: this.Pinned ? "rotate(45deg)" : "" }} /> </button> +                    <FontAwesomeIcon icon="thumbtack" size="lg" style={{ transition: "transform 0.1s", transform: this.Pinned ? "rotate(45deg)" : "" }} /></button>,              ] : [                  <button key="5" className="antimodeMenu-button" title="Delete Anchor" onPointerDown={this.deleteClicked}>                      <FontAwesomeIcon icon="trash-alt" size="lg" /></button>, diff --git a/src/client/views/pdf/PDFViewer.scss b/src/client/views/pdf/PDFViewer.scss index cfe0b3d4b..86c73bfee 100644 --- a/src/client/views/pdf/PDFViewer.scss +++ b/src/client/views/pdf/PDFViewer.scss @@ -12,14 +12,14 @@      //     transform-origin: top left;      // }      .textLayer { -         +        opacity: unset;          mix-blend-mode: multiply;// bcz: makes text fuzzy!          span {              padding-right: 5px;              padding-bottom: 4px;          }      } -    .textLayer ::selection { background: yellow; } // should match the backgroundColor in createAnnotation() +    .textLayer ::selection { background: #ACCEF7; } // should match the backgroundColor in createAnnotation()      .textLayer .highlight {          background-color: yellow;      } diff --git a/src/client/views/pdf/PDFViewer.tsx b/src/client/views/pdf/PDFViewer.tsx index 8185dd67f..98f64edec 100644 --- a/src/client/views/pdf/PDFViewer.tsx +++ b/src/client/views/pdf/PDFViewer.tsx @@ -115,8 +115,8 @@ export class PDFViewer extends ViewBoxAnnotatableComponent<IViewerProps, PdfDocu      private _downX: number = 0;      private _downY: number = 0;      private _coverPath: any; +    private _lastSearch = false;      private _viewerIsSetup = false; -    private _lastSearch: string = "";      @computed get allAnnotations() {          return DocListCast(this.dataDoc[this.props.fieldKey + "-annotations"]). @@ -150,27 +150,11 @@ export class PDFViewer extends ViewBoxAnnotatableComponent<IViewerProps, PdfDocu          runInAction(() => this._showWaiting = this._showCover = true);          this.props.startupLive && this.setupPdfJsViewer();          this._mainCont.current && (this._mainCont.current.scrollTop = this.layoutDoc._scrollTop || 0); -        this._searchReactionDisposer = reaction(() => this.Document.searchMatch, search => { -            if (search) { -                this.search(Doc.SearchQuery(), false); -                this._lastSearch = Doc.SearchQuery(); -            } -            else { -                setTimeout(() => this._lastSearch === "mxytzlaf" && this.search("mxytzlaf", true), 200); // bcz: how do we clear search highlights? -                this._lastSearch && (this._lastSearch = "mxytzlaf"); -            } -        }, { fireImmediately: true }); - -        this._searchReactionDisposer2 = reaction(() => this.Document.searchMatch2, search => { -            if (search) { -                this.search(Doc.SearchQuery(), true); -                this._lastSearch = Doc.SearchQuery(); -            } -            else { -                setTimeout(() => this._lastSearch === "mxytzlaf" && this.search("mxytzlaf", true), 200); // bcz: how do we clear search highlights? -                this._lastSearch && (this._lastSearch = "mxytzlaf"); -            } -        }, { fireImmediately: true }); +        this._searchReactionDisposer = reaction(() => this.Document.searchMatch, +            m => { +                if (m) (this._lastSearch = true) && this.search(Doc.SearchQuery(), true); +                else !(this._lastSearch = false) && setTimeout(() => !this._lastSearch && this.search("", false, true), 200); +            }, { fireImmediately: true });          this._selectionReactionDisposer = reaction(() => this.props.isSelected(),              selected => { @@ -305,7 +289,7 @@ export class PDFViewer extends ViewBoxAnnotatableComponent<IViewerProps, PdfDocu          let minY = Number.MAX_VALUE;          if ((this._savedAnnotations.values()[0][0] as any).marqueeing) {              const anno = this._savedAnnotations.values()[0][0]; -            const annoDoc = Docs.Create.FreeformDocument([], { backgroundColor: color, _LODdisable: true, title: "Annotation on " + this.Document.title }); +            const annoDoc = Docs.Create.FreeformDocument([], { backgroundColor: color, title: "Annotation on " + this.Document.title });              if (anno.style.left) annoDoc.x = parseInt(anno.style.left);              if (anno.style.top) annoDoc.y = parseInt(anno.style.top);              if (anno.style.height) annoDoc._height = parseInt(anno.style.height); @@ -399,7 +383,7 @@ export class PDFViewer extends ViewBoxAnnotatableComponent<IViewerProps, PdfDocu                  div.style.top = (parseInt(div.style.top)/*+ this.getScrollFromPage(page)*/).toString();              }              this._annotationLayer.current.append(div); -            div.style.backgroundColor = "yellow"; +            div.style.backgroundColor = "#ACCEF7";              div.style.opacity = "0.5";              const savedPage = this._savedAnnotations.getValue(page);              if (savedPage) { @@ -413,11 +397,12 @@ export class PDFViewer extends ViewBoxAnnotatableComponent<IViewerProps, PdfDocu      }      @action -    search = (searchString: string, fwd: boolean) => { -        if (!searchString) { +    search = (searchString: string, fwd: boolean, clear: boolean = false) => { +        if (clear) { +            this._pdfViewer.findController.executeCommand('reset', {}); +        } else if (!searchString) {              fwd ? this.nextAnnotation() : this.prevAnnotation(); -        } -        else if (this._pdfViewer.pageViewsReady) { +        } else if (this._pdfViewer.pageViewsReady) {              this._pdfViewer.findController.executeCommand('findagain', {                  caseSensitive: false,                  findPrevious: !fwd, @@ -686,7 +671,7 @@ export class PDFViewer extends ViewBoxAnnotatableComponent<IViewerProps, PdfDocu      panelWidth = () => (this.Document.scrollHeight || this.Document._nativeHeight || 0);      panelHeight = () => this._pageSizes.length && this._pageSizes[0] ? this._pageSizes[0].width : (this.Document._nativeWidth || 0);      @computed get overlayLayer() { -        return <div className={`pdfViewer-overlay${Doc.GetSelectedTool() !== InkTool.None || SnappingManager.GetIsDragging() ? "-inking" : ""}`} id="overlay" +        return <div className={`pdfViewerDash-overlay${Doc.GetSelectedTool() !== InkTool.None || SnappingManager.GetIsDragging() ? "-inking" : ""}`} id="overlay"              style={{ transform: `scale(${this._zoomed})` }}>              <CollectionFreeFormView {...this.props}                  LibraryPath={this.props.ContainingCollectionView?.props.LibraryPath ?? emptyPath} diff --git a/src/client/views/search/SearchBox.tsx b/src/client/views/search/SearchBox.tsx index d1e1818c2..5960a0502 100644 --- a/src/client/views/search/SearchBox.tsx +++ b/src/client/views/search/SearchBox.tsx @@ -21,7 +21,7 @@ import { DocumentView } from '../nodes/DocumentView';  import { SelectionManager } from '../../util/SelectionManager';  import { FilterQuery } from 'mongodb';  import { CollectionLinearView } from '../collections/CollectionLinearView'; -import { CurrentUserUtils } from  '../../util/CurrentUserUtils'; +import { CurrentUserUtils } from '../../util/CurrentUserUtils';  import { CollectionDockingView } from '../collections/CollectionDockingView';  import { ScriptField } from '../../../fields/ScriptField'; @@ -30,7 +30,7 @@ import { List } from '../../../fields/List';  import { faSearch, faFilePdf, faFilm, faImage, faObjectGroup, faStickyNote, faMusic, faLink, faChartBar, faGlobeAsia, faBan, faVideo, faCaretDown } from '@fortawesome/free-solid-svg-icons';  import { Transform } from '../../util/Transform';  import { MainView } from "../MainView"; -import { Scripting,_scriptingGlobals } from '../../util/Scripting'; +import { Scripting, _scriptingGlobals } from '../../util/Scripting';  import { CollectionView, CollectionViewType } from '../collections/CollectionView';  import { ViewBoxBaseComponent } from "../DocComponent";  import { documentSchema } from "../../../fields/documentSchemas"; @@ -54,11 +54,11 @@ export enum Keys {      DATA = "data"  } -export interface filterData{ +export interface filterData {      deletedDocsStatus: boolean;      authorFieldStatus: boolean; -    titleFieldStatus:boolean; -    basicWordStatus:boolean; +    titleFieldStatus: boolean; +    basicWordStatus: boolean;      icons: string[];  } @@ -70,7 +70,7 @@ const SearchBoxDocument = makeInterface(documentSchema, searchSchema);  export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDocument>(SearchBoxDocument) {      @computed get _searchString() { return this.layoutDoc.searchQuery; } -    @computed set _searchString(value) { this.layoutDoc.searchQuery=(value); } +    @computed set _searchString(value) { this.layoutDoc.searchQuery = (value); }      @observable private _resultsOpen: boolean = false;      @observable private _searchbarOpen: boolean = false;      @observable private _results: [Doc, string[], string[]][] = []; @@ -94,7 +94,7 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc      private _curRequest?: Promise<any> = undefined;      public static LayoutString(fieldKey: string) { return FieldView.LayoutString(SearchBox, fieldKey); } -    private new_buckets: { [characterName: string]: number} = {}; +    private new_buckets: { [characterName: string]: number } = {};      //if true, any keywords can be used. if false, all keywords are required.      //this also serves as an indicator if the word status filter is applied      @observable private _basicWordStatus: boolean = false; @@ -104,36 +104,38 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc      @observable private newAssign: boolean = true;      constructor(props: any) { -         +          super(props);          SearchBox.Instance = this; -        if (!_scriptingGlobals.hasOwnProperty("handleNodeChange")){ -        Scripting.addGlobal(this.handleNodeChange); +        if (!_scriptingGlobals.hasOwnProperty("handleNodeChange")) { +            Scripting.addGlobal(this.handleNodeChange);          } -        if (!_scriptingGlobals.hasOwnProperty("handleKeyChange")){ +        if (!_scriptingGlobals.hasOwnProperty("handleKeyChange")) {              Scripting.addGlobal(this.handleKeyChange);          } -        if (!_scriptingGlobals.hasOwnProperty("handleWordQueryChange")){ +        if (!_scriptingGlobals.hasOwnProperty("handleWordQueryChange")) {              Scripting.addGlobal(this.handleWordQueryChange);          } -        if (!_scriptingGlobals.hasOwnProperty("updateIcon")){ +        if (!_scriptingGlobals.hasOwnProperty("updateIcon")) {              Scripting.addGlobal(this.updateIcon);          } -        if (!_scriptingGlobals.hasOwnProperty("updateTitleStatus")){ +        if (!_scriptingGlobals.hasOwnProperty("updateTitleStatus")) {              Scripting.addGlobal(this.updateTitleStatus);          } -        if (!_scriptingGlobals.hasOwnProperty("updateAuthorStatus")){ +        if (!_scriptingGlobals.hasOwnProperty("updateAuthorStatus")) {              Scripting.addGlobal(this.updateAuthorStatus);          } -        if (!_scriptingGlobals.hasOwnProperty("updateDeletedStatus")){ +        if (!_scriptingGlobals.hasOwnProperty("updateDeletedStatus")) {              Scripting.addGlobal(this.updateDeletedStatus);          }          this.resultsScrolled = this.resultsScrolled.bind(this); -       new PrefetchProxy(Docs.Create.SearchItemBoxDocument({ title: "search item template",  -       backgroundColor: "transparent", _xMargin: 5, _height: 46, isTemplateDoc: true, isTemplateForField: "data" })); +        new PrefetchProxy(Docs.Create.SearchItemBoxDocument({ +            title: "search item template", +            backgroundColor: "transparent", _xMargin: 5, _height: 46, isTemplateDoc: true, isTemplateForField: "data" +        }));          if (!this.searchItemTemplate) { // create exactly one presElmentBox template to use by any and all presentations. @@ -146,28 +148,28 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc                  { field: "string", data: Doc.name, container: Doc.name });          }      } -    @observable setupButtons =false; +    @observable setupButtons = false;      componentDidMount = () => { -        if (this.setupButtons==false){ +        if (this.setupButtons == false) {              this.setupDocTypeButtons();              this.setupKeyButtons();              this.setupDefaultButtons(); -        runInAction(()=>this.setupButtons==true); -    } +            runInAction(() => this.setupButtons == true); +        }          if (this.inputRef.current) {              this.inputRef.current.focus(); -            runInAction( () => {this._searchbarOpen = true}); +            runInAction(() => { this._searchbarOpen = true });          } -        if (this.rootDoc.searchQuery&& this.newAssign) { +        if (this.rootDoc.searchQuery && this.newAssign) {              const sq = this.rootDoc.searchQuery;              runInAction(() => { -            // this._deletedDocsStatus=this.props.filterQuery!.deletedDocsStatus; -            // this._authorFieldStatus=this.props.filterQuery!.authorFieldStatus -            // this._titleFieldStatus=this.props.filterQuery!.titleFieldStatus; -            // this._basicWordStatus=this.props.filterQuery!.basicWordStatus; -            // this._icons=this.props.filterQuery!.icons; -            this.newAssign=false; +                // this._deletedDocsStatus=this.props.filterQuery!.deletedDocsStatus; +                // this._authorFieldStatus=this.props.filterQuery!.authorFieldStatus +                // this._titleFieldStatus=this.props.filterQuery!.titleFieldStatus; +                // this._basicWordStatus=this.props.filterQuery!.basicWordStatus; +                // this._icons=this.props.filterQuery!.icons; +                this.newAssign = false;              });              runInAction(() => {                  this.layoutDoc._searchString = StrCast(sq); @@ -196,8 +198,8 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc      enter = (e: React.KeyboardEvent) => {          if (e.key === "Enter") { -            if (this._icons!==this._allIcons){ -            runInAction(()=>{this.expandedBucket=false}); +            if (this._icons !== this._allIcons) { +                runInAction(() => { this.expandedBucket = false });              }              this.submitSearch();          } @@ -271,21 +273,21 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc      @action      filterDocsByType(docs: Doc[]) {          const finalDocs: Doc[] = []; -        const blockedTypes:string[]= ["preselement","docholder","collection","search","searchitem", "script", "fonticonbox", "button", "label"]; +        const blockedTypes: string[] = ["preselement", "docholder", "collection", "search", "searchitem", "script", "fonticonbox", "button", "label"];          docs.forEach(doc => {              const layoutresult = Cast(doc.type, "string"); -            if (layoutresult && !blockedTypes.includes(layoutresult)){ -            if (layoutresult && this._icons.includes(layoutresult)) { -                finalDocs.push(doc); +            if (layoutresult && !blockedTypes.includes(layoutresult)) { +                if (layoutresult && this._icons.includes(layoutresult)) { +                    finalDocs.push(doc); +                }              } -        }          });          return finalDocs;      }      addCollectionFilter(query: string): string {          const collections: Doc[] = this.getCurCollections(); -         +          console.log(collections);          const oldWords = query.split(" "); @@ -374,31 +376,31 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc      get fieldFiltersApplied() { return !(this._authorFieldStatus && this._titleFieldStatus); } -    @observable expandedBucket:boolean=false; +    @observable expandedBucket: boolean = false;      @action -    submitSearch = async (reset?:boolean) => { +    submitSearch = async (reset?: boolean) => {          this.checkIcons(); -        if (reset){ -            this.layoutDoc._searchString=""; +        if (reset) { +            this.layoutDoc._searchString = "";          }          this.dataDoc[this.fieldKey] = new List<Doc>([]); -        this.buckets=[]; -        this.new_buckets={}; +        this.buckets = []; +        this.new_buckets = {};          const query = StrCast(this.layoutDoc._searchString);          this.getFinalQuery(query);          this._results = [];          this._resultsSet.clear();          this._isSearch = []; -        this._isSorted=[]; +        this._isSorted = [];          this._visibleElements = [];          this._visibleDocuments = []; -        if (StrCast(this.props.Document.searchQuery)){ -            if (this._timeout){clearTimeout(this._timeout); this._timeout=undefined}; -            this._timeout= setTimeout(()=>{ +        if (StrCast(this.props.Document.searchQuery)) { +            if (this._timeout) { clearTimeout(this._timeout); this._timeout = undefined }; +            this._timeout = setTimeout(() => {                  console.log("Resubmitting search");                  this.submitSearch();              }, 60000); -            } +        }          if (query !== "") {              this._endIndex = 12; @@ -414,84 +416,84 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc              });          }      } -    @observable _timeout:any=undefined; -     +    @observable _timeout: any = undefined; +      @observable firststring: string = "";      @observable secondstring: string = ""; -    @observable bucketcount:number[]=[]; +    @observable bucketcount: number[] = []; -    @action private makenewbuckets(){ +    @action private makenewbuckets() {          console.log("new"); -        let highcount=0; -        let secondcount=0; -        this.firststring=""; -        this.secondstring=""; -        this.buckets=[]; -        this.bucketcount=[]; +        let highcount = 0; +        let secondcount = 0; +        this.firststring = ""; +        this.secondstring = ""; +        this.buckets = []; +        this.bucketcount = [];          this.dataDoc[this.fieldKey] = new List<Doc>([]); -        for (var key in this.new_buckets){ -            if (this.new_buckets[key]>highcount){ -                secondcount===highcount; -                this.secondstring=this.firststring; -                highcount=this.new_buckets[key]; -                this.firststring= key; +        for (var key in this.new_buckets) { +            if (this.new_buckets[key] > highcount) { +                secondcount === highcount; +                this.secondstring = this.firststring; +                highcount = this.new_buckets[key]; +                this.firststring = key;              } -            else if (this.new_buckets[key]>secondcount){ -                secondcount=this.new_buckets[key]; -                this.secondstring= key; +            else if (this.new_buckets[key] > secondcount) { +                secondcount = this.new_buckets[key]; +                this.secondstring = key;              }          } -        let bucket = Docs.Create.StackingDocument([],{ _viewType:CollectionViewType.Stacking, ignoreClick: true, forceActive: true, lockedPosition: true,title: `default bucket`}); +        let bucket = Docs.Create.StackingDocument([], { _viewType: CollectionViewType.Stacking, ignoreClick: true, forceActive: true, lockedPosition: true, title: `default bucket` });          bucket._viewType === CollectionViewType.Stacking; -        bucket._height=185; +        bucket._height = 185;          bucket.bucketfield = "results"; -        bucket.isBucket=true; +        bucket.isBucket = true;          Doc.AddDocToList(this.dataDoc, this.props.fieldKey, bucket);          this.buckets!.push(bucket); -        this.bucketcount[0]=0; -         -        if (this.firststring!==""){ -        let firstbucket = Docs.Create.StackingDocument([],{ _viewType:CollectionViewType.Stacking, ignoreClick: true, forceActive: true, lockedPosition: true, title: this.firststring }); -        firstbucket._height=185; - -        firstbucket._viewType === CollectionViewType.Stacking; -        firstbucket.bucketfield = this.firststring; -        firstbucket.isBucket=true; -        Doc.AddDocToList(this.dataDoc, this.props.fieldKey, firstbucket); -        this.buckets!.push(firstbucket); -        this.bucketcount[1]=0; +        this.bucketcount[0] = 0; + +        if (this.firststring !== "") { +            let firstbucket = Docs.Create.StackingDocument([], { _viewType: CollectionViewType.Stacking, ignoreClick: true, forceActive: true, lockedPosition: true, title: this.firststring }); +            firstbucket._height = 185; + +            firstbucket._viewType === CollectionViewType.Stacking; +            firstbucket.bucketfield = this.firststring; +            firstbucket.isBucket = true; +            Doc.AddDocToList(this.dataDoc, this.props.fieldKey, firstbucket); +            this.buckets!.push(firstbucket); +            this.bucketcount[1] = 0;          } -        if (this.secondstring!==""){ -        let secondbucket = Docs.Create.StackingDocument([],{ _viewType:CollectionViewType.Stacking, ignoreClick: true, forceActive: true, lockedPosition: true, title: this.secondstring }); -        secondbucket._height=185; -        secondbucket._viewType === CollectionViewType.Stacking; -        secondbucket.bucketfield = this.secondstring; -        secondbucket.isBucket=true; -        Doc.AddDocToList(this.dataDoc, this.props.fieldKey, secondbucket); -        this.buckets!.push(secondbucket); -        this.bucketcount[2]=0; +        if (this.secondstring !== "") { +            let secondbucket = Docs.Create.StackingDocument([], { _viewType: CollectionViewType.Stacking, ignoreClick: true, forceActive: true, lockedPosition: true, title: this.secondstring }); +            secondbucket._height = 185; +            secondbucket._viewType === CollectionViewType.Stacking; +            secondbucket.bucketfield = this.secondstring; +            secondbucket.isBucket = true; +            Doc.AddDocToList(this.dataDoc, this.props.fieldKey, secondbucket); +            this.buckets!.push(secondbucket); +            this.bucketcount[2] = 0;          } -        let webbucket = Docs.Create.StackingDocument([],{ _viewType:CollectionViewType.Stacking, childDropAction: "alias", ignoreClick: true, lockedPosition: true, title: this.secondstring }); -        webbucket._height=185; +        let webbucket = Docs.Create.StackingDocument([], { _viewType: CollectionViewType.Stacking, childDropAction: "alias", ignoreClick: true, lockedPosition: true, title: this.secondstring }); +        webbucket._height = 185;          webbucket._viewType === CollectionViewType.Stacking;          webbucket.bucketfield = "webs"; -        webbucket.isBucket=true; -        let old = Cast(this.props.Document.webbucket,Doc) as Doc; -        let old2=Cast(this.props.Document.bing,Doc) as Doc; -        if (old){ +        webbucket.isBucket = true; +        let old = Cast(this.props.Document.webbucket, Doc) as Doc; +        let old2 = Cast(this.props.Document.bing, Doc) as Doc; +        if (old) {              console.log("Cleanup"); -            Doc.RemoveDocFromList(old, this.props.fieldKey,old2); +            Doc.RemoveDocFromList(old, this.props.fieldKey, old2);          }          const textDoc = Docs.Create.WebDocument(`https://bing.com/search?q=${this.layoutDoc._searchString}`, { -                    _width: 200,  _nativeHeight: 962, _nativeWidth: 800, isAnnotating: false, -                    title: "bing", UseCors: true -                }); -        this.props.Document.bing=textDoc; +            _width: 200, _nativeHeight: 962, _nativeWidth: 800, isAnnotating: false, +            title: "bing", UseCors: true +        }); +        this.props.Document.bing = textDoc;          this.props.Document.webbucket = webbucket;          Doc.AddDocToList(this.dataDoc, this.props.fieldKey, webbucket);          Doc.AddDocToList(webbucket, this.props.fieldKey, textDoc); @@ -500,7 +502,7 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc      } -    @observable buckets:Doc[]|undefined; +    @observable buckets: Doc[] | undefined;      getAllResults = async (query: string) => {          return SearchUtil.Search(query, true, { fq: this.filterQuery, start: 0, rows: 10000000 }); @@ -515,7 +517,7 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc          // fq: type_t:collection OR {!join from=id to=proto_i}type_t:collection   q:text_t:hello          const query = [baseExpr, includeDeleted, includeIcons].join(" AND ").replace(/AND $/, "");          return query; -        } +    }      getDataStatus() { return this._deletedDocsStatus; } @@ -529,7 +531,7 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc          }          this.lockPromise = new Promise(async res => {              while (this._results.length <= this._endIndex && (this._numTotalResults === -1 || this._maxSearchIndex < this._numTotalResults)) { -                this._curRequest = SearchUtil.Search(query, true, {fq: this.filterQuery, start: this._maxSearchIndex, rows: this.NumResults, hl: true, "hl.fl": "*",}).then(action(async (res: SearchUtil.DocSearchResult) => { +                this._curRequest = SearchUtil.Search(query, true, { fq: this.filterQuery, start: this._maxSearchIndex, rows: this.NumResults, hl: true, "hl.fl": "*", }).then(action(async (res: SearchUtil.DocSearchResult) => {                      // happens at the beginning                      if (res.numFound !== this._numTotalResults && this._numTotalResults === -1) {                          this._numTotalResults = res.numFound; @@ -542,34 +544,34 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc                      const highlights: typeof res.highlighting = {};                      docs.forEach((doc, index) => highlights[doc[Id]] = highlightList[index]);                      const filteredDocs = this.filterDocsByType(docs); -                     +                      runInAction(() => { -                        filteredDocs.forEach((doc,i) => { +                        filteredDocs.forEach((doc, i) => {                              const index = this._resultsSet.get(doc);                              const highlight = highlights[doc[Id]];                              const line = lines.get(doc[Id]) || [];                              const hlights = highlight ? Object.keys(highlight).map(key => key.substring(0, key.length - 2)) : []; -                            doc? console.log(Cast(doc.context, Doc)) : null; -                            if (this.findCommonElements(hlights)){ +                            doc ? console.log(Cast(doc.context, Doc)) : null; +                            if (this.findCommonElements(hlights)) {                              } -                            else{ +                            else {                                  const layoutresult = Cast(doc.type, "string"); -                                if (layoutresult){ -                                if(this.new_buckets[layoutresult]===undefined){ -                                    this.new_buckets[layoutresult]=1; +                                if (layoutresult) { +                                    if (this.new_buckets[layoutresult] === undefined) { +                                        this.new_buckets[layoutresult] = 1; +                                    } +                                    else { +                                        this.new_buckets[layoutresult] = this.new_buckets[layoutresult] + 1; +                                    }                                  } -                                else { -                                    this.new_buckets[layoutresult]=this.new_buckets[layoutresult]+1; +                                if (index === undefined) { +                                    this._resultsSet.set(doc, this._results.length); +                                    this._results.push([doc, hlights, line]); +                                } else { +                                    this._results[index][1].push(...hlights); +                                    this._results[index][2].push(...line);                                  }                              } -                            if (index === undefined) { -                                this._resultsSet.set(doc, this._results.length); -                                this._results.push([doc, hlights, line]); -                            } else { -                                this._results[index][1].push(...hlights); -                                this._results[index][2].push(...line); -                            } -                        }                          });                      }); @@ -579,8 +581,8 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc                  await this._curRequest;              } -            if (this._numTotalResults>3 && this.expandedBucket===false){ -            this.makenewbuckets(); +            if (this._numTotalResults > 3 && this.expandedBucket === false) { +                this.makenewbuckets();              }              this.resultsScrolled();              res(); @@ -590,9 +592,9 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc      collectionRef = React.createRef<HTMLSpanElement>();      startDragCollection = async () => { -    const res = await this.getAllResults(this.getFinalQuery(StrCast(this.layoutDoc._searchString))); -       const filtered = this.filterDocsByType(res.docs); -       const docs = filtered.map(doc => { +        const res = await this.getAllResults(this.getFinalQuery(StrCast(this.layoutDoc._searchString))); +        const filtered = this.filterDocsByType(res.docs); +        const docs = filtered.map(doc => {              const isProto = Doc.GetT(doc, "isPrototype", "boolean", true);              if (isProto) {                  return Doc.MakeDelegate(doc); @@ -623,14 +625,14 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc                  y += 300;              }          } -        const filter : filterData = { +        const filter: filterData = {              deletedDocsStatus: this._deletedDocsStatus,              authorFieldStatus: this._authorFieldStatus,              titleFieldStatus: this._titleFieldStatus,              basicWordStatus: this._basicWordStatus,              icons: this._icons,          } -        return Docs.Create.SearchDocument({ _autoHeight: true, _viewType: CollectionViewType.Stacking , title: StrCast(this.layoutDoc._searchString), searchQuery: StrCast(this.layoutDoc._searchString) }); +        return Docs.Create.SearchDocument({ _autoHeight: true, _viewType: CollectionViewType.Stacking, title: StrCast(this.layoutDoc._searchString), searchQuery: StrCast(this.layoutDoc._searchString) });      }      @action.bound @@ -653,7 +655,7 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc          this._results = [];          this._resultsSet.clear();          this._visibleElements = []; -        this._visibleDocuments=[]; +        this._visibleDocuments = [];          this._numTotalResults = -1;          this._endIndex = -1;          this._curRequest = undefined; @@ -667,14 +669,14 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc          const itemHght = 53;          const startIndex = Math.floor(Math.max(0, scrollY / itemHght));          //const endIndex = Math.ceil(Math.min(this._numTotalResults - 1, startIndex + (this._resultsRef.current.getBoundingClientRect().height / itemHght))); -        const endIndex= 30; +        const endIndex = 30;          this._endIndex = endIndex === -1 ? 12 : endIndex; -        this._endIndex=30; +        this._endIndex = 30;          if ((this._numTotalResults === 0 || this._results.length === 0) && this._openNoResults) {              this._visibleElements = [<div className="no-result">No Search Results</div>];              //this._visibleDocuments= Docs.Create. -            let noResult= Docs.Create.TextDocument("",{title:"noResult"}) -            noResult.isBucket =false; +            let noResult = Docs.Create.TextDocument("", { title: "noResult" }) +            noResult.isBucket = false;              Doc.AddDocToList(this.dataDoc, this.props.fieldKey, noResult);              return;          } @@ -700,7 +702,7 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc              if (i < startIndex || i > endIndex) {                  if (this._isSearch[i] !== "placeholder") {                      this._isSearch[i] = "placeholder"; -                    this._isSorted[i]="placeholder"; +                    this._isSorted[i] = "placeholder";                      this._visibleElements[i] = <div className="searchBox-placeholder" key={`searchBox-placeholder-${i}`}>Loading...</div>;                  }              } @@ -713,92 +715,92 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc                          if (result) {                              const highlights = Array.from([...Array.from(new Set(result[1]).values())]);                              let lines = new List<string>(result[2]); -                            result[0]._height=46; -                            result[0].lines=lines; -                            result[0].highlighting=highlights.join(", "); +                            result[0]._height = 46; +                            result[0].lines = lines; +                            result[0].highlighting = highlights.join(", ");                              this._visibleDocuments[i] = result[0]; -                            this._isSearch[i] = "search";            -                            if (this._numTotalResults>3 && this.expandedBucket===false){ +                            this._isSearch[i] = "search"; +                            if (this._numTotalResults > 3 && this.expandedBucket === false) {                                  let doctype = StrCast(result[0].type);                                  console.log(doctype); -                                if (doctype=== this.firststring){ -                                if (this.bucketcount[1]<3){ -                                result[0].parent= this.buckets![1]; -                                Doc.AddDocToList(this.buckets![1], this.props.fieldKey, result[0]); -                                this.bucketcount[1]+=1; +                                if (doctype === this.firststring) { +                                    if (this.bucketcount[1] < 3) { +                                        result[0].parent = this.buckets![1]; +                                        Doc.AddDocToList(this.buckets![1], this.props.fieldKey, result[0]); +                                        this.bucketcount[1] += 1; +                                    }                                  } +                                else if (doctype === this.secondstring) { +                                    if (this.bucketcount[2] < 3) { +                                        result[0].parent = this.buckets![2]; +                                        Doc.AddDocToList(this.buckets![2], this.props.fieldKey, result[0]); +                                        this.bucketcount[2] += 1; +                                    }                                  } -                                else if (doctype=== this.secondstring){ -                                if (this.bucketcount[2]<3){ -                                result[0].parent= this.buckets![2]; -                                Doc.AddDocToList(this.buckets![2], this.props.fieldKey, result[0]); -                                this.bucketcount[2]+=1; +                                else if (this.bucketcount[0] < 3) { +                                    //Doc.AddDocToList(this.buckets![0], this.props.fieldKey, result[0]); +                                    //this.bucketcount[0]+=1; +                                    Doc.AddDocToList(this.dataDoc, this.props.fieldKey, result[0]);                                  } -                                } -                                else if (this.bucketcount[0]<3){ -                                //Doc.AddDocToList(this.buckets![0], this.props.fieldKey, result[0]); -                                //this.bucketcount[0]+=1; -                                Doc.AddDocToList(this.dataDoc, this.props.fieldKey, result[0]); -                                }                                  }                              else {                                  Doc.AddDocToList(this.dataDoc, this.props.fieldKey, result[0]);                              } -                    } +                        }                      }                      else {                          result = this._results[i];                          if (result) {                              const highlights = Array.from([...Array.from(new Set(result[1]).values())]);                              let lines = new List<string>(result[2]); -                            result[0]._height=46; -                            result[0].lines= lines; -                            result[0].highlighting=highlights.join(", "); -                            if(i<this._visibleDocuments.length){ -                            this._visibleDocuments[i]=result[0]; -                            this._isSearch[i] = "search"; -                            if (this._numTotalResults>3 && this.expandedBucket===false){ - -                                if (StrCast(result[0].type)=== this.firststring){ -                                if (this.bucketcount[1]<3){ -                                result[0].parent= this.buckets![1]; -                                Doc.AddDocToList(this.buckets![1], this.props.fieldKey, result[0]); -                                this.bucketcount[1]+=1; -                                } -                                } -                                else if (StrCast(result[0].type)=== this.secondstring){ -                                if (this.bucketcount[2]<3){ -                                result[0].parent= this.buckets![2]; -                                Doc.AddDocToList(this.buckets![2], this.props.fieldKey, result[0]); -                                this.bucketcount[2]+=1; +                            result[0]._height = 46; +                            result[0].lines = lines; +                            result[0].highlighting = highlights.join(", "); +                            if (i < this._visibleDocuments.length) { +                                this._visibleDocuments[i] = result[0]; +                                this._isSearch[i] = "search"; +                                if (this._numTotalResults > 3 && this.expandedBucket === false) { + +                                    if (StrCast(result[0].type) === this.firststring) { +                                        if (this.bucketcount[1] < 3) { +                                            result[0].parent = this.buckets![1]; +                                            Doc.AddDocToList(this.buckets![1], this.props.fieldKey, result[0]); +                                            this.bucketcount[1] += 1; +                                        } +                                    } +                                    else if (StrCast(result[0].type) === this.secondstring) { +                                        if (this.bucketcount[2] < 3) { +                                            result[0].parent = this.buckets![2]; +                                            Doc.AddDocToList(this.buckets![2], this.props.fieldKey, result[0]); +                                            this.bucketcount[2] += 1; +                                        } +                                    } +                                    else if (this.bucketcount[0] < 3) { +                                        //Doc.AddDocToList(this.buckets![0], this.props.fieldKey, result[0]); +                                        //this.bucketcount[0]+=1; +                                        Doc.AddDocToList(this.dataDoc, this.props.fieldKey, result[0]); +                                    }                                  } +                                else { +                                    Doc.AddDocToList(this.dataDoc, this.props.fieldKey, result[0]);                                  } -                                else if (this.bucketcount[0]<3){ -                                //Doc.AddDocToList(this.buckets![0], this.props.fieldKey, result[0]); -                                //this.bucketcount[0]+=1; -                                Doc.AddDocToList(this.dataDoc, this.props.fieldKey, result[0]); -                                }     -                            } -                            else { -                                Doc.AddDocToList(this.dataDoc, this.props.fieldKey, result[0]);                              }                          } -                        }                      }                  }              }          } -        if (this._numTotalResults>3 && this.expandedBucket===false){ -        if (this.buckets![0]){ -        this.buckets![0]._height = this.bucketcount[0]*55 + 25; -        } -        if (this.buckets![1]){ -        this.buckets![1]._height = this.bucketcount[1]*55 + 25; -        } -        if (this.buckets![2]){   -        this.buckets![2]._height = this.bucketcount[2]*55 + 25; +        if (this._numTotalResults > 3 && this.expandedBucket === false) { +            if (this.buckets![0]) { +                this.buckets![0]._height = this.bucketcount[0] * 55 + 25; +            } +            if (this.buckets![1]) { +                this.buckets![1]._height = this.bucketcount[1] * 55 + 25; +            } +            if (this.buckets![2]) { +                this.buckets![2]._height = this.bucketcount[2] * 55 + 25; +            }          } -    }          if (this._maxSearchIndex >= this._numTotalResults) {              this._visibleElements.length = this._results.length;              this._visibleDocuments.length = this._results.length; @@ -806,11 +808,11 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc          }      } -    findCommonElements(arr2:string[]) {  -        let arr1= ["layout", "data"]; -        return arr1.some(item => arr2.includes(item))  -    }  -     +    findCommonElements(arr2: string[]) { +        let arr1 = ["layout", "data"]; +        return arr1.some(item => arr2.includes(item)) +    } +      @computed      get resFull() { return this._numTotalResults <= 8; } @@ -819,16 +821,16 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc      //if true, any keywords can be used. if false, all keywords are required.      @action.bound -    handleWordQueryChange =  async() => { +    handleWordQueryChange = async () => {          this._collectionStatus = !this._collectionStatus;          if (this._collectionStatus) {              let doc = await Cast(this.props.Document.keywords, Doc) -            doc!.backgroundColor= "grey"; +            doc!.backgroundColor = "grey";          }          else {              let doc = await Cast(this.props.Document.keywords, Doc) -            doc!.backgroundColor= "black"; +            doc!.backgroundColor = "black";          }      } @@ -839,13 +841,13 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc          if (this._nodeStatus) {              this.expandSection(`node${this.props.Document[Id]}`);              let doc = await Cast(this.props.Document.nodes, Doc) -            doc!.backgroundColor= "grey"; +            doc!.backgroundColor = "grey";          }          else {              this.collapseSection(`node${this.props.Document[Id]}`);              let doc = await Cast(this.props.Document.nodes, Doc) -            doc!.backgroundColor= "black"; +            doc!.backgroundColor = "black";          }      } @@ -855,12 +857,12 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc          if (this._keyStatus) {              this.expandSection(`key${this.props.Document[Id]}`);              let doc = await Cast(this.props.Document.keys, Doc) -            doc!.backgroundColor= "grey"; +            doc!.backgroundColor = "grey";          }          else {              this.collapseSection(`key${this.props.Document[Id]}`);              let doc = await Cast(this.props.Document.keys, Doc) -            doc!.backgroundColor= "black"; +            doc!.backgroundColor = "black";          }      } @@ -949,80 +951,83 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc      }      @action.bound -    updateTitleStatus= async () =>{ this._titleFieldStatus = !this._titleFieldStatus;  -        if (this._titleFieldStatus){ +    updateTitleStatus = async () => { +        this._titleFieldStatus = !this._titleFieldStatus; +        if (this._titleFieldStatus) {              let doc = await Cast(this.props.Document.title, Doc) -            doc!.backgroundColor= "grey"; +            doc!.backgroundColor = "grey";          } -        else{ +        else {              let doc = await Cast(this.props.Document.title, Doc) -            doc!.backgroundColor= "black"; +            doc!.backgroundColor = "black";          }      }      @action.bound -    updateAuthorStatus=async () => { this._authorFieldStatus = !this._authorFieldStatus;  -    if (this._authorFieldStatus){ -        let doc = await Cast(this.props.Document.author, Doc) -        doc!.backgroundColor= "grey"; -    } -    else{ -        let doc = await Cast(this.props.Document.author, Doc) -        doc!.backgroundColor= "black"; -    } +    updateAuthorStatus = async () => { +        this._authorFieldStatus = !this._authorFieldStatus; +        if (this._authorFieldStatus) { +            let doc = await Cast(this.props.Document.author, Doc) +            doc!.backgroundColor = "grey"; +        } +        else { +            let doc = await Cast(this.props.Document.author, Doc) +            doc!.backgroundColor = "black"; +        }      }      @action.bound -    updateDeletedStatus=async() =>{ this._deletedDocsStatus = !this._deletedDocsStatus;  -        if (this._deletedDocsStatus){ +    updateDeletedStatus = async () => { +        this._deletedDocsStatus = !this._deletedDocsStatus; +        if (this._deletedDocsStatus) {              let doc = await Cast(this.props.Document.deleted, Doc) -            doc!.backgroundColor= "grey"; +            doc!.backgroundColor = "grey";          } -        else{ +        else {              let doc = await Cast(this.props.Document.deleted, Doc) -            doc!.backgroundColor= "black"; +            doc!.backgroundColor = "black";          }      }      addButtonDoc = (doc: Doc) => Doc.AddDocToList(CurrentUserUtils.UserDocument.expandingButtons as Doc, "data", doc);      remButtonDoc = (doc: Doc) => Doc.RemoveDocFromList(CurrentUserUtils.UserDocument.expandingButtons as Doc, "data", doc);      moveButtonDoc = (doc: Doc, targetCollection: Doc | undefined, addDocument: (document: Doc) => boolean) => this.remButtonDoc(doc) && addDocument(doc); -     +      @computed get docButtons() {          const nodeBtns = this.props.Document.nodeButtons;          let width = () => NumCast(this.props.Document._width);          // if (StrCast(this.props.Document.title)==="sidebar search stack"){ -            width = MainView.Instance.flyoutWidthFunc; +        width = MainView.Instance.flyoutWidthFunc;          // }             if (nodeBtns instanceof Doc) { -            return <div id="hi" style={{height:"100px",}}> +            return <div id="hi" style={{ height: "100px", }}>                  <DocumentView -                docFilters={returnEmptyFilter} -                Document={nodeBtns} -                DataDoc={undefined} -                LibraryPath={emptyPath} -                addDocument={undefined} -                addDocTab={returnFalse} -                rootSelected={returnTrue} -                pinToPres={emptyFunction} -                onClick={undefined} -                removeDocument={undefined} -                ScreenToLocalTransform={this.getTransform} -                ContentScaling={returnOne} -                PanelWidth={width} -                PanelHeight={() => 100} -                renderDepth={0} -                backgroundColor={returnEmptyString} -                focus={emptyFunction} -                parentActive={returnTrue} -                whenActiveChanged={emptyFunction} -                bringToFront={emptyFunction} -                ContainingCollectionView={undefined} -                ContainingCollectionDoc={undefined} -                NativeHeight={()=>100} -                NativeWidth={width} -            /> +                    docFilters={returnEmptyFilter} +                    Document={nodeBtns} +                    DataDoc={undefined} +                    LibraryPath={emptyPath} +                    addDocument={undefined} +                    addDocTab={returnFalse} +                    rootSelected={returnTrue} +                    pinToPres={emptyFunction} +                    onClick={undefined} +                    removeDocument={undefined} +                    ScreenToLocalTransform={this.getTransform} +                    ContentScaling={returnOne} +                    PanelWidth={width} +                    PanelHeight={() => 100} +                    renderDepth={0} +                    backgroundColor={returnEmptyString} +                    focus={emptyFunction} +                    parentActive={returnTrue} +                    whenActiveChanged={emptyFunction} +                    bringToFront={emptyFunction} +                    ContainingCollectionView={undefined} +                    ContainingCollectionDoc={undefined} +                    NativeHeight={() => 100} +                    NativeWidth={width} +                />              </div>;          }          return (null); @@ -1032,36 +1037,36 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc          const nodeBtns = this.props.Document.keyButtons;          let width = () => NumCast(this.props.Document._width);          // if (StrCast(this.props.Document.title)==="sidebar search stack"){ -            width = MainView.Instance.flyoutWidthFunc; +        width = MainView.Instance.flyoutWidthFunc;          // }          if (nodeBtns instanceof Doc) { -            return <div id="hi" style={{height:"35px",}}> +            return <div id="hi" style={{ height: "35px", }}>                  <DocumentView -                docFilters={returnEmptyFilter} -                Document={nodeBtns} -                DataDoc={undefined} -                LibraryPath={emptyPath} -                addDocument={undefined} -                addDocTab={returnFalse} -                rootSelected={returnTrue} -                pinToPres={emptyFunction} -                onClick={undefined} -                removeDocument={undefined} -                ScreenToLocalTransform={this.getTransform} -                ContentScaling={returnOne} -                PanelWidth={width} -                PanelHeight={() => 100} -                renderDepth={0} -                backgroundColor={returnEmptyString} -                focus={emptyFunction} -                parentActive={returnTrue} -                whenActiveChanged={emptyFunction} -                bringToFront={emptyFunction} -                ContainingCollectionView={undefined} -                ContainingCollectionDoc={undefined} -                NativeHeight={()=>100} -                NativeWidth={width} -            /> +                    docFilters={returnEmptyFilter} +                    Document={nodeBtns} +                    DataDoc={undefined} +                    LibraryPath={emptyPath} +                    addDocument={undefined} +                    addDocTab={returnFalse} +                    rootSelected={returnTrue} +                    pinToPres={emptyFunction} +                    onClick={undefined} +                    removeDocument={undefined} +                    ScreenToLocalTransform={this.getTransform} +                    ContentScaling={returnOne} +                    PanelWidth={width} +                    PanelHeight={() => 100} +                    renderDepth={0} +                    backgroundColor={returnEmptyString} +                    focus={emptyFunction} +                    parentActive={returnTrue} +                    whenActiveChanged={emptyFunction} +                    bringToFront={emptyFunction} +                    ContainingCollectionView={undefined} +                    ContainingCollectionDoc={undefined} +                    NativeHeight={() => 100} +                    NativeWidth={width} +                />              </div>;          }          return (null); @@ -1071,81 +1076,83 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc          const defBtns = this.props.Document.defaultButtons;          let width = () => NumCast(this.props.Document._width);          // if (StrCast(this.props.Document.title)==="sidebar search stack"){ -            width = MainView.Instance.flyoutWidthFunc; +        width = MainView.Instance.flyoutWidthFunc;          // }          if (defBtns instanceof Doc) { -            return <div id="hi" style={{height:"35px",}}> +            return <div id="hi" style={{ height: "35px", }}>                  <DocumentView -                docFilters={returnEmptyFilter} - -                Document={defBtns} -                DataDoc={undefined} -                LibraryPath={emptyPath} -                addDocument={undefined} -                addDocTab={returnFalse} -                rootSelected={returnTrue} -                pinToPres={emptyFunction} -                onClick={undefined} -                removeDocument={undefined} -                ScreenToLocalTransform={this.getTransform} -                ContentScaling={returnOne} -                PanelWidth={width} -                PanelHeight={() => 100} -                renderDepth={0} -                backgroundColor={returnEmptyString} -                focus={emptyFunction} -                parentActive={returnTrue} -                whenActiveChanged={emptyFunction} -                bringToFront={emptyFunction} -                ContainingCollectionView={undefined} -                ContainingCollectionDoc={undefined} -                NativeHeight={()=>100} -                NativeWidth={width} -            /> +                    docFilters={returnEmptyFilter} + +                    Document={defBtns} +                    DataDoc={undefined} +                    LibraryPath={emptyPath} +                    addDocument={undefined} +                    addDocTab={returnFalse} +                    rootSelected={returnTrue} +                    pinToPres={emptyFunction} +                    onClick={undefined} +                    removeDocument={undefined} +                    ScreenToLocalTransform={this.getTransform} +                    ContentScaling={returnOne} +                    PanelWidth={width} +                    PanelHeight={() => 100} +                    renderDepth={0} +                    backgroundColor={returnEmptyString} +                    focus={emptyFunction} +                    parentActive={returnTrue} +                    whenActiveChanged={emptyFunction} +                    bringToFront={emptyFunction} +                    ContainingCollectionView={undefined} +                    ContainingCollectionDoc={undefined} +                    NativeHeight={() => 100} +                    NativeWidth={width} +                />              </div>;          }          return (null);      }      @action.bound -    updateIcon= async (icon: string) =>{ -        if (this._icons.includes(icon)){ +    updateIcon = async (icon: string) => { +        if (this._icons.includes(icon)) {              _.pull(this._icons, icon);              let cap = icon.charAt(0).toUpperCase() + icon.slice(1)              console.log(cap);              let doc = await Cast(this.props.Document[cap], Doc) -            doc!.backgroundColor= "black"; +            doc!.backgroundColor = "black";          } -        else{ +        else {              this._icons.push(icon);              let cap = icon.charAt(0).toUpperCase() + icon.slice(1)              let doc = await Cast(this.props.Document[cap], Doc) -            doc!.backgroundColor= "grey"; +            doc!.backgroundColor = "grey";          }      }      @action.bound -    checkIcons = async ()=>{ -        for (let i=0; i<this._allIcons.length; i++){ -         +    checkIcons = async () => { +        for (let i = 0; i < this._allIcons.length; i++) { +              let cap = this._allIcons[i].charAt(0).toUpperCase() + this._allIcons[i].slice(1)              let doc = await Cast(this.props.Document[cap], Doc) -            if (this._icons.includes(this._allIcons[i])){ -                doc!.backgroundColor= "grey"; +            if (this._icons.includes(this._allIcons[i])) { +                doc!.backgroundColor = "grey";              } -            else{ -                doc!.backgroundColor= "black"; +            else { +                doc!.backgroundColor = "black";              } -    } +        }      }      setupDocTypeButtons() {          let doc = this.props.Document; -        const ficon = (opts: DocumentOptions) => new PrefetchProxy(Docs.Create.FontIconDocument({ ...opts,   -        dropAction: "alias", removeDropProperties: new List<string>(["dropAction"]), _nativeWidth: 100, _nativeHeight: 100, _width: 100, -         _height: 100 })) as any as Doc; +        const ficon = (opts: DocumentOptions) => new PrefetchProxy(Docs.Create.FontIconDocument({ +            ...opts, +            dropAction: "alias", removeDropProperties: new List<string>(["dropAction"]), _nativeWidth: 100, _nativeHeight: 100, _width: 100, +            _height: 100 +        })) as any as Doc;          doc.Audio = ficon({ onClick: ScriptField.MakeScript(`updateIcon("audio")`), title: "music button", icon: "music" }); -        doc.Collection  = ficon({ onClick: ScriptField.MakeScript(`updateIcon("collection")`), title: "col button", icon: "object-group" }); +        doc.Collection = ficon({ onClick: ScriptField.MakeScript(`updateIcon("collection")`), title: "col button", icon: "object-group" });          doc.Image = ficon({ onClick: ScriptField.MakeScript(`updateIcon("image")`), title: "image button", icon: "image" });          doc.Link = ficon({ onClick: ScriptField.MakeScript(`updateIcon("link")`), title: "link button", icon: "link" });          doc.Pdf = ficon({ onClick: ScriptField.MakeScript(`updateIcon("pdf")`), title: "pdf button", icon: "file-pdf" }); @@ -1153,61 +1160,63 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc          doc.Video = ficon({ onClick: ScriptField.MakeScript(`updateIcon("video")`), title: "vid button", icon: "video" });          doc.Web = ficon({ onClick: ScriptField.MakeScript(`updateIcon("web")`), title: "web button", icon: "globe-asia" }); -        let buttons = [doc.None as Doc, doc.Audio as Doc, doc.Collection as Doc,  +        let buttons = [doc.None as Doc, doc.Audio as Doc, doc.Collection as Doc,          doc.Image as Doc, doc.Link as Doc, doc.Pdf as Doc, doc.Rtf as Doc, doc.Video as Doc, doc.Web as Doc];          const dragCreators = Docs.Create.MasonryDocument(buttons, { -            _width: 500, backgroundColor:"#121721", _autoHeight: true, columnWidth: 35, ignoreClick: true, lockedPosition: true, _chromeStatus: "disabled", title: "buttons", +            _width: 500, backgroundColor: "#121721", _autoHeight: true, _columnWidth: 35, ignoreClick: true, lockedPosition: true, _chromeStatus: "disabled", title: "buttons",              dropConverter: ScriptField.MakeScript("convertToButtons(dragData)", { dragData: DragManager.DocumentDragData.name }), _yMargin: 5          }); -        doc.nodeButtons= dragCreators; +        doc.nodeButtons = dragCreators;          this.checkIcons()      }      setupKeyButtons() {          let doc = this.props.Document; -        const button = (opts: DocumentOptions) => new PrefetchProxy( Docs.Create.ButtonDocument({...opts, +        const button = (opts: DocumentOptions) => new PrefetchProxy(Docs.Create.ButtonDocument({ +            ...opts,              _width: 35, _height: 30, -            borderRounding: "16px", border:"1px solid grey", color:"white", hovercolor: "rgb(170, 170, 163)", letterSpacing: "2px", +            borderRounding: "16px", border: "1px solid grey", color: "white", hovercolor: "rgb(170, 170, 163)", letterSpacing: "2px",              _fontSize: 7, -        }))as any as Doc; -        doc.title=button({ backgroundColor:"grey", title: "Title", onClick:ScriptField.MakeScript("updateTitleStatus(self)")}); -        doc.deleted=button({ title: "Deleted", onClick:ScriptField.MakeScript("updateDeletedStatus(self)")}); -        doc.author = button({ backgroundColor:"grey", title: "Author", onClick:ScriptField.MakeScript("updateAuthorStatus(self)")}); +        })) as any as Doc; +        doc.title = button({ backgroundColor: "grey", title: "Title", onClick: ScriptField.MakeScript("updateTitleStatus(self)") }); +        doc.deleted = button({ title: "Deleted", onClick: ScriptField.MakeScript("updateDeletedStatus(self)") }); +        doc.author = button({ backgroundColor: "grey", title: "Author", onClick: ScriptField.MakeScript("updateAuthorStatus(self)") });          let buttons = [doc.title as Doc, doc.deleted as Doc, doc.author as Doc];          const dragCreators = Docs.Create.MasonryDocument(buttons, { -            _width: 500, backgroundColor:"#121721", _autoHeight: true, columnWidth: 50, ignoreClick: true, lockedPosition: true, _chromeStatus: "disabled", title: "buttons",_yMargin: 5 +            _width: 500, backgroundColor: "#121721", _autoHeight: true, _columnWidth: 50, ignoreClick: true, lockedPosition: true, _chromeStatus: "disabled", title: "buttons", _yMargin: 5              //dropConverter: ScriptField.MakeScript("convertToButtons(dragData)", { dragData: DragManager.DocumentDragData.name }),           }); -        doc.keyButtons= dragCreators; +        doc.keyButtons = dragCreators;      }      setupDefaultButtons() {          let doc = this.props.Document; -        const button = (opts: DocumentOptions) => new PrefetchProxy( Docs.Create.ButtonDocument({...opts, +        const button = (opts: DocumentOptions) => new PrefetchProxy(Docs.Create.ButtonDocument({ +            ...opts,              _width: 35, _height: 30, -            borderRounding: "16px", border:"1px solid grey", color:"white",  +            borderRounding: "16px", border: "1px solid grey", color: "white",              //hovercolor: "rgb(170, 170, 163)",               letterSpacing: "2px",              _fontSize: 7, -        }))as any as Doc; -        doc.keywords=button({ title: "Keywords", onClick:ScriptField.MakeScript("handleWordQueryChange(self)")}); -        doc.keys=button({ title: "Keys", onClick:ScriptField.MakeScript(`handleKeyChange(self)`)}); -        doc.nodes = button({ title: "Nodes", onClick:ScriptField.MakeScript("handleNodeChange(self)")}); +        })) as any as Doc; +        doc.keywords = button({ title: "Keywords", onClick: ScriptField.MakeScript("handleWordQueryChange(self)") }); +        doc.keys = button({ title: "Keys", onClick: ScriptField.MakeScript(`handleKeyChange(self)`) }); +        doc.nodes = button({ title: "Nodes", onClick: ScriptField.MakeScript("handleNodeChange(self)") });          let buttons = [doc.keywords as Doc, doc.keys as Doc, doc.nodes as Doc];          const dragCreators = Docs.Create.MasonryDocument(buttons, { -            _width: 500, backgroundColor:"#121721", _autoHeight: true, columnWidth: 60, ignoreClick: true, lockedPosition: true, _chromeStatus: "disabled", title: "buttons",_yMargin: 5 +            _width: 500, backgroundColor: "#121721", _autoHeight: true, _columnWidth: 60, ignoreClick: true, lockedPosition: true, _chromeStatus: "disabled", title: "buttons", _yMargin: 5              //dropConverter: ScriptField.MakeScript("convertToButtons(dragData)", { dragData: DragManager.DocumentDragData.name }),           }); -        doc.defaultButtons= dragCreators; +        doc.defaultButtons = dragCreators;      }      @computed get searchItemTemplate() { return Cast(Doc.UserDoc().searchItemTemplate, Doc, null); } -    childLayoutTemplate = () => this.layoutDoc._viewType === CollectionViewType.Stacking ? this.searchItemTemplate: undefined; +    childLayoutTemplate = () => this.layoutDoc._viewType === CollectionViewType.Stacking ? this.searchItemTemplate : undefined;      getTransform = () => { -    return this.props.ScreenToLocalTransform().translate(-5, -65);// listBox padding-left and pres-box-cont minHeight +        return this.props.ScreenToLocalTransform().translate(-5, -65);// listBox padding-left and pres-box-cont minHeight      }      panelHeight = () => {          return this.props.PanelHeight() - 50; @@ -1221,16 +1230,16 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc      }      //Make id layour document      render() { -        if (this.expandedBucket  === true){ -            this.props.Document._gridGap=5; +        if (this.expandedBucket === true) { +            this.props.Document._gridGap = 5;          }          else { -            this.props.Document._gridGap=10; +            this.props.Document._gridGap = 10;          } -        this.props.Document._searchDoc=true; +        this.props.Document._searchDoc = true;          return ( -            <div style={{pointerEvents:"all"}}className="searchBox-container"> +            <div style={{ pointerEvents: "all" }} className="searchBox-container">                  <div className="searchBox-bar">                      <span className="searchBox-barChild searchBox-collection" onPointerDown={SetupDrag(this.collectionRef, () => StrCast(this.layoutDoc._searchString) ? this.startDragCollection() : undefined)} ref={this.collectionRef} title="Drag Results as Collection"> @@ -1239,9 +1248,9 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc                      <input value={StrCast(this.layoutDoc._searchString)} onChange={this.onChange} type="text" placeholder="Search..." id="search-input" ref={this.inputRef}                          className="searchBox-barChild searchBox-input" onPointerDown={this.openSearch} onKeyPress={this.enter} onFocus={this.openSearch}                          style={{ width: this._searchbarOpen ? "500px" : "100px" }} /> -                    <button className="searchBox-barChild searchBox-filter" style={{transform:"none"}} title="Advanced Filtering Options" onClick={() => this.handleFilterChange()}><FontAwesomeIcon icon="ellipsis-v" color="white" /></button> +                    <button className="searchBox-barChild searchBox-filter" style={{ transform: "none" }} title="Advanced Filtering Options" onClick={() => this.handleFilterChange()}><FontAwesomeIcon icon="ellipsis-v" color="white" /></button>                  </div> -                <div id={`filterhead${this.props.Document[Id]}`} className="filter-form" style={this._filterOpen && this._numTotalResults >0 ? {overflow:"visible"} : {overflow:"hidden"}}> +                <div id={`filterhead${this.props.Document[Id]}`} className="filter-form" style={this._filterOpen && this._numTotalResults > 0 ? { overflow: "visible" } : { overflow: "hidden" }}>                      <div id={`filterhead2${this.props.Document[Id]}`} className="filter-header"  >                          {this.defaultButtons}                      </div> @@ -1253,15 +1262,15 @@ export class SearchBox extends ViewBoxBaseComponent<FieldViewProps, SearchBoxDoc                      </div>                  </div>                  <CollectionView {...this.props} -                        Document={this.props.Document} -                        PanelHeight={this.panelHeight} -                        moveDocument={returnFalse} -                        NativeHeight={()=>400} -                        childLayoutTemplate={this.childLayoutTemplate} -                        addDocument={undefined} -                        removeDocument={returnFalse} -                        focus={this.selectElement} -                        ScreenToLocalTransform={Transform.Identity} /> +                    Document={this.props.Document} +                    PanelHeight={this.panelHeight} +                    moveDocument={returnFalse} +                    NativeHeight={() => 400} +                    childLayoutTemplate={this.childLayoutTemplate} +                    addDocument={undefined} +                    removeDocument={returnFalse} +                    focus={this.selectElement} +                    ScreenToLocalTransform={Transform.Identity} />                  <div className="searchBox-results" onScroll={this.resultsScrolled} style={{                      display: this._resultsOpen ? "flex" : "none",                      height: this.resFull ? "auto" : this.resultHeight, @@ -1278,7 +1287,7 @@ Scripting.addGlobal(function lookupSearchBoxField(container: Doc, field: string,      // if (field === 'presCollapsedHeight') return container._viewType === CollectionViewType.Stacking ? 50 : 46;      // if (field === 'presStatus') return container.presStatus;      // if (field === '_itemIndex') return container._itemIndex; -        if (field == "query") return container._searchString; +    if (field == "query") return container._searchString;      return undefined;  }); diff --git a/src/fields/DateField.ts b/src/fields/DateField.ts index a925148c2..bee62663e 100644 --- a/src/fields/DateField.ts +++ b/src/fields/DateField.ts @@ -29,6 +29,10 @@ export class DateField extends ObjectField {      [ToString]() {          return this.date.toISOString();      } + +    getDate() { +        return this.date; +    }  }  Scripting.addGlobal(function d(...dateArgs: any[]) { diff --git a/src/fields/Doc.ts b/src/fields/Doc.ts index adaa1d193..b2fc10b99 100644 --- a/src/fields/Doc.ts +++ b/src/fields/Doc.ts @@ -96,6 +96,7 @@ export const AclSym = Symbol("Acl");  export const AclPrivate = Symbol("AclOwnerOnly");  export const AclReadonly = Symbol("AclReadOnly");  export const AclAddonly = Symbol("AclAddonly"); +export const AclReadWrite = Symbol("AclReadWrite");  export const UpdatingFromServer = Symbol("UpdatingFromServer");  const CachedUpdates = Symbol("Cached updates"); @@ -113,6 +114,8 @@ export function fetchProto(doc: Doc) {              case "addOnly":                  doc[AclSym] = AclAddonly;                  break; +            case "write": +                doc[AclSym] = AclReadWrite;          }      } @@ -942,20 +945,27 @@ export namespace Doc {      // filters document in a container collection:      // all documents with the specified value for the specified key are included/excluded       // based on the modifiers :"check", "x", undefined -    export function setDocFilter(container: Doc, key: string, value: any, modifiers?: "check" | "x" | undefined) { +    export function setDocFilter(container: Doc, key: string, value: any, modifiers?: "match" | "check" | "x" | undefined) {          const docFilters = Cast(container._docFilters, listSpec("string"), []); -        for (let i = 0; i < docFilters.length; i += 3) { -            if (docFilters[i] === key && docFilters[i + 1] === value) { -                docFilters.splice(i, 3); -                break; +        runInAction(() => { +            for (let i = 0; i < docFilters.length; i += 3) { +                if (docFilters[i] === key && (docFilters[i + 1] === value || modifiers === "match")) { +                    if (docFilters[i + 2] === modifiers && modifiers && docFilters[i + 1] === value) return; +                    docFilters.splice(i, 3); +                    break; +                }              } -        } -        if (typeof modifiers === "string") { -            docFilters.push(key); -            docFilters.push(value); -            docFilters.push(modifiers); -            container._docFilters = new List<string>(docFilters); -        } +            if (typeof modifiers === "string") { +                if (!docFilters.length && modifiers === "match" && value === undefined) { +                    container._docFilters = undefined; +                } else { +                    docFilters.push(key); +                    docFilters.push(value); +                    docFilters.push(modifiers); +                    container._docFilters = new List<string>(docFilters); +                } +            } +        });      }      export function readDocRangeFilter(doc: Doc, key: string) {          const docRangeFilters = Cast(doc._docRangeFilters, listSpec("string"), []); @@ -1165,5 +1175,5 @@ Scripting.addGlobal(function selectedDocs(container: Doc, excludeCollections: bo              (!excludeCollections || d.type !== DocumentType.COL || !Cast(d.data, listSpec(Doc), null)));      return docs.length ? new List(docs) : prevValue;  }); -Scripting.addGlobal(function setDocFilter(container: Doc, key: string, value: any, modifiers?: "check" | "x" | undefined) { Doc.setDocFilter(container, key, value, modifiers); }); +Scripting.addGlobal(function setDocFilter(container: Doc, key: string, value: any, modifiers?: "match" | "check" | "x" | undefined) { Doc.setDocFilter(container, key, value, modifiers); });  Scripting.addGlobal(function setDocFilterRange(container: Doc, key: string, range: number[]) { Doc.setDocFilterRange(container, key, range); }); diff --git a/src/fields/RichTextUtils.ts b/src/fields/RichTextUtils.ts index 7c7bf3e12..a590c88c4 100644 --- a/src/fields/RichTextUtils.ts +++ b/src/fields/RichTextUtils.ts @@ -392,7 +392,7 @@ export namespace RichTextUtils {                      const { attrs } = mark;                      switch (converted) {                          case "link": -                            let url = attrs.allHrefs.length ? attrs.allHrefs[0].href : ""; +                            let url = attrs.allLinks.length ? attrs.allLinks[0].href : "";                              const delimiter = "/doc/";                              const alreadyShared = "?sharing=true";                              if (new RegExp(window.location.origin + delimiter).test(url) && !url.endsWith(alreadyShared)) { diff --git a/src/fields/documentSchemas.ts b/src/fields/documentSchemas.ts index 40dadf5a8..97f62c9d4 100644 --- a/src/fields/documentSchemas.ts +++ b/src/fields/documentSchemas.ts @@ -2,6 +2,8 @@ import { makeInterface, createSchema, listSpec } from "./Schema";  import { ScriptField } from "./ScriptField";  import { Doc } from "./Doc";  import { DateField } from "./DateField"; +import { SchemaHeaderField } from "./SchemaHeaderField"; +import { Schema } from "prosemirror-model";  export const documentSchema = createSchema({      // content properties @@ -43,10 +45,16 @@ export const documentSchema = createSchema({      _showTitleHover: "string",  // the showTitle should be shown only on hover      _showAudio: "boolean",      // whether to show the audio record icon on documents      _freeformLayoutEngine: "string",// the string ID for the layout engine to use to layout freeform view documents -    _LODdisable: "boolean",     // whether to disbale LOD switching for CollectionFreeFormViews +    _freeformLOD: "boolean",    // whether to enable LOD switching for CollectionFreeFormViews      _pivotField: "string",      // specifies which field key should be used as the timeline/pivot axis      _replacedChrome: "string",  // what the default chrome is replaced with. Currently only supports the value of 'replaced' for PresBox's.      _chromeStatus: "string",    // determines the state of the collection chrome. values allowed are 'replaced', 'enabled', 'disabled', 'collapsed' +    _columnsFill: "boolean",    // whether documents in a stacking view column should be sized to fill the column +    _columnsSort: "string",     // how a document should be sorted "ascending", "descending", undefined (none)    +    _columnsStack: "boolean",   // whether a stacking document stacks vertically (as opposed to masonry horizontal) +    _columnsHideIfEmpty: "boolean",   // whether empty stacking view column headings should be hidden +    _columnHeaders: listSpec(SchemaHeaderField), // header descriptions for stacking/masonry +    _schemaHeaders: listSpec(SchemaHeaderField), // header descriptions for schema views      _fontSize: "number",      _fontFamily: "string",      _sidebarWidthPercent: "string", // percent of text window width taken up by sidebar diff --git a/src/fields/util.ts b/src/fields/util.ts index c4affb2d7..2dc21c987 100644 --- a/src/fields/util.ts +++ b/src/fields/util.ts @@ -78,7 +78,7 @@ const _setterImpl = action(function (target: any, prop: string | symbol | number          } else {              target.__fields[prop] = value;          } -        if (typeof value === "object" && !(value instanceof ObjectField)) debugger; +        //if (typeof value === "object" && !(value instanceof ObjectField)) debugger;          if (writeToServer) {              if (value === undefined) target[Update]({ '$unset': { ["fields." + prop]: "" } });              else target[Update]({ '$set': { ["fields." + prop]: value instanceof ObjectField ? SerializationHelper.Serialize(value) : (value === undefined ? null : value) } }); @@ -108,7 +108,7 @@ export function OVERRIDE_ACL(val: boolean) {  }  const layoutProps = ["panX", "panY", "width", "height", "nativeWidth", "nativeHeight", "fitWidth", "fitToBox", -    "LODdisable", "chromeStatus", "viewType", "gridGap", "xMargin", "yMargin", "autoHeight"]; +    "chromeStatus", "viewType", "gridGap", "xMargin", "yMargin", "autoHeight"];  export function setter(target: any, in_prop: string | symbol | number, value: any, receiver: any): boolean {      let prop = in_prop;      if (target[AclSym] && !_overrideAcl && !DocServer.PlaygroundFields.includes(in_prop.toString())) return true; diff --git a/src/scraping/buxton/scraper.py b/src/scraping/buxton/scraper.py index ed122e544..312915e2d 100644 --- a/src/scraping/buxton/scraper.py +++ b/src/scraping/buxton/scraper.py @@ -91,14 +91,13 @@ def write_collection(parse_results, display_fields, storage_key, viewType):              "zIndex": 2,              "libraryBrush": False,              "_viewType": viewType, -            "_LODdisable": True          },          "__type": "Doc"      }      fields["proto"] = protofy(common_proto_id)      fields[storage_key] = listify(proxify_guids(view_guids)) -    fields["schemaColumns"] = listify(display_fields) +    fields["_columnHeaders"] = listify(display_fields)      fields["author"] = "Bill Buxton"      fields["creationDate"] = {          "date": datetime.datetime.utcnow().microsecond, | 
