diff options
| author | Andy Rickert <andrew_rickert@brown.edu> | 2020-04-29 16:23:30 -0700 | 
|---|---|---|
| committer | Andy Rickert <andrew_rickert@brown.edu> | 2020-04-29 16:23:30 -0700 | 
| commit | ddf0902be470f6557695627fc65103c2d10e42f7 (patch) | |
| tree | 38311ac28f3f253462b9f867220fdee732f7a336 /src/server/authentication | |
| parent | 9aab1f5e7dc7438dfa8a93afe03bd5746315c994 (diff) | |
| parent | dadbb74ffa56a0dc55745ce972e7b13925629b7b (diff) | |
merge w master
Diffstat (limited to 'src/server/authentication')
| -rw-r--r-- | src/server/authentication/models/current_user_utils.ts | 743 | 
1 files changed, 491 insertions, 252 deletions
| diff --git a/src/server/authentication/models/current_user_utils.ts b/src/server/authentication/models/current_user_utils.ts index 3ca6b1b3a..ba5ada6c4 100644 --- a/src/server/authentication/models/current_user_utils.ts +++ b/src/server/authentication/models/current_user_utils.ts @@ -3,7 +3,7 @@ import * as rp from 'request-promise';  import { DocServer } from "../../../client/DocServer";  import { Docs, DocumentOptions } from "../../../client/documents/Documents";  import { UndoManager } from "../../../client/util/UndoManager"; -import { Doc, DocListCast } from "../../../new_fields/Doc"; +import { Doc, DocListCast, DocListCastAsync } from "../../../new_fields/Doc";  import { List } from "../../../new_fields/List";  import { listSpec } from "../../../new_fields/Schema";  import { ScriptField, ComputedField } from "../../../new_fields/ScriptField"; @@ -17,7 +17,10 @@ import { CollectionViewType } from "../../../client/views/collections/Collection  import { makeTemplate } from "../../../client/util/DropConverter";  import { RichTextField } from "../../../new_fields/RichTextField";  import { PrefetchProxy } from "../../../new_fields/Proxy"; -import { FormattedTextBox } from "../../../client/views/nodes/FormattedTextBox"; +import { FormattedTextBox } from "../../../client/views/nodes/formattedText/FormattedTextBox"; +import { MainView } from "../../../client/views/MainView"; +import { DocumentType } from "../../../client/documents/DocumentTypes"; +import { SchemaHeaderField } from "../../../new_fields/SchemaHeaderField";  export class CurrentUserUtils {      private static curr_id: string; @@ -28,111 +31,307 @@ export class CurrentUserUtils {      public static get MainDocId() { return this.mainDocId; }      public static set MainDocId(id: string | undefined) { this.mainDocId = id; }      @computed public static get UserDocument() { return Doc.UserDoc(); } -    @computed public static get ActivePen() { return Doc.UserDoc().activePen instanceof Doc && (Doc.UserDoc().activePen as Doc).pen as Doc; } +    @computed public static get ActivePen() { return Doc.UserDoc().activePen instanceof Doc && (Doc.UserDoc().activePen as Doc).inkPen as Doc; }      @observable public static GuestTarget: Doc | undefined;      @observable public static GuestWorkspace: Doc | undefined;      @observable public static GuestMobile: Doc | undefined; -    static setupDefaultDocTemplates(doc: Doc, buttons?: string[]) { -        const taskStatusValues = [{ title: "todo", _backgroundColor: "blue", color: "white" }, -        { title: "in progress", _backgroundColor: "yellow", color: "black" }, -        { title: "completed", _backgroundColor: "green", color: "white" } -        ]; -        const noteTemplates = [ -            Docs.Create.TextDocument("", { title: "text", style: "Note", isTemplateDoc: true, backgroundColor: "yellow" }), -            Docs.Create.TextDocument("", { title: "text", style: "Idea", isTemplateDoc: true, backgroundColor: "pink" }), -            Docs.Create.TextDocument("", { title: "text", style: "Topic", isTemplateDoc: true, backgroundColor: "lightBlue" }), -            Docs.Create.TextDocument("", { title: "text", style: "Person", isTemplateDoc: true, backgroundColor: "lightGreen" }), -            Docs.Create.TextDocument("", { -                title: "text", style: "Todo", isTemplateDoc: true, backgroundColor: "orange", _autoHeight: false, -                layout: FormattedTextBox.LayoutString("Todo"), _height: 100, _showCaption: "caption", caption: RichTextField.DashField("taskStatus") -            }) +    // sets up the default User Templates - slideView, queryView, descriptionView +    static setupUserTemplateButtons(doc: Doc) { +        if (doc["template-button-query"] === undefined) { +            const queryTemplate = Docs.Create.MulticolumnDocument( +                [ +                    Docs.Create.SearchDocument({ 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 } +            ); +            queryTemplate.isTemplateDoc = makeTemplate(queryTemplate); +            doc["template-button-query"] = CurrentUserUtils.ficon({ +                onDragStart: ScriptField.MakeFunction('getCopy(this.dragFactory, true)'), +                dragFactory: new PrefetchProxy(queryTemplate) as any as Doc, +                removeDropProperties: new List<string>(["dropAction"]), title: "query view", icon: "question-circle" +            }); +        } + +        if (doc["template-button-slides"] === undefined) { +            const slideTemplate = Docs.Create.MultirowDocument( +                [ +                    Docs.Create.MulticolumnDocument([], { title: "data", _height: 200 }), +                    Docs.Create.TextDocument("", { title: "text", _height: 100 }) +                ], +                { _width: 400, _height: 300, title: "slideView", _chromeStatus: "disabled", _xMargin: 3, _yMargin: 3, hideFilterView: true } +            ); +            slideTemplate.isTemplateDoc = makeTemplate(slideTemplate); +            doc["template-button-slides"] = CurrentUserUtils.ficon({ +                onDragStart: ScriptField.MakeFunction('getCopy(this.dragFactory, true)'), +                dragFactory: new PrefetchProxy(slideTemplate) as any as Doc, +                removeDropProperties: new List<string>(["dropAction"]), title: "presentation slide", icon: "address-card" +            }); +        } + +        if (doc["template-button-description"] === undefined) { +            const descriptionTemplate = Docs.Create.TextDocument("", { title: "header", _height: 100 }); +            Doc.GetProto(descriptionTemplate).layout = "<div><FormattedTextBox {...props} background='orange' height='50px' fieldKey={'header'}/><FormattedTextBox {...props} height='calc(100% - 50px)' fieldKey={'text'}/></div>"; +            descriptionTemplate.isTemplateDoc = makeTemplate(descriptionTemplate, true, "descriptionView"); + +            doc["template-button-description"] = CurrentUserUtils.ficon({ +                onDragStart: ScriptField.MakeFunction('getCopy(this.dragFactory, true)'), +                dragFactory: new PrefetchProxy(descriptionTemplate) as any as Doc, +                removeDropProperties: new List<string>(["dropAction"]), title: "description view", icon: "window-maximize" +            }); +        } + +        if (doc["template-button-detail"] === undefined) { +            const { TextDocument, MasonryDocument, CarouselDocument } = Docs.Create; + +            const carousel = CarouselDocument([], { title: "data", _height: 350, _itemIndex: 0, backgroundColor: "#9b9b9b3F" }); + +            const details = TextDocument("", { title: "details", _height: 350, _autoHeight: true }); +            const short = TextDocument("", { title: "shortDescription", treeViewOpen: true, treeViewExpandedView: "layout", _height: 50, _autoHeight: true }); +            const long = TextDocument("", { title: "longDescription", treeViewOpen: false, treeViewExpandedView: "layout", _height: 350, _autoHeight: true }); + +            const buxtonFieldKeys = ["year", "originalPrice", "degreesOfFreedom", "company", "attribute", "primaryKey", "secondaryKey", "dimensions"]; +            const detailedTemplate = { +                doc: { +                    type: "doc", content: buxtonFieldKeys.map(fieldKey => ({ +                        type: "paragraph", +                        content: [{ type: "dashField", attrs: { fieldKey } }] +                    })) +                }, +                selection: { type: "text", anchor: 1, head: 1 }, +                storedMarks: [] +            }; +            details.text = new RichTextField(JSON.stringify(detailedTemplate), buxtonFieldKeys.join(" ")); + +            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 descriptionWrapper = MasonryDocument([details, short, long], { ...shared, ...descriptionWrapperOpts }); +            descriptionWrapper.sectionHeaders = new List<SchemaHeaderField>([ +                new SchemaHeaderField("[Long Description]", "LemonChiffon", undefined, undefined, undefined, true), +                new SchemaHeaderField("[Details]", "lightBlue", undefined, undefined, undefined, true), +            ]); +            const detailView = Docs.Create.StackingDocument([carousel, descriptionWrapper], { ...shared, ...detailViewOpts }); +            detailView.isTemplateDoc = makeTemplate(detailView); + +            details.title = "Details"; +            short.title = "A Short Description"; +            long.title = "Long Description"; + +            doc["template-button-detail"] = CurrentUserUtils.ficon({ +                onDragStart: ScriptField.MakeFunction('getCopy(this.dragFactory, true)'), +                dragFactory: new PrefetchProxy(detailView) as any as Doc, +                removeDropProperties: new List<string>(["dropAction"]), title: "detail view", icon: "window-maximize" +            }); +        } + +        if (doc["template-buttons"] === undefined) { +            doc["template-buttons"] = new PrefetchProxy(Docs.Create.MasonryDocument([doc["template-button-slides"] as Doc, doc["template-button-description"] as Doc, +            doc["template-button-query"] as Doc, doc["template-button-detail"] as Doc], { +                title: "Compound Item Creators", _xMargin: 0, _showTitle: "title", +                _autoHeight: true, _width: 500, columnWidth: 35, ignoreClick: true, lockedPosition: true, _chromeStatus: "disabled", +                dropConverter: ScriptField.MakeScript("convertToButtons(dragData)", { dragData: DragManager.DocumentDragData.name }), +            })); +        } else { +            DocListCast(Cast(doc["template-buttons"], Doc, null)?.data); // prefetch templates +        } +        return doc["template-buttons"] as Doc; +    } + +    // setup the different note type skins +    static setupNoteTemplates(doc: Doc) { +        if (doc["template-note-Note"] === undefined) { +            const noteView = Docs.Create.TextDocument("", { title: "text", style: "Note", isTemplateDoc: true, backgroundColor: "yellow" }); +            noteView.isTemplateDoc = makeTemplate(noteView, true, "Note"); +            doc["template-note-Note"] = new PrefetchProxy(noteView); +        } +        if (doc["template-note-Idea"] === undefined) { +            const noteView = Docs.Create.TextDocument("", { title: "text", style: "Idea", backgroundColor: "pink" }); +            noteView.isTemplateDoc = makeTemplate(noteView, true, "Idea"); +            doc["template-note-Idea"] = new PrefetchProxy(noteView); +        } +        if (doc["template-note-Topic"] === undefined) { +            const noteView = Docs.Create.TextDocument("", { title: "text", style: "Topic", backgroundColor: "lightBlue" }); +            noteView.isTemplateDoc = makeTemplate(noteView, true, "Topic"); +            doc["template-note-Topic"] = new PrefetchProxy(noteView); +        } +        if (doc["template-note-Todo"] === undefined) { +            const noteView = Docs.Create.TextDocument("", { +                title: "text", style: "Todo", backgroundColor: "orange", _autoHeight: false, _height: 100, _showCaption: "caption", +                layout: FormattedTextBox.LayoutString("Todo"), caption: RichTextField.DashField("taskStatus") +            }); +            noteView.isTemplateDoc = makeTemplate(noteView, true, "Todo"); +            doc["template-note-Todo"] = new PrefetchProxy(noteView); +        } +        const taskStatusValues = [ +            { title: "todo", _backgroundColor: "blue", color: "white" }, +            { title: "in progress", _backgroundColor: "yellow", color: "black" }, +            { title: "completed", _backgroundColor: "green", color: "white" }          ]; -        doc.fieldTypes = Docs.Create.TreeDocument([], { title: "field enumerations" }); -        Doc.addFieldEnumerations(Doc.GetProto(noteTemplates[4]), "taskStatus", taskStatusValues); -        doc.noteTypes = new PrefetchProxy(Docs.Create.TreeDocument(noteTemplates.map(nt => makeTemplate(nt, true, StrCast(nt.style)) ? nt : nt), { title: "Note Layouts", _height: 75 })); +        if (doc.fieldTypes === undefined) { +            doc.fieldTypes = Docs.Create.TreeDocument([], { title: "field enumerations" }); +            Doc.addFieldEnumerations(Doc.GetProto(doc["template-note-Todo"] as any as Doc), "taskStatus", taskStatusValues); +        } + +        if (doc["template-notes"] === undefined) { +            doc["template-notes"] = new PrefetchProxy(Docs.Create.TreeDocument([doc["template-note-Note"] as any as Doc, +            doc["template-note-Idea"] as any as Doc, doc["template-note-Topic"] as any as Doc, doc["template-note-Todo"] as any as Doc], +                { title: "Note Layouts", _height: 75 })); +        } else { +            const curNoteTypes = Cast(doc["template-notes"], Doc, null); +            const requiredTypes = [doc["template-note-Note"] as any as Doc, doc["template-note-Idea"] as any as Doc, +            doc["template-note-Topic"] as any as Doc, doc["template-note-Todo"] as any as Doc]; +            DocListCastAsync(curNoteTypes.data).then(async curNotes => { +                await Promise.all(curNotes!); +                requiredTypes.map(ntype => Doc.AddDocToList(curNoteTypes, "data", ntype)); +            }); +        } + +        return doc["template-notes"] as Doc; +    } + +    // creates Note templates, and initial "user" templates +    static setupDocTemplates(doc: Doc) { +        const noteTemplates = CurrentUserUtils.setupNoteTemplates(doc); +        const userTemplateBtns = CurrentUserUtils.setupUserTemplateButtons(doc); +        const clickTemplates = CurrentUserUtils.setupClickEditorTemplates(doc); +        if (doc.templateDocs === undefined) { +            doc.templateDocs = new PrefetchProxy(Docs.Create.TreeDocument([noteTemplates, userTemplateBtns, clickTemplates], { +                title: "template layouts", _xPadding: 0, +                dropConverter: ScriptField.MakeScript("convertToButtons(dragData)", { dragData: DragManager.DocumentDragData.name }) +            })); +        }      } -    static setupDefaultIconTypes(doc: Doc, buttons?: string[]) { -        doc.iconView = new PrefetchProxy(Docs.Create.TextDocument("", { title: "icon", _width: 150, _height: 30, isTemplateDoc: true, onClick: ScriptField.MakeScript("deiconifyView(this)") })); -        Doc.GetProto(doc.iconView as any as Doc).icon = new RichTextField('{"doc":{"type":"doc","content":[{"type":"paragraph","attrs":{"align":null,"color":null,"id":null,"indent":null,"inset":null,"lineSpacing":null,"paddingBottom":null,"paddingTop":null},"content":[{"type":"dashField","attrs":{"fieldKey":"title","docid":""}}]}]},"selection":{"type":"text","anchor":2,"head":2},"storedMarks":[]}', ""); -        doc.isTemplateDoc = makeTemplate(doc.iconView as any as Doc); -        doc.iconImageView = new PrefetchProxy(Docs.Create.ImageDocument("http://www.cs.brown.edu/~bcz/face.gif", { title: "data", _width: 50, isTemplateDoc: true, onClick: ScriptField.MakeScript("deiconifyView(self)") })); -        doc.isTemplateDoc = makeTemplate(doc.iconImageView as any as Doc, true, "image_icon"); -        doc.iconColView = new PrefetchProxy(Docs.Create.TreeDocument([], { title: "data", _width: 180, _height: 80, isTemplateDoc: true, onClick: ScriptField.MakeScript("deiconifyView(self)") })); -        doc.isTemplateDoc = makeTemplate(doc.iconColView as any as Doc, true, "collection_icon"); -        doc.iconViews = Docs.Create.TreeDocument([doc.iconView as any as Doc, doc.iconImageView as any as Doc, doc.iconColView as any as Doc], { title: "icon types", _height: 75 }); + +    // setup templates for different document types when they are iconified from Document Decorations +    static setupDefaultIconTemplates(doc: Doc) { +        if (doc["template-icon-view"] === undefined) { +            const iconView = Docs.Create.TextDocument("", { +                title: "icon", _width: 150, _height: 30, isTemplateDoc: true, +                onClick: ScriptField.MakeScript("deiconifyView(self)") +            }); +            Doc.GetProto(iconView).icon = new RichTextField('{"doc":{"type":"doc","content":[{"type":"paragraph","attrs":{"align":null,"color":null,"id":null,"indent":null,"inset":null,"lineSpacing":null,"paddingBottom":null,"paddingTop":null},"content":[{"type":"dashField","attrs":{"fieldKey":"title","docid":""}}]}]},"selection":{"type":"text","anchor":2,"head":2},"storedMarks":[]}', ""); +            iconView.isTemplateDoc = makeTemplate(iconView); +            doc["template-icon-view"] = new PrefetchProxy(iconView); +        } +        if (doc["template-icon-view-rtf"] === undefined) { +            const iconRtfView = Docs.Create.LabelDocument({ title: "icon_" + DocumentType.RTF, textTransform: "unset", letterSpacing: "unset", _width: 150, _height: 70, _xPadding: 10, _yPadding: 10, isTemplateDoc: true, onClick: ScriptField.MakeScript("deiconifyView(self)") }); +            iconRtfView.isTemplateDoc = makeTemplate(iconRtfView, true, "icon_" + DocumentType.RTF); +            doc["template-icon-view-rtf"] = new PrefetchProxy(iconRtfView); +        } +        if (doc["template-icon-view-img"] === undefined) { +            const iconImageView = Docs.Create.ImageDocument("http://www.cs.brown.edu/~bcz/face.gif", { +                title: "data", _width: 50, isTemplateDoc: true, +                onClick: ScriptField.MakeScript("deiconifyView(self)") +            }); +            iconImageView.isTemplateDoc = makeTemplate(iconImageView, true, "icon_" + DocumentType.IMG); +            doc["template-icon-view-img"] = new PrefetchProxy(iconImageView); +        } +        if (doc["template-icon-view-col"] === undefined) { +            const iconColView = Docs.Create.TreeDocument([], { title: "data", _width: 180, _height: 80, onClick: ScriptField.MakeScript("deiconifyView(self)") }); +            iconColView.isTemplateDoc = makeTemplate(iconColView, true, "icon_" + DocumentType.COL); +            doc["template-icon-view-col"] = new PrefetchProxy(iconColView); +        } +        if (doc["template-icons"] === undefined) { +            doc["template-icons"] = new PrefetchProxy(Docs.Create.TreeDocument([doc["template-icon-view"] as Doc, doc["template-icon-view-img"] as Doc, +            doc["template-icon-view-col"] as Doc, doc["template-icon-view-rtf"] as Doc], { title: "icon templates", _height: 75 })); +        } else { +            const templateIconsDoc = Cast(doc["template-icons"], Doc, null); +            DocListCastAsync(templateIconsDoc).then(list => templateIconsDoc.data = new List<Doc>([doc["template-icon-view"] as Doc, doc["template-icon-view-img"] as Doc, +            doc["template-icon-view-col"] as Doc, doc["template-icon-view-rtf"] as Doc])); +        } +        return doc["template-icons"] as Doc;      } -    // setup the "creator" buttons for the sidebar-- eg. the default set of draggable document creation tools -    static setupCreatorButtons(doc: Doc, alreadyCreatedButtons?: string[]) { -        const emptyPresentation = Docs.Create.PresDocument(new List<Doc>(), { title: "Presentation", _viewType: CollectionViewType.Stacking, _LODdisable: true, _chromeStatus: "replaced", _showTitle: "title", boxShadow: "0 0" }); -        const emptyCollection = Docs.Create.FreeformDocument([], { _nativeWidth: undefined, _nativeHeight: undefined, _LODdisable: true, _width: 150, _height: 100, title: "freeform" }); -        doc.activePen = doc; -        const docProtoData: { title: string, icon: string, drag?: string, ignoreClick?: boolean, click?: string, ischecked?: string, activePen?: Doc, backgroundColor?: string, dragFactory?: Doc }[] = [ -            { title: "collection", icon: "folder", click: 'openOnRight(getCopy(this.dragFactory, true))', drag: 'getCopy(this.dragFactory, true)', dragFactory: emptyCollection }, -            { title: "preview", icon: "expand", ignoreClick: true, drag: 'Docs.Create.DocumentDocument(ComputedField.MakeFunction("selectedDocs(this,this.excludeCollections,[_last_])?.[0]"), { _width: 250, _height: 250, title: "container" })' }, -            { title: "web page", icon: "globe-asia", ignoreClick: true, drag: 'Docs.Create.WebDocument("https://en.wikipedia.org/wiki/Hedgehog", {_width: 300, _height: 300, title: "New Webpage" })' }, -            { title: "cat image", 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" })' }, -            { title: "screenshot", icon: "photo-video", ignoreClick: true, drag: 'Docs.Create.ScreenshotDocument("", { _width: 400, _height: 200, title: "screen snapshot" })' }, -            { title: "webcam", icon: "video", ignoreClick: true, drag: 'Docs.Create.WebCamDocument("", { _width: 400, _height: 400, title: "a test cam" })' }, -            { title: "record", icon: "microphone", ignoreClick: true, drag: `Docs.Create.AudioDocument("${nullAudio}", { _width: 200, title: "ready to record audio" })` }, -            { title: "clickable button", icon: "bolt", ignoreClick: true, drag: 'Docs.Create.ButtonDocument({ _width: 150, _height: 50, title: "Button" })' }, -            { title: "presentation", icon: "tv", click: 'openOnRight(Doc.UserDoc().curPresentation = getCopy(this.dragFactory, true))', drag: `Doc.UserDoc().curPresentation = getCopy(this.dragFactory,true)`, dragFactory: emptyPresentation }, -            { title: "script", icon: "terminal", ignoreClick: true, drag: 'Docs.Create.ScriptingDocument(undefined, { _width: 200, _height: 250 title: "untitled script" })' }, -            { title: "import folder", icon: "cloud-upload-alt", ignoreClick: true, drag: 'Docs.Create.DirectoryImportDocument({ title: "Directory Import", _width: 400, _height: 400 })' }, -            { title: "mobile view", icon: "phone", ignoreClick: true, drag: 'Doc.UserDoc().activeMobile' }, -            { title: "use pen", icon: "pen-nib", click: 'activatePen(this.activePen.pen = sameDocs(this.activePen.pen, this) ? undefined : this,2, this.backgroundColor)', backgroundColor: "blue", ischecked: `sameDocs(this.activePen.pen,  this)`, activePen: doc }, -            { title: "use highlighter", icon: "highlighter", click: 'activateBrush(this.activePen.pen = sameDocs(this.activePen.pen, this) ? undefined : this,20,this.backgroundColor)', backgroundColor: "yellow", ischecked: `sameDocs(this.activePen.pen, this)`, activePen: doc }, -            { title: "use stamp", icon: "stamp", click: 'activateStamp(this.activePen.pen = sameDocs(this.activePen.pen, this) ? undefined : this)', backgroundColor: "orange", ischecked: `sameDocs(this.activePen.pen, this)`, activePen: doc }, -            { title: "use eraser", icon: "eraser", click: 'activateEraser(this.activePen.pen = sameDocs(this.activePen.pen, this) ? undefined : this);', ischecked: `sameDocs(this.activePen.pen, this)`, backgroundColor: "pink", activePen: doc }, -            { title: "use drag", icon: "mouse-pointer", click: 'deactivateInk();this.activePen.pen = this;', ischecked: `sameDocs(this.activePen.pen, this)`, backgroundColor: "white", activePen: doc }, -            { title: "query", icon: "bolt", ignoreClick: true, drag: 'Docs.Create.SearchDocument({ _width: 200, title: "an image of a cat" })' }, -            // { title: "buxton", icon: "cloud-upload-alt", ignoreClick: true, drag: "Docs.Create.Buxton()" }, +    static creatorBtnDescriptors(doc: Doc): { +        title: string, label: string, icon: string, drag?: string, ignoreClick?: boolean, +        click?: string, ischecked?: string, activePen?: Doc, backgroundColor?: string, dragFactory?: Doc +    }[] { +        if (doc.emptyPresentation === undefined) { +            doc.emptyPresentation = Docs.Create.PresDocument(new List<Doc>(), +                { title: "Presentation", _viewType: CollectionViewType.Stacking, _LODdisable: true, _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" }); +        } +        return [ +            { 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", ignoreClick: true, drag: 'Docs.Create.WebDocument("", { title: "New Webpage" })' }, +            { 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" })' }, +            { title: "Drag a screenshot", label: "Grab", icon: "photo-video", ignoreClick: true, drag: 'Docs.Create.ScreenshotDocument("", { _width: 400, _height: 200, title: "screen snapshot" })' }, +            { title: "Drag a webcam", label: "Cam", icon: "video", ignoreClick: true, drag: 'Docs.Create.WebCamDocument("", { _width: 400, _height: 400, title: "a test cam" })' }, +            { title: "Drag a audio recorder", label: "Audio", icon: "microphone", ignoreClick: true, drag: `Docs.Create.AudioDocument("${nullAudio}", { _width: 200, title: "ready to record audio" })` }, +            { title: "Drag a clickable button", label: "Btn", icon: "bolt", ignoreClick: true, drag: 'Docs.Create.ButtonDocument({ _width: 150, _height: 50, _xPadding:10, _yPadding: 10, title: "Button" })' }, +            { title: "Drag a presentation view", label: "Prezi", icon: "tv", click: 'openOnRight(Doc.UserDoc().activePresentation = getCopy(this.dragFactory, true))', drag: `Doc.UserDoc().activePresentation = getCopy(this.dragFactory,true)`, dragFactory: doc.emptyPresentation as Doc }, +            { title: "Drag a search box", label: "Query", icon: "search", ignoreClick: true, drag: 'Docs.Create.SearchDocument({ _width: 200, title: "an image of a cat" })' }, +            { title: "Drag a scripting box", label: "Script", icon: "terminal", ignoreClick: true, drag: 'Docs.Create.ScriptingDocument(undefined, { _width: 200, _height: 250 title: "untitled script" })' }, +            { title: "Drag an import folder", label: "Load", icon: "cloud-upload-alt", ignoreClick: true, drag: 'Docs.Create.DirectoryImportDocument({ title: "Directory Import", _width: 400, _height: 400 })' }, +            { title: "Drag a mobile view", label: "Phone", icon: "phone", ignoreClick: true, drag: 'Doc.UserDoc().activeMobile' }, +            { title: "Drag an instance of the device collection", label: "Buxton", icon: "globe-asia", ignoreClick: true, drag: 'Docs.Create.Buxton()' }, +            // { title: "use pen", icon: "pen-nib", click: 'activatePen(this.activePen.inkPen = sameDocs(this.activePen.inkPen, this) ? undefined : this,2, this.backgroundColor)', backgroundColor: "blue", ischecked: `sameDocs(this.activePen.inkPen,  this)`, activePen: doc }, +            // { title: "use highlighter", icon: "highlighter", click: 'activateBrush(this.activePen.inkPen = sameDocs(this.activePen.inkPen, this) ? undefined : this,20,this.backgroundColor)', backgroundColor: "yellow", ischecked: `sameDocs(this.activePen.inkPen, this)`, activePen: doc }, +            // { title: "use stamp", icon: "stamp", click: 'activateStamp(this.activePen.inkPen = sameDocs(this.activePen.inkPen, this) ? undefined : this)', backgroundColor: "orange", ischecked: `sameDocs(this.activePen.inkPen, this)`, activePen: doc }, +            // { title: "use eraser", icon: "eraser", click: 'activateEraser(this.activePen.inkPen = sameDocs(this.activePen.inkPen, this) ? undefined : this);', ischecked: `sameDocs(this.activePen.inkPen, this)`, backgroundColor: "pink", activePen: doc }, +            // { title: "use drag", icon: "mouse-pointer", click: 'deactivateInk();this.activePen.inkPen = this;', ischecked: `sameDocs(this.activePen.inkPen, this)`, backgroundColor: "white", activePen: doc }, +            { title: "Drag a document previewer", label: "Prev", icon: "expand", ignoreClick: true, drag: 'Docs.Create.DocumentDocument(ComputedField.MakeFunction("selectedDocs(this,this.excludeCollections,[_last_])?.[0]"), { _width: 250, _height: 250, title: "container" })' }, +            { title: "Drag a Calculator REPL", label: "repl", icon: "calculator", click: 'addOverlayWindow("ScriptingRepl", { x: 300, y: 100, width: 200, height: 200, title: "Scripting REPL" })' }, +            { title: "query", icon: "bolt", label: "Col", ignoreClick: true, drag: 'Docs.Create.SearchDocument({ _width: 200, title: "an image of a cat" })' }, +          ]; -        return docProtoData.filter(d => !alreadyCreatedButtons?.includes(d.title)).map(data => Docs.Create.FontIconDocument({ -            _nativeWidth: 100, _nativeHeight: 100, _width: 100, _height: 100, -            icon: data.icon, -            title: data.title, -            ignoreClick: data.ignoreClick, -            dropAction: data.click ? "copy" : undefined, -            onDragStart: data.drag ? ScriptField.MakeFunction(data.drag) : undefined, -            onClick: data.click ? ScriptField.MakeScript(data.click) : undefined, -            ischecked: data.ischecked ? ComputedField.MakeFunction(data.ischecked) : undefined, -            activePen: data.activePen, -            backgroundColor: data.backgroundColor, removeDropProperties: new List<string>(["dropAction"]), -            dragFactory: data.dragFactory, -        })); +      } -    static async updateCreatorButtons(doc: Doc) { -        const toolsBtn = await Cast(doc.ToolsBtn, Doc); -        if (toolsBtn) { -            const stackingDoc = await Cast(toolsBtn.sourcePanel, Doc); -            if (stackingDoc) { -                const stackdocs = await Cast(stackingDoc.data, listSpec(Doc)); -                if (stackdocs) { -                    const dragset = await Cast(stackdocs[0], Doc); -                    if (dragset) { -                        const dragdocs = await Cast(dragset.data, listSpec(Doc)); -                        if (dragdocs) { -                            const dragDocs = await Promise.all(dragdocs); -                            this.setupCreatorButtons(doc, dragDocs.map(d => StrCast(d.title))).map(nb => Doc.AddDocToList(dragset, "data", nb)); -                        } -                    } -                } +    // setup the "creator" buttons for the sidebar-- eg. the default set of draggable document creation tools +    static async setupCreatorButtons(doc: Doc) { +        let alreadyCreatedButtons: string[] = []; +        const dragCreatorSet = await Cast(doc.myItemCreators, Doc, null); +        if (dragCreatorSet) { +            const dragCreators = await Cast(dragCreatorSet.data, listSpec(Doc)); +            if (dragCreators) { +                const dragDocs = await Promise.all(dragCreators); +                alreadyCreatedButtons = dragDocs.map(d => StrCast(d.title));              }          } +        const creatorBtns = CurrentUserUtils.creatorBtnDescriptors(doc).filter(d => !alreadyCreatedButtons?.includes(d.title)). +            map(data => Docs.Create.FontIconDocument({ +                _nativeWidth: 100, _nativeHeight: 100, _width: 100, _height: 100, +                icon: data.icon, +                title: data.title, +                label: data.label, +                ignoreClick: data.ignoreClick, +                dropAction: data.click ? "copy" : undefined, +                onDragStart: data.drag ? ScriptField.MakeFunction(data.drag) : undefined, +                onClick: data.click ? ScriptField.MakeScript(data.click) : undefined, +                ischecked: data.ischecked ? ComputedField.MakeFunction(data.ischecked) : undefined, +                activePen: data.activePen, +                backgroundColor: data.backgroundColor, removeDropProperties: new List<string>(["dropAction"]), +                dragFactory: data.dragFactory, +            })); + +        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", +                dropConverter: ScriptField.MakeScript("convertToButtons(dragData)", { dragData: DragManager.DocumentDragData.name }), +            })); +        } else { +            creatorBtns.forEach(nb => Doc.AddDocToList(doc.myItemCreators as Doc, "data", nb)); +        } +        return doc.myItemCreators as Doc;      }      static setupMobileButtons(doc: Doc, buttons?: string[]) {          const docProtoData: { title: string, icon: string, drag?: string, ignoreClick?: boolean, click?: string, ischecked?: string, activePen?: Doc, backgroundColor?: string, dragFactory?: Doc }[] = [              { title: "record", icon: "microphone", ignoreClick: true, click: "FILL" }, -            { title: "use pen", icon: "pen-nib", click: 'activatePen(this.activePen.pen = sameDocs(this.activePen.pen, this) ? undefined : this,2, this.backgroundColor)', backgroundColor: "blue", ischecked: `sameDocs(this.activePen.pen,  this)`, activePen: doc }, -            { title: "use highlighter", icon: "highlighter", click: 'activateBrush(this.activePen.pen = sameDocs(this.activePen.pen, this) ? undefined : this,20,this.backgroundColor)', backgroundColor: "yellow", ischecked: `sameDocs(this.activePen.pen, this)`, activePen: doc }, -            { title: "use eraser", icon: "eraser", click: 'activateEraser(this.activePen.pen = sameDocs(this.activePen.pen, this) ? undefined : this);', ischecked: `sameDocs(this.activePen.pen, this)`, backgroundColor: "pink", activePen: doc }, -            { title: "use drag", icon: "mouse-pointer", click: 'deactivateInk();this.activePen.pen = this;', ischecked: `sameDocs(this.activePen.pen, this)`, backgroundColor: "white", activePen: doc }, -            // { title: "draw", icon: "pen-nib", click: 'switchMobileView(setupMobileInkingDoc, renderMobileInking, onSwitchMobileInking);', ischecked: `sameDocs(this.activePen.pen, this)`, backgroundColor: "red", activePen: doc }, +            { title: "use pen", icon: "pen-nib", click: 'activatePen(this.activePen.inkPen = sameDocs(this.activePen.inkPen, this) ? undefined : this,2, this.backgroundColor)', backgroundColor: "blue", ischecked: `sameDocs(this.activePen.inkPen,  this)`, activePen: doc }, +            { title: "use highlighter", icon: "highlighter", click: 'activateBrush(this.activePen.inkPen = sameDocs(this.activePen.inkPen, this) ? undefined : this,20,this.backgroundColor)', backgroundColor: "yellow", ischecked: `sameDocs(this.activePen.inkPen, this)`, activePen: doc }, +            { title: "use eraser", icon: "eraser", click: 'activateEraser(this.activePen.inkPen = sameDocs(this.activePen.inkPen, this) ? undefined : this);', ischecked: `sameDocs(this.activePen.inkPen, this)`, backgroundColor: "pink", activePen: doc }, +            { title: "use drag", icon: "mouse-pointer", click: 'deactivateInk();this.activePen.inkPen = this;', ischecked: `sameDocs(this.activePen.inkPen, this)`, backgroundColor: "white", activePen: doc }, +            // { title: "draw", icon: "pen-nib", click: 'switchMobileView(setupMobileInkingDoc, renderMobileInking, onSwitchMobileInking);', ischecked: `sameDocs(this.activePen.inkPen, this)`, backgroundColor: "red", activePen: doc },              { title: "upload", icon: "upload", click: 'switchMobileView(setupMobileUploadDoc, renderMobileUpload, onSwitchMobileUpload);', backgroundColor: "orange" },              // { title: "upload", icon: "upload", click: 'uploadImageMobile();', backgroundColor: "cyan" },          ]; @@ -146,11 +345,11 @@ export class CurrentUserUtils {      static setupThumbButtons(doc: Doc) {          const docProtoData: { title: string, icon: string, drag?: string, ignoreClick?: boolean, pointerDown?: string, pointerUp?: string, ischecked?: string, clipboard?: Doc, activePen?: Doc, backgroundColor?: string, dragFactory?: Doc }[] = [ -            { title: "use pen", icon: "pen-nib", pointerUp: "resetPen()", pointerDown: 'setPen(2, this.backgroundColor)', backgroundColor: "blue", ischecked: `sameDocs(this.activePen.pen,  this)`, activePen: doc }, -            { title: "use highlighter", icon: "highlighter", pointerUp: "resetPen()", pointerDown: 'setPen(20, this.backgroundColor)', backgroundColor: "yellow", ischecked: `sameDocs(this.activePen.pen, this)`, activePen: doc }, -            { title: "notepad", icon: "clipboard", pointerUp: "GestureOverlay.Instance.closeFloatingDoc()", pointerDown: 'GestureOverlay.Instance.openFloatingDoc(this.clipboard)', clipboard: Docs.Create.FreeformDocument([], { _width: 300, _height: 300 }), backgroundColor: "orange", ischecked: `sameDocs(this.activePen.pen, this)`, activePen: doc }, -            { title: "interpret text", icon: "font", pointerUp: "setToolglass('none')", pointerDown: "setToolglass('inktotext')", backgroundColor: "orange", ischecked: `sameDocs(this.activePen.pen, this)`, activePen: doc }, -            { title: "ignore gestures", icon: "signature", pointerUp: "setToolglass('none')", pointerDown: "setToolglass('ignoregesture')", backgroundColor: "green", ischecked: `sameDocs(this.activePen.pen, this)`, activePen: doc }, +            { title: "use pen", icon: "pen-nib", pointerUp: "resetPen()", pointerDown: 'setPen(2, this.backgroundColor)', backgroundColor: "blue", ischecked: `sameDocs(this.activePen.inkPen,  this)`, activePen: doc }, +            { title: "use highlighter", icon: "highlighter", pointerUp: "resetPen()", pointerDown: 'setPen(20, this.backgroundColor)', backgroundColor: "yellow", ischecked: `sameDocs(this.activePen.inkPen, this)`, activePen: doc }, +            { title: "notepad", icon: "clipboard", pointerUp: "GestureOverlay.Instance.closeFloatingDoc()", pointerDown: 'GestureOverlay.Instance.openFloatingDoc(this.clipboard)', clipboard: Docs.Create.FreeformDocument([], { _width: 300, _height: 300 }), backgroundColor: "orange", ischecked: `sameDocs(this.activePen.inkPen, this)`, activePen: doc }, +            { title: "interpret text", icon: "font", pointerUp: "setToolglass('none')", pointerDown: "setToolglass('inktotext')", backgroundColor: "orange", ischecked: `sameDocs(this.activePen.inkPen, this)`, activePen: doc }, +            { title: "ignore gestures", icon: "signature", pointerUp: "setToolglass('none')", pointerDown: "setToolglass('ignoregesture')", backgroundColor: "green", ischecked: `sameDocs(this.activePen.inkPen, this)`, activePen: doc },          ];          return docProtoData.map(data => Docs.Create.FontIconDocument({              _nativeWidth: 10, _nativeHeight: 10, _width: 10, _height: 10, title: data.title, icon: data.icon, @@ -200,206 +399,245 @@ export class CurrentUserUtils {          });      } -    // setup the Creator button which will display the creator panel.  This panel will include the drag creators and the color picker.  when clicked, this panel will be displayed in the target container (ie, sidebarContainer)   -    static setupToolsPanel(sidebarContainer: Doc, doc: Doc) { +    // setup the Creator button which will display the creator panel.  This panel will include the drag creators and the color picker.  +    // when clicked, this panel will be displayed in the target container (ie, sidebarContainer)   +    static async setupToolsBtnPanel(doc: Doc, sidebarContainer: Doc) {          // setup a masonry view of all he creators -        const dragCreators = Docs.Create.MasonryDocument(CurrentUserUtils.setupCreatorButtons(doc), { -            _width: 500, _autoHeight: true, columnWidth: 35, ignoreClick: true, lockedPosition: true, _chromeStatus: "disabled", title: "buttons", -            dropConverter: ScriptField.MakeScript("convertToButtons(dragData)", { dragData: DragManager.DocumentDragData.name }), _yMargin: 5 -        }); +        const creatorBtns = await CurrentUserUtils.setupCreatorButtons(doc); +        const templateBtns = CurrentUserUtils.setupUserTemplateButtons(doc); + +        if (doc.myCreators === undefined) { +            doc.myCreators = new PrefetchProxy(Docs.Create.StackingDocument([creatorBtns, templateBtns], { +                title: "all Creators", _yMargin: 0, _autoHeight: true, _xMargin: 0, +                _width: 500, ignoreClick: true, lockedPosition: true, _chromeStatus: "disabled", +            })); +        }          // setup a color picker -        const color = Docs.Create.ColorDocument({ -            title: "color picker", _width: 300, dropAction: "alias", forceActive: true, removeDropProperties: new List<string>(["dropAction", "forceActive"]) -        }); +        if (doc.myColorPicker === undefined) { +            const color = Docs.Create.ColorDocument({ +                title: "color picker", _width: 300, dropAction: "alias", forceActive: true, removeDropProperties: new List<string>(["dropAction", "forceActive"]) +            }); +            doc.myColorPicker = new PrefetchProxy(color); +        } -        return Docs.Create.ButtonDocument({ -            _width: 35, _height: 25, title: "Tools", fontSize: 10, targetContainer: sidebarContainer, -            letterSpacing: "0px", textTransform: "unset", borderRounding: "5px 5px 0px 0px", boxShadow: "3px 3px 0px rgb(34, 34, 34)", -            sourcePanel: Docs.Create.StackingDocument([dragCreators, color], { -                _width: 500, lockedPosition: true, _chromeStatus: "disabled", title: "tools stack", forceActive: true -            }), -            onClick: ScriptField.MakeScript("this.targetContainer.proto = this.sourcePanel"), -        }); +        if (doc["tabs-button-tools"] === undefined) { +            doc["tabs-button-tools"] = new PrefetchProxy(Docs.Create.ButtonDocument({ +                _width: 35, _height: 25, title: "Tools", _fontSize: 10, +                letterSpacing: "0px", textTransform: "unset", borderRounding: "5px 5px 0px 0px", boxShadow: "3px 3px 0px rgb(34, 34, 34)", +                sourcePanel: new PrefetchProxy(Docs.Create.StackingDocument([doc.myCreators as Doc, doc.myColorPicker as Doc], { +                    _width: 500, lockedPosition: true, _chromeStatus: "disabled", title: "tools stack", forceActive: true +                })) as any as Doc, +                targetContainer: new PrefetchProxy(sidebarContainer) as any as Doc, +                onClick: ScriptField.MakeScript("this.targetContainer.proto = this.sourcePanel"), +            })); +        } +        (doc["tabs-button-tools"] as Doc).sourcePanel; // prefetch sourcePanel +        return doc["tabs-button-tools"] as Doc;      } -    // setup the Library button which will display the library panel.  This panel includes a collection of workspaces, documents, and recently closed views -    static setupLibraryPanel(sidebarContainer: Doc, doc: Doc) { +    static setupWorkspaces(doc: Doc) {          // setup workspaces library item -        doc.workspaces = Docs.Create.TreeDocument([], { -            title: "WORKSPACES", _height: 100, forceActive: true, boxShadow: "0 0", lockedPosition: true, -        }); - -        doc.documents = Docs.Create.TreeDocument([], { -            title: "DOCUMENTS", _height: 42, forceActive: true, boxShadow: "0 0", treeViewPreventOpen: true, lockedPosition: true, -        }); +        if (doc.myWorkspaces === undefined) { +            doc.myWorkspaces = new PrefetchProxy(Docs.Create.TreeDocument([], { +                title: "WORKSPACES", _height: 100, forceActive: true, boxShadow: "0 0", lockedPosition: true, +            })); +        } +        const newWorkspace = ScriptField.MakeScript(`createNewWorkspace()`); +        (doc.myWorkspaces as Doc).contextMenuScripts = new List<ScriptField>([newWorkspace!]); +        (doc.myWorkspaces as Doc).contextMenuLabels = new List<string>(["Create New Workspace"]); +        return doc.myWorkspaces as Doc; +    } +    static setupDocumentCollection(doc: Doc) { +        if (doc.myDocuments === undefined) { +            doc.myDocuments = new PrefetchProxy(Docs.Create.TreeDocument([], { +                title: "DOCUMENTS", _height: 42, forceActive: true, boxShadow: "0 0", treeViewPreventOpen: true, lockedPosition: true, +            })); +        } +        return doc.myDocuments as Doc; +    } +    static setupRecentlyClosed(doc: Doc) {          // setup Recently Closed library item -        doc.recentlyClosed = Docs.Create.TreeDocument([], { -            title: "RECENTLY CLOSED", _height: 75, forceActive: true, boxShadow: "0 0", treeViewPreventOpen: true, lockedPosition: true, -        }); +        if (doc.myRecentlyClosed === undefined) { +            doc.myRecentlyClosed = new PrefetchProxy(Docs.Create.TreeDocument([], { +                title: "RECENTLY CLOSED", _height: 75, forceActive: true, boxShadow: "0 0", treeViewPreventOpen: true, lockedPosition: true, +            })); +        } +        // this is equivalent to using PrefetchProxies to make sure the recentlyClosed doc is ready +        PromiseValue(Cast(doc.myRecentlyClosed, Doc)).then(recent => recent && PromiseValue(recent.data).then(DocListCast)); +        const clearAll = ScriptField.MakeScript(`self.data = new List([])`); +        (doc.myRecentlyClosed as Doc).contextMenuScripts = new List<ScriptField>([clearAll!]); +        (doc.myRecentlyClosed as Doc).contextMenuLabels = new List<string>(["Clear All"]); -        return Docs.Create.ButtonDocument({ -            _width: 50, _height: 25, title: "Library", fontSize: 10, -            letterSpacing: "0px", textTransform: "unset", borderRounding: "5px 5px 0px 0px", boxShadow: "3px 3px 0px rgb(34, 34, 34)", -            sourcePanel: Docs.Create.TreeDocument([doc.workspaces as Doc, doc.documents as Doc, Docs.Prototypes.MainLinkDocument(), doc, doc.recentlyClosed as Doc], { -                title: "Library", _xMargin: 5, _yMargin: 5, _gridGap: 5, forceActive: true, childDropAction: "place", lockedPosition: true, boxShadow: "0 0", dontRegisterChildren: true -            }), -            targetContainer: sidebarContainer, -            onClick: ScriptField.MakeScript("this.targetContainer.proto = this.sourcePanel;") -        }); +        return doc.myRecentlyClosed as Doc; +    } +    // setup the Library button which will display the library panel.  This panel includes a collection of workspaces, documents, and recently closed views +    static setupLibraryPanel(doc: Doc, sidebarContainer: Doc) { +        const workspaces = CurrentUserUtils.setupWorkspaces(doc); +        const documents = CurrentUserUtils.setupDocumentCollection(doc); +        const recentlyClosed = CurrentUserUtils.setupRecentlyClosed(doc); + +        if (doc["tabs-button-library"] === undefined) { +            doc["tabs-button-library"] = new PrefetchProxy(Docs.Create.ButtonDocument({ +                _width: 50, _height: 25, title: "Library", _fontSize: 10, +                letterSpacing: "0px", textTransform: "unset", borderRounding: "5px 5px 0px 0px", boxShadow: "3px 3px 0px rgb(34, 34, 34)", +                sourcePanel: new PrefetchProxy(Docs.Create.TreeDocument([workspaces, documents, recentlyClosed, doc], { +                    title: "Library", _xMargin: 5, _yMargin: 5, _gridGap: 5, forceActive: true, childDropAction: "move", lockedPosition: true, boxShadow: "0 0", dontRegisterChildren: true +                })) as any as Doc, +                targetContainer: new PrefetchProxy(sidebarContainer) as any as Doc, +                onClick: ScriptField.MakeScript("this.targetContainer.proto = this.sourcePanel;") +            })); +        } +        return doc["tabs-button-library"] as Doc;      }      // setup the Search button which will display the search panel.   -    static setupSearchPanel(sidebarContainer: Doc) { - -        return 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: Docs.Create.SearchDocument({ title: "search stack", }), -            targetContainer: sidebarContainer, -            lockedPosition: true, -            onClick: ScriptField.MakeScript("this.targetContainer.proto = this.sourcePanel") -        }); +    static setupSearchBtnPanel(doc: Doc, sidebarContainer: Doc) { +        if (doc["tabs-button-search"] === undefined) { +            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({ title: "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, +                onClick: ScriptField.MakeScript("this.targetContainer.proto = this.sourcePanel") +            })); +        } +        return doc["tabs-button-search"] as Doc;      } -    // setup the list of sidebar mode buttons which determine what is displayed in the sidebar -    static setupSidebarButtons(doc: Doc) { -        const sidebarContainer = new Doc(); -        doc.sidebarContainer = new PrefetchProxy(sidebarContainer); -        sidebarContainer._chromeStatus = "disabled"; -        sidebarContainer.onClick = ScriptField.MakeScript("freezeSidebar()"); +    static setupSidebarContainer(doc: Doc) { +        if (doc["tabs-panelContainer"] === undefined) { +            const sidebarContainer = new Doc(); +            sidebarContainer._chromeStatus = "disabled"; +            sidebarContainer.onClick = ScriptField.MakeScript("freezeSidebar()"); +            doc["tabs-panelContainer"] = new PrefetchProxy(sidebarContainer); +        } +        return doc["tabs-panelContainer"] as Doc; +    } -        doc.ToolsBtn = new PrefetchProxy(this.setupToolsPanel(sidebarContainer, doc)); -        doc.LibraryBtn = new PrefetchProxy(this.setupLibraryPanel(sidebarContainer, doc)); -        doc.SearchBtn = new PrefetchProxy(this.setupSearchPanel(sidebarContainer)); +    // setup the list of sidebar mode buttons which determine what is displayed in the sidebar +    static async setupSidebarButtons(doc: Doc) { +        const sidebarContainer = CurrentUserUtils.setupSidebarContainer(doc); +        const toolsBtn = await CurrentUserUtils.setupToolsBtnPanel(doc, sidebarContainer); +        const libraryBtn = CurrentUserUtils.setupLibraryPanel(doc, sidebarContainer); +        const searchBtn = CurrentUserUtils.setupSearchBtnPanel(doc, sidebarContainer);          // Finally, setup the list of buttons to display in the sidebar -        doc.sidebarButtons = new PrefetchProxy(Docs.Create.StackingDocument([doc.SearchBtn as any as Doc, doc.LibraryBtn as any as Doc, doc.ToolsBtn as any as Doc], { -            _width: 500, _height: 80, boxShadow: "0 0", _pivotField: "title", hideHeadings: true, ignoreClick: true, _chromeStatus: "view-mode", -            title: "sidebar btn row stack", backgroundColor: "dimGray", -        })); +        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", +                title: "sidebar btn row stack", backgroundColor: "dimGray", +            })); +            (toolsBtn.onClick as ScriptField).script.run({ this: toolsBtn }); +        }      } -    // forceActive: true +    static blist = (opts: DocumentOptions, docs: Doc[]) => new PrefetchProxy(Docs.Create.LinearDocument(docs, { +        ...opts, +        _gridGap: 5, _xMargin: 5, _yMargin: 5, _height: 42, _width: 100, boxShadow: "0 0", forceActive: true, +        dropConverter: ScriptField.MakeScript("convertToButtons(dragData)", { dragData: DragManager.DocumentDragData.name }), +        backgroundColor: "black", treeViewPreventOpen: true, lockedPosition: true, _chromeStatus: "disabled", linearViewIsExpanded: true +    })) as any as Doc + +    static 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 +      /// sets up the default list of buttons to be shown in the expanding button menu at the bottom of the Dash window -    static setupExpandingButtons(doc: Doc) { -        const queryTemplate = Docs.Create.MulticolumnDocument( -            [ -                Docs.Create.SearchDocument({ title: "query", _height: 200, forceActive:true }), -                Docs.Create.FreeformDocument([], { title: "data", _height: 100, _LODdisable: true, forceActive: true }) -            ], -            { _width: 400, _height: 300, title: "queryView", _chromeStatus: "disabled", _xMargin: 3, _yMargin: 3, _autoHeight: false, forceActive: true, hideFilterView: true }); -        queryTemplate.isTemplateDoc = makeTemplate(queryTemplate); -        const slideTemplate = Docs.Create.MultirowDocument( -            [ -                Docs.Create.MulticolumnDocument([], { title: "data", _height: 200, forceActive: true }), -                Docs.Create.TextDocument("", { title: "text", _height: 100, forceActive: true }) -            ], -            { _width: 400, _height: 300, title: "slideView", _chromeStatus: "disabled", _xMargin: 3, _yMargin: 3, _autoHeight: false, forceActive: true, hideFilterView: true }); -        slideTemplate.isTemplateDoc = makeTemplate(slideTemplate); -        const descriptionTemplate = Docs.Create.TextDocument("", { title: "text", _height: 100, _showTitle: "title" }); -        Doc.GetProto(descriptionTemplate).layout = FormattedTextBox.LayoutString("description"); -        descriptionTemplate.isTemplateDoc = makeTemplate(descriptionTemplate, true, "descriptionView"); - -        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 blist = (opts: DocumentOptions, docs: Doc[]) => new PrefetchProxy(Docs.Create.LinearDocument(docs, { -            ...opts, -            _gridGap: 5, _xMargin: 5, _yMargin: 5, _height: 42, _width: 100, boxShadow: "0 0", forceActive: true, -            dropConverter: ScriptField.MakeScript("convertToButtons(dragData)", { dragData: DragManager.DocumentDragData.name }), -            backgroundColor: "black", treeViewPreventOpen: true, lockedPosition: true, _chromeStatus: "disabled", linearViewIsExpanded: true -        })) as any as Doc; - -        doc.undoBtn = ficon({ onClick: ScriptField.MakeScript("undo()"), title: "undo button", icon: "undo-alt" }); -        doc.redoBtn = ficon({ onClick: ScriptField.MakeScript("redo()"), title: "redo button", icon: "redo-alt" }); -        doc.slidesBtn = ficon({ onDragStart: ScriptField.MakeFunction('getCopy(this.dragFactory, true)'), dragFactory: slideTemplate, removeDropProperties: new List<string>(["dropAction"]), title: "presentation slide", icon: "sticky-note" }); -        doc.descriptionBtn = ficon({ onDragStart: ScriptField.MakeFunction('getCopy(this.dragFactory, true)'), dragFactory: descriptionTemplate, removeDropProperties: new List<string>(["dropAction"]), title: "description view", icon: "sticky-note" }); -        doc.queryBtn = ficon({ onDragStart: ScriptField.MakeFunction('getCopy(this.dragFactory, true)'), dragFactory: queryTemplate, removeDropProperties: new List<string>(["dropAction"]), title: "query view", icon: "sticky-note" }); -        doc.templateButtons = blist({ title: "template buttons", ignoreClick: true }, [doc.slidesBtn as Doc, doc.descriptionBtn as Doc, doc.queryBtn as Doc]); -        doc.expandingButtons = blist({ title: "expanding buttons", ignoreClick: true }, [doc.undoBtn as Doc, doc.redoBtn as Doc, doc.templateButtons as Doc]); -        doc.templateDocs = new PrefetchProxy(Docs.Create.TreeDocument([doc.noteTypes as Doc, doc.templateButtons as Doc, doc.clickFuncs as Doc], { -            title: "template layouts", _xPadding: 0, -            dropConverter: ScriptField.MakeScript("convertToButtons(dragData)", { dragData: DragManager.DocumentDragData.name }) -        })); +    static setupDockedButtons(doc: Doc) { +        if (doc["dockedBtn-pen"] === undefined) { +            doc["dockedBtn-pen"] = CurrentUserUtils.ficon({ +                onClick: ScriptField.MakeScript("activatePen(this.activePen.inkPen = sameDocs(this.activePen.inkPen, this) ? undefined : this,2, this.backgroundColor)"), +                author: "systemTemplates", title: "ink mode", icon: "pen-nib", ischecked: ComputedField.MakeFunction(`sameDocs(this.activePen.inkPen,  this)`), activePen: doc +            }); +        } +        if (doc["dockedBtn-undo"] === undefined) { +            doc["dockedBtn-undo"] = CurrentUserUtils.ficon({ onClick: ScriptField.MakeScript("undo()"), title: "undo button", icon: "undo-alt" }); +        } +        if (doc["dockedBtn-redo"] === undefined) { +            doc["dockedBtn-redo"] = CurrentUserUtils.ficon({ onClick: ScriptField.MakeScript("redo()"), title: "redo button", icon: "redo-alt" }); +        } +        if (doc.dockedBtns === undefined) { +            doc.dockedBtns = CurrentUserUtils.blist({ title: "docked buttons", ignoreClick: true }, [doc["dockedBtn-undo"] as Doc, doc["dockedBtn-redo"] as Doc, doc["dockedBtn-pen"] as Doc]); +        }      } -      // sets up the default set of documents to be shown in the Overlay layer      static setupOverlays(doc: Doc) { -        doc.overlays = new PrefetchProxy(Docs.Create.FreeformDocument([], { title: "Overlays", backgroundColor: "#aca3a6" })); +        if (doc.myOverlayDocuments === undefined) { +            doc.myOverlayDocuments = new PrefetchProxy(Docs.Create.FreeformDocument([], { title: "overlay documents", backgroundColor: "#aca3a6" })); +        }      }      // the initial presentation Doc to use      static setupDefaultPresentation(doc: Doc) {          doc.searchItemTemplate = new PrefetchProxy(Docs.Create.SearchItemBoxDocument({ title: "search item template", backgroundColor: "transparent", _xMargin: 5, _height: 46, isTemplateDoc: true, isTemplateForField: "data" })); - -        doc.presentationTemplate = new PrefetchProxy(Docs.Create.PresElementBoxDocument({ title: "pres element template", backgroundColor: "transparent", _xMargin: 5, _height: 46, isTemplateDoc: true, isTemplateForField: "data" })); -        doc.curPresentation = Docs.Create.PresDocument(new List<Doc>(), { title: "Presentation", _viewType: CollectionViewType.Stacking, _LODdisable: true, _chromeStatus: "replaced", _showTitle: "title", boxShadow: "0 0" }); -    } - -    static setupMobileUploads(doc: Doc) { -        doc.optionalRightCollection = new PrefetchProxy(Docs.Create.StackingDocument([], { title: "New mobile uploads" })); +        if (doc["template-presentation"] === undefined) { +            doc["template-presentation"] = new PrefetchProxy(Docs.Create.PresElementBoxDocument({ +                title: "pres element template", backgroundColor: "transparent", _xMargin: 5, _height: 46, isTemplateDoc: true, isTemplateForField: "data" +            })); +        } +        if (doc.activePresentation === undefined) { +            doc.activePresentation = Docs.Create.PresDocument(new List<Doc>(), { +                title: "Presentation", _viewType: CollectionViewType.Stacking, +                _LODdisable: true, _chromeStatus: "replaced", _showTitle: "title", boxShadow: "0 0" +            }); +        }      } -    static setupChildClicks(doc: Doc) { -        const openInTarget = Docs.Create.ScriptingDocument(ScriptField.MakeScript( -            "docCast(thisContainer.target).then((target) => { target && docCast(this.source).then((source) => { target.proto.data = new List([source || this]); } ); } )", -            { target: Doc.name }), { title: "On Child Clicked (open in target)", _width: 300, _height: 200 }); -        const onClick = Docs.Create.ScriptingDocument(undefined, { title: "onClick", "onClick-rawScript": "console.log('click')", isTemplateDoc: true, isTemplateForField: "onClick", _width: 300, _height: 200 }, "onClick"); -        const onCheckedClick = Docs.Create.ScriptingDocument(undefined, -            { title: "onCheckedClick", "onCheckedClick-rawScript": "console.log(heading + checked + containingTreeView)", "onCheckedClick-params": new List<string>(["heading", "checked", "containingTreeView"]), isTemplateDoc: true, isTemplateForField: "onCheckedClick", _width: 300, _height: 200 }, "onCheckedClick"); -        doc.childClickFuncs = Docs.Create.TreeDocument([openInTarget], { title: "on Child Click function templates" }); -        doc.clickFuncs = Docs.Create.TreeDocument([onClick, onCheckedClick], { title: "onClick funcs" }); +    static setupRightSidebar(doc: Doc) { +        if (doc.rightSidebarCollection === undefined) { +            doc.rightSidebarCollection = new PrefetchProxy(Docs.Create.StackingDocument([], { title: "Right Sidebar" })); +        }      } -    static updateUserDocument(doc: Doc) { -        doc.title = Doc.CurrentUserEmail; -        new InkingControl(); -        (doc.iconTypes === undefined) && CurrentUserUtils.setupDefaultIconTypes(doc); -        (doc.noteTypes === undefined) && CurrentUserUtils.setupDefaultDocTemplates(doc); -        (doc.childClickFuncs === undefined) && CurrentUserUtils.setupChildClicks(doc); -        (doc.optionalRightCollection === undefined) && CurrentUserUtils.setupMobileUploads(doc); -        (doc.overlays === undefined) && CurrentUserUtils.setupOverlays(doc); -        (doc.expandingButtons === undefined) && CurrentUserUtils.setupExpandingButtons(doc); -        (doc.curPresentation === undefined) && CurrentUserUtils.setupDefaultPresentation(doc); -        (doc.sidebarButtons === undefined) && CurrentUserUtils.setupSidebarButtons(doc); +    static setupClickEditorTemplates(doc: Doc) { +        if (doc.childClickFuncs === undefined) { +            const openInTarget = Docs.Create.ScriptingDocument(ScriptField.MakeScript( +                "docCast(thisContainer.target).then((target) => { target && docCast(this.source).then((source) => { target.proto.data = new List([source || this]); } ); } )", +                { target: Doc.name }), { title: "On Child Clicked (open in target)", _width: 300, _height: 200 }); +            doc.childClickFuncs = Docs.Create.TreeDocument([openInTarget], { title: "on Child Click function templates" }); +        }          // this is equivalent to using PrefetchProxies to make sure all the childClickFuncs have been retrieved.          PromiseValue(Cast(doc.childClickFuncs, Doc)).then(func => func && PromiseValue(func.data).then(DocListCast)); -        // this is equivalent to using PrefetchProxies to make sure the recentlyClosed doc is ready -        PromiseValue(Cast(doc.recentlyClosed, Doc)).then(recent => recent && PromiseValue(recent.data).then(DocListCast)); -        // this is equivalent to using PrefetchProxies to make sure all the sidebarButtons and noteType internal Doc's have been retrieved. -        PromiseValue(Cast(doc.noteTypes, Doc)).then(noteTypes => noteTypes && PromiseValue(noteTypes.data).then(DocListCast)); + +        if (doc.clickFuncs === undefined) { +            const onClick = Docs.Create.ScriptingDocument(undefined, { +                title: "onClick", "onClick-rawScript": "console.log('click')", +                isTemplateDoc: true, isTemplateForField: "onClick", _width: 300, _height: 200 +            }, "onClick"); +            const onCheckedClick = Docs.Create.ScriptingDocument(undefined, { +                title: "onCheckedClick", "onCheckedClick-rawScript": "console.log(heading + checked + containingTreeView)", "onCheckedClick-params": new List<string>(["heading", "checked", "containingTreeView"]), isTemplateDoc: true, isTemplateForField: "onCheckedClick", _width: 300, _height: 200 +            }, "onCheckedClick"); +            doc.clickFuncs = Docs.Create.TreeDocument([onClick, onCheckedClick], { title: "onClick funcs" }); +        }          PromiseValue(Cast(doc.clickFuncs, Doc)).then(func => func && PromiseValue(func.data).then(DocListCast)); -        PromiseValue(Cast(doc.childClickFuncs, Doc)).then(func => func && PromiseValue(func.data).then(DocListCast)); -        PromiseValue(Cast(doc.sidebarButtons, Doc)).then(stackingDoc => { -            stackingDoc && PromiseValue(Cast(stackingDoc.data, listSpec(Doc))).then(sidebarButtons => { -                sidebarButtons && sidebarButtons.map((sidebarBtn, i) => { -                    sidebarBtn && PromiseValue(Cast(sidebarBtn, Doc)).then(async btn => { -                        btn && btn.sourcePanel && btn.targetContainer && i === 1 && (btn.onClick as ScriptField).script.run({ this: btn }); -                    }); -                }); -            }); -        }); + +        return doc.clickFuncs as Doc; +    } + +    static async updateUserDocument(doc: Doc) { +        new InkingControl(); +        doc.title = Doc.CurrentUserEmail; +        doc.activePen = doc; +        this.setupDefaultIconTemplates(doc);  // creates a set of icon templates triggered by the document deoration icon +        this.setupDocTemplates(doc); // sets up the template menu of templates +        this.setupRightSidebar(doc);  // sets up the right sidebar collection for mobile upload documents and sharing +        this.setupOverlays(doc);  // documents in overlay layer  +        this.setupDockedButtons(doc);  // the bottom bar of font icons +        this.setupDefaultPresentation(doc); // presentation that's initially triggered +        await this.setupSidebarButtons(doc); // the pop-out left sidebar of tools/panels +        doc.globalLinkDatabase = Docs.Prototypes.MainLinkDocument();          // 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.undoBtn && reaction(() => UndoManager.undoStack.slice(), () => Doc.GetProto(doc.undoBtn as Doc).opacity = UndoManager.CanUndo() ? 1 : 0.4, { fireImmediately: true }); -        doc.redoBtn && reaction(() => UndoManager.redoStack.slice(), () => Doc.GetProto(doc.redoBtn as Doc).opacity = UndoManager.CanRedo() ? 1 : 0.4, { fireImmediately: true }); +        doc["dockedBtn-undo"] && reaction(() => UndoManager.undoStack.slice(), () => Doc.GetProto(doc["dockedBtn-undo"] as Doc).opacity = UndoManager.CanUndo() ? 1 : 0.4, { fireImmediately: true }); +        doc["dockedBtn-redo"] && reaction(() => UndoManager.redoStack.slice(), () => Doc.GetProto(doc["dockedBtn-redo"] as Doc).opacity = UndoManager.CanRedo() ? 1 : 0.4, { fireImmediately: true }); -        this.updateCreatorButtons(doc);          return doc;      } - -    public static IsDocPinned(doc: Doc) { -        //add this new doc to props.Document -        const curPres = Cast(CurrentUserUtils.UserDocument.curPresentation, Doc) as Doc; -        if (curPres) { -            return DocListCast(curPres.data).findIndex((val) => Doc.AreProtosEqual(val, doc)) !== -1; -        } -        return false; -    } -      public static async loadCurrentUser() {          return rp.get(Utils.prepend("/getCurrentUser")).then(response => {              if (response) { @@ -427,3 +665,4 @@ export class CurrentUserUtils {  Scripting.addGlobal(function setupMobileInkingDoc(userDoc: Doc) { return CurrentUserUtils.setupMobileInkingDoc(userDoc); });  Scripting.addGlobal(function setupMobileUploadDoc(userDoc: Doc) { return CurrentUserUtils.setupMobileUploadDoc(userDoc); }); +Scripting.addGlobal(function createNewWorkspace() { return MainView.Instance.createNewWorkspace(); });
\ No newline at end of file | 
