diff options
| author | andrewdkim <adkim414@gmail.com> | 2019-09-28 14:58:17 -0400 | 
|---|---|---|
| committer | andrewdkim <adkim414@gmail.com> | 2019-09-28 14:58:17 -0400 | 
| commit | 19178bad14cda27aa36932f500ec06d684b843ef (patch) | |
| tree | 6d7538358b730068cfa7848bbb7c6f05b06d314b /src | |
| parent | bf8907cfc3e005f2ce6756820d9b3f9de35f1807 (diff) | |
| parent | a80f0867032a4735b319c87c1c7c045f062a7d4f (diff) | |
merge from master
Diffstat (limited to 'src')
39 files changed, 1194 insertions, 1460 deletions
| diff --git a/src/Utils.ts b/src/Utils.ts index 65eb3cffd..4fac53c7d 100644 --- a/src/Utils.ts +++ b/src/Utils.ts @@ -235,6 +235,21 @@ export function timenow() {      return now.toLocaleDateString() + ' ' + h + ':' + m + ' ' + ampm;  } +export function aggregateBounds(boundsList: { x: number, y: number, width: number, height: number }[]) { +    return boundsList.reduce((bounds, b) => { +        var [sptX, sptY] = [b.x, b.y]; +        let [bptX, bptY] = [sptX + b.width, sptY + b.height]; +        return { +            x: Math.min(sptX, bounds.x), y: Math.min(sptY, bounds.y), +            r: Math.max(bptX, bounds.r), b: Math.max(bptY, bounds.b) +        }; +    }, { x: Number.MAX_VALUE, y: Number.MAX_VALUE, r: -Number.MAX_VALUE, b: -Number.MAX_VALUE }); +} +export function intersectRect(r1: { left: number, top: number, width: number, height: number }, +    r2: { left: number, top: number, width: number, height: number }) { +    return !(r2.left > r1.left + r1.width || r2.left + r2.width < r1.left || r2.top > r1.top + r1.height || r2.top + r2.height < r1.top); +} +  export function percent2frac(percent: string) {      return Number(percent.substr(0, percent.length - 1)) / 100;  } diff --git a/src/client/documents/Documents.ts b/src/client/documents/Documents.ts index 4ae770e25..392dca373 100644 --- a/src/client/documents/Documents.ts +++ b/src/client/documents/Documents.ts @@ -95,12 +95,13 @@ export namespace Docs {      export namespace Prototypes { -        type LayoutSource = { LayoutString: () => string }; +        type LayoutSource = { LayoutString: (ext?: string) => string };          type CollectionLayoutSource = { LayoutString: (fieldStr: string, fieldExt?: string) => string };          type CollectionViewType = [CollectionLayoutSource, string, string?];          type PrototypeTemplate = {              layout: {                  view: LayoutSource, +                ext?: string, // optional extension field for layout source                  collectionView?: CollectionViewType              },              options?: Partial<DocumentOptions> @@ -144,7 +145,7 @@ export namespace Docs {                  options: { height: 32 }              }],              [DocumentType.PDF, { -                layout: { view: PDFBox, collectionView: [CollectionPDFView, data, anno] as CollectionViewType }, +                layout: { view: PDFBox, ext: anno },                  options: { nativeWidth: 1200, curPage: 1 }              }],              [DocumentType.ICON, { @@ -254,7 +255,7 @@ export namespace Docs {              // synthesize the default options, the type and title from computed values and              // whatever options pertain to this specific prototype              let options = { title: title, type: type, baseProto: true, ...defaultOptions, ...(template.options || {}) }; -            let primary = layout.view.LayoutString(); +            let primary = layout.view.LayoutString(layout.ext);              let collectionView = layout.collectionView;              if (collectionView) {                  options.layout = collectionView[0].LayoutString(collectionView[1], collectionView[2]); diff --git a/src/client/util/DocumentManager.ts b/src/client/util/DocumentManager.ts index a3c7429b9..e60ab09bb 100644 --- a/src/client/util/DocumentManager.ts +++ b/src/client/util/DocumentManager.ts @@ -132,9 +132,7 @@ export class DocumentManager {          let doc = Doc.GetProto(docDelegate);          const contextDoc = await Cast(doc.annotationOn, Doc);          if (contextDoc) { -            const page = NumCast(doc.page, linkPage || 0); -            const curPage = NumCast(contextDoc.curPage, page); -            if (page !== curPage) contextDoc.curPage = page; +            contextDoc.panY = doc.y;          }          let docView: DocumentView | null; @@ -188,13 +186,8 @@ export class DocumentManager {      @action      zoomIntoScale = (docDelegate: Doc, scale: number) => { -        let doc = Doc.GetProto(docDelegate); - -        let docView: DocumentView | null; -        docView = DocumentManager.Instance.getDocumentView(doc); -        if (docView) { -            docView.props.zoomToScale(scale); -        } +        let docView = DocumentManager.Instance.getDocumentView(Doc.GetProto(docDelegate)); +        docView && docView.props.zoomToScale(scale);      }      getScaleOfDocView = (docDelegate: Doc) => { diff --git a/src/client/util/DragManager.ts b/src/client/util/DragManager.ts index 56496c99b..ddc8fb62c 100644 --- a/src/client/util/DragManager.ts +++ b/src/client/util/DragManager.ts @@ -349,8 +349,8 @@ export namespace DragManager {          let xs: number[] = [];          let ys: number[] = []; -        const docs: Doc[] = -            dragData instanceof DocumentDragData ? dragData.draggedDocuments : dragData instanceof AnnotationDragData ? [dragData.dragDocument] : []; +        const docs = dragData instanceof DocumentDragData ? dragData.draggedDocuments : +            dragData instanceof AnnotationDragData ? [dragData.dragDocument] : [];          let dragElements = eles.map(ele => {              const w = ele.offsetWidth,                  h = ele.offsetHeight; @@ -379,22 +379,20 @@ export namespace DragManager {              dragElement.style.width = `${rect.width / scaleX}px`;              dragElement.style.height = `${rect.height / scaleY}px`; -            // bcz: if PDFs are rendered with svg's, then this code isn't needed -            // bcz: PDFs don't show up if you clone them when rendered using a canvas.  -            //      however, PDF's have a thumbnail field that contains an image of their canvas. -            //      So we replace the pdf's canvas with the image thumbnail -            // if (docs.length) { -            //     var pdfBox = dragElement.getElementsByClassName("pdfBox-cont")[0] as HTMLElement; -            //     let thumbnail = docs[0].GetT(KeyStore.Thumbnail, ImageField); -            //     if (pdfBox && pdfBox.childElementCount && thumbnail) { -            //         let img = new Image(); -            //         img.src = thumbnail.toString(); -            //         img.style.position = "absolute"; -            //         img.style.width = `${rect.width / scaleX}px`; -            //         img.style.height = `${rect.height / scaleY}px`; -            //         pdfBox.replaceChild(img, pdfBox.children[0]) -            //     } -            // } +            if (docs.length) { +                var pdfBox = dragElement.getElementsByTagName("canvas"); +                var pdfBoxSrc = ele.getElementsByTagName("canvas"); +                Array.from(pdfBox).map((pb, i) => pb.getContext('2d')!.drawImage(pdfBoxSrc[i], 0, 0)); +                var pdfView = dragElement.getElementsByClassName("pdfViewer-viewer"); +                var pdfViewSrc = ele.getElementsByClassName("pdfViewer-viewer"); +                let tops = Array.from(pdfViewSrc).map(p => p.scrollTop); +                let oldopacity = dragElement.style.opacity; +                dragElement.style.opacity = "0"; +                setTimeout(() => { +                    dragElement.style.opacity = oldopacity; +                    Array.from(pdfView).map((v, i) => v.scrollTo({ top: tops[i] })); +                }, 0); +            }              let set = dragElement.getElementsByTagName('*');              if (dragElement.hasAttribute("style")) (dragElement as any).style.pointerEvents = "none";              // tslint:disable-next-line: prefer-for-of @@ -418,8 +416,8 @@ export namespace DragManager {                  hideSource = options.hideSource();              }          } -        eles.map(ele => (ele.hidden = hideSource) && -            (ele.parentElement && ele.parentElement.className.indexOf("collectionFreeFormDocumentView") !== -1 && (ele.parentElement.hidden = hideSource))); + +        eles.map(ele => ele.hidden = hideSource);          let lastX = downX;          let lastY = downY; @@ -447,12 +445,9 @@ export namespace DragManager {              );          }; -        let hideDragElements = () => { +        let hideDragShowOriginalElements = () => {              dragElements.map(dragElement => dragElement.parentNode === dragDiv && dragDiv.removeChild(dragElement)); -            eles.map(ele => { -                ele.hidden = false; -                (ele.parentElement && ele.parentElement.className.indexOf("collectionFreeFormDocumentView") !== -1 && (ele.parentElement.hidden = false)); -            }); +            eles.map(ele => ele.hidden = false);          };          let endDrag = () => {              document.removeEventListener("pointermove", moveHandler, true); @@ -463,12 +458,12 @@ export namespace DragManager {          };          AbortDrag = () => { -            hideDragElements(); +            hideDragShowOriginalElements();              SelectionManager.SetIsDragging(false);              endDrag();          };          const upHandler = (e: PointerEvent) => { -            hideDragElements(); +            hideDragShowOriginalElements();              dispatchDrag(eles, e, dragData, options, finishDrag);              SelectionManager.SetIsDragging(false);              endDrag(); diff --git a/src/client/util/SelectionManager.ts b/src/client/util/SelectionManager.ts index 4c97a1056..a02a270ee 100644 --- a/src/client/util/SelectionManager.ts +++ b/src/client/util/SelectionManager.ts @@ -88,20 +88,4 @@ export namespace SelectionManager {      export function SelectedDocuments(): Array<DocumentView> {          return manager.SelectedDocuments.slice();      } -    export function ViewsSortedHorizontally(): DocumentView[] { -        let sorted = SelectionManager.SelectedDocuments().slice().sort((doc1, doc2) => { -            if (NumCast(doc1.props.Document.x) > NumCast(doc2.props.Document.x)) return 1; -            if (NumCast(doc1.props.Document.x) < NumCast(doc2.props.Document.x)) return -1; -            return 0; -        }); -        return sorted; -    } -    export function ViewsSortedVertically(): DocumentView[] { -        let sorted = SelectionManager.SelectedDocuments().slice().sort((doc1, doc2) => { -            if (NumCast(doc1.props.Document.y) > NumCast(doc2.props.Document.y)) return 1; -            if (NumCast(doc1.props.Document.y) < NumCast(doc2.props.Document.y)) return -1; -            return 0; -        }); -        return sorted; -    }  } diff --git a/src/client/views/DocumentButtonBar.tsx b/src/client/views/DocumentButtonBar.tsx index b482e3298..9ca54f738 100644 --- a/src/client/views/DocumentButtonBar.tsx +++ b/src/client/views/DocumentButtonBar.tsx @@ -23,6 +23,7 @@ import React = require("react");  import { DocumentView } from './nodes/DocumentView';  import { ParentDocSelector } from './collections/ParentDocumentSelector';  import { CollectionDockingView } from './collections/CollectionDockingView'; +import { DocumentDecorations } from './DocumentDecorations';  const higflyout = require("@hig/flyout");  export const { anchorPoints } = higflyout;  export const Flyout = higflyout.default; @@ -225,7 +226,7 @@ export class DocumentButtonBar extends React.Component<{ views: DocumentView[],          return (              <div className={"linkButtonWrapper"}>                  <div title={`${published ? "Push" : "Publish"} to Google Docs`} className="linkButton-linker" onClick={() => { -                    DocumentDecorations.hasPushedHack = false; +                    DocumentButtonBar.hasPushedHack = false;                      this.targetDoc[Pushes] = NumCast(this.targetDoc[Pushes]) + 1;                  }}>                      <FontAwesomeIcon className="documentdecorations-icon" icon={icon} size={published ? "sm" : "xs"} /> @@ -259,7 +260,7 @@ export class DocumentButtonBar extends React.Component<{ views: DocumentView[],                              window.open(`https://docs.google.com/document/d/${dataDoc[GoogleRef]}/edit`);                          } else {                              this.clearPullColor(); -                            DocumentDecorations.hasPulledHack = false; +                            DocumentButtonBar.hasPulledHack = false;                              this.targetDoc[Pulls] = NumCast(this.targetDoc[Pulls]) + 1;                              dataDoc.unchanged && runInAction(() => this.isAnimatingFetch = true);                          } diff --git a/src/client/views/DocumentDecorations.tsx b/src/client/views/DocumentDecorations.tsx index 7ec316bf9..cacaddcc8 100644 --- a/src/client/views/DocumentDecorations.tsx +++ b/src/client/views/DocumentDecorations.tsx @@ -617,7 +617,7 @@ export class DocumentDecorations extends React.Component<{}, { value: string }>              </div>              <div className="documentDecorations-container" style={{                  width: (bounds.r - bounds.x + this._resizeBorderWidth) + "px", -                height: (bounds.b - bounds.y + this._resizeBorderWidth + this._linkBoxHeight + this._titleHeight) + "px", +                height: (bounds.b - bounds.y + this._resizeBorderWidth + this._linkBoxHeight + this._titleHeight + 3) + "px",                  left: bounds.x - this._resizeBorderWidth / 2,                  top: bounds.y - this._resizeBorderWidth / 2 - this._titleHeight,                  opacity: this._opacity diff --git a/src/client/views/InkingCanvas.scss b/src/client/views/InkingCanvas.scss index 2d1142f38..b3edad459 100644 --- a/src/client/views/InkingCanvas.scss +++ b/src/client/views/InkingCanvas.scss @@ -1,7 +1,7 @@  @import "globalCssVariables";  .inkingCanvas { -    opacity: 0.99; +   // opacity: 0.99;      touch-action: none;      .jsx-parser { diff --git a/src/client/views/MainView.tsx b/src/client/views/MainView.tsx index 1526dad34..a772f9523 100644 --- a/src/client/views/MainView.tsx +++ b/src/client/views/MainView.tsx @@ -234,7 +234,7 @@ export class MainView extends React.Component {          } else {              DocServer.GetRefField(CurrentUserUtils.MainDocId).then(field => {                  field instanceof Doc ? this.openWorkspace(field) : -                    this.createNewWorkspace(CurrentUserUtils.MainDocId) +                    this.createNewWorkspace(CurrentUserUtils.MainDocId);              });          }      } @@ -375,8 +375,9 @@ export class MainView extends React.Component {      }      flyoutWidthFunc = () => this.flyoutWidth;      addDocTabFunc = (doc: Doc, data: Opt<Doc>, where: string) => { -        if (where === "close") +        if (where === "close") {              return CollectionDockingView.CloseRightSplit(doc); +        }          if (doc.dockingConfig) {              this.openWorkspace(doc);              return true; @@ -568,7 +569,7 @@ export class MainView extends React.Component {          let next = () => PresBox.CurrentPresentation.next();          let back = () => PresBox.CurrentPresentation.back();          let startOrResetPres = () => PresBox.CurrentPresentation.startOrResetPres(); -        let closePresMode = action(() => { PresBox.CurrentPresentation.presMode = false; this.addDocTabFunc(PresBox.CurrentPresentation.props.Document); }); +        let closePresMode = action(() => { PresBox.CurrentPresentation.presMode = false; this.addDocTabFunc(PresBox.CurrentPresentation.props.Document, undefined, "onRight"); });          return !PresBox.CurrentPresentation || !PresBox.CurrentPresentation.presMode ? (null) : <PresModeMenu next={next} back={back} presStatus={PresBox.CurrentPresentation.presStatus} startOrResetPres={startOrResetPres} closePresMode={closePresMode} > </PresModeMenu>;      } diff --git a/src/client/views/TemplateMenu.tsx b/src/client/views/TemplateMenu.tsx index e4ef8313d..9e5e62e03 100644 --- a/src/client/views/TemplateMenu.tsx +++ b/src/client/views/TemplateMenu.tsx @@ -117,13 +117,13 @@ export class TemplateMenu extends React.Component<TemplateMenuProps> {      @action      toggleChrome = (): void => {          this.props.docs.map(dv => { -            let layout = dv.Document.layout instanceof Doc ? dv.Document.layout as Doc : dv.Document; +            let layout = dv.Document.layout instanceof Doc ? dv.Document.layout : dv.Document;              layout.chromeStatus = (layout.chromeStatus !== "disabled" ? "disabled" : "enabled");          });      }      render() { -        let layout = this.props.docs[0].Document.layout instanceof Doc ? this.props.docs[0].Document.layout as Doc : this.props.docs[0].Document; +        let layout = this.props.docs[0].Document.layout instanceof Doc ? this.props.docs[0].Document.layout : this.props.docs[0].Document;          let templateMenu: Array<JSX.Element> = [];          this.props.templates.forEach((checked, template) =>              templateMenu.push(<TemplateToggle key={template.Name} template={template} checked={checked} toggle={this.toggleTemplate} />)); diff --git a/src/client/views/collections/CollectionBaseView.tsx b/src/client/views/collections/CollectionBaseView.tsx index 56d12bd84..0168c466f 100644 --- a/src/client/views/collections/CollectionBaseView.tsx +++ b/src/client/views/collections/CollectionBaseView.tsx @@ -20,7 +20,8 @@ export enum CollectionViewType {      Docking,      Tree,      Stacking, -    Masonry +    Masonry, +    Pivot,  }  export namespace CollectionViewType { @@ -32,7 +33,8 @@ export namespace CollectionViewType {          ["docking", CollectionViewType.Docking],          ["tree", CollectionViewType.Tree],          ["stacking", CollectionViewType.Stacking], -        ["masonry", CollectionViewType.Masonry] +        ["masonry", CollectionViewType.Masonry], +        ["pivot", CollectionViewType.Pivot]      ]);      export const valueOf = (value: string) => { diff --git a/src/client/views/collections/CollectionDockingView.tsx b/src/client/views/collections/CollectionDockingView.tsx index 8fcba99e3..b6bc4b4ba 100644 --- a/src/client/views/collections/CollectionDockingView.tsx +++ b/src/client/views/collections/CollectionDockingView.tsx @@ -9,7 +9,7 @@ import * as ReactDOM from 'react-dom';  import Measure from "react-measure";  import * as GoldenLayout from "../../../client/goldenLayout";  import { DateField } from '../../../new_fields/DateField'; -import { Doc, DocListCast, Field, Opt } from "../../../new_fields/Doc"; +import { Doc, DocListCast, Field, Opt, WidthSym, HeightSym } from "../../../new_fields/Doc";  import { Id } from '../../../new_fields/FieldSymbols';  import { List } from '../../../new_fields/List';  import { FieldId } from "../../../new_fields/RefField"; @@ -30,6 +30,7 @@ import "./CollectionDockingView.scss";  import { SubCollectionViewProps } from "./CollectionSubView";  import React = require("react");  import { ButtonSelector } from './ParentDocumentSelector'; +import { DocumentType } from '../../documents/DocumentTypes';  library.add(faFile);  @observer @@ -595,12 +596,19 @@ export class DockedFrameRenderer extends React.Component<DockedFrameProps> {      }      panelWidth = () => this._document!.ignoreAspect ? this._panelWidth : Math.min(this._panelWidth, Math.max(NumCast(this._document!.width), this.nativeWidth())); -    panelHeight = () => this._document!.ignoreAspect ? this._panelHeight : Math.min(this._panelHeight, Math.max(NumCast(this._document!.height), NumCast(this._document!.nativeHeight, this._panelHeight))); +    panelHeight = () => this._document!.ignoreAspect ? this._panelHeight : Math.min(this._panelHeight, Math.max(NumCast(this._document!.height), this.nativeHeight()));      nativeWidth = () => !this._document!.ignoreAspect ? NumCast(this._document!.nativeWidth) || this._panelWidth : 0;      nativeHeight = () => !this._document!.ignoreAspect ? NumCast(this._document!.nativeHeight) || this._panelHeight : 0;      contentScaling = () => { +        if (this._document!.type === DocumentType.PDF) { +            if (this._panelHeight / NumCast(this._document!.nativeHeight) > this._panelWidth / NumCast(this._document!.nativeWidth)) { +                return this._panelWidth / NumCast(this._document!.nativeWidth); +            } else { +                return this._panelHeight / NumCast(this._document!.nativeHeight); +            } +        }          const nativeH = this.nativeHeight();          const nativeW = this.nativeWidth();          if (!nativeW || !nativeH) return 1; @@ -619,6 +627,7 @@ export class DockedFrameRenderer extends React.Component<DockedFrameProps> {      get previewPanelCenteringOffset() { return this.nativeWidth() && !BoolCast(this._document!.ignoreAspect) ? (this._panelWidth - this.nativeWidth() / this.ScreenToLocalTransform().Scale) / 2 : 0; }      addDocTab = (doc: Doc, dataDoc: Opt<Doc>, location: string) => { +        SelectionManager.DeselectAll();          if (doc.dockingConfig) {              MainView.Instance.openWorkspace(doc);              return true; diff --git a/src/client/views/collections/CollectionPDFView.scss b/src/client/views/collections/CollectionPDFView.scss index 50201bae8..62ec8a5be 100644 --- a/src/client/views/collections/CollectionPDFView.scss +++ b/src/client/views/collections/CollectionPDFView.scss @@ -1,26 +1,4 @@ -.collectionPdfView-buttonTray { -    top: 15px; -    left: 20px; -    position: relative; -    transform-origin: left top; -    position: absolute; -} -.collectionPdfView-thumb { -    width: 25px; -    height: 25px; -    transform-origin: left top; -    position: absolute; -    background: darkgray; -} - -.collectionPdfView-slider { -    width: 25px; -    height: 25px; -    transform-origin: left top; -    position: absolute; -    background: lightgray; -}  .collectionPdfView-cont {      width: 100%; @@ -29,28 +7,5 @@      top: 0;      left: 0;      z-index: -1; +    overflow: hidden !important;  } - -.collectionPdfView-cont-dragging { -    span { -        user-select: none; -    } -} - -.collectionPdfView-backward { -    color: white; -    font-size: 24px; -    top: 0px; -    left: 0px; -    position: absolute; -    background-color: rgba(50, 50, 50, 0.2); -} - -.collectionPdfView-forward { -    color: white; -    font-size: 24px; -    top: 0px; -    left: 45px; -    position: absolute; -    background-color: rgba(50, 50, 50, 0.2); -}
\ No newline at end of file diff --git a/src/client/views/collections/CollectionPDFView.tsx b/src/client/views/collections/CollectionPDFView.tsx index 8eda4d9ee..cc8142ec0 100644 --- a/src/client/views/collections/CollectionPDFView.tsx +++ b/src/client/views/collections/CollectionPDFView.tsx @@ -1,10 +1,9 @@ -import { computed } from "mobx"; +import { trace } from "mobx";  import { observer } from "mobx-react";  import { Id } from "../../../new_fields/FieldSymbols";  import { emptyFunction } from "../../../Utils";  import { ContextMenu } from "../ContextMenu";  import { FieldView, FieldViewProps } from "../nodes/FieldView"; -import { PDFBox } from "../nodes/PDFBox";  import { CollectionBaseView, CollectionRenderProps, CollectionViewType } from "./CollectionBaseView";  import { CollectionFreeFormView } from "./collectionFreeForm/CollectionFreeFormView";  import "./CollectionPDFView.scss"; @@ -17,35 +16,18 @@ export class CollectionPDFView extends React.Component<FieldViewProps> {          return FieldView.LayoutString(CollectionPDFView, fieldKey, fieldExt);      } -    private _pdfBox?: PDFBox; -    private _buttonTray: React.RefObject<HTMLDivElement> = React.createRef(); - -    @computed -    get uIButtons() { -        return ( -            <div className="collectionPdfView-buttonTray" ref={this._buttonTray} key="tray" style={{ height: "100%" }}> -                <button className="collectionPdfView-backward" onClick={() => this._pdfBox && this._pdfBox.BackPage()}>{"<"}</button> -                <button className="collectionPdfView-forward" onClick={() => this._pdfBox && this._pdfBox.ForwardPage()}>{">"}</button> -            </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: "PDFOptions", event: emptyFunction, icon: "file-pdf" });          }      } -    setPdfBox = (pdfBox: PDFBox) => { this._pdfBox = pdfBox; }; -      subView = (_type: CollectionViewType, renderProps: CollectionRenderProps) => { -        return (<> -            <CollectionFreeFormView {...this.props} {...renderProps} setPdfBox={this.setPdfBox} CollectionView={this} chromeCollapsed={true} /> -            {renderProps.active() ? this.uIButtons : (null)} -        </>); +        return (<CollectionFreeFormView {...this.props} {...renderProps} CollectionView={this} chromeCollapsed={true} />);      }      render() { +        trace();          return (              <CollectionBaseView {...this.props} className={"collectionPdfView-cont"} onContextMenu={this.onContextMenu}>                  {this.subView} diff --git a/src/client/views/collections/CollectionSchemaCells.tsx b/src/client/views/collections/CollectionSchemaCells.tsx index 4dac27e60..179e44266 100644 --- a/src/client/views/collections/CollectionSchemaCells.tsx +++ b/src/client/views/collections/CollectionSchemaCells.tsx @@ -34,7 +34,7 @@ export interface CellProps {      row: number;      col: number;      rowProps: CellInfo; -    CollectionView: CollectionView | CollectionPDFView | CollectionVideoView; +    CollectionView: Opt<CollectionView | CollectionPDFView | CollectionVideoView>;      ContainingCollection: Opt<CollectionView | CollectionPDFView | CollectionVideoView>;      Document: Doc;      fieldKey: string; @@ -151,7 +151,7 @@ export class CollectionSchemaCell extends React.Component<CellProps> {              fieldExt: "",              ruleProvider: undefined,              ContainingCollectionView: this.props.CollectionView, -            ContainingCollectionDoc: this.props.CollectionView.props.Document, +            ContainingCollectionDoc: this.props.CollectionView && this.props.CollectionView.props.Document,              isSelected: returnFalse,              select: emptyFunction,              renderDepth: this.props.renderDepth + 1, @@ -301,7 +301,7 @@ export class CollectionSchemaCheckboxCell extends CollectionSchemaCell {      render() {          let reference = React.createRef<HTMLDivElement>();          let onItemDown = (e: React.PointerEvent) => { -            (!this.props.CollectionView.props.isSelected() ? undefined : +            (!this.props.CollectionView || !this.props.CollectionView.props.isSelected() ? undefined :                  SetupDrag(reference, () => this._document, this.props.moveDocument, this.props.Document.schemaDoc ? "copy" : undefined)(e));          };          return ( diff --git a/src/client/views/collections/CollectionSchemaView.tsx b/src/client/views/collections/CollectionSchemaView.tsx index 7bd2a1971..8d931f812 100644 --- a/src/client/views/collections/CollectionSchemaView.tsx +++ b/src/client/views/collections/CollectionSchemaView.tsx @@ -246,7 +246,7 @@ export interface SchemaTableProps {      PanelHeight: () => number;      PanelWidth: () => number;      childDocs?: Doc[]; -    CollectionView: CollectionView | CollectionPDFView | CollectionVideoView; +    CollectionView: Opt<CollectionView | CollectionPDFView | CollectionVideoView>;      ContainingCollectionView: Opt<CollectionView | CollectionPDFView | CollectionVideoView>;      ContainingCollectionDoc: Opt<Doc>;      fieldKey: string; @@ -804,7 +804,7 @@ export class SchemaTable extends React.Component<SchemaTableProps> {          csv.substring(0, csv.length - 1);          let dbName = StrCast(this.props.Document.title);          let res = await Gateway.Instance.PostSchema(csv, dbName); -        if (self.props.CollectionView.props.addDocument) { +        if (self.props.CollectionView && self.props.CollectionView.props.addDocument) {              let schemaDoc = await Docs.Create.DBDocument("https://www.cs.brown.edu/" + dbName, { title: dbName }, { dbDoc: self.props.Document });              if (schemaDoc) {                  //self.props.CollectionView.props.addDocument(schemaDoc, false); diff --git a/src/client/views/collections/CollectionStackingView.tsx b/src/client/views/collections/CollectionStackingView.tsx index ccf131797..597f3f745 100644 --- a/src/client/views/collections/CollectionStackingView.tsx +++ b/src/client/views/collections/CollectionStackingView.tsx @@ -42,7 +42,7 @@ export class CollectionStackingView extends CollectionSubView(doc => doc) {      @computed get gridGap() { return NumCast(this.props.Document.gridGap, 10); }      @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.sectionFilter && (this.props.CollectionView.props.Document.chromeStatus !== 'view-mode' && this.props.CollectionView.props.Document.chromeStatus !== 'disabled')); } +    @computed get showAddAGroup() { return (this.sectionFilter && this.props.ContainingCollectionDoc && (this.props.ContainingCollectionDoc.chromeStatus !== 'view-mode' && this.props.ContainingCollectionDoc.chromeStatus !== 'disabled')); }      @computed get columnWidth() {          return Math.min(this.props.PanelWidth() / (this.props as any).ContentScaling() - 2 * this.xMargin,              this.isStackingView ? Number.MAX_VALUE : NumCast(this.props.Document.columnWidth, 250)); @@ -347,7 +347,7 @@ export class CollectionStackingView extends CollectionSubView(doc => doc) {      }      onToggle = (checked: Boolean) => { -        this.props.CollectionView.props.Document.chromeStatus = checked ? "collapsed" : "view-mode"; +        this.props.ContainingCollectionDoc && (this.props.ContainingCollectionDoc.chromeStatus = checked ? "collapsed" : "view-mode");      }      onContextMenu = (e: React.MouseEvent): void => { @@ -391,10 +391,10 @@ export class CollectionStackingView extends CollectionSubView(doc => doc) {                          style={{ width: this.columnWidth / this.numGroupColumns - 10, marginTop: 10 }}>                          <EditableView {...editableViewProps} />                      </div>} -                {this.props.CollectionView.props.Document.chromeStatus !== 'disabled' ? <Switch +                {this.props.ContainingCollectionDoc && this.props.ContainingCollectionDoc.chromeStatus !== 'disabled' ? <Switch                      onChange={this.onToggle}                      onClick={this.onToggle} -                    defaultChecked={this.props.CollectionView.props.Document.chromeStatus !== 'view-mode'} +                    defaultChecked={this.props.ContainingCollectionDoc.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 b3b7b40dd..240adf428 100644 --- a/src/client/views/collections/CollectionStackingViewFieldColumn.tsx +++ b/src/client/views/collections/CollectionStackingViewFieldColumn.tsx @@ -266,7 +266,7 @@ export class CollectionStackingViewFieldColumn extends React.Component<CSVFieldC                  style={{                      width: (style.columnWidth) /                          ((uniqueHeadings.length + -                            ((this.props.parent.props.CollectionView.props.Document.chromeStatus !== 'view-mode' && this.props.parent.props.CollectionView.props.Document.chromeStatus !== 'disabled') ? 1 : 0)) || 1) +                            ((this.props.parent.props.ContainingCollectionDoc && this.props.parent.props.ContainingCollectionDoc.chromeStatus !== 'view-mode' && this.props.parent.props.ContainingCollectionDoc.chromeStatus !== 'disabled') ? 1 : 0)) || 1)                  }}>                  {/* the default bucket (no key value) has a tooltip that describes what it is.                      Further, it does not have a color and cannot be deleted. */} @@ -297,7 +297,7 @@ export class CollectionStackingViewFieldColumn extends React.Component<CSVFieldC              </div> : (null);          for (let i = 0; i < cols; i++) templatecols += `${style.columnWidth / style.numGroupColumns}px `;          return ( -            <div className="collectionStackingViewFieldColumn" key={heading} style={{ width: `${100 / ((uniqueHeadings.length + ((this.props.parent.props.CollectionView.props.Document.chromeStatus !== 'view-mode' && this.props.parent.props.CollectionView.props.Document.chromeStatus !== 'disabled') ? 1 : 0)) || 1)}%`, background: this._background }} +            <div className="collectionStackingViewFieldColumn" key={heading} style={{ width: `${100 / ((uniqueHeadings.length + ((this.props.parent.props.ContainingCollectionDoc && this.props.parent.props.ContainingCollectionDoc.chromeStatus !== 'view-mode' && this.props.parent.props.ContainingCollectionDoc.chromeStatus !== 'disabled') ? 1 : 0)) || 1)}%`, background: this._background }}                  ref={this.createColumnDropRef} onPointerEnter={this.pointerEntered} onPointerLeave={this.pointerLeave}>                  {headingView}                  <div key={`${heading}-stack`} className={`collectionStackingView-masonry${singleColumn ? "Single" : "Grid"}`} @@ -315,7 +315,7 @@ export class CollectionStackingViewFieldColumn extends React.Component<CSVFieldC                      {this.children(this.props.docList)}                      {singleColumn ? (null) : this.props.parent.columnDragger}                  </div> -                {(this.props.parent.props.CollectionView.props.Document.chromeStatus !== 'view-mode' && this.props.parent.props.CollectionView.props.Document.chromeStatus !== 'disabled') ? +                {(this.props.parent.props.ContainingCollectionDoc && this.props.parent.props.ContainingCollectionDoc.chromeStatus !== 'view-mode' && this.props.parent.props.ContainingCollectionDoc.chromeStatus !== 'disabled') ?                      <div key={`${heading}-add-document`} className="collectionStackingView-addDocumentButton"                          style={{ width: style.columnWidth / style.numGroupColumns }}>                          <EditableView {...newEditableViewProps} /> diff --git a/src/client/views/collections/CollectionSubView.tsx b/src/client/views/collections/CollectionSubView.tsx index c11dd6150..069269b06 100644 --- a/src/client/views/collections/CollectionSubView.tsx +++ b/src/client/views/collections/CollectionSubView.tsx @@ -1,12 +1,12 @@  import { action, computed, IReactionDisposer, reaction } from "mobx";  import * as rp from 'request-promise';  import CursorField from "../../../new_fields/CursorField"; -import { Doc, DocListCast } from "../../../new_fields/Doc"; +import { Doc, DocListCast, Opt } from "../../../new_fields/Doc";  import { Id } from "../../../new_fields/FieldSymbols";  import { List } from "../../../new_fields/List";  import { listSpec } from "../../../new_fields/Schema";  import { ScriptField } from "../../../new_fields/ScriptField"; -import { BoolCast, Cast } from "../../../new_fields/Types"; +import { Cast } from "../../../new_fields/Types";  import { CurrentUserUtils } from "../../../server/authentication/models/current_user_utils";  import { RouteStore } from "../../../server/RouteStore";  import { Utils } from "../../../Utils"; @@ -30,10 +30,11 @@ export interface CollectionViewProps extends FieldViewProps {      PanelWidth: () => number;      PanelHeight: () => number;      chromeCollapsed: boolean; +    setPreviewCursor?: (func: (x: number, y: number, drag: boolean) => void) => void;  }  export interface SubCollectionViewProps extends CollectionViewProps { -    CollectionView: CollectionView | CollectionPDFView | CollectionVideoView; +    CollectionView: Opt<CollectionView | CollectionPDFView | CollectionVideoView>;      ruleProvider: Doc | undefined;  } diff --git a/src/client/views/collections/CollectionView.tsx b/src/client/views/collections/CollectionView.tsx index 5f4742834..d3072ff1e 100644 --- a/src/client/views/collections/CollectionView.tsx +++ b/src/client/views/collections/CollectionView.tsx @@ -18,6 +18,7 @@ import { CollectionSchemaView } from "./CollectionSchemaView";  import { CollectionStackingView } from './CollectionStackingView';  import { CollectionTreeView } from "./CollectionTreeView";  import { CollectionViewBaseChrome } from './CollectionViewChromes'; +import { AddCustomFreeFormLayout } from './collectionFreeForm/CollectionFreeFormLayoutEngines';  export const COLLECTION_BORDER_WIDTH = 2;  library.add(faTh, faTree, faSquare, faProjectDiagram, faSignature, faThList, faFingerprint, faColumns, faEllipsisV, faImage, faEye as any, faCopy); @@ -59,8 +60,10 @@ export class CollectionView extends React.Component<FieldViewProps> {              case CollectionViewType.Tree: return (<CollectionTreeView chromeCollapsed={this._collapsed} key="collview" {...props} CollectionView={this} />);              case CollectionViewType.Stacking: { this.props.Document.singleColumn = true; return (<CollectionStackingView chromeCollapsed={this._collapsed} key="collview" {...props} CollectionView={this} />); }              case CollectionViewType.Masonry: { this.props.Document.singleColumn = false; return (<CollectionStackingView chromeCollapsed={this._collapsed} key="collview" {...props} CollectionView={this} />); } +            case CollectionViewType.Pivot: { this.props.Document.freeformLayoutEngine = "pivot"; return (<CollectionFreeFormView chromeCollapsed={this._collapsed} key="collview" {...props} CollectionView={this} />); }              case CollectionViewType.Freeform:              default: +                this.props.Document.freeformLayoutEngine = undefined;                  return (<CollectionFreeFormView chromeCollapsed={this._collapsed} key="collview" {...props} CollectionView={this} />);          }          return (null); @@ -89,7 +92,7 @@ export class CollectionView extends React.Component<FieldViewProps> {          if (!this.isAnnotationOverlay && !e.isPropagationStopped() && this.props.Document[Id] !== CurrentUserUtils.MainDocId) { // need to test this because GoldenLayout causes a parallel hierarchy in the React DOM for its children and the main document view7              let existingVm = ContextMenu.Instance.findByDescription("View Modes...");              let subItems: ContextMenuProps[] = existingVm && "subitems" in existingVm ? existingVm.subitems : []; -            subItems.push({ description: "Freeform", event: () => { this.props.Document.viewType = CollectionViewType.Freeform; delete this.props.Document.usePivotLayout; }, icon: "signature" }); +            subItems.push({ description: "Freeform", event: () => { this.props.Document.viewType = CollectionViewType.Freeform; }, icon: "signature" });              if (CollectionBaseView.InSafeMode()) {                  ContextMenu.Instance.addItem({ description: "Test Freeform", event: () => this.props.Document.viewType = CollectionViewType.Invalid, icon: "project-diagram" });              } @@ -103,10 +106,10 @@ export class CollectionView extends React.Component<FieldViewProps> {                  }, icon: "ellipsis-v"              });              subItems.push({ description: "Masonry", event: () => this.props.Document.viewType = CollectionViewType.Masonry, icon: "columns" }); +            subItems.push({ description: "Pivot", event: () => this.props.Document.viewType = CollectionViewType.Pivot, icon: "columns" });              switch (this.props.Document.viewType) {                  case CollectionViewType.Freeform: { -                    subItems.push({ description: "Custom", icon: "fingerprint", event: CollectionFreeFormView.AddCustomLayout(this.props.Document, this.props.fieldKey) }); -                    subItems.push({ description: "Pivot", icon: "copy", event: () => this.props.Document.usePivotLayout = true }); +                    subItems.push({ description: "Custom", icon: "fingerprint", event: AddCustomFreeFormLayout(this.props.Document, this.props.fieldKey) });                      break;                  }              } diff --git a/src/client/views/collections/CollectionViewChromes.tsx b/src/client/views/collections/CollectionViewChromes.tsx index 20786f690..cefa9eebc 100644 --- a/src/client/views/collections/CollectionViewChromes.tsx +++ b/src/client/views/collections/CollectionViewChromes.tsx @@ -260,7 +260,7 @@ export class CollectionViewBaseChrome extends React.Component<CollectionViewChro      @observable private pivotKeyDisplay = this.pivotKey;      getPivotInput = () => { -        if (!this.document.usePivotLayout) { +        if (StrCast(this.document.freeformLayoutEngine) !== "pivot") {              return (null);          }          return (<input className="collectionViewBaseChrome-viewSpecsInput" @@ -396,6 +396,7 @@ export class CollectionViewBaseChrome extends React.Component<CollectionViewChro                              <option className="collectionViewBaseChrome-viewOption" onPointerDown={stopPropagation} value="4">Tree View</option>                              <option className="collectionViewBaseChrome-viewOption" onPointerDown={stopPropagation} value="5">Stacking View</option>                              <option className="collectionViewBaseChrome-viewOption" onPointerDown={stopPropagation} value="6">Masonry View</option> +                            <option className="collectionViewBaseChrome-viewOption" onPointerDown={stopPropagation} value="7">Pivot View</option>                          </select>                          <div className="collectionViewBaseChrome-viewSpecs" style={{ display: collapsed ? "none" : "grid" }}>                              <input className="collectionViewBaseChrome-viewSpecsInput" diff --git a/src/client/views/collections/collectionFreeForm/CollectionFreeFormLayoutEngines.tsx b/src/client/views/collections/collectionFreeForm/CollectionFreeFormLayoutEngines.tsx new file mode 100644 index 000000000..886692172 --- /dev/null +++ b/src/client/views/collections/collectionFreeForm/CollectionFreeFormLayoutEngines.tsx @@ -0,0 +1,117 @@ +import { Doc, Field, FieldResult } from "../../../../new_fields/Doc"; +import { NumCast, StrCast, Cast } from "../../../../new_fields/Types"; +import { ScriptBox } from "../../ScriptBox"; +import { CompileScript } from "../../../util/Scripting"; +import { ScriptField } from "../../../../new_fields/ScriptField"; +import { OverlayView, OverlayElementOptions } from "../../OverlayView"; +import { emptyFunction } from "../../../../Utils"; +import React = require("react"); + +interface PivotData { +    type: string; +    text: string; +    x: number; +    y: number; +    width: number; +    height: number; +    fontSize: number; +} + +export interface ViewDefBounds { +    x: number; +    y: number; +    z?: number; +    width: number; +    height: number; +    transition?: string; +} + +export interface ViewDefResult { +    ele: JSX.Element; +    bounds?: ViewDefBounds; +} + +export function computePivotLayout(pivotDoc: Doc, childDocs: Doc[], childPairs: { layout: Doc, data?: Doc }[], viewDefsToJSX: (views: any) => ViewDefResult[]) { +    let layoutPoolData: Map<{ layout: Doc, data?: Doc }, any> = new Map(); +    const pivotAxisWidth = NumCast(pivotDoc.pivotWidth, 200); +    const pivotColumnGroups = new Map<FieldResult<Field>, Doc[]>(); + +    for (const doc of childDocs) { +        const val = doc[StrCast(pivotDoc.pivotField, "title")]; +        if (val) { +            !pivotColumnGroups.get(val) && pivotColumnGroups.set(val, []); +            pivotColumnGroups.get(val)!.push(doc); +        } +    } + +    const minSize = Array.from(pivotColumnGroups.entries()).reduce((min, pair) => Math.min(min, pair[1].length), Infinity); +    const numCols = NumCast(pivotDoc.pivotNumColumns, Math.ceil(Math.sqrt(minSize))); +    const docMap = new Map<Doc, ViewDefBounds>(); +    const groupNames: PivotData[] = []; + +    let x = 0; +    pivotColumnGroups.forEach((val, key) => { +        let y = 0; +        let xCount = 0; +        groupNames.push({ +            type: "text", +            text: String(key), +            x, +            y: pivotAxisWidth + 50, +            width: pivotAxisWidth * 1.25 * numCols, +            height: 100, +            fontSize: NumCast(pivotDoc.pivotFontSize, 10) +        }); +        for (const doc of val) { +            docMap.set(doc, { +                x: x + xCount * pivotAxisWidth * 1.25, +                y: -y, +                width: pivotAxisWidth, +                height: doc.nativeWidth ? (NumCast(doc.nativeHeight) / NumCast(doc.nativeWidth)) * pivotAxisWidth : pivotAxisWidth +            }); +            xCount++; +            if (xCount >= numCols) { +                xCount = 0; +                y += pivotAxisWidth * 1.25; +            } +        } +        x += pivotAxisWidth * 1.25 * (numCols + 1); +    }); + +    childPairs.map(pair => { +        let defaultPosition = { +            x: NumCast(pair.layout.x), +            y: NumCast(pair.layout.y), +            z: NumCast(pair.layout.z), +            width: NumCast(pair.layout.width), +            height: NumCast(pair.layout.height) +        }; +        const pos = docMap.get(pair.layout) || defaultPosition; +        layoutPoolData.set(pair, { transition: "transform 1s", ...pos }); +    }); +    return { map: layoutPoolData, elements: viewDefsToJSX(groupNames) }; +} + +export function AddCustomFreeFormLayout(doc: Doc, dataKey: string): () => void { +    return () => { +        let addOverlay = (key: "arrangeScript" | "arrangeInit", options: OverlayElementOptions, params?: Record<string, string>, requiredType?: string) => { +            let overlayDisposer: () => void = emptyFunction; // filled in below after we have a reference to the scriptingBox +            const scriptField = Cast(doc[key], ScriptField); +            let scriptingBox = <ScriptBox initialText={scriptField && scriptField.script.originalScript} +                // tslint:disable-next-line: no-unnecessary-callback-wrapper +                onCancel={() => overlayDisposer()}  // don't get rid of the function wrapper-- we don't want to use the current value of overlayDiposer, but the one set below +                onSave={(text, onError) => { +                    const script = CompileScript(text, { params, requiredType, typecheck: false }); +                    if (!script.compiled) { +                        onError(script.errors.map(error => error.messageText).join("\n")); +                    } else { +                        doc[key] = new ScriptField(script); +                        overlayDisposer(); +                    } +                }} />; +            overlayDisposer = OverlayView.Instance.addWindow(scriptingBox, options); +        }; +        addOverlay("arrangeInit", { x: 400, y: 100, width: 400, height: 300, title: "Layout Initialization" }, { collection: "Doc", docs: "Doc[]" }, undefined); +        addOverlay("arrangeScript", { x: 400, y: 500, width: 400, height: 300, title: "Layout Script" }, { doc: "Doc", index: "number", collection: "Doc", state: "any", docs: "Doc[]" }, "{x: number, y: number, width?: number, height?: number}"); +    }; +} diff --git a/src/client/views/collections/collectionFreeForm/CollectionFreeFormView.tsx b/src/client/views/collections/collectionFreeForm/CollectionFreeFormView.tsx index 5157d0c75..225f67b3e 100644 --- a/src/client/views/collections/collectionFreeForm/CollectionFreeFormView.tsx +++ b/src/client/views/collections/collectionFreeForm/CollectionFreeFormView.tsx @@ -1,22 +1,23 @@  import { library } from "@fortawesome/fontawesome-svg-core";  import { faEye } from "@fortawesome/free-regular-svg-icons";  import { faBraille, faChalkboard, faCompass, faCompressArrowsAlt, faExpandArrowsAlt, faPaintBrush, faTable, faUpload } from "@fortawesome/free-solid-svg-icons"; -import { action, computed, IReactionDisposer, observable, reaction, trace } from "mobx"; +import { action, computed, observable } from "mobx";  import { observer } from "mobx-react"; -import { Doc, DocListCastAsync, Field, FieldResult, HeightSym, Opt, WidthSym, DocListCast } from "../../../../new_fields/Doc"; +import { Doc, DocListCast, HeightSym, Opt, WidthSym } from "../../../../new_fields/Doc";  import { Id } from "../../../../new_fields/FieldSymbols";  import { InkField, StrokeData } from "../../../../new_fields/InkField";  import { createSchema, makeInterface } from "../../../../new_fields/Schema";  import { ScriptField } from "../../../../new_fields/ScriptField"; -import { BoolCast, Cast, FieldValue, NumCast, StrCast, PromiseValue, DateCast } from "../../../../new_fields/Types"; -import { emptyFunction, returnEmptyString, returnOne, Utils } from "../../../../Utils"; +import { BoolCast, Cast, DateCast, NumCast, StrCast } from "../../../../new_fields/Types"; +import { CurrentUserUtils } from "../../../../server/authentication/models/current_user_utils"; +import { aggregateBounds, emptyFunction, intersectRect, returnEmptyString, returnOne, Utils } from "../../../../Utils";  import { CognitiveServices } from "../../../cognitive_services/CognitiveServices"; +import { DocServer } from "../../../DocServer";  import { Docs } from "../../../documents/Documents";  import { DocumentType } from "../../../documents/DocumentTypes";  import { DocumentManager } from "../../../util/DocumentManager";  import { DragManager } from "../../../util/DragManager";  import { HistoryUtil } from "../../../util/History"; -import { CompileScript } from "../../../util/Scripting";  import { SelectionManager } from "../../../util/SelectionManager";  import { Transform } from "../../../util/Transform";  import { undoBatch, UndoManager } from "../../../util/UndoManager"; @@ -26,12 +27,12 @@ import { ContextMenuProps } from "../../ContextMenuItem";  import { InkingCanvas } from "../../InkingCanvas";  import { CollectionFreeFormDocumentView, positionSchema } from "../../nodes/CollectionFreeFormDocumentView";  import { DocumentContentsView } from "../../nodes/DocumentContentsView"; -import { DocumentViewProps, documentSchema } from "../../nodes/DocumentView"; +import { documentSchema, DocumentViewProps } from "../../nodes/DocumentView"; +import { FormattedTextBox } from "../../nodes/FormattedTextBox";  import { pageSchema } from "../../nodes/ImageBox"; -import { OverlayElementOptions, OverlayView } from "../../OverlayView";  import PDFMenu from "../../pdf/PDFMenu"; -import { ScriptBox } from "../../ScriptBox";  import { CollectionSubView } from "../CollectionSubView"; +import { computePivotLayout, ViewDefResult } from "./CollectionFreeFormLayoutEngines";  import { CollectionFreeFormLinksView } from "./CollectionFreeFormLinksView";  import { CollectionFreeFormRemoteCursors } from "./CollectionFreeFormRemoteCursors";  import "./CollectionFreeFormView.scss"; @@ -40,9 +41,6 @@ import React = require("react");  import v5 = require("uuid/v5");  import { Timeline } from "../../animationtimeline/Timeline";  import { number } from "prop-types"; -import { DocServer } from "../../../DocServer"; -import { FormattedTextBox } from "../../nodes/FormattedTextBox"; -import { CurrentUserUtils } from "../../../../server/authentication/models/current_user_utils";  library.add(faEye as any, faTable, faPaintBrush, faExpandArrowsAlt, faCompressArrowsAlt, faCompass, faUpload, faBraille, faChalkboard); @@ -54,132 +52,10 @@ export const panZoomSchema = createSchema({      arrangeInit: ScriptField,      useClusters: "boolean",      isRuleProvider: "boolean", -    fitToBox: "boolean" +    fitToBox: "boolean", +    panTransformType: "string",  }); -export interface ViewDefBounds { -    x: number; -    y: number; -    z?: number; -    width: number; -    height: number; -} - -export interface ViewDefResult { -    ele: JSX.Element; -    bounds?: ViewDefBounds; -} - -export namespace PivotView { - -    export interface PivotData { -        type: string; -        text: string; -        x: number; -        y: number; -        width: number; -        height: number; -        fontSize: number; -    } - -    export const elements = (target: CollectionFreeFormView) => { -        let collection = target.Document; -        const field = StrCast(collection.pivotField) || "title"; -        const width = NumCast(collection.pivotWidth) || 200; -        const groups = new Map<FieldResult<Field>, Doc[]>(); - -        for (const doc of target.childDocs) { -            const val = doc[field]; -            if (val === undefined) continue; - -            const l = groups.get(val); -            if (l) { -                l.push(doc); -            } else { -                groups.set(val, [doc]); -            } -        } - -        let minSize = Infinity; - -        groups.forEach((val, key) => minSize = Math.min(minSize, val.length)); - -        const numCols = NumCast(collection.pivotNumColumns) || Math.ceil(Math.sqrt(minSize)); -        const fontSize = NumCast(collection.pivotFontSize); - -        const docMap = new Map<Doc, ViewDefBounds>(); -        const groupNames: PivotData[] = []; - -        let x = 0; -        groups.forEach((val, key) => { -            let y = 0; -            let xCount = 0; -            groupNames.push({ -                type: "text", -                text: String(key), -                x, -                y: width + 50, -                width: width * 1.25 * numCols, -                height: 100, fontSize: fontSize -            }); -            for (const doc of val) { -                docMap.set(doc, { -                    x: x + xCount * width * 1.25, -                    y: -y, -                    width, -                    height: width -                }); -                xCount++; -                if (xCount >= numCols) { -                    xCount = 0; -                    y += width * 1.25; -                } -            } -            x += width * 1.25 * (numCols + 1); -        }); - -        let elements = target.viewDefsToJSX(groupNames); -        let docViews = target.childDocs.reduce((prev, doc) => { -            let minim = BoolCast(doc.isMinimized); -            if (minim === undefined || !minim) { -                let defaultPosition = (): ViewDefBounds => { -                    return { -                        x: NumCast(doc.x), -                        y: NumCast(doc.y), -                        z: NumCast(doc.z), -                        width: NumCast(doc.width), -                        height: NumCast(doc.height) -                    }; -                }; -                const pos = docMap.get(doc) || defaultPosition(); -                prev.push({ -                    ele: <CollectionFreeFormDocumentView -                        key={doc[Id]} -                        x={pos.x} -                        y={pos.y} -                        width={pos.width} -                        height={pos.height} -                        transition={"transform 1s"} -                        jitterRotation={NumCast(target.props.Document.jitterRotation)} -                        {...target.getChildDocumentViewProps(doc)} -                    />, -                    bounds: { -                        x: pos.x, -                        y: pos.y, -                        z: pos.z, -                        width: NumCast(pos.width), -                        height: NumCast(pos.height) -                    } -                }); -            } -            return prev; -        }, elements); - -        return docViews; -    }; - -} -  type PanZoomDocument = makeInterface<[typeof panZoomSchema, typeof documentSchema, typeof positionSchema, typeof pageSchema]>;  const PanZoomDocument = makeInterface(panZoomSchema, documentSchema, positionSchema, pageSchema); @@ -187,50 +63,26 @@ const PanZoomDocument = makeInterface(panZoomSchema, documentSchema, positionSch  export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {      private _lastX: number = 0;      private _lastY: number = 0; -    private get _pwidth() { return this.props.PanelWidth(); } -    private get _pheight() { return this.props.PanelHeight(); } -    private _timelineRef = React.createRef<Timeline>(); -    private get parentScaling() { -        return (this.props as any).ContentScaling && this.fitToBox && !this.isAnnotationOverlay ? (this.props as any).ContentScaling() : 1; -    } - -    ComputeContentBounds(boundsList: { x: number, y: number, width: number, height: number }[]) { -        let bounds = boundsList.reduce((bounds, b) => { -            var [sptX, sptY] = [b.x, b.y]; -            let [bptX, bptY] = [sptX + NumCast(b.width, 1), sptY + NumCast(b.height, 1)]; -            return { -                x: Math.min(sptX, bounds.x), y: Math.min(sptY, bounds.y), -                r: Math.max(bptX, bounds.r), b: Math.max(bptY, bounds.b) -            }; -        }, { x: Number.MAX_VALUE, y: Number.MAX_VALUE, r: -Number.MAX_VALUE, b: -Number.MAX_VALUE }); -        return bounds; -    } - -    @computed get actualContentBounds() { -        return this.fitToBox && !this.isAnnotationOverlay ? this.ComputeContentBounds(this.elements.filter(e => e.bounds && !e.bounds.z).map(e => e.bounds!)) : undefined; -    } - -    @computed get contentBounds() { -        let bounds = this.actualContentBounds; -        let res = { -            panX: bounds ? (bounds.x + bounds.r) / 2 : this.Document.panX || 0, -            panY: bounds ? (bounds.y + bounds.b) / 2 : this.Document.panY || 0, -            scale: (bounds ? Math.min(this.props.PanelHeight() / (bounds.b - bounds.y), this.props.PanelWidth() / (bounds.r - bounds.x)) : this.Document.scale || 1) / this.parentScaling -        }; -        if (res.scale === 0) res.scale = 1; -        return res; -    } - -    @computed get fitToBox() { return this.props.fitToBox || this.Document.fitToBox; } -    @computed get nativeWidth() { return this.fitToBox ? 0 : this.Document.nativeWidth || 0; } -    @computed get nativeHeight() { return this.fitToBox ? 0 : this.Document.nativeHeight || 0; } -    public get isAnnotationOverlay() { return this.props.fieldExt ? true : false; } // fieldExt will be "" or "annotation". should maybe generalize this, or make it more specific (ie, 'annotation' instead of 'fieldExt') +    private _clusterDistance: number = 75; +    private _hitCluster = false; +    @observable _clusterSets: (Doc[])[] = []; +    @observable _timelineRef = React.createRef<Timeline>();  + +    @computed get fitToContent() { return (this.props.fitToBox || this.Document.fitToBox) && !this.isAnnotationOverlay; } +    @computed get parentScaling() { return this.props.ContentScaling && this.fitToContent && !this.isAnnotationOverlay ? this.props.ContentScaling() : 1; } +    @computed get contentBounds() { return aggregateBounds(this.elements.filter(e => e.bounds && !e.bounds.z).map(e => e.bounds!)); } +    @computed get nativeWidth() { return this.fitToContent ? 0 : this.Document.nativeWidth || 0; } +    @computed get nativeHeight() { return this.fitToContent ? 0 : this.Document.nativeHeight || 0; } +    private get isAnnotationOverlay() { return this.props.fieldExt ? true : false; } // fieldExt will be "" or "annotation". should maybe generalize this, or make it more specific (ie, 'annotation' instead of 'fieldExt')      private get borderWidth() { return this.isAnnotationOverlay ? 0 : COLLECTION_BORDER_WIDTH; } -    private panX = () => this.contentBounds.panX; -    private panY = () => this.contentBounds.panY; -    private zoomScaling = () => this.contentBounds.scale; -    private centeringShiftX = () => !this.nativeWidth && !this.isAnnotationOverlay ? this._pwidth / 2 / this.parentScaling : 0;  // shift so pan position is at center of window for non-overlay collections -    private centeringShiftY = () => !this.nativeHeight && !this.isAnnotationOverlay ? this._pheight / 2 / this.parentScaling : 0;// shift so pan position is at center of window for non-overlay collections +    private easing = () => this.props.Document.panTransformType === "Ease"; +    private panX = () => this.fitToContent ? (this.contentBounds.x + this.contentBounds.r) / 2 : this.Document.panX || 0; +    private panY = () => this.fitToContent ? (this.contentBounds.y + this.contentBounds.b) / 2 : this.Document.panY || 0; +    private zoomScaling = () => (1 / this.parentScaling) * (this.fitToContent ? +        Math.min(this.props.PanelHeight() / (this.contentBounds.b - this.contentBounds.y), this.props.PanelWidth() / (this.contentBounds.r - this.contentBounds.x)) : +        this.Document.scale || 1); +    private centeringShiftX = () => !this.nativeWidth && !this.isAnnotationOverlay ? this.props.PanelWidth() / 2 / this.parentScaling : 0;  // shift so pan position is at center of window for non-overlay collections +    private centeringShiftY = () => !this.nativeHeight && !this.isAnnotationOverlay ? this.props.PanelHeight() / 2 / this.parentScaling : 0;// shift so pan position is at center of window for non-overlay collections      private getTransform = (): Transform => this.props.ScreenToLocalTransform().translate(-this.borderWidth + 1, -this.borderWidth + 1).translate(-this.centeringShiftX(), -this.centeringShiftY()).transform(this.getLocalTransform());      private getTransformOverlay = (): Transform => this.props.ScreenToLocalTransform().translate(-this.borderWidth + 1, -this.borderWidth + 1);      private getContainerTransform = (): Transform => this.props.ScreenToLocalTransform().translate(-this.borderWidth, -this.borderWidth); @@ -248,47 +100,31 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {          this.addDocument(newBox, false);      }      private addDocument = (newBox: Doc, allowDuplicates: boolean) => { -        this.props.addDocument(newBox, false); -        this.bringToFront(newBox); -        this.updateCluster(newBox); -        return true; +        let added = this.props.addDocument(newBox, false); +        added && this.bringToFront(newBox); +        added && this.updateCluster(newBox); +        return added;      }      private selectDocuments = (docs: Doc[]) => {          SelectionManager.DeselectAll(); -        docs.map(doc => DocumentManager.Instance.getDocumentView(doc)).filter(dv => dv).map(dv => -            SelectionManager.SelectDoc(dv!, true)); +        docs.map(doc => DocumentManager.Instance.getDocumentView(doc)).map(dv => dv && SelectionManager.SelectDoc(dv, true));      } +    public isCurrent(doc: Doc) { return !this.props.Document.isMinimized && (Math.abs(NumCast(doc.page, -1) - NumCast(this.Document.curPage, -1)) < 1.5 || NumCast(doc.page, -1) === -1); } +      public getActiveDocuments = () => { -        const curPage = FieldValue(this.Document.curPage, -1); -        return this.childLayoutPairs.filter(pair => { -            var page = NumCast(pair.layout!.page, -1); -            return page === curPage || page === -1; -        }).map(pair => pair.layout); +        return this.childLayoutPairs.filter(pair => this.isCurrent(pair.layout)).map(pair => pair.layout);      }      @computed get fieldExtensionDoc() {          return Doc.fieldExtensionDoc(this.props.DataDoc || this.props.Document, this.props.fieldKey);      } -    intersectRect(r1: { left: number, top: number, width: number, height: number }, -        r2: { left: number, top: number, width: number, height: number }) { -        return !(r2.left > r1.left + r1.width || r2.left + r2.width < r1.left || r2.top > r1.top + r1.height || r2.top + r2.height < r1.top); -    } -    _clusterDistance = 75; -    boundsOverlap(doc: Doc, doc2: Doc) { -        var x2 = NumCast(doc2.x) - this._clusterDistance; -        var y2 = NumCast(doc2.y) - this._clusterDistance; -        var w2 = NumCast(doc2.width) + this._clusterDistance; -        var h2 = NumCast(doc2.height) + this._clusterDistance; -        var x = NumCast(doc.x) - this._clusterDistance; -        var y = NumCast(doc.y) - this._clusterDistance; -        var w = NumCast(doc.width) + this._clusterDistance; -        var h = NumCast(doc.height) + this._clusterDistance; -        if (doc.z === doc2.z && this.intersectRect({ left: x, top: y, width: w, height: h }, { left: x2, top: y2, width: w2, height: h2 })) { -            return true; -        } -        return false; +    @action +    onDrop = (e: React.DragEvent): void => { +        var pt = this.getTransform().transformPoint(e.pageX, e.pageY); +        super.onDrop(e, { x: pt[0], y: pt[1] });      } +      @undoBatch      @action      drop = (e: Event, de: DragManager.DropEvent) => { @@ -304,7 +140,7 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {                      let y = (z ? ypo : yp) - de.data.offset[1];                      let dropX = NumCast(de.data.droppedDocuments[0].x);                      let dropY = NumCast(de.data.droppedDocuments[0].y); -                    de.data.droppedDocuments.forEach(d => { +                    de.data.droppedDocuments.forEach(action((d: Doc) => {                          d.x = x + NumCast(d.x) - dropX;                          d.y = y + NumCast(d.y) - dropY;                          if (!NumCast(d.width)) { @@ -316,7 +152,7 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {                              d.height = nw && nh ? nh / nw * NumCast(d.width) : 300;                          }                          this.bringToFront(d); -                    }); +                    }));                      de.data.droppedDocuments.length === 1 && this.updateCluster(de.data.droppedDocuments[0]);                  } @@ -339,18 +175,18 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {          return false;      } -    tryDragCluster(e: PointerEvent) { -        let probe = this.getTransform().transformPoint(e.clientX, e.clientY); -        let cluster = this.childLayoutPairs.map(pair => pair.layout).reduce((cluster, cd) => { +    pickCluster(probe: number[]) { +        return this.childLayoutPairs.map(pair => pair.layout).reduce((cluster, cd) => {              let cx = NumCast(cd.x) - this._clusterDistance;              let cy = NumCast(cd.y) - this._clusterDistance;              let cw = NumCast(cd.width) + 2 * this._clusterDistance;              let ch = NumCast(cd.height) + 2 * this._clusterDistance; -            if (!cd.z && this.intersectRect({ left: cx, top: cy, width: cw, height: ch }, { left: probe[0], top: probe[1], width: 1, height: 1 })) { -                return NumCast(cd.cluster); -            } -            return cluster; +            return !cd.z && intersectRect({ left: cx, top: cy, width: cw, height: ch }, { left: probe[0], top: probe[1], width: 1, height: 1 }) ? +                NumCast(cd.cluster) : cluster;          }, -1); +    } +    tryDragCluster(e: PointerEvent) { +        let cluster = this.pickCluster(this.getTransform().transformPoint(e.clientX, e.clientY));          if (cluster !== -1) {              let eles = this.childLayoutPairs.map(pair => pair.layout).filter(cd => NumCast(cd.cluster) === cluster); @@ -375,36 +211,12 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {          return false;      } -    @observable sets: (Doc[])[] = [];      @undoBatch -    @action      updateClusters(useClusters: boolean) { -        this.Document.useClusters = useClusters; -        this.sets.length = 0; -        this.childLayoutPairs.map(pair => pair.layout).map(c => { -            let included = []; -            for (let i = 0; i < this.sets.length; i++) { -                for (let member of this.sets[i]) { -                    if (this.boundsOverlap(c, member)) { -                        included.push(i); -                        break; -                    } -                } -            } -            if (included.length === 0) { -                this.sets.push([c]); -            } else if (included.length === 1) { -                this.sets[included[0]].push(c); -            } else { -                this.sets[included[0]].push(c); -                for (let s = 1; s < included.length; s++) { -                    this.sets[included[0]].push(...this.sets[included[s]]); -                    this.sets[included[s]].length = 0; -                } -            } -        }); -        this.sets.map((set, i) => set.map(member => member.cluster = i)); +        this.props.Document.useClusters = useClusters; +        this._clusterSets.length = 0; +        this.childLayoutPairs.map(pair => pair.layout).map(c => this.updateCluster(c));      }      @undoBatch @@ -412,28 +224,28 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {      updateCluster(doc: Doc) {          let childLayouts = this.childLayoutPairs.map(pair => pair.layout);          if (this.props.Document.useClusters) { -            this.sets.map(set => Doc.IndexOf(doc, set) !== -1 && set.splice(Doc.IndexOf(doc, set), 1)); +            this._clusterSets.map(set => Doc.IndexOf(doc, set) !== -1 && set.splice(Doc.IndexOf(doc, set), 1));              let preferredInd = NumCast(doc.cluster);              doc.cluster = -1; -            this.sets.map((set, i) => set.map(member => { -                if (doc.cluster === -1 && Doc.IndexOf(member, childLayouts) !== -1 && this.boundsOverlap(doc, member)) { +            this._clusterSets.map((set, i) => set.map(member => { +                if (doc.cluster === -1 && Doc.IndexOf(member, childLayouts) !== -1 && Doc.overlapping(doc, member, this._clusterDistance)) {                      doc.cluster = i;                  }              })); -            if (doc.cluster === -1 && preferredInd !== -1 && (!this.sets[preferredInd] || !this.sets[preferredInd].filter(member => Doc.IndexOf(member, childLayouts) !== -1).length)) { +            if (doc.cluster === -1 && preferredInd !== -1 && (!this._clusterSets[preferredInd] || !this._clusterSets[preferredInd].filter(member => Doc.IndexOf(member, childLayouts) !== -1).length)) {                  doc.cluster = preferredInd;              } -            this.sets.map((set, i) => { +            this._clusterSets.map((set, i) => {                  if (doc.cluster === -1 && !set.filter(member => Doc.IndexOf(member, childLayouts) !== -1).length) {                      doc.cluster = i;                  }              });              if (doc.cluster === -1) { -                doc.cluster = this.sets.length; -                this.sets.push([doc]); +                doc.cluster = this._clusterSets.length; +                this._clusterSets.push([doc]);              } else { -                for (let i = this.sets.length; i <= doc.cluster; i++) !this.sets[i] && this.sets.push([]); -                this.sets[doc.cluster].push(doc); +                for (let i = this._clusterSets.length; i <= doc.cluster; i++) !this._clusterSets[i] && this._clusterSets.push([]); +                this._clusterSets[doc.cluster].push(doc);              }          }      } @@ -442,13 +254,13 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {          let clusterColor = "";          let cluster = NumCast(doc.cluster);          if (this.Document.useClusters) { -            if (this.sets.length <= cluster) { +            if (this._clusterSets.length <= cluster) {                  setTimeout(() => this.updateCluster(doc), 0);              } else {                  // choose a cluster color from a palette                  let colors = ["#da42429e", "#31ea318c", "#8c4000", "#4a7ae2c4", "#d809ff", "#ff7601", "#1dffff", "yellow", "#1b8231f2", "#000000ad"];                  clusterColor = colors[cluster % colors.length]; -                let set = this.sets.length > cluster ? this.sets[cluster].filter(s => s.backgroundColor && (s.backgroundColor !== s.defaultBackgroundColor)) : undefined; +                let set = this._clusterSets[cluster] && this._clusterSets[cluster].filter(s => s.backgroundColor && (s.backgroundColor !== s.defaultBackgroundColor));                  // override the cluster color with an explicitly set color on a non-background document.  then override that with an explicitly set color on a background document                  set && set.filter(s => !s.isBackground).map(s => clusterColor = StrCast(s.backgroundColor));                  set && set.filter(s => s.isBackground).map(s => clusterColor = StrCast(s.backgroundColor)); @@ -459,6 +271,7 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {      @action      onPointerDown = (e: React.PointerEvent): void => { +        this._hitCluster = this.props.Document.useClusters ? this.pickCluster(this.getTransform().transformPoint(e.clientX, e.clientY)) !== -1 : false;          if (e.button === 0 && !e.shiftKey && !e.altKey && (!this.isAnnotationOverlay || this.zoomScaling() !== 1) && this.props.active()) {              document.removeEventListener("pointermove", this.onPointerMove);              document.removeEventListener("pointerup", this.onPointerUp); @@ -476,8 +289,8 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {      @action      onPointerMove = (e: PointerEvent): void => { -        if (!e.cancelBubble) { -            if (this.props.Document.useClusters && this.tryDragCluster(e)) { +        if (!e.cancelBubble && !this.isAnnotationOverlay) { +            if (this._hitCluster && this.tryDragCluster(e)) {                  e.stopPropagation(); // doesn't actually stop propagation since all our listeners are listening to events on 'document'  however it does mark the event as cancelBubble=true which we test for in the move event handlers                  e.preventDefault();                  document.removeEventListener("pointermove", this.onPointerMove); @@ -512,8 +325,8 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {                  }                  let cscale = this.props.ContainingCollectionDoc ? NumCast(this.props.ContainingCollectionDoc.scale) : 1; -                let panelDim = this.props.ScreenToLocalTransform().transformDirection(this._pwidth / this.zoomScaling() * cscale, -                    this._pheight / this.zoomScaling() * cscale); +                let panelDim = this.props.ScreenToLocalTransform().transformDirection(this.props.PanelWidth() / this.zoomScaling() * cscale, +                    this.props.PanelHeight() / this.zoomScaling() * cscale);                  if (ranges[0][0] - dx > (this.panX() + panelDim[0] / 2)) x = ranges[0][1] + panelDim[0] / 2;                  if (ranges[0][1] - dx < (this.panX() - panelDim[0] / 2)) x = ranges[0][0] - panelDim[0] / 2;                  if (ranges[1][0] - dy > (this.panY() + panelDim[1] / 2)) y = ranges[1][1] + panelDim[1] / 2; @@ -529,7 +342,7 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {      @action      onPointerWheel = (e: React.WheelEvent): void => { -        if (BoolCast(this.props.Document.lockedPosition)) return; +        if (this.props.Document.lockedPosition || this.isAnnotationOverlay) return;          if (!e.ctrlKey && this.props.Document.scrollHeight !== undefined) { // things that can scroll vertically should do that instead of zooming              e.stopPropagation();          } @@ -546,13 +359,12 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {              let safeScale = Math.min(Math.max(0.15, localTransform.Scale), 40);              this.props.Document.scale = Math.abs(safeScale);              this.setPan(-localTransform.TranslateX / safeScale, -localTransform.TranslateY / safeScale); -            e.preventDefault();          }      }      @action      setPan(panX: number, panY: number) { -        if (!BoolCast(this.props.Document.lockedPosition)) { +        if (!this.props.Document.lockedPosition) {              this.props.Document.panTransformType = "None";              var scale = this.getLocalTransform().inverse().Scale;              const newPanX = Math.min((1 - 1 / scale) * this.nativeWidth, Math.max(0, panX)); @@ -562,12 +374,6 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {          }      } -    @action -    onDrop = (e: React.DragEvent): void => { -        var pt = this.getTransform().transformPoint(e.pageX, e.pageY); -        super.onDrop(e, { x: pt[0], y: pt[1] }); -    } -      bringToFront = (doc: Doc, sendToBack?: boolean) => {          if (sendToBack || doc.isBackground) {              doc.zIndex = 0; @@ -605,48 +411,32 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {          newState.initializers![this.Document[Id]] = { panX: newPanX, panY: newPanY };          HistoryUtil.pushState(newState); -        let px = this.Document.panX; -        let py = this.Document.panY; -        let s = this.Document.scale; -        this.setPan(newPanX, newPanY); +        let savedState = { px: this.Document.panX, py: this.Document.panY, s: this.Document.scale, pt: this.Document.panTransformType }; -        this.props.Document.panTransformType = "Ease"; +        this.setPan(newPanX, newPanY); +        this.Document.panTransformType = "Ease";          this.props.focus(this.props.Document); -        if (willZoom) { -            this.setScaleToZoom(doc, scale); -        } -        console.log("Focused " + this.Document.title + " " + s); +        willZoom && this.setScaleToZoom(doc, scale); +          afterFocus && setTimeout(() => {              if (afterFocus && afterFocus()) { -                console.log("UnFocused " + this.Document.title + " " + s); -                this.Document.panX = px; -                this.Document.panY = py; -                this.Document.scale = s; +                this.Document.panX = savedState.px; +                this.Document.panY = savedState.py; +                this.Document.scale = savedState.s; +                this.Document.panTransformType = savedState.pt;              }          }, 1000);      }      setScaleToZoom = (doc: Doc, scale: number = 0.5) => { -        let p = this.props; -        let PanelHeight = p.PanelHeight(); -        let panelWidth = p.PanelWidth(); - -        let docHeight = NumCast(doc.height); -        let docWidth = NumCast(doc.width); -        let targetHeight = scale * PanelHeight; -        let targetWidth = scale * panelWidth; - -        let maxScaleX: number = targetWidth / docWidth; -        let maxScaleY: number = targetHeight / docHeight; -        let maxApplicableScale = Math.min(maxScaleX, maxScaleY); -        this.Document.scale = maxApplicableScale; +        this.Document.scale = scale * Math.min(this.props.PanelWidth() / NumCast(doc.width), this.props.PanelHeight() / NumCast(doc.height));      }      zoomToScale = (scale: number) => {          this.Document.scale = scale;      } -    getScale = () => this.Document.scale ? this.Document.scale : 1; +    getScale = () => this.Document.scale || 1;      getChildDocumentViewProps(childLayout: Doc, childData?: Doc): DocumentViewProps {          return { @@ -663,7 +453,7 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {              PanelHeight: childLayout[HeightSym],              ContentScaling: returnOne,              ContainingCollectionView: this.props.CollectionView, -            ContainingCollectionDoc: this.props.CollectionView.props.Document, +            ContainingCollectionDoc: this.props.ContainingCollectionDoc,              focus: this.focusDocument,              backgroundColor: this.getClusterColor,              parentActive: this.props.active, @@ -690,7 +480,7 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {              PanelHeight: layoutDoc[HeightSym],              ContentScaling: returnOne,              ContainingCollectionView: this.props.CollectionView, -            ContainingCollectionDoc: this.props.CollectionView.props.Document, +            ContainingCollectionDoc: this.props.ContainingCollectionDoc,              focus: this.focusDocument,              backgroundColor: returnEmptyString,              parentActive: this.props.active, @@ -713,110 +503,109 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {      }      viewDefsToJSX = (views: any[]) => { -        let elements: ViewDefResult[] = []; -        if (Array.isArray(views)) { -            elements = views.reduce<typeof elements>((prev, ele) => { -                const jsx = this.viewDefToJSX(ele); -                jsx && prev.push(jsx); -                return prev; -            }, elements); -        } -        return elements; +        return !Array.isArray(views) ? [] : views.filter(ele => this.viewDefToJSX(ele)).map(ele => this.viewDefToJSX(ele)!);      }      private viewDefToJSX(viewDef: any): Opt<ViewDefResult> {          if (viewDef.type === "text") { -            const text = Cast(viewDef.text, "string"); +            const text = Cast(viewDef.text, "string"); // don't use NumCast, StrCast, etc since we want to test for undefined below              const x = Cast(viewDef.x, "number");              const y = Cast(viewDef.y, "number");              const z = Cast(viewDef.z, "number");              const width = Cast(viewDef.width, "number");              const height = Cast(viewDef.height, "number");              const fontSize = Cast(viewDef.fontSize, "number"); -            if ([text, x, y, width, height].some(val => val === undefined)) { -                return undefined; -            } - -            return { -                ele: <div className="collectionFreeform-customText" style={{ -                    transform: `translate(${x}px, ${y}px)`, -                    width, height, fontSize -                }}>{text}</div>, bounds: { x: x!, y: y!, z: z, width: width!, height: height! } -            }; +            return [text, x, y, width, height].some(val => val === undefined) ? undefined : +                { +                    ele: <div className="collectionFreeform-customText" style={{ width, height, fontSize, transform: `translate(${x}px, ${y}px)` }}> +                        {text} +                    </div>, +                    bounds: { x: x!, y: y!, z: z, width: width!, height: height! } +                };          }      } -    @computed.struct -    get elements() { -        if (this.Document.usePivotLayout) return PivotView.elements(this); -        let curPage = FieldValue(this.Document.curPage, -1); -        const initScript = this.Document.arrangeInit; -        let state: any = undefined; -        let pairs = this.childLayoutPairs; -        let elements: ViewDefResult[] = []; -        if (initScript) { -            const initResult = initScript.script.run({ docs: pairs.map(pair => pair.layout), collection: this.Document }, console.log); -            if (initResult.success) { -                const result = initResult.result; -                const { state: scriptState, views } = result; -                state = scriptState; -                elements = this.viewDefsToJSX(views); -            } +    lookupLayout = (doc: Doc, dataDoc?: Doc) => { +        let data: any = undefined; +        let computedElementData: { map: Map<{ layout: Doc, data?: Doc | undefined }, any>, elements: ViewDefResult[] }; +        switch (this.Document.freeformLayoutEngine) { +            case "pivot": computedElementData = this.doPivotLayout; break; +            default: computedElementData = this.doFreeformLayout; break;          } -        let docviews = pairs.reduce((prev, pair) => { -            var page = NumCast(pair.layout.page, -1); -            if (!pair.layout.isMinimized && ((Math.abs(Math.round(page) - Math.round(curPage)) < 3) || page === -1)) { -                const pos = this.getCalculatedPositions({ doc: pair.layout, index: prev.length, collection: this.Document, docs: pairs.map(pair => pair.layout), state }); -                state = pos.state === undefined ? state : pos.state; -                prev.push({ -                    ele: <CollectionFreeFormDocumentView key={pair.layout[Id]} -                        ruleProvider={this.Document.isRuleProvider ? this.props.Document : this.props.ruleProvider} -                        jitterRotation={NumCast(this.props.Document.jitterRotation)} -                        transition={pos.transition} x={pos.x} y={pos.y} width={pos.width} height={pos.height} -                        {...this.getChildDocumentViewProps(pair.layout, pair.data)} />, -                    bounds: { x: pos.x || 0, y: pos.y || 0, z: pos.z, width: pos.width || 0, height: pos.height || 0 } -                }); +        computedElementData.map.forEach((value: any, key: { layout: Doc, data?: Doc }) => { +            if (key.layout === doc && key.data === dataDoc) { +                data = value;              } -            // } -            return prev; -        }, elements); +        }); +        return data && { x: data.x, y: data.y, z: data.z, width: data.width, height: data.height, transition: data.transition }; +    } -        return docviews; +    @computed +    get doPivotLayout() { +        return computePivotLayout(this.props.Document, this.childDocs, +            this.childLayoutPairs.filter(pair => this.isCurrent(pair.layout)), this.viewDefsToJSX);      } -    @computed.struct -    get views() { -        return this.elements.filter(ele => ele.bounds && !ele.bounds.z).map(ele => ele.ele); +    @computed +    get doFreeformLayout() { +        let layoutPoolData: Map<{ layout: Doc, data?: Doc }, any> = new Map(); +        let layoutDocs = this.childLayoutPairs.map(pair => pair.layout); +        const initResult = this.Document.arrangeInit && this.Document.arrangeInit.script.run({ docs: layoutDocs, collection: this.Document }, console.log); +        let state = initResult && initResult.success ? initResult.result.scriptState : undefined; +        let elements = initResult && initResult.success ? this.viewDefsToJSX(initResult.result.views) : []; + +        this.childLayoutPairs.filter(pair => this.isCurrent(pair.layout)).map((pair, i) => { +            const pos = this.getCalculatedPositions({ doc: pair.layout, index: i, collection: this.Document, docs: layoutDocs, state }); +            state = pos.state === undefined ? state : pos.state; +            layoutPoolData.set(pair, pos); +        }); +        return { map: layoutPoolData, elements: elements };      } -    @computed.struct -    get overlayViews() { -        return this.elements.filter(ele => ele.bounds && ele.bounds.z).map(ele => ele.ele); + +    @computed +    get doLayoutComputation() { +        let computedElementData: { map: Map<{ layout: Doc, data?: Doc | undefined }, any>, elements: ViewDefResult[] }; +        switch (this.Document.freeformLayoutEngine) { +            case "pivot": computedElementData = this.doPivotLayout; break; +            default: computedElementData = this.doFreeformLayout; break; +        } +        this.childLayoutPairs.filter(pair => this.isCurrent(pair.layout)).forEach(pair => +            computedElementData.elements.push({ +                ele: <CollectionFreeFormDocumentView key={pair.layout[Id]} dataProvider={this.lookupLayout} +                    ruleProvider={this.Document.isRuleProvider ? this.props.Document : this.props.ruleProvider} +                    jitterRotation={NumCast(this.props.Document.jitterRotation)} {...this.getChildDocumentViewProps(pair.layout, pair.data)} />, +                bounds: this.lookupLayout(pair.layout, pair.data) +            })); + +        return computedElementData;      } +    @computed.struct get elements() { return this.doLayoutComputation.elements; } +    @computed.struct get views() { return this.elements.filter(ele => ele.bounds && !ele.bounds.z).map(ele => ele.ele); } +    @computed.struct get overlayViews() { return this.elements.filter(ele => ele.bounds && ele.bounds.z).map(ele => ele.ele); } +      @action      onCursorMove = (e: React.PointerEvent) => {          super.setCursorPosition(this.getTransform().transformPoint(e.clientX, e.clientY));      } -    arrangeContents = async () => { -        const docs = await DocListCastAsync(this.Document[this.props.fieldKey]); +    layoutDocsInGrid = () => {          UndoManager.RunInBatch(() => { -            if (docs) { -                let startX = this.Document.panX || 0; -                let x = startX; -                let y = this.Document.panY || 0; -                let i = 0; -                const width = Math.max(...docs.map(doc => NumCast(doc.width))); -                const height = Math.max(...docs.map(doc => NumCast(doc.height))); -                for (const doc of docs) { -                    doc.x = x; -                    doc.y = y; -                    x += width + 20; -                    if (++i === 6) { -                        i = 0; -                        x = startX; -                        y += height + 20; -                    } +            const docs = DocListCast(this.Document[this.props.fieldKey]); +            let startX = this.Document.panX || 0; +            let x = startX; +            let y = this.Document.panY || 0; +            let i = 0; +            const width = Math.max(...docs.map(doc => NumCast(doc.width))); +            const height = Math.max(...docs.map(doc => NumCast(doc.height))); +            for (const doc of docs) { +                doc.x = x; +                doc.y = y; +                x += width + 20; +                if (++i === 6) { +                    i = 0; +                    x = startX; +                    y += height + 20;                  }              }          }, "arrange contents"); @@ -852,10 +641,10 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {          }          this._timelineRef.current!.timelineContextMenu(e.nativeEvent);          layoutItems.push({ description: "reset view", event: () => { this.props.Document.panX = this.props.Document.panY = 0; this.props.Document.scale = 1; }, icon: "compress-arrows-alt" }); -        layoutItems.push({ description: `${this.fitToBox ? "Unset" : "Set"} Fit To Container`, event: async () => this.Document.fitToBox = !this.fitToBox, icon: !this.fitToBox ? "expand-arrows-alt" : "compress-arrows-alt" }); +        layoutItems.push({ description: `${this.fitToContent ? "Unset" : "Set"} Fit To Container`, event: async () => this.Document.fitToBox = !this.fitToContent, icon: !this.fitToContent ? "expand-arrows-alt" : "compress-arrows-alt" });          layoutItems.push({ description: `${this.Document.useClusters ? "Uncluster" : "Use Clusters"}`, event: () => this.updateClusters(!this.Document.useClusters), icon: "braille" });          layoutItems.push({ description: `${this.Document.isRuleProvider ? "Stop Auto Format" : "Auto Format"}`, event: this.autoFormat, icon: "chalkboard" }); -        layoutItems.push({ description: "Arrange contents in grid", event: this.arrangeContents, icon: "table" }); +        layoutItems.push({ description: "Arrange contents in grid", event: this.layoutDocsInGrid, icon: "table" });          layoutItems.push({ description: "Analyze Strokes", event: this.analyzeStrokes, icon: "paint-brush" });          layoutItems.push({ description: "Jitter Rotation", event: action(() => this.props.Document.jitterRotation = 10), icon: "paint-brush" });          layoutItems.push({ @@ -910,54 +699,26 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {          <CollectionFreeFormBackgroundView key="backgroundView" {...this.props} {...this.getDocumentViewProps(this.props.Document)} />,          ...this.views      ] - -    public static AddCustomLayout(doc: Doc, dataKey: string): () => void { -        return () => { -            let addOverlay = (key: "arrangeScript" | "arrangeInit", options: OverlayElementOptions, params?: Record<string, string>, requiredType?: string) => { -                let overlayDisposer: () => void = emptyFunction; -                const script = Cast(doc[key], ScriptField); -                let originalText: string | undefined = undefined; -                if (script) originalText = script.script.originalScript; -                // tslint:disable-next-line: no-unnecessary-callback-wrapper -                let scriptingBox = <ScriptBox initialText={originalText} onCancel={() => overlayDisposer()} onSave={(text, onError) => { -                    const script = CompileScript(text, { -                        params, -                        requiredType, -                        typecheck: false -                    }); -                    if (!script.compiled) { -                        onError(script.errors.map(error => error.messageText).join("\n")); -                        return; -                    } -                    doc[key] = new ScriptField(script); -                    overlayDisposer(); -                }} />; -                overlayDisposer = OverlayView.Instance.addWindow(scriptingBox, options); -            }; -            addOverlay("arrangeInit", { x: 400, y: 100, width: 400, height: 300, title: "Layout Initialization" }, { collection: "Doc", docs: "Doc[]" }, undefined); -            addOverlay("arrangeScript", { x: 400, y: 500, width: 400, height: 300, title: "Layout Script" }, { doc: "Doc", index: "number", collection: "Doc", state: "any", docs: "Doc[]" }, "{x: number, y: number, width?: number, height?: number}"); -        }; -    }      render() {          // update the actual dimensions of the collection so that they can inquired (e.g., by a minimap) -        this.props.Document.fitX = this.actualContentBounds && this.actualContentBounds.x; -        this.props.Document.fitY = this.actualContentBounds && this.actualContentBounds.y; -        this.props.Document.fitW = this.actualContentBounds && (this.actualContentBounds.r - this.actualContentBounds.x); -        this.props.Document.fitH = this.actualContentBounds && (this.actualContentBounds.b - this.actualContentBounds.y); +        this.props.Document.fitX = this.contentBounds && this.contentBounds.x; +        this.props.Document.fitY = this.contentBounds && this.contentBounds.y; +        this.props.Document.fitW = this.contentBounds && (this.contentBounds.r - this.contentBounds.x); +        this.props.Document.fitH = this.contentBounds && (this.contentBounds.b - this.contentBounds.y);          // if fieldExt is set, then children will be stored in the extension document for the fieldKey.           // otherwise, they are stored in fieldKey.  All annotations to this document are stored in the extension document          Doc.UpdateDocumentExtensionForField(this.props.DataDoc || this.props.Document, this.props.fieldKey); -        const easing = () => this.props.Document.panTransformType === "Ease";          return (              <div className={"collectionfreeformview-container"} ref={this.createDropTarget} onWheel={this.onPointerWheel} +                style={{ pointerEvents: SelectionManager.GetIsDragging() ? "all" : undefined, height: this.isAnnotationOverlay ? (NumCast(this.props.Document.scrollHeight) ? NumCast(this.props.Document.scrollHeight) : "100%") : this.props.PanelHeight() }}                  onPointerDown={this.onPointerDown} onPointerMove={this.onCursorMove} onDrop={this.onDrop.bind(this)} onContextMenu={this.onContextMenu}>                  <MarqueeView container={this} activeDocuments={this.getActiveDocuments} selectDocuments={this.selectDocuments} isSelected={this.props.isSelected} -                    addDocument={this.addDocument} removeDocument={this.props.removeDocument} addLiveTextDocument={this.addLiveTextBox} -                    getContainerTransform={this.getContainerTransform} getTransform={this.getTransform}> +                    addDocument={this.addDocument} removeDocument={this.props.removeDocument} addLiveTextDocument={this.addLiveTextBox} setPreviewCursor={this.props.setPreviewCursor} +                    getContainerTransform={this.getContainerTransform} getTransform={this.getTransform} isAnnotationOverlay={this.isAnnotationOverlay}>                      <CollectionFreeFormViewPannableContents centeringShiftX={this.centeringShiftX} centeringShiftY={this.centeringShiftY} -                        easing={easing} zoomScaling={this.zoomScaling} panX={this.panX} panY={this.panY}> +                        easing={this.easing} zoomScaling={this.zoomScaling} panX={this.panX} panY={this.panY}>                          <CollectionFreeFormLinksView {...this.props} key="freeformLinks"> -                            <InkingCanvas getScreenTransform={this.getTransform} Document={this.props.Document} AnnotationDocument={this.fieldExtensionDoc} inkFieldKey={this._inkKey} > +                            <InkingCanvas getScreenTransform={this.getTransform} Document={this.props.Document} AnnotationDocument={this.fieldExtensionDoc} inkFieldKey={"ink"} >                                  {this.childViews}                              </InkingCanvas>                          </CollectionFreeFormLinksView> @@ -974,23 +735,18 @@ export class CollectionFreeFormView extends CollectionSubView(PanZoomDocument) {  @observer  class CollectionFreeFormOverlayView extends React.Component<DocumentViewProps & { isSelected: () => boolean }> { -    @computed get overlayView() { -        return (<DocumentContentsView {...this.props} layoutKey={"overlayLayout"} -            renderDepth={this.props.renderDepth} isSelected={this.props.isSelected} select={emptyFunction} />); -    }      render() { -        return this.overlayView; +        return <DocumentContentsView {...this.props} layoutKey={"overlayLayout"} +            renderDepth={this.props.renderDepth} isSelected={this.props.isSelected} select={emptyFunction} />;      }  }  @observer  class CollectionFreeFormBackgroundView extends React.Component<DocumentViewProps & { isSelected: () => boolean }> { -    @computed get backgroundView() { -        return (<DocumentContentsView {...this.props} layoutKey={"backgroundLayout"} -            renderDepth={this.props.renderDepth} isSelected={this.props.isSelected} select={emptyFunction} />); -    }      render() { -        return this.props.Document.backgroundLayout ? this.backgroundView : (null); +        return !this.props.Document.backgroundLayout ? (null) : +            (<DocumentContentsView {...this.props} layoutKey={"backgroundLayout"} +                renderDepth={this.props.renderDepth} isSelected={this.props.isSelected} select={emptyFunction} />);      }  } diff --git a/src/client/views/collections/collectionFreeForm/MarqueeView.tsx b/src/client/views/collections/collectionFreeForm/MarqueeView.tsx index bbea4a555..44611869e 100644 --- a/src/client/views/collections/collectionFreeForm/MarqueeView.tsx +++ b/src/client/views/collections/collectionFreeForm/MarqueeView.tsx @@ -30,6 +30,8 @@ interface MarqueeViewProps {      removeDocument: (doc: Doc) => boolean;      addLiveTextDocument: (doc: Doc) => void;      isSelected: () => boolean; +    isAnnotationOverlay: boolean; +    setPreviewCursor?: (func: (x: number, y: number, drag: boolean) => void) => void;  }  @observer @@ -43,6 +45,10 @@ export class MarqueeView extends React.Component<MarqueeViewProps>      @observable _visible: boolean = false;      _commandExecuted = false; +    componentDidMount() { +        this.props.setPreviewCursor && this.props.setPreviewCursor(this.setPreviewCursor); +    } +      @action      cleanupInteractions = (all: boolean = false) => {          if (all) { @@ -145,15 +151,10 @@ export class MarqueeView extends React.Component<MarqueeViewProps>      }      @action      onPointerDown = (e: React.PointerEvent): void => { -        this._downX = this._lastX = e.pageX; -        this._downY = this._lastY = e.pageY; -        this._commandExecuted = false; -        PreviewCursor.Visible = false; -        this.cleanupInteractions(true); +        this._downX = this._lastX = e.clientX; +        this._downY = this._lastY = e.clientY;          if (e.button === 2 || (e.button === 0 && e.altKey)) { -            document.addEventListener("pointermove", this.onPointerMove, true); -            document.addEventListener("pointerup", this.onPointerUp, true); -            document.addEventListener("keydown", this.marqueeCommand, true); +            this.setPreviewCursor(e.clientX, e.clientY, true);              if (e.altKey) {                  //e.stopPropagation(); // bcz: removed so that you can alt-click on button in a collection to switch link following behaviors.                  e.preventDefault(); @@ -176,6 +177,8 @@ export class MarqueeView extends React.Component<MarqueeViewProps>                  e.stopPropagation();                  e.preventDefault();              } +        } else { +            this.cleanupInteractions(true); // stop listening for events if another lower-level handle (e.g. another Marquee) has stopPropagated this          }          if (e.altKey) {              e.preventDefault(); @@ -202,11 +205,28 @@ export class MarqueeView extends React.Component<MarqueeViewProps>          }      } +    setPreviewCursor = (x: number, y: number, drag: boolean) => { +        if (drag) { +            this._downX = this._lastX = x; +            this._downY = this._lastY = y; +            this._commandExecuted = false; +            PreviewCursor.Visible = false; +            this.cleanupInteractions(true); +            document.addEventListener("pointermove", this.onPointerMove, true); +            document.addEventListener("pointerup", this.onPointerUp, true); +            document.addEventListener("keydown", this.marqueeCommand, true); +        } else { +            this._downX = x; +            this._downY = y; +            PreviewCursor.Show(x, y, this.onKeyPress, this.props.addLiveTextDocument, this.props.getTransform, this.props.addDocument); +        } +    } +      @action      onClick = (e: React.MouseEvent): void => {          if (Math.abs(e.clientX - this._downX) < Utils.DRAG_THRESHOLD &&              Math.abs(e.clientY - this._downY) < Utils.DRAG_THRESHOLD) { -            PreviewCursor.Show(e.clientX, e.clientY, this.onKeyPress, this.props.addLiveTextDocument, this.props.getTransform, this.props.addDocument); +            this.setPreviewCursor(e.clientX, e.clientY, false);              // let the DocumentView stopPropagation of this event when it selects this document          } else {  // why do we get a click event when the cursor have moved a big distance?              // let's cut it off here so no one else has to deal with it. @@ -297,8 +317,8 @@ export class MarqueeView extends React.Component<MarqueeViewProps>                  y: bounds.top,                  panX: 0,                  panY: 0, -                backgroundColor: this.props.container.isAnnotationOverlay ? undefined : chosenColor, -                defaultBackgroundColor: this.props.container.isAnnotationOverlay ? undefined : chosenColor, +                backgroundColor: this.props.isAnnotationOverlay ? undefined : chosenColor, +                defaultBackgroundColor: this.props.isAnnotationOverlay ? undefined : chosenColor,                  width: bounds.width,                  height: bounds.height,                  title: "a nested collection", diff --git a/src/client/views/nodes/CollectionFreeFormDocumentView.tsx b/src/client/views/nodes/CollectionFreeFormDocumentView.tsx index fcf483659..af2651dc8 100644 --- a/src/client/views/nodes/CollectionFreeFormDocumentView.tsx +++ b/src/client/views/nodes/CollectionFreeFormDocumentView.tsx @@ -12,6 +12,7 @@ import { Doc, WidthSym, HeightSym } from "../../../new_fields/Doc";  import { random } from "animejs";  export interface CollectionFreeFormDocumentViewProps extends DocumentViewProps { +    dataProvider?: (doc: Doc, dataDoc?: Doc) => { x: number, y: number, width: number, height: number, z: number, transition?: string } | undefined;      x?: number;      y?: number;      width?: number; @@ -32,14 +33,14 @@ export const PositionDocument = makeInterface(documentSchema, positionSchema);  @observer  export class CollectionFreeFormDocumentView extends DocComponent<CollectionFreeFormDocumentViewProps, PositionDocument>(PositionDocument) {      _disposer: IReactionDisposer | undefined = undefined; -    @computed get transform() { return `scale(${this.props.ContentScaling()}) translate(${this.X}px, ${this.Y}px) rotate(${random(-1, 1) * this.props.jitterRotation}deg)`; } -    @computed get X() { return this._animPos !== undefined ? this._animPos[0] : this.renderScriptDim ? this.renderScriptDim.x : this.props.x !== undefined ? this.props.x : this.Document.x || 0; } -    @computed get Y() { return this._animPos !== undefined ? this._animPos[1] : this.renderScriptDim ? this.renderScriptDim.y : this.props.y !== undefined ? this.props.y : this.Document.y || 0; } -    @computed get width() { return this.renderScriptDim ? this.renderScriptDim.width : this.props.width !== undefined ? this.props.width : this.props.Document[WidthSym](); } -    @computed get height() { return this.renderScriptDim ? this.renderScriptDim.height : this.props.height !== undefined ? this.props.height : this.props.Document[HeightSym](); } +    get transform() { return `scale(${this.props.ContentScaling()}) translate(${this.X}px, ${this.Y}px) rotate(${random(-1, 1) * this.props.jitterRotation}deg)`; } +    get X() { return this._animPos !== undefined ? this._animPos[0] : this.renderScriptDim ? this.renderScriptDim.x : this.props.x !== undefined ? this.props.x : this.dataProvider ? this.dataProvider.x : this.Document.x || 0; } +    get Y() { return this._animPos !== undefined ? this._animPos[1] : this.renderScriptDim ? this.renderScriptDim.y : this.props.y !== undefined ? this.props.y : this.dataProvider ? this.dataProvider.y : this.Document.y || 0; } +    get width() { return this.renderScriptDim ? this.renderScriptDim.width : this.props.width !== undefined ? this.props.width : this.props.dataProvider && this.dataProvider ? this.dataProvider.width : this.props.Document[WidthSym](); } +    get height() { return this.renderScriptDim ? this.renderScriptDim.height : this.props.height !== undefined ? this.props.height : this.props.dataProvider && this.dataProvider ? this.dataProvider.height : this.props.Document[HeightSym](); } +    @computed get dataProvider() { return this.props.dataProvider && this.props.dataProvider(this.props.Document, this.props.DataDoc) ? this.props.dataProvider(this.props.Document, this.props.DataDoc) : undefined; }      @computed get nativeWidth() { return FieldValue(this.Document.nativeWidth, 0); }      @computed get nativeHeight() { return FieldValue(this.Document.nativeHeight, 0); } -    @computed get scaleToOverridingWidth() { return this.width / FieldValue(this.Document.width, this.width); }      @computed get renderScriptDim() {          if (this.Document.renderScript) { @@ -72,11 +73,12 @@ export class CollectionFreeFormDocumentView extends DocComponent<CollectionFreeF      panelHeight = () => this.props.PanelHeight();      getTransform = (): Transform => this.props.ScreenToLocalTransform()          .translate(-this.X, -this.Y) -        .scale(1 / this.contentScaling()).scale(1 / this.scaleToOverridingWidth) +        .scale(1 / this.contentScaling())      borderRounding = () => {          let ruleRounding = this.props.ruleProvider ? StrCast(this.props.ruleProvider["ruleRounding_" + this.Document.heading]) : undefined; -        let br = StrCast(((this.layoutDoc.layout as Doc) || this.Document).borderRounding); +        let ld = this.layoutDoc.layout instanceof Doc ? this.layoutDoc.layout as Doc : undefined; +        let br = StrCast((ld || this.props.Document).borderRounding);          br = !br && ruleRounding ? ruleRounding : br;          if (br.endsWith("%")) {              let nativeDim = Math.min(NumCast(this.layoutDoc.nativeWidth), NumCast(this.layoutDoc.nativeHeight)); @@ -98,6 +100,9 @@ export class CollectionFreeFormDocumentView extends DocComponent<CollectionFreeF      @observable _animPos: number[] | undefined = undefined; +    finalPanelWidh = () => this.dataProvider ? this.dataProvider.width : this.panelWidth(); +    finalPanelHeight = () => this.dataProvider ? this.dataProvider.height : this.panelHeight(); +      render() {          return (              <div className="collectionFreeFormDocumentView-container" @@ -110,7 +115,8 @@ export class CollectionFreeFormDocumentView extends DocComponent<CollectionFreeF                                          StrCast(this.layoutDoc.boxShadow, ""),                      borderRadius: this.borderRounding(),                      transform: this.transform, -                    //transition: this.Document.isAnimating !== undefined ? ".5s ease-in" : this.props.transition ? this.props.transition : StrCast(this.layoutDoc.transition), +                    //CHANGE +                    transition: this.Document.isAnimating !== undefined ? ".5s ease-in" : this.props.transition ? this.props.transition : this.dataProvider ? this.dataProvider.transition : StrCast(this.layoutDoc.transition),                      width: this.width,                      height: this.height,                      zIndex: this.Document.zIndex || 0, @@ -119,8 +125,8 @@ export class CollectionFreeFormDocumentView extends DocComponent<CollectionFreeF                      ContentScaling={this.contentScaling}                      ScreenToLocalTransform={this.getTransform}                      backgroundColor={this.clusterColorFunc} -                    PanelWidth={this.panelWidth} -                    PanelHeight={this.panelHeight} +                    PanelWidth={this.finalPanelWidh} +                    PanelHeight={this.finalPanelHeight}                  />              </div>          ); diff --git a/src/client/views/nodes/DocumentView.tsx b/src/client/views/nodes/DocumentView.tsx index 132d4ba91..ced1fa4df 100644 --- a/src/client/views/nodes/DocumentView.tsx +++ b/src/client/views/nodes/DocumentView.tsx @@ -258,7 +258,7 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu                  this._hitTemplateDrag = true;              }          } -        if (this.active) e.stopPropagation(); // events stop at the lowest document that is active.   +        if (this.active && e.button === 0 && !this.Document.lockedPosition) e.stopPropagation(); // events stop at the lowest document that is active.  if right dragging, we let it go through though to allow for context menu clicks. PointerMove callbacks should remove themselves if the move event gets stopPropagated by a lower-level handler (e.g, marquee drag);          document.removeEventListener("pointermove", this.onPointerMove);          document.removeEventListener("pointerup", this.onPointerUp);          document.addEventListener("pointermove", this.onPointerMove); @@ -266,11 +266,11 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu      }      onPointerMove = (e: PointerEvent): void => {          if (e.cancelBubble && this.active) { -            document.removeEventListener("pointermove", this.onPointerMove); +            document.removeEventListener("pointermove", this.onPointerMove); // stop listening to pointerMove if something else has stopPropagated it (e.g., the MarqueeView)          } -        else if (!e.cancelBubble && this.active) { +        else if (!e.cancelBubble && (SelectionManager.IsSelected(this) || this.props.parentActive()) && !this.Document.lockedPosition) {              if (Math.abs(this._downX - e.clientX) > 3 || Math.abs(this._downY - e.clientY) > 3) { -                if (!e.altKey && !this.topMost && e.buttons === 1 && !BoolCast(this.Document.lockedPosition)) { +                if (!e.altKey && !this.topMost && e.buttons === 1) {                      document.removeEventListener("pointermove", this.onPointerMove);                      document.removeEventListener("pointerup", this.onPointerUp);                      this.startDragging(this._downX, this._downY, e.ctrlKey || e.altKey ? "alias" : undefined, this._hitTemplateDrag); @@ -300,7 +300,7 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu              await swapViews(this.props.Document, "", "layoutNative");              let options = { title: "data", width: (this.Document.width || 0), x: -(this.Document.width || 0) / 2, y: - (this.Document.height || 0) / 2, }; -            let fieldTemplate = this.Document.type === DocumentType.TEXT ? Docs.Create.TextDocument(options) : +            let fieldTemplate = this.Document.type === DocumentType.TEXT ? Docs.Create.TextDocument(options) : this.Document.type === DocumentType.PDF ? Docs.Create.PdfDocument("http://www.msn.com", options) :                  this.Document.type === DocumentType.VID ? Docs.Create.VideoDocument("http://www.cs.brown.edu", options) :                      Docs.Create.ImageDocument("http://www.cs.brown.edu", options); @@ -605,7 +605,7 @@ export class DocumentView extends DocComponent<DocumentViewProps, Document>(Docu              ruleColor && !colorSet ? ruleColor : StrCast(this.layoutDoc.backgroundColor) || this.props.backgroundColor(this.Document);          const nativeWidth = this.nativeWidth > 0 && !this.Document.ignoreAspect ? `${this.nativeWidth}px` : "100%"; -        const nativeHeight = this.Document.ignoreAspect ? this.props.PanelHeight() / this.props.ContentScaling() : this.nativeHeight > 0 ? `${this.nativeHeight}px` : "100%"; +        const nativeHeight = this.Document.ignoreAspect ? this.props.PanelHeight() / this.props.ContentScaling() : this.nativeHeight > 0 ? `${this.nativeHeight}px` : nativeWidth !== "100%" ? nativeWidth : "100%";          const showOverlays = this.props.showOverlays ? this.props.showOverlays(this.Document) : undefined;          const showTitle = showOverlays && "title" in showOverlays ? showOverlays.title : this.getLayoutPropStr("showTitle");          const showCaption = showOverlays && "caption" in showOverlays ? showOverlays.caption : this.getLayoutPropStr("showCaption"); diff --git a/src/client/views/nodes/FieldView.tsx b/src/client/views/nodes/FieldView.tsx index 49fc2263d..ec1b03a40 100644 --- a/src/client/views/nodes/FieldView.tsx +++ b/src/client/views/nodes/FieldView.tsx @@ -50,7 +50,6 @@ export interface FieldViewProps {      PanelWidth: () => number;      PanelHeight: () => number;      setVideoBox?: (player: VideoBox) => void; -    setPdfBox?: (player: PDFBox) => void;      ContentScaling: () => number;      ChromeHeight?: () => number;  } diff --git a/src/client/views/nodes/FormattedTextBox.scss b/src/client/views/nodes/FormattedTextBox.scss index 435f5c055..45e516015 100644 --- a/src/client/views/nodes/FormattedTextBox.scss +++ b/src/client/views/nodes/FormattedTextBox.scss @@ -164,13 +164,13 @@ ol { counter-reset: deci1 0;}  .upper-alpha-ol {counter-reset: ualph; p { display: inline }; font-size: 18 }  .lower-roman-ol {counter-reset: lroman;  p { display: inline }; font-size: 14; }  .lower-alpha-ol {counter-reset: lalpha;  p { display: inline }; font-size: 10;} -.decimal1:before { content: counter(deci1) ")"; counter-increment: deci1; display:inline-block; min-width: 30;} -.decimal2:before { content: counter(deci1) "." counter(deci2) ")"; counter-increment: deci2; display:inline-block; min-width: 35} -.decimal3:before { content: counter(deci1) "." counter(deci2) "." counter(deci3) ")"; counter-increment: deci3; display:inline-block; min-width: 35} -.decimal4:before { content: counter(deci1) "." counter(deci2) "." counter(deci3) "." counter(deci4) ")"; counter-increment: deci4; display:inline-block; min-width: 40} -.decimal5:before { content: counter(deci1) "." counter(deci2) "." counter(deci3) "." counter(deci4) "." counter(deci5) ")"; counter-increment: deci5; display:inline-block; min-width: 40} -.decimal6:before { content: counter(deci1) "." counter(deci2) "." counter(deci3) "." counter(deci4) "." counter(deci5) "." counter(deci6) ")"; counter-increment: deci6; display:inline-block; min-width: 45} -.decimal7:before { content: counter(deci1) "." counter(deci2) "." counter(deci3) "." counter(deci4) "." counter(deci5) "." counter(deci6) "." counter(deci7) ")"; counter-increment: deci7; display:inline-block; min-width: 50} -.upper-alpha:before { content: counter(deci1) "." counter(ualph, upper-alpha) ")"; counter-increment: ualph; display:inline-block; min-width: 35 } -.lower-roman:before { content: counter(deci1) "." counter(ualph, upper-alpha) "." counter(lroman, lower-roman) ")"; counter-increment: lroman;display:inline-block; min-width: 50 } -.lower-alpha:before { content: counter(deci1) "." counter(ualph, upper-alpha) "." counter(lroman, lower-roman) "." counter(lalpha, lower-alpha) ")"; counter-increment: lalpha; display:inline-block; min-width: 35} +.decimal1:before { content: counter(deci1) ") "; counter-increment: deci1; display:inline-block; min-width: 30;} +.decimal2:before { content: counter(deci1) "." counter(deci2) ") "; counter-increment: deci2; display:inline-block; min-width: 35} +.decimal3:before { content: counter(deci1) "." counter(deci2) "." counter(deci3) ") "; counter-increment: deci3; display:inline-block; min-width: 35} +.decimal4:before { content: counter(deci1) "." counter(deci2) "." counter(deci3) "." counter(deci4) ") "; counter-increment: deci4; display:inline-block; min-width: 40} +.decimal5:before { content: counter(deci1) "." counter(deci2) "." counter(deci3) "." counter(deci4) "." counter(deci5) ") "; counter-increment: deci5; display:inline-block; min-width: 40} +.decimal6:before { content: counter(deci1) "." counter(deci2) "." counter(deci3) "." counter(deci4) "." counter(deci5) "." counter(deci6) ") "; counter-increment: deci6; display:inline-block; min-width: 45} +.decimal7:before { content: counter(deci1) "." counter(deci2) "." counter(deci3) "." counter(deci4) "." counter(deci5) "." counter(deci6) "." counter(deci7) ") "; counter-increment: deci7; display:inline-block; min-width: 50} +.upper-alpha:before { content: counter(deci1) "." counter(ualph, upper-alpha) ") "; counter-increment: ualph; display:inline-block; min-width: 35 } +.lower-roman:before { content: counter(deci1) "." counter(ualph, upper-alpha) "." counter(lroman, lower-roman) ") "; counter-increment: lroman;display:inline-block; min-width: 50 } +.lower-alpha:before { content: counter(deci1) "." counter(ualph, upper-alpha) "." counter(lroman, lower-roman) "." counter(lalpha, lower-alpha) ") "; counter-increment: lalpha; display:inline-block; min-width: 35} diff --git a/src/client/views/nodes/FormattedTextBox.tsx b/src/client/views/nodes/FormattedTextBox.tsx index 47b64e260..923dd1544 100644 --- a/src/client/views/nodes/FormattedTextBox.tsx +++ b/src/client/views/nodes/FormattedTextBox.tsx @@ -153,7 +153,7 @@ export class FormattedTextBox extends DocComponent<(FieldViewProps & FormattedTe                          }                      });                      return { frag: Fragment.fromArray(nodes), start: start }; -                } +                };                  let findLinkNode = (node: Node, editor: EditorView) => {                      if (!node.isText) {                          const content = findLinkFrag(node.content, editor); @@ -162,7 +162,7 @@ export class FormattedTextBox extends DocComponent<(FieldViewProps & FormattedTe                      const marks = [...node.marks];                      const linkIndex = marks.findIndex(mark => mark.type === editor.state.schema.marks.link);                      return linkIndex !== -1 && scrollToLinkID === marks[linkIndex].attrs.href.replace(/.*\/doc\//, "") ? node : undefined; -                } +                };                  let start = -1; @@ -748,7 +748,7 @@ export class FormattedTextBox extends DocComponent<(FieldViewProps & FormattedTe                      let ref = editorView.domAtPos(editorView.state.selection.from);                      let refNode = ref.node as any;                      while (refNode && !("getBoundingClientRect" in refNode)) refNode = refNode.parentElement; -                    let r1 = refNode && (refNode as any).getBoundingClientRect(); +                    let r1 = refNode && refNode.getBoundingClientRect();                      let r3 = self._ref.current!.getBoundingClientRect();                      r1 && (self._ref.current!.scrollTop += (r1.top - r3.top) * self.props.ScreenToLocalTransform().Scale);                      return true; diff --git a/src/client/views/nodes/PDFBox.scss b/src/client/views/nodes/PDFBox.scss index c88a94c28..2917c81cb 100644 --- a/src/client/views/nodes/PDFBox.scss +++ b/src/client/views/nodes/PDFBox.scss @@ -1,77 +1,67 @@  .pdfBox-cont,  .pdfBox-cont-interactive { -    display: flex; +    display: inline-block;      flex-direction: row;      height: 100%; -    overflow-y: scroll; -    overflow-x: hidden; -    .pdfBox-scrollHack { -        pointer-events: none; -    } +    width:100%; +    overflow: hidden; +    position:absolute; +    z-index: -1;  }  .pdfBox-cont {      pointer-events: none; -    .pdfPage-textlayer { -        span { -            pointer-events: none !important; -            user-select: none;  +    .collectionFreeFormView-none { +        pointer-events: none; +    } +    .pdfViewer-text { +        .textLayer { +            span { +                user-select: none; +            }          }      }  }  .pdfBox-cont-interactive {      pointer-events: all; -    .pdfPage-textlayer { -        span { -            pointer-events: all !important; -            user-select: text; +    .pdfViewer-text { +        .textLayer { +            span { +                user-select: text; +            }          }      }  } -.react-pdf__Page { -    transform-origin: left top; -    position: absolute; -    top: 0; -    left: 0; -} - -.react-pdf__Page__textContent span { -    user-select: text; -} - -.react-pdf__Document { -    position: absolute; -} -  .pdfBox-settingsCont {      position: absolute;      right: 0; -    top: 0; +    top: 3; +    pointer-events: all;      .pdfBox-settingsButton {          border-bottom-left-radius: 50%;          display: flex;          justify-content: space-evenly;          align-items: center; -        height: 70px; +        height: 30px;          background: none;          padding: 0;          .pdfBox-settingsButton-arrow {              width: 0;              height: 0; -            border-top: 25px solid transparent; -            border-bottom: 25px solid transparent; -            border-right: 25px solid #121721; +            border-top: 15px solid transparent; +            border-bottom: 15px solid transparent; +            border-right: 15px solid #121721;              transition: all 0.5s;          }          .pdfBox-settingsButton-iconCont {              background: #121721; -            height: 50px; +            height: 30px;              width: 70px;              display: flex;              justify-content: center; @@ -86,16 +76,15 @@      }      .pdfBox-settingsFlyout { -        width: 600px;          position: absolute;          background: #323232;          box-shadow: 3px 3px 3px rgba(0, 0, 0, 0.25); -        left: -400px; +        right: 20px;          border-radius: 7px;          padding: 20px;          display: flex;          flex-direction: column; -        font-size: 30px; +        font-size: 14px;          transition: all 0.5s;          .pdfBox-settingsFlyout-title { @@ -108,4 +97,69 @@              grid-template-columns: 47.5% 5% 47.5%;          }      } -}
\ No newline at end of file +} + +.pdfBox-overlayCont { +    position: absolute; +    width: 100%; +    height: 40px; +    background: #121721; +    bottom: 0; +    display: flex; +    justify-content: center; +    align-items: center; +    padding: 20px; +    overflow: hidden; +    transition: left .5s; +    pointer-events: all; + +    .pdfBox-searchBar { +        width: 70%; +        font-size: 14px; +    } +} + +.pdfBox-overlayButton { +    border-bottom-left-radius: 50%; +    display: flex; +    justify-content: space-evenly; +    align-items: center; +    height: 30px; +    background: none; +    padding: 0; +    position: absolute; +    pointer-events: all; + +    .pdfBox-overlayButton-arrow { +        width: 0; +        height: 0; +        border-top: 15px solid transparent; +        border-bottom: 15px solid transparent; +        border-right: 15px solid #121721; +        transition: all 0.5s; +    } + +    .pdfBox-overlayButton-iconCont, +    .pdfBox-nextIcon,  +    .pdfBox-prevIcon { +        background: #121721; +        height: 30px; +        width: 70px; +        display: flex; +        justify-content: center; +        align-items: center; +        margin-left: -2px; +        border-radius: 3px; +    } +} + +.pdfBox-overlayButton:hover { +    background: none; +} + +.pdfBox-nextIcon { +    left: 20; top: 5; height: 30px; position: absolute; +} +.pdfBox-prevIcon { +    left: 50; top: 5; height: 30px; position: absolute; +} diff --git a/src/client/views/nodes/PDFBox.tsx b/src/client/views/nodes/PDFBox.tsx index 764051d62..0fcbaaa7c 100644 --- a/src/client/views/nodes/PDFBox.tsx +++ b/src/client/views/nodes/PDFBox.tsx @@ -1,12 +1,12 @@  import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'; -import { action, computed, IReactionDisposer, observable, reaction, runInAction } from 'mobx'; +import { action, computed, IReactionDisposer, observable, reaction, runInAction, untracked, trace } from 'mobx';  import { observer } from "mobx-react";  import * as Pdfjs from "pdfjs-dist";  import "pdfjs-dist/web/pdf_viewer.css";  import 'react-image-lightbox/style.css';  import { Doc, Opt, WidthSym } from "../../../new_fields/Doc";  import { makeInterface } from "../../../new_fields/Schema"; -import { ComputedField, ScriptField } from '../../../new_fields/ScriptField'; +import { ScriptField } from '../../../new_fields/ScriptField';  import { Cast, NumCast } from "../../../new_fields/Types";  import { PdfField } from "../../../new_fields/URLField";  import { KeyCodes } from '../../northstar/utils/KeyCodes'; @@ -19,82 +19,56 @@ import { FieldView, FieldViewProps } from './FieldView';  import { pageSchema } from "./ImageBox";  import "./PDFBox.scss";  import React = require("react"); +import { undoBatch } from '../../util/UndoManager';  type PdfDocument = makeInterface<[typeof documentSchema, typeof panZoomSchema, typeof pageSchema]>;  const PdfDocument = makeInterface(documentSchema, panZoomSchema, pageSchema);  @observer  export class PDFBox extends DocComponent<FieldViewProps, PdfDocument>(PdfDocument) { -    public static LayoutString() { return FieldView.LayoutString(PDFBox); } -    private _mainCont: React.RefObject<HTMLDivElement> = React.createRef(); -    private _reactionDisposer?: IReactionDisposer; +    public static LayoutString(fieldExt?: string) { return FieldView.LayoutString(PDFBox, "data", fieldExt); }      private _keyValue: string = "";      private _valueValue: string = "";      private _scriptValue: string = ""; +    private _searchString: string = ""; +    private _isChildActive = false; +    private _pdfViewer: PDFViewer | undefined;      private _keyRef: React.RefObject<HTMLInputElement> = React.createRef();      private _valueRef: React.RefObject<HTMLInputElement> = React.createRef();      private _scriptRef: React.RefObject<HTMLInputElement> = React.createRef(); +    @observable private _searching: boolean = false;      @observable private _flyout: boolean = false; -    @observable private _alt = false;      @observable private _pdf: Opt<Pdfjs.PDFDocumentProxy>; +    @observable private _pageControls = false;      @computed get extensionDoc() { return Doc.fieldExtensionDoc(this.dataDoc, this.props.fieldKey); } -      @computed get dataDoc() { return this.props.DataDoc && this.props.Document.isTemplate ? this.props.DataDoc : Doc.GetProto(this.props.Document); }      componentDidMount() { -        this.props.setPdfBox && this.props.setPdfBox(this); - -        this.props.Document.curPage = ComputedField.MakeFunction("Math.floor(Number(this.panY) / Number(this.nativeHeight) + 1)"); -          const pdfUrl = Cast(this.dataDoc[this.props.fieldKey], PdfField);          if (pdfUrl instanceof PdfField) {              Pdfjs.getDocument(pdfUrl.url.pathname).promise.then(pdf => runInAction(() => this._pdf = pdf));          } -        this._reactionDisposer = reaction( -            () => this.Document.panY, -            () => this._mainCont.current && this._mainCont.current.scrollTo({ top: this.Document.panY || 0, behavior: "auto" }) -        ); -    } - -    componentWillUnmount() { -        this._reactionDisposer && this._reactionDisposer(); -    } - -    public GetPage() { -        return Math.floor((this.Document.panY || 0) / (this.Document.nativeHeight || 0)) + 1; -    } - -    @action -    public BackPage() { -        let cp = Math.ceil((this.Document.panY || 0) / (this.Document.nativeHeight || 0)) + 1; -        cp = cp - 1; -        if (cp > 0) { -            this.Document.panY = (cp - 1) * (this.Document.nativeHeight || 0); -        }      } - -    @action -    public GotoPage = (p: number) => { -        if (p > 0 && p <= NumCast(this.dataDoc.numPages)) { -            this.Document.panY = (p - 1) * (this.Document.nativeHeight || 0); +    loaded = (nw: number, nh: number, np: number) => { +        this.dataDoc.numPages = np; +        if (!this.Document.nativeWidth || !this.Document.nativeHeight || !this.Document.scrollHeight) { +            let oldaspect = (this.Document.nativeHeight || 0) / (this.Document.nativeWidth || 1); +            this.Document.nativeWidth = nw * 96 / 72; +            this.Document.nativeHeight = this.Document.nativeHeight ? nw * 96 / 72 * oldaspect : nh * 96 / 72;          } +        this.Document.height = this.Document[WidthSym]() * (nh / nw);      } -    @action -    public ForwardPage() { -        let cp = this.GetPage() + 1; -        if (cp <= NumCast(this.dataDoc.numPages)) { -            this.Document.panY = (cp - 1) * (this.Document.nativeHeight || 0); -        } -    } - -    @action -    setPanY = (y: number) => { -        this.Document.panY = y; -    } +    public search(string: string, fwd: boolean) { this._pdfViewer && this._pdfViewer.search(string, fwd); } +    public prevAnnotation() { this._pdfViewer && this._pdfViewer.prevAnnotation(); } +    public nextAnnotation() { this._pdfViewer && this._pdfViewer.nextAnnotation(); } +    public backPage() { this._pdfViewer!.gotoPage(NumCast(this.props.Document.curPage) - 1); } +    public gotoPage = (p: number) => { this._pdfViewer!.gotoPage(p); }; +    public forwardPage() { this._pdfViewer!.gotoPage(NumCast(this.props.Document.curPage) + 1); } +    @undoBatch      @action      private applyFilter = () => {          let scriptText = this._scriptValue ? this._scriptValue : @@ -102,10 +76,6 @@ export class PDFBox extends DocComponent<FieldViewProps, PdfDocument>(PdfDocumen          this.props.Document.filterScript = ScriptField.MakeFunction(scriptText);      } -    scrollTo = (y: number) => { -        this._mainCont.current && this._mainCont.current.scrollTo({ top: Math.max(y - (this._mainCont.current.offsetHeight / 2), 0), behavior: "auto" }); -    } -      private resetFilters = () => {          this._keyValue = this._valueValue = this._scriptValue = "";          this._keyRef.current && (this._keyRef.current.value = ""); @@ -117,83 +87,103 @@ export class PDFBox extends DocComponent<FieldViewProps, PdfDocument>(PdfDocumen      private newValueChange = (e: React.ChangeEvent<HTMLInputElement>) => this._valueValue = e.currentTarget.value;      private newScriptChange = (e: React.ChangeEvent<HTMLInputElement>) => this._scriptValue = e.currentTarget.value; +    whenActiveChanged = (isActive: boolean) => this.props.whenActiveChanged(this._isChildActive = isActive); +    active = () => this.props.isSelected() || this._isChildActive || this.props.renderDepth === 0; +    setPdfViewer = (pdfViewer: PDFViewer) => { this._pdfViewer = pdfViewer; }; +    searchStringChanged = (e: React.ChangeEvent<HTMLInputElement>) => this._searchString = e.currentTarget.value; +      settingsPanel() { +        let pageBtns = <> +            <button className="pdfBox-overlayButton-iconCont" key="back" title="Page Back" +                onPointerDown={(e) => e.stopPropagation()} +                onClick={() => this.backPage()} +                style={{ left: 50, top: 5, height: "30px", position: "absolute", pointerEvents: "all" }}> +                <FontAwesomeIcon style={{ color: "white" }} icon={"arrow-left"} size="sm" /> +            </button> +            <button className="pdfBox-overlayButton-iconCont" key="fwd" title="Page Forward" +                onPointerDown={(e) => e.stopPropagation()} +                onClick={() => this.forwardPage()} +                style={{ left: 80, top: 5, height: "30px", position: "absolute", pointerEvents: "all" }}> +                <FontAwesomeIcon style={{ color: "white" }} icon={"arrow-right"} size="sm" /> +            </button> +        </>;          return !this.props.active() ? (null) : -            (<div className="pdfBox-settingsCont" onPointerDown={(e) => e.stopPropagation()}> -                <button className="pdfBox-settingsButton" onClick={action(() => this._flyout = !this._flyout)} title="Open Annotation Settings" -                    style={{ marginTop: `${this.Document.panY || 0}px` }}> -                    <div className="pdfBox-settingsButton-arrow" -                        style={{ -                            borderTop: `25px solid ${this._flyout ? "#121721" : "transparent"}`, -                            borderBottom: `25px solid ${this._flyout ? "#121721" : "transparent"}`, -                            borderRight: `25px solid ${this._flyout ? "transparent" : "#121721"}`, -                            transform: `scaleX(${this._flyout ? -1 : 1})` -                        }} /> -                    <div className="pdfBox-settingsButton-iconCont"> -                        <FontAwesomeIcon style={{ color: "white" }} icon="cog" size="3x" /> -                    </div> +            (<div className="pdfBox-ui" onKeyDown={e => e.keyCode === KeyCodes.BACKSPACE || e.keyCode === KeyCodes.DELETE ? e.stopPropagation() : true} +                onPointerDown={e => e.stopPropagation()} style={{ display: this.active() ? "flex" : "none", position: "absolute", width: "100%", height: "100%", zIndex: 1, pointerEvents: "none" }}> +                <div className="pdfBox-overlayCont" key="cont" onPointerDown={(e) => e.stopPropagation()} style={{ left: `${this._searching ? 0 : 100}%` }}> +                    <button className="pdfBox-overlayButton" title="Open Search Bar" /> +                    <input className="pdfBox-searchBar" placeholder="Search" onChange={this.searchStringChanged} onKeyDown={e => e.keyCode === KeyCodes.ENTER && this.search(this._searchString, !e.shiftKey)} /> +                    <button title="Search" onClick={e => this.search(this._searchString, !e.shiftKey)}> +                        <FontAwesomeIcon icon="search" size="sm" color="white" /></button> +                    <button className="pdfBox-prevIcon " title="Previous Annotation" onClick={e => this.prevAnnotation()} > +                        <FontAwesomeIcon style={{ color: "white" }} icon={"arrow-up"} size="sm" /> +                    </button> +                    <button className="pdfBox-nextIcon" title="Next Annotation" onClick={e => this.nextAnnotation()} > +                        <FontAwesomeIcon style={{ color: "white" }} icon={"arrow-down"} size="sm" /> +                    </button> +                </div> +                <button className="pdfBox-overlayButton" key="search" onClick={action(() => this._searching = !this._searching)} title="Open Search Bar" style={{ bottom: 8, right: 0 }}> +                    <div className="pdfBox-overlayButton-arrow" onPointerDown={(e) => e.stopPropagation()}></div> +                    <div className="pdfBox-overlayButton-iconCont" onPointerDown={(e) => e.stopPropagation()}> +                        <FontAwesomeIcon style={{ color: "white", padding: 5 }} icon={this._searching ? "times" : "search"} size="3x" /></div>                  </button> -                <div className="pdfBox-settingsFlyout" style={{ left: `${this._flyout ? -600 : 100}px` }} > -                    <div className="pdfBox-settingsFlyout-title"> -                        Annotation View Settings -                    </div> -                    <div className="pdfBox-settingsFlyout-kvpInput"> -                        <input placeholder="Key" className="pdfBox-settingsFlyout-input" onChange={this.newKeyChange} -                            style={{ gridColumn: 1 }} ref={this._keyRef} /> -                        <input placeholder="Value" className="pdfBox-settingsFlyout-input" onChange={this.newValueChange} -                            style={{ gridColumn: 3 }} ref={this._valueRef} /> -                    </div> -                    <div className="pdfBox-settingsFlyout-kvpInput"> -                        <input placeholder="Custom Script" onChange={this.newScriptChange} style={{ gridColumn: "1 / 4" }} ref={this._scriptRef} /> -                    </div> -                    <div className="pdfBox-settingsFlyout-kvpInput"> -                        <button style={{ gridColumn: 1 }} onClick={this.resetFilters}> -                            <FontAwesomeIcon style={{ color: "white" }} icon="trash" size="lg" /> -                              Reset Filters -                        </button> -                        <button style={{ gridColumn: 3 }} onClick={this.applyFilter}> -                            <FontAwesomeIcon style={{ color: "white" }} icon="check" size="lg" /> -                              Apply -                        </button> +                <input value={`${NumCast(this.props.Document.curPage)}`} +                    onChange={e => this.gotoPage(Number(e.currentTarget.value))} +                    style={{ left: 20, top: 5, height: "30px", width: "30px", position: "absolute", pointerEvents: "all" }} +                    onClick={action(() => this._pageControls = !this._pageControls)} /> +                {this._pageControls ? pageBtns : (null)} +                <div className="pdfBox-settingsCont" key="settings" onPointerDown={(e) => e.stopPropagation()}> +                    <button className="pdfBox-settingsButton" onClick={action(() => this._flyout = !this._flyout)} title="Open Annotation Settings" > +                        <div className="pdfBox-settingsButton-arrow" style={{ transform: `scaleX(${this._flyout ? -1 : 1})` }} /> +                        <div className="pdfBox-settingsButton-iconCont"> +                            <FontAwesomeIcon style={{ color: "white", padding: 5 }} icon="cog" size="3x" /> +                        </div> +                    </button> +                    <div className="pdfBox-settingsFlyout" style={{ right: `${this._flyout ? 20 : -600}px` }} > +                        <div className="pdfBox-settingsFlyout-title"> +                            Annotation View Settings +                        </div> +                        <div className="pdfBox-settingsFlyout-kvpInput"> +                            <input placeholder="Key" className="pdfBox-settingsFlyout-input" onChange={this.newKeyChange} style={{ gridColumn: 1 }} ref={this._keyRef} /> +                            <input placeholder="Value" className="pdfBox-settingsFlyout-input" onChange={this.newValueChange} style={{ gridColumn: 3 }} ref={this._valueRef} /> +                        </div> +                        <div className="pdfBox-settingsFlyout-kvpInput"> +                            <input placeholder="Custom Script" onChange={this.newScriptChange} style={{ gridColumn: "1 / 4" }} ref={this._scriptRef} /> +                        </div> +                        <div className="pdfBox-settingsFlyout-kvpInput"> +                            <button style={{ gridColumn: 1 }} onClick={this.resetFilters}> +                                <FontAwesomeIcon style={{ color: "white" }} icon="trash" size="lg" /> +                                  Reset Filters +                            </button> +                            <button style={{ gridColumn: 3 }} onClick={this.applyFilter}> +                                <FontAwesomeIcon style={{ color: "white" }} icon="check" size="lg" /> +                                  Apply +                            </button> +                        </div>                      </div>                  </div>              </div>);      } -    loaded = (nw: number, nh: number, np: number) => { -        this.dataDoc.numPages = np; -        if (!this.Document.nativeWidth || !this.Document.nativeHeight || !this.Document.scrollHeight) { -            let oldaspect = (this.Document.nativeHeight || 0) / (this.Document.nativeWidth || 1); -            this.Document.nativeWidth = nw; -            this.Document.nativeHeight = this.Document.nativeHeight ? nw * oldaspect : nh; -            this.Document.height = this.Document[WidthSym]() * (nh / nw); -            this.Document.scrollHeight = np * this.Document.nativeHeight; -        } -    } - -    @action -    onScroll = (e: React.UIEvent<HTMLDivElement>) => { -        if (e.currentTarget && this.props.ContainingCollectionDoc) { -            this.props.Document.panTransformType = "None"; -            this.Document.panY = e.currentTarget.scrollTop; -        } -    } - -      render() {          const pdfUrl = Cast(this.dataDoc[this.props.fieldKey], PdfField); -        let classname = "pdfBox-cont" + (this.props.active() && !InkingControl.Instance.selectedTool && !this._alt ? "-interactive" : ""); +        let classname = "pdfBox-cont" + (InkingControl.Instance.selectedTool || !this.active ? "" : "-interactive");          return (!(pdfUrl instanceof PdfField) || !this._pdf ?              <div>{`pdf, ${this.dataDoc[this.props.fieldKey]}, not found`}</div> : -            <div className={classname} -                onScroll={this.onScroll} -                style={{ marginTop: `${(this.Document.panY || 0)}px` }} -                ref={this._mainCont}> -                <div className="pdfBox-scrollHack" style={{ height: NumCast(this.props.Document.scrollHeight) + ((this.Document.nativeHeight || 0) - (this.Document.nativeHeight || 0) / (this.Document.scale || 1)) }} /> -                <PDFViewer pdf={this._pdf} url={pdfUrl.url.pathname} active={this.props.active} scrollTo={this.scrollTo} loaded={this.loaded} panY={this.Document.panY || 0} -                    Document={this.props.Document} DataDoc={this.dataDoc} -                    addDocTab={this.props.addDocTab} setPanY={this.setPanY} GoToPage={this.GotoPage} +            <div className={classname} onPointerDown={(e: React.PointerEvent) => { +                let hit = document.elementFromPoint(e.clientX, e.clientY); +                if (hit && hit.localName === "span" && this.props.isSelected()) { +                    e.button === 0 && e.stopPropagation(); +                } +            }}> +                <PDFViewer {...this.props} pdf={this._pdf} url={pdfUrl.url.pathname} active={this.props.active} loaded={this.loaded} +                    setPdfViewer={this.setPdfViewer} ContainingCollectionView={this.props.ContainingCollectionView} +                    renderDepth={this.props.renderDepth} PanelHeight={this.props.PanelHeight} PanelWidth={this.props.PanelWidth} +                    Document={this.props.Document} DataDoc={this.dataDoc} ContentScaling={this.props.ContentScaling} +                    addDocTab={this.props.addDocTab} GoToPage={this.gotoPage}                      pinToPres={this.props.pinToPres} addDocument={this.props.addDocument} +                    ScreenToLocalTransform={this.props.ScreenToLocalTransform} +                    isSelected={this.props.isSelected} whenActiveChanged={this.whenActiveChanged}                      fieldKey={this.props.fieldKey} fieldExtensionDoc={this.extensionDoc} />                  {this.settingsPanel()}              </div>); diff --git a/src/client/views/nodes/PresBox.tsx b/src/client/views/nodes/PresBox.tsx index e376fbddb..5afd85430 100644 --- a/src/client/views/nodes/PresBox.tsx +++ b/src/client/views/nodes/PresBox.tsx @@ -161,10 +161,8 @@ export class PresBox extends React.Component<FieldViewProps> { //FieldViewProps?              if (zoomOut || this.presElementsMappings.get(docAtCurrent)!.showButton) {                  let prevScale = NumCast(this.childrenDocs[prevSelected].viewScale, null);                  let curScale = DocumentManager.Instance.getScaleOfDocView(this.childrenDocs[current]); -                if (prevScale !== undefined) { -                    if (prevScale !== curScale) { -                        DocumentManager.Instance.zoomIntoScale(docAtCurrent, prevScale); -                    } +                if (prevScale !== undefined && prevScale !== curScale) { +                    DocumentManager.Instance.zoomIntoScale(docAtCurrent, prevScale);                  }              }          } diff --git a/src/client/views/pdf/PDFMenu.tsx b/src/client/views/pdf/PDFMenu.tsx index 3ed81faef..2202351ee 100644 --- a/src/client/views/pdf/PDFMenu.tsx +++ b/src/client/views/pdf/PDFMenu.tsx @@ -1,11 +1,10 @@  import React = require("react");  import "./PDFMenu.scss"; -import { observable, action, runInAction } from "mobx"; +import { observable, action, } from "mobx";  import { observer } from "mobx-react";  import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";  import { emptyFunction, returnFalse } from "../../../Utils";  import { Doc } from "../../../new_fields/Doc"; -import { handleBackspace } from "../nodes/PDFBox";  @observer  export default class PDFMenu extends React.Component { @@ -238,8 +237,8 @@ export default class PDFMenu extends React.Component {                  <button key="6" className="pdfMenu-button" title="Pin to Presentation" onPointerDown={this.PinToPres}>                      <FontAwesomeIcon icon="map-pin" size="lg" /></button>,                  <div key="7" className="pdfMenu-addTag" > -                    <input onKeyDown={handleBackspace} onChange={this.keyChanged} placeholder="Key" style={{ gridColumn: 1 }} /> -                    <input onKeyDown={handleBackspace} onChange={this.valueChanged} placeholder="Value" style={{ gridColumn: 3 }} /> +                    <input onChange={this.keyChanged} placeholder="Key" style={{ gridColumn: 1 }} /> +                    <input onChange={this.valueChanged} placeholder="Value" style={{ gridColumn: 3 }} />                  </div>,                  <button key="8" className="pdfMenu-button" title={`Add tag: ${this._keyValue} with value: ${this._valueValue}`} onPointerDown={this.addTag}>                      <FontAwesomeIcon style={{ transition: "all .2s" }} color={this._added ? "#42f560" : "white"} icon="check" size="lg" /></button>, diff --git a/src/client/views/pdf/PDFViewer.scss b/src/client/views/pdf/PDFViewer.scss index a2f3911c5..8027e93a3 100644 --- a/src/client/views/pdf/PDFViewer.scss +++ b/src/client/views/pdf/PDFViewer.scss @@ -1,93 +1,66 @@ - -.pdfViewer-viewer { -    pointer-events:inherit; +  +.pdfViewer-viewer, .pdfViewer-viewer-zoomed { +    pointer-events: inherit;      width: 100%; -    .pdfViewer-visibleElements { -        .pdfPage-cont { -            .pdfPage-textLayer { -                div { -                    user-select: text; -                } -                span { -                    color: transparent; -                    position: absolute; -                    white-space: pre; -                    cursor: text; -                    -webkit-transform-origin: 0% 0%; -                    transform-origin: 0% 0%; -                } -            } -        } +    height: 100%; +    position: absolute; +    overflow-y: auto; +    overflow-x: hidden; +     +    // .canvasWrapper { +    //     transform: scale(0.75); +    //     transform-origin: top left; +    // } +    // .textLayer { +    //     transform: scale(0.75); +    //     transform-origin: top left; +    // } + +    .page { +        position: relative;      } -    .pdfViewer-text { -        transform: scale(1.5); -        transform-origin: top left; +    .collectionfreeformview-container { +        pointer-events: none; +    } + +    .pdfViewer-dragAnnotationBox { +        position:absolute; +        background-color: transparent; +        opacity: 0.1;      } +    .pdfViewer-overlay { +        transform: scale(2.14359); +        transform-origin: left top; +        position: absolute; +        top: 0px; +        left: 0px; +        display: inline-block; +        width:100%; +    }      .pdfViewer-annotationLayer {          position: absolute;          top: 0;          width: 100%;          pointer-events: none; +          .pdfPage-annotationBox {              position: absolute;              background-color: red;              opacity: 0.1;          }      } - -    .pdfViewer-overlayCont { -        position: absolute; -        width: 100%; -        height: 100px; -        background: #121721; -        bottom: 0; -        display: flex; -        justify-content: center; -        align-items: center; -        padding: 20px; -        overflow: hidden; -        transition: left .5s; -        .pdfViewer-overlaySearchBar { -            width: 20%; -            height: 100%; -            font-size: 30px; -            padding: 5px; -        } -    } - -    .pdfViewer-overlayButton { -        border-bottom-left-radius: 50%; -        display: flex; -        justify-content: space-evenly; -        align-items: center; -        height: 70px; -        background: none; -        padding: 0; +    .pdfViewer-waiting { +        width: 70%; +        height: 70%; +        margin : 15%; +        transition: 0.4s opacity ease; +        opacity: 0.7;           position: absolute; - -        .pdfViewer-overlayButton-arrow { -            width: 0; -            height: 0; -            border-top: 25px solid transparent; -            border-bottom: 25px solid transparent; -            border-right: 25px solid #121721; -            transition: all 0.5s; -        } - -        .pdfViewer-overlayButton-iconCont { -            background: #121721; -            height: 50px; -            width: 70px; -            display: flex; -            justify-content: center; -            align-items: center; -            margin-left: -2px; -            border-radius: 3px; -        } -    } - -    .pdfViewer-overlayButton:hover { -        background: none; +        z-index: 10;      }  } +.pdfViewer-viewer-zoomed { +    overflow-x: scroll; +} +   
\ No newline at end of file diff --git a/src/client/views/pdf/PDFViewer.tsx b/src/client/views/pdf/PDFViewer.tsx index c94b4e3a4..5ad4ffd48 100644 --- a/src/client/views/pdf/PDFViewer.tsx +++ b/src/client/views/pdf/PDFViewer.tsx @@ -1,28 +1,34 @@ -import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; -import { action, computed, IReactionDisposer, observable, reaction, runInAction, trace } from "mobx"; +import { action, computed, IReactionDisposer, observable, reaction, trace, runInAction } from "mobx";  import { observer } from "mobx-react";  import * as Pdfjs from "pdfjs-dist";  import "pdfjs-dist/web/pdf_viewer.css"; -import * as rp from "request-promise";  import { Dictionary } from "typescript-collections"; -import { Doc, DocListCast, FieldResult } from "../../../new_fields/Doc"; +import { Doc, DocListCast, FieldResult, WidthSym, Opt, HeightSym } from "../../../new_fields/Doc";  import { Id } from "../../../new_fields/FieldSymbols";  import { List } from "../../../new_fields/List"; +import { listSpec } from "../../../new_fields/Schema";  import { ScriptField } from "../../../new_fields/ScriptField";  import { Cast, NumCast, StrCast } from "../../../new_fields/Types"; -import { Utils, numberRange } from "../../../Utils"; +import { emptyFunction, returnOne, Utils } from "../../../Utils";  import { DocServer } from "../../DocServer";  import { Docs, DocUtils } from "../../documents/Documents"; -import { KeyCodes } from "../../northstar/utils/KeyCodes"; -import { CompileScript, CompiledScript } from "../../util/Scripting"; +import { DragManager } from "../../util/DragManager"; +import { CompiledScript, CompileScript } from "../../util/Scripting"; +import { Transform } from "../../util/Transform"; +import { CollectionFreeFormView } from "../collections/collectionFreeForm/CollectionFreeFormView";  import Annotation from "./Annotation"; -import Page from "./Page"; +import PDFMenu from "./PDFMenu";  import "./PDFViewer.scss";  import React = require("react"); -import requestPromise = require("request-promise"); +import * as rp from "request-promise"; +import { CollectionPDFView } from "../collections/CollectionPDFView"; +import { CollectionVideoView } from "../collections/CollectionVideoView"; +import { CollectionView } from "../collections/CollectionView"; +import { SelectionManager } from "../../util/SelectionManager";  const PDFJSViewer = require("pdfjs-dist/web/pdf_viewer"); +const pdfjsLib = require("pdfjs-dist"); -export const scale = 2; +pdfjsLib.GlobalWorkerOptions.workerSrc = `/assets/pdf.worker.js`;  interface IViewerProps {      pdf: Pdfjs.PDFDocumentProxy; @@ -31,15 +37,22 @@ interface IViewerProps {      DataDoc?: Doc;      fieldExtensionDoc: Doc;      fieldKey: string; +    fieldExt: string; +    PanelWidth: () => number; +    PanelHeight: () => number; +    ContentScaling: () => number; +    renderDepth: number; +    isSelected: () => boolean;      loaded: (nw: number, nh: number, np: number) => void; -    panY: number; -    scrollTo: (y: number) => void;      active: () => boolean; -    setPanY?: (n: number) => void;      GoToPage?: (n: number) => void;      addDocTab: (document: Doc, dataDoc: Doc | undefined, where: string) => boolean;      pinToPres: (document: Doc) => void;      addDocument?: (doc: Doc, allowDuplicates?: boolean) => boolean; +    setPdfViewer: (view: PDFViewer) => void; +    ScreenToLocalTransform: () => Transform; +    ContainingCollectionView: Opt<CollectionView | CollectionPDFView | CollectionVideoView>; +    whenActiveChanged: (isActive: boolean) => void;  }  /** @@ -47,36 +60,37 @@ interface IViewerProps {   */  @observer  export class PDFViewer extends React.Component<IViewerProps> { -    @observable.shallow private _visibleElements: JSX.Element[] = []; // _visibleElements is the array of JSX elements that gets rendered -    @observable private _isPage: string[] = [];// _isPage is an array that tells us whether or not an index is rendered as a page or as a placeholder      @observable private _pageSizes: { width: number, height: number }[] = [];      @observable private _annotations: Doc[] = [];      @observable private _savedAnnotations: Dictionary<number, HTMLDivElement[]> = new Dictionary<number, HTMLDivElement[]>();      @observable private _script: CompiledScript = CompileScript("return true") as CompiledScript; -    @observable private _searching: boolean = false;      @observable private Index: number = -1; - -    private _pageBuffer: number = 1; +    @observable private _marqueeX: number = 0; +    @observable private _marqueeY: number = 0; +    @observable private _marqueeWidth: number = 0; +    @observable private _marqueeHeight: number = 0; +    @observable private _marqueeing: boolean = false; +    @observable private _showWaiting = true; +    @observable private _showCover = false; +    @observable private _zoomed = 1; + +    public pdfViewer: any; +    private _isChildActive = false; +    private _setPreviewCursor: undefined | ((x: number, y: number, drag: boolean) => void);      private _annotationLayer: React.RefObject<HTMLDivElement> = React.createRef();      private _reactionDisposer?: IReactionDisposer; +    private _selectionReactionDisposer?: IReactionDisposer;      private _annotationReactionDisposer?: IReactionDisposer;      private _filterReactionDisposer?: IReactionDisposer;      private _viewer: React.RefObject<HTMLDivElement> = React.createRef();      private _mainCont: React.RefObject<HTMLDivElement> = React.createRef(); -    public _pdfViewer: any; -    private _pdfFindController: any; -    private _searchString: string = ""; +    private _marquee: React.RefObject<HTMLDivElement> = React.createRef();      private _selectionText: string = ""; - -    @computed get panY(): number { return this.props.panY; } - -    // startIndex: where to start rendering pages -    @computed get startIndex(): number { return Math.max(0, this.getPageFromScroll(this.panY) - this._pageBuffer); } - -    // endIndex: where to end rendering pages -    @computed get endIndex(): number { -        return Math.min(this.props.pdf.numPages - 1, this.getPageFromScroll(this.panY + (this._pageSizes[0] ? this._pageSizes[0].height : 0)) + this._pageBuffer); -    } +    private _startX: number = 0; +    private _startY: number = 0; +    private _downX: number = 0; +    private _downY: number = 0; +    private _coverPath: any;      @computed get allAnnotations() {          return DocListCast(this.props.fieldExtensionDoc.annotations).filter( @@ -87,42 +101,23 @@ export class PDFViewer extends React.Component<IViewerProps> {          return this._annotations.filter(anno => this._script.run({ this: anno }, console.log, true).result);      } -    componentDidUpdate = (prevProps: IViewerProps) => this.panY !== prevProps.panY && this.renderPages(); -      componentDidMount = async () => { -        await this.initialLoad(); - -        this._reactionDisposer = reaction( -            () => [this.props.active(), this.startIndex, this._pageSizes.length ? this.endIndex : 0], -            () => this.renderPages(), -            { fireImmediately: true }); - -        this._annotationReactionDisposer = reaction( -            () => this.props.fieldExtensionDoc && DocListCast(this.props.fieldExtensionDoc.annotations), -            annotations => annotations && annotations.length && this.renderAnnotations(annotations, true), -            { fireImmediately: true }); - -        this._filterReactionDisposer = reaction( -            () => ({ scriptField: Cast(this.props.Document.filterScript, ScriptField), annos: this._annotations.slice() }), -            action(({ scriptField, annos }: { scriptField: FieldResult<ScriptField>, annos: Doc[] }) => { -                let oldScript = this._script.originalScript; -                this._script = scriptField && scriptField.script.compiled ? scriptField.script : CompileScript("return true") as CompiledScript; -                if (this._script.originalScript !== oldScript) { -                    this.Index = -1; -                } -                annos.forEach(d => d.opacity = this._script.run({ this: d }, console.log, 1).result ? 1 : 0); -            }), -            { fireImmediately: true } -        ); - -        document.removeEventListener("copy", this.copy); -        document.addEventListener("copy", this.copy); +        // change the address to be the file address of the PNG version of each page +        // file address of the pdf +        this._coverPath = JSON.parse(await rp.get(Utils.prepend(`/thumbnail${this.props.url.substring("files/".length, this.props.url.length - ".pdf".length)}-${NumCast(this.props.Document.curPage, 1)}.PNG`))); +        runInAction(() => this._showWaiting = this._showCover = true); +        this._selectionReactionDisposer = reaction(() => this.props.isSelected(), () => { +            this.setupPdfJsViewer(); +            this._selectionReactionDisposer && this._selectionReactionDisposer(); +            this._selectionReactionDisposer = undefined; +        })      }      componentWillUnmount = () => {          this._reactionDisposer && this._reactionDisposer();          this._annotationReactionDisposer && this._annotationReactionDisposer();          this._filterReactionDisposer && this._filterReactionDisposer(); +        this._selectionReactionDisposer && this._selectionReactionDisposer();          document.removeEventListener("copy", this.copy);      } @@ -143,42 +138,77 @@ export class PDFViewer extends React.Component<IViewerProps> {          }      } -    searchStringChanged = (e: React.ChangeEvent<HTMLInputElement>) => this._searchString = e.currentTarget.value; - -    pageLoaded = (page: Pdfjs.PDFPageViewport): void => this.props.loaded(page.width, page.height, this.props.pdf.numPages); -      setSelectionText = (text: string) => this._selectionText = text; -    getIndex = () => this.Index; -      @action      initialLoad = async () => {          if (this._pageSizes.length === 0) { -            this._isPage = Array<string>(this.props.pdf.numPages);              this._pageSizes = Array<{ width: number, height: number }>(this.props.pdf.numPages); -            this._visibleElements = Array<JSX.Element>(this.props.pdf.numPages);              await Promise.all(this._pageSizes.map<Pdfjs.PDFPromise<any>>((val, i) =>                  this.props.pdf.getPage(i + 1).then(action((page: Pdfjs.PDFPageProxy) => {                      this._pageSizes.splice(i, 1, { -                        width: (page.view[page.rotate === 0 || page.rotate === 180 ? 2 : 3] - page.view[page.rotate === 0 || page.rotate === 180 ? 0 : 1]) * scale, -                        height: (page.view[page.rotate === 0 || page.rotate === 180 ? 3 : 2] - page.view[page.rotate === 0 || page.rotate === 180 ? 1 : 0]) * scale +                        width: (page.view[page.rotate === 0 || page.rotate === 180 ? 2 : 3] - page.view[page.rotate === 0 || page.rotate === 180 ? 0 : 1]), +                        height: (page.view[page.rotate === 0 || page.rotate === 180 ? 3 : 2] - page.view[page.rotate === 0 || page.rotate === 180 ? 1 : 0])                      }); -                    this._visibleElements.splice(i, 1, -                        <div key={`${this.props.url}-placeholder-${i + 1}`} className="pdfviewer-placeholder" -                            style={{ width: this._pageSizes[i].width, height: this._pageSizes[i].height }}> -                            "PAGE IS LOADING... " -                </div>); -                    this.getPlaceholderPage(i); +                    i === this.props.pdf.numPages - 1 && this.props.loaded((page.view[page.rotate === 0 || page.rotate === 180 ? 2 : 3] - page.view[page.rotate === 0 || page.rotate === 180 ? 0 : 1]), +                        (page.view[page.rotate === 0 || page.rotate === 180 ? 3 : 2] - page.view[page.rotate === 0 || page.rotate === 180 ? 1 : 0]), i);                  })))); -            this.props.loaded(Math.max(...this._pageSizes.map(i => i.width)), this._pageSizes[0].height, this.props.pdf.numPages); - -            let startY = NumCast(this.props.Document.startY, NumCast(this.props.Document.panY)); -            this.props.setPanY && this.props.setPanY(startY); -            this.props.scrollTo(startY); +            Doc.GetProto(this.props.Document).scrollHeight = this._pageSizes.reduce((size, page) => size + page.height, 0);          }      }      @action +    setupPdfJsViewer = async () => { +        this._showWaiting = true; +        this.props.setPdfViewer(this); +        await this.initialLoad(); + +        this._annotationReactionDisposer = reaction( +            () => this.props.fieldExtensionDoc && DocListCast(this.props.fieldExtensionDoc.annotations), +            annotations => annotations && annotations.length && this.renderAnnotations(annotations, true), +            { fireImmediately: true }); + +        this._filterReactionDisposer = reaction( +            () => ({ scriptField: Cast(this.props.Document.filterScript, ScriptField), annos: this._annotations.slice() }), +            action(({ scriptField, annos }: { scriptField: FieldResult<ScriptField>, annos: Doc[] }) => { +                let oldScript = this._script.originalScript; +                this._script = scriptField && scriptField.script.compiled ? scriptField.script : CompileScript("return true") as CompiledScript; +                if (this._script.originalScript !== oldScript) { +                    this.Index = -1; +                } +                annos.forEach(d => d.opacity = this._script.run({ this: d }, console.log, 1).result ? 1 : 0); +            }), +            { fireImmediately: true } +        ); +        this._reactionDisposer = reaction( +            () => this.props.Document.panY, +            () => this._mainCont.current && this._mainCont.current.scrollTo({ top: NumCast(this.props.Document.panY) || 0, behavior: "auto" }) +        ); + +        document.removeEventListener("copy", this.copy); +        document.addEventListener("copy", this.copy); +        document.addEventListener("pagesinit", action(() => { +            this.pdfViewer.currentScaleValue = this._zoomed = 1; +            this.gotoPage(NumCast(this.props.Document.curPage, 1)); +        })); +        document.addEventListener("pagerendered", action(() => this._showCover = this._showWaiting = false)); +        var pdfLinkService = new PDFJSViewer.PDFLinkService(); +        let pdfFindController = new PDFJSViewer.PDFFindController({ +            linkService: pdfLinkService, +        }); +        this.pdfViewer = new PDFJSViewer.PDFViewer({ +            container: this._mainCont.current, +            viewer: this._viewer.current, +            linkService: pdfLinkService, +            findController: pdfFindController, +            renderer: "canvas", +        }); +        pdfLinkService.setViewer(this.pdfViewer); +        pdfLinkService.setDocument(this.props.pdf, null); +        this.pdfViewer.setDocument(this.props.pdf); +    } + +    @action      makeAnnotationDocument = (sourceDoc: Doc | undefined, color: string, createLink: boolean = true): Doc => {          let mainAnnoDoc = Docs.Create.InstanceFromProto(new Doc(), "", {});          let mainAnnoDocProto = Doc.GetProto(mainAnnoDoc); @@ -198,8 +228,9 @@ export class PDFViewer extends React.Component<IViewerProps> {              annoDocs.push(annoDoc);              annoDoc.isButton = true;              anno.remove(); -            this.props.addDocument && this.props.addDocument(annoDoc, false); +            // this.props.addDocument && this.props.addDocument(annoDoc, false);              mainAnnoDoc = annoDoc; +            mainAnnoDocProto.y = annoDoc.y;              mainAnnoDocProto = Doc.GetProto(annoDoc);          } else {              this._savedAnnotations.forEach((key: number, value: HTMLDivElement[]) => value.map(anno => { @@ -231,61 +262,6 @@ export class PDFViewer extends React.Component<IViewerProps> {      }      @action -    getPlaceholderPage = (page: number) => { -        if (this._isPage[page] !== "none") { -            this._isPage[page] = "none"; -            this._visibleElements[page] = ( -                <div key={`${this.props.url}-placeholder-${page + 1}`} className="pdfviewer-placeholder" -                    style={{ width: this._pageSizes[page].width, height: this._pageSizes[page].height }}> -                    "PAGE IS LOADING... " -                </div>); -        } -    } - -    @action -    getRenderedPage = (page: number) => { -        if (this._isPage[page] !== "page") { -            this._isPage[page] = "page"; -            this._visibleElements[page] = (<Page {...this.props} -                size={this._pageSizes[page]} -                numPages={this.props.pdf.numPages} -                setSelectionText={this.setSelectionText} -                page={page} -                key={`${this.props.url}-rendered-${page + 1}`} -                name={`${this.props.pdf.fingerprint + `-page${page + 1}`}`} -                pageLoaded={this.pageLoaded} -                renderAnnotations={this.renderAnnotations} -                createAnnotation={this.createAnnotation} -                sendAnnotations={this.receiveAnnotations} -                makeAnnotationDocuments={this.makeAnnotationDocument} -                getScrollFromPage={this.getScrollFromPage} />); -        } -    } - -    // change the address to be the file address of the PNG version of each page -    // file address of the pdf -    @action -    getPageImage = async (page: number) => { -        if (this._isPage[page] !== "image") { -            this._isPage[page] = "image"; -            try { -                let res = JSON.parse(await rp.get(Utils.prepend(`/thumbnail${this.props.url.substring("files/".length, this.props.url.length - ".pdf".length)}-${page + 1}.PNG`))); -                runInAction(() => this._visibleElements[page] = -                    <img key={res.path} src={res.path} onError={() => this.getRenderedPage(page)} -                        style={{ width: `${parseInt(res.width) * scale}px`, height: `${parseInt(res.height) * scale}px` }} />); -            } catch (e) { -                console.log(e); -            } -        } -    } - -    renderPages = () => { -        numberRange(this.props.pdf.numPages).filter(p => this._isPage[p] !== undefined).map(i => -            (i < this.startIndex || i > this.endIndex) ? this.getPlaceholderPage(i) : // pages outside of the pdf use empty stand-in divs -                this.props.active() ? this.getRenderedPage(i) : this.getPageImage(i)); -    } - -    @action      renderAnnotations = (annotations: Doc[], removeOldAnnotations: boolean): void => {          if (removeOldAnnotations) {              this._annotations = annotations; @@ -297,23 +273,30 @@ export class PDFViewer extends React.Component<IViewerProps> {      }      @action -    prevAnnotation = (e: React.MouseEvent) => { -        e.stopPropagation(); +    prevAnnotation = () => {          this.Index = Math.max(this.Index - 1, 0); -        let scrollToAnnotation = this.allAnnotations.sort((a, b) => NumCast(a.y) - NumCast(b.y))[this.Index]; -        this.allAnnotations.forEach(d => Doc.UnBrushDoc(d)); -        Doc.BrushDoc(scrollToAnnotation); -        this.props.scrollTo(NumCast(scrollToAnnotation.y)); +        this.scrollToAnnotation(this.allAnnotations.sort((a, b) => NumCast(a.y) - NumCast(b.y))[this.Index]);      }      @action -    nextAnnotation = (e: React.MouseEvent) => { -        e.stopPropagation(); +    nextAnnotation = () => {          this.Index = Math.min(this.Index + 1, this.allAnnotations.length - 1); -        let scrollToAnnotation = this.allAnnotations.sort((a, b) => NumCast(a.y) - NumCast(b.y))[this.Index]; +        this.scrollToAnnotation(this.allAnnotations.sort((a, b) => NumCast(a.y) - NumCast(b.y))[this.Index]); +    } + +    @action +    gotoPage = (p: number) => { +        this.pdfViewer && this.pdfViewer.scrollPageIntoView({ pageNumber: Math.min(Math.max(1, p), this._pageSizes.length) }); +    } + +    @action +    scrollToAnnotation = (scrollToAnnotation: Doc) => {          this.allAnnotations.forEach(d => Doc.UnBrushDoc(d)); +        let windowHgt = this.props.PanelHeight() / this.props.ContentScaling(); +        let scrollRange = this._mainCont.current!.scrollHeight - windowHgt; +        let pgScroll = scrollRange / this._pageSizes.length; +        this._mainCont.current!.scrollTo(0, NumCast(scrollToAnnotation.y) - pgScroll / 2);          Doc.BrushDoc(scrollToAnnotation); -        this.props.scrollTo(NumCast(scrollToAnnotation.y));      }      sendAnnotations = (page: number) => { @@ -330,6 +313,11 @@ export class PDFViewer extends React.Component<IViewerProps> {          }      } +    @action +    onScroll = (e: React.UIEvent<HTMLElement>) => { +        this.pdfViewer && (this.props.Document.curPage = this.pdfViewer.currentPageNumber); +    } +      // get the page index that the vertical offset passed in is on      getPageFromScroll = (vOffset: number) => {          let index = 0; @@ -340,15 +328,11 @@ export class PDFViewer extends React.Component<IViewerProps> {          return index;      } -    getScrollFromPage = (index: number): number => { -        return numberRange(Math.min(this.props.pdf.numPages, index)).reduce((counter, i) => counter + this._pageSizes[i].height, 0); -    } -      @action      createAnnotation = (div: HTMLDivElement, page: number) => {          if (this._annotationLayer.current) {              if (div.style.top) { -                div.style.top = (parseInt(div.style.top) + this.getScrollFromPage(page)).toString(); +                div.style.top = (parseInt(div.style.top)/*+ this.getScrollFromPage(page)*/).toString();              }              this._annotationLayer.current.append(div);              let savedPage = this._savedAnnotations.getValue(page); @@ -363,11 +347,14 @@ export class PDFViewer extends React.Component<IViewerProps> {      }      @action -    search = (searchString: string) => { -        if (this._pdfViewer._pageViewsReady) { -            this._pdfFindController.executeCommand('findagain', { +    search = (searchString: string, fwd: boolean) => { +        if (!searchString) { +            fwd ? this.nextAnnotation() : this.prevAnnotation(); +        } +        else if (this.pdfViewer._pageViewsReady) { +            this.pdfViewer.findController.executeCommand('findagain', {                  caseSensitive: false, -                findPrevious: undefined, +                findPrevious: !fwd,                  highlightAll: true,                  phraseSearch: true,                  query: searchString @@ -375,122 +362,357 @@ export class PDFViewer extends React.Component<IViewerProps> {          }          else if (this._mainCont.current) {              let executeFind = () => { -                this._pdfFindController.executeCommand('find', { +                this.pdfViewer.findController.executeCommand('find', {                      caseSensitive: false, -                    findPrevious: undefined, +                    findPrevious: !fwd,                      highlightAll: true,                      phraseSearch: true,                      query: searchString                  }); -            } +            };              this._mainCont.current.addEventListener("pagesloaded", executeFind);              this._mainCont.current.addEventListener("pagerendered", executeFind);          }      } -      @action -    toggleSearch = (e: React.MouseEvent) => { -        e.stopPropagation(); -        this._searching = !this._searching; - -        if (this._searching) { -            if (!this._pdfFindController && this._mainCont.current && this._viewer.current && !this._pdfViewer) { -                let simpleLinkService = new SimpleLinkService(this); -                this._pdfViewer = new PDFJSViewer.PDFViewer({ -                    container: this._mainCont.current, -                    viewer: this._viewer.current, -                    linkService: simpleLinkService -                }) -                simpleLinkService.setPDFJSview(this._pdfViewer); -                this._mainCont.current.addEventListener("pagesinit", () => this._pdfViewer.currentScaleValue = 1); -                this._mainCont.current.addEventListener("pagerendered", () => console.log("rendered")); -                this._pdfViewer.setDocument(this.props.pdf); -                this._pdfFindController = new PDFJSViewer.PDFFindController(this._pdfViewer); -                this._pdfViewer.findController = this._pdfFindController; +    onPointerDown = (e: React.PointerEvent): void => { +        // if alt+left click, drag and annotate +        this._downX = e.clientX; +        this._downY = e.clientY; +        if (NumCast(this.props.Document.scale, 1) !== 1) return; +        if (e.button !== 0 && this.active()) { +            this._setPreviewCursor && this._setPreviewCursor(e.clientX, e.clientY, true); +        } +        this._marqueeing = false; +        if (!e.altKey && e.button === 0 && this.active()) { +            PDFMenu.Instance.StartDrag = this.startDrag; +            PDFMenu.Instance.Highlight = this.highlight; +            PDFMenu.Instance.Snippet = this.createSnippet; +            PDFMenu.Instance.Status = "pdf"; +            PDFMenu.Instance.fadeOut(true); +            if (e.target && (e.target as any).parentElement.className === "textLayer") { +                if (!e.ctrlKey) { +                    this.receiveAnnotations([], -1); +                }              } +            else { +                // set marquee x and y positions to the spatially transformed position +                if (this._mainCont.current) { +                    let boundingRect = this._mainCont.current.getBoundingClientRect(); +                    this._startX = this._marqueeX = (e.clientX - boundingRect.left) * (this._mainCont.current.offsetWidth / boundingRect.width); +                    this._startY = this._marqueeY = (e.clientY - boundingRect.top) * (this._mainCont.current.offsetHeight / boundingRect.height) + this._mainCont.current.scrollTop; +                } +                this._marqueeing = true; +                let marquees = this._mainCont.current!.getElementsByClassName("pdfViewer-dragAnnotationBox"); +                if (marquees && marquees.length) { // make a copy of the marquee +                    let marquee = marquees[0] as HTMLDivElement; +                    marquee.style.opacity = "0.2"; +                } +                this.receiveAnnotations([], -1); +            } +            document.removeEventListener("pointermove", this.onSelectMove); +            document.addEventListener("pointermove", this.onSelectMove); +            document.removeEventListener("pointerup", this.onSelectEnd); +            document.addEventListener("pointerup", this.onSelectEnd);          }      } -    @computed get visibleElementWrapper() { -        trace(); -        return this._visibleElements; + +    @action +    onSelectMove = (e: PointerEvent): void => { +        if (this._marqueeing && this._mainCont.current) { +            // transform positions and find the width and height to set the marquee to +            let boundingRect = this._mainCont.current.getBoundingClientRect(); +            this._marqueeWidth = ((e.clientX - boundingRect.left) * (this._mainCont.current.offsetWidth / boundingRect.width)) - this._startX; +            this._marqueeHeight = ((e.clientY - boundingRect.top) * (this._mainCont.current.offsetHeight / boundingRect.height)) - this._startY + this._mainCont.current.scrollTop; +            this._marqueeX = Math.min(this._startX, this._startX + this._marqueeWidth); +            this._marqueeY = Math.min(this._startY, this._startY + this._marqueeHeight); +            this._marqueeWidth = Math.abs(this._marqueeWidth); +            this._marqueeHeight = Math.abs(this._marqueeHeight); +            e.stopPropagation(); +            e.preventDefault(); +        } +        else if (e.target && (e.target as any).parentElement === this._mainCont.current) { +            e.stopPropagation(); +        }      } -    render() { -        return (<div className="pdfViewer-viewer" ref={this._mainCont} > -            <div className="pdfViewer-visibleElements" style={this._searching ? { position: "absolute", top: 0 } : {}}> -                {this.visibleElementWrapper} -            </div> -            <div className="pdfViewer-text" ref={this._viewer} /> -            <div className="pdfViewer-annotationLayer" style={{ height: NumCast(this.props.Document.nativeHeight) }} ref={this._annotationLayer}> -                {this.nonDocAnnotations.sort((a, b) => NumCast(a.y) - NumCast(b.y)).map((anno, index) => -                    <Annotation {...this.props} anno={anno} key={`${anno[Id]}-annotation`} />)} -            </div> -            <div className="pdfViewer-overlayCont" onPointerDown={(e) => e.stopPropagation()} -                style={{ bottom: -this.props.panY, left: `${this._searching ? 0 : 100}%` }}> -                <button className="pdfViewer-overlayButton" title="Open Search Bar" /> -                <input className="pdfViewer-overlaySearchBar" placeholder="Search" onChange={this.searchStringChanged} -                    onKeyDown={(e: React.KeyboardEvent) => e.keyCode === KeyCodes.ENTER ? this.search(this._searchString) : e.keyCode === KeyCodes.BACKSPACE ? e.stopPropagation() : true} /> -                <button title="Search" onClick={() => this.search(this._searchString)}> -                    <FontAwesomeIcon icon="search" size="3x" color="white" /></button> -            </div> -            <button className="pdfViewer-overlayButton" onClick={this.prevAnnotation} title="Previous Annotation" -                style={{ bottom: -this.props.panY + 280, right: 10, display: this.props.active() ? "flex" : "none" }}> -                <div className="pdfViewer-overlayButton-iconCont" onPointerDown={(e) => e.stopPropagation()}> -                    <FontAwesomeIcon style={{ color: "white" }} icon={"arrow-up"} size="3x" /></div> -            </button> -            <button className="pdfViewer-overlayButton" onClick={this.nextAnnotation} title="Next Annotation" -                style={{ bottom: -this.props.panY + 200, right: 10, display: this.props.active() ? "flex" : "none" }}> -                <div className="pdfViewer-overlayButton-iconCont" onPointerDown={(e) => e.stopPropagation()}> -                    <FontAwesomeIcon style={{ color: "white" }} icon={"arrow-down"} size="3x" /></div> -            </button> -            <button className="pdfViewer-overlayButton" onClick={this.toggleSearch} title="Open Search Bar" -                style={{ bottom: -this.props.panY + 10, right: 0, display: this.props.active() ? "flex" : "none" }}> -                <div className="pdfViewer-overlayButton-arrow" onPointerDown={(e) => e.stopPropagation()}></div> -                <div className="pdfViewer-overlayButton-iconCont" onPointerDown={(e) => e.stopPropagation()}> -                    <FontAwesomeIcon style={{ color: "white" }} icon={this._searching ? "times" : "search"} size="3x" /></div> -            </button> -        </div >); +    @action +    createTextAnnotation = (sel: Selection, selRange: Range) => { +        if (this._mainCont.current) { +            let boundingRect = this._mainCont.current.getBoundingClientRect(); +            let clientRects = selRange.getClientRects(); +            for (let i = 0; i < clientRects.length; i++) { +                let rect = clientRects.item(i); +                if (rect/* && rect.width !== this._mainCont.current.getBoundingClientRect().width && rect.height !== this._mainCont.current.getBoundingClientRect().height / this.props.pdf.numPages*/) { +                    let scaleY = this._mainCont.current.offsetHeight / boundingRect.height; +                    let scaleX = this._mainCont.current.offsetWidth / boundingRect.width; +                    if (rect.width !== this._mainCont.current.clientWidth) { +                        let annoBox = document.createElement("div"); +                        annoBox.className = "pdfPage-annotationBox"; +                        // transforms the positions from screen onto the pdf div +                        annoBox.style.top = ((rect.top - boundingRect.top) * scaleY + this._mainCont.current.scrollTop).toString(); +                        annoBox.style.left = ((rect.left - boundingRect.left) * scaleX).toString(); +                        annoBox.style.width = (rect.width * this._mainCont.current.offsetWidth / boundingRect.width).toString(); +                        annoBox.style.height = (rect.height * this._mainCont.current.offsetHeight / boundingRect.height).toString(); +                        this.createAnnotation(annoBox, this.getPageFromScroll(rect.top)); +                    } +                } +            } +        } +        let text = selRange.cloneContents().textContent; +        text && this.setSelectionText(text); + +        // clear selection +        if (sel.empty) {  // Chrome +            sel.empty(); +        } else if (sel.removeAllRanges) {  // Firefox +            sel.removeAllRanges(); +        }      } -} -export enum AnnotationTypes { Region } +    @action +    onSelectEnd = (e: PointerEvent): void => { +        if (this._marqueeing) { +            if (this._marqueeWidth > 10 || this._marqueeHeight > 10) { +                let marquees = this._mainCont.current!.getElementsByClassName("pdfViewer-dragAnnotationBox"); +                if (marquees && marquees.length) { // make a copy of the marquee +                    let copy = document.createElement("div"); +                    let marquee = marquees[0] as HTMLDivElement; +                    let style = marquee.style; +                    copy.style.left = style.left; +                    copy.style.top = style.top; +                    copy.style.width = style.width; +                    copy.style.height = style.height; +                    copy.style.border = style.border; +                    copy.style.opacity = style.opacity; +                    copy.className = "pdfPage-annotationBox"; +                    this.createAnnotation(copy, this.getPageFromScroll(this._marqueeY)); +                    marquee.style.opacity = "0"; +                } -class SimpleLinkService { -    _viewer: PDFViewer; -    _pdfjsView: any; +                if (!e.ctrlKey) { +                    PDFMenu.Instance.Status = "snippet"; +                    PDFMenu.Instance.Marquee = { left: this._marqueeX, top: this._marqueeY, width: this._marqueeWidth, height: this._marqueeHeight }; +                } +                PDFMenu.Instance.jumpTo(e.clientX, e.clientY); +            } -    constructor(viewer: PDFViewer) { -        this._viewer = viewer; +            this._marqueeHeight = this._marqueeWidth = 0; +        } +        else { +            let sel = window.getSelection(); +            if (sel && sel.type === "Range") { +                let selRange = sel.getRangeAt(0); +                this.createTextAnnotation(sel, selRange); +                PDFMenu.Instance.jumpTo(e.clientX, e.clientY); +            } +        } + +        if (PDFMenu.Instance.Highlighting) { +            this.highlight(undefined, "goldenrod"); +        } +        else { +            PDFMenu.Instance.StartDrag = this.startDrag; +            PDFMenu.Instance.Highlight = this.highlight; +        } +        document.removeEventListener("pointermove", this.onSelectMove); +        document.removeEventListener("pointerup", this.onSelectEnd);      } -    setPDFJSview(v: any) { this._pdfjsView = v; } -    navigateTo() { } +    @action +    highlight = (targetDoc: Doc | undefined, color: string) => { +        // creates annotation documents for current highlights +        let annotationDoc = this.makeAnnotationDocument(targetDoc, color, false); +        Doc.AddDocToList(this.props.fieldExtensionDoc, this.props.fieldExt, annotationDoc); +        return annotationDoc; +    } -    getDestinationHash() { return "#"; } +    /** +     * This is temporary for creating annotations from highlights. It will +     * start a drag event and create or put the necessary info into the drag event. +     */ +    @action +    startDrag = (e: PointerEvent, ele: HTMLElement): void => { +        e.preventDefault(); +        e.stopPropagation(); +        let targetDoc = Docs.Create.TextDocument({ width: 200, height: 200, title: "New Annotation" }); +        targetDoc.targetPage = this.getPageFromScroll(this._marqueeY); +        let annotationDoc = this.highlight(undefined, "red"); +        annotationDoc.linkedToDoc = false; +        let dragData = new DragManager.AnnotationDragData(this.props.Document, annotationDoc, targetDoc); +        DragManager.StartAnnotationDrag([ele], dragData, e.pageX, e.pageY, { +            handlers: { +                dragComplete: () => { +                    if (!annotationDoc.linkedToDoc) { +                        let annotations = DocListCast(annotationDoc.annotations); +                        annotations && annotations.forEach(anno => anno.target = targetDoc); +                        DocUtils.MakeLink(annotationDoc, targetDoc, dragData.targetContext, `Annotation from ${StrCast(this.props.Document.title)}`); +                    } +                } +            }, +            hideSource: false +        }); +    } -    getAnchorUrl() { return "#"; } +    createSnippet = (marquee: { left: number, top: number, width: number, height: number }): void => { +        let view = Doc.MakeAlias(this.props.Document); +        let data = Doc.MakeDelegate(Doc.GetProto(this.props.Document)); +        data.title = StrCast(data.title) + "_snippet"; +        view.proto = data; +        view.nativeHeight = marquee.height; +        view.height = (this.props.Document[WidthSym]() / NumCast(this.props.Document.nativeWidth)) * marquee.height; +        view.nativeWidth = this.props.Document.nativeWidth; +        view.startY = marquee.top; +        view.width = this.props.Document[WidthSym](); +        DragManager.StartDocumentDrag([], new DragManager.DocumentDragData([view]), 0, 0); +    } -    setHash() { } +    // this is called with the document that was dragged and the collection to move it into. +    // if the target collection is the same as this collection, then the move will be allowed. +    // otherwise, the document being moved must be able to be removed from its container before +    // moving it into the target.   +    @action.bound +    moveDocument(doc: Doc, targetCollection: Doc, addDocument: (doc: Doc) => boolean): boolean { +        if (Doc.AreProtosEqual(this.props.Document, targetCollection)) { +            return true; +        } +        return this.removeDocument(doc) ? addDocument(doc) : false; +    } -    isPageVisible(page: number) { return true; } -    executeNamedAction() { } +    @action.bound +    removeDocument(doc: Doc): boolean { +        //TODO This won't create the field if it doesn't already exist +        let targetDataDoc = this.props.fieldExtensionDoc; +        let targetField = this.props.fieldExt; +        let value = Cast(targetDataDoc[targetField], listSpec(Doc), []); +        let index = value.reduce((p, v, i) => (v instanceof Doc && v === doc) ? i : p, -1); +        index = index !== -1 ? index : value.reduce((p, v, i) => (v instanceof Doc && Doc.AreProtosEqual(v, doc)) ? i : p, -1); +        index !== -1 && value.splice(index, 1); +        return true; +    } +    scrollXf = () => { +        return this._mainCont.current ? this.props.ScreenToLocalTransform().translate(0, this._mainCont.current.scrollTop) : this.props.ScreenToLocalTransform(); +    } +    setPreviewCursor = (func?: (x: number, y: number, drag: boolean) => void) => { +        this._setPreviewCursor = func; +    } +    onClick = (e: React.MouseEvent) => { +        this._setPreviewCursor && +            e.button === 0 && +            Math.abs(e.clientX - this._downX) < 3 && +            Math.abs(e.clientY - this._downY) < 3 && +            this._setPreviewCursor(e.clientX, e.clientY, false); +    } +    whenActiveChanged = (isActive: boolean) => { +        this._isChildActive = isActive; +        this.props.whenActiveChanged(isActive); +    } +    active = () => { +        return this.props.isSelected() || this._isChildActive || this.props.renderDepth === 0; +    } -    cachePageRef() { } +    getCoverImage = () => { +        if (!this.props.Document[HeightSym]()) { +            setTimeout(() => { +                this.props.Document.height = this.props.Document[WidthSym]() * this._coverPath.height / this._coverPath.width; +                this.props.Document.nativeHeight = nativeWidth * this._coverPath.height / this._coverPath.width; +            }, 0); +        } +        let nativeWidth = NumCast(this.props.Document.nativeWidth); +        let nativeHeight = NumCast(this.props.Document.nativeHeight); +        return <img key={this._coverPath.path} src={this._coverPath.path} onLoad={action(() => this._showWaiting = false)} +            style={{ position: "absolute", display: "inline-block", top: 0, left: 0, width: `${nativeWidth}px`, height: `${nativeHeight}px` }} />; +    } -    get pagesCount() { return this._viewer._pdfViewer.pagesCount; } -    get page() { return NumCast(this._viewer.props.Document.curPage); } -    set page(p: number) { -        this._pdfjsView._ensurePdfPageLoaded(this._pdfjsView._pages[p - 1]).then(() => { -            this._pdfjsView.renderingQueue.renderView(this._pdfjsView._pages[p - 1]); -            if (this._viewer.props.GoToPage) -                this._viewer.props.GoToPage(p); -        }); +    @action +    onZoomWheel = (e: React.WheelEvent) => { +        e.stopPropagation(); +        if (e.ctrlKey) { +            let curScale = Number(this.pdfViewer.currentScaleValue); +            this.pdfViewer.currentScaleValue = Math.max(1, Math.min(10, curScale + curScale * e.deltaY / 1000)); +            this._zoomed = Number(this.pdfViewer.currentScaleValue); +        } +    } + +    @computed get annotationLayer() { +        trace(); +        return <div className="pdfViewer-annotationLayer" style={{ height: NumCast(this.props.Document.nativeHeight) }} ref={this._annotationLayer}> +            {this.nonDocAnnotations.sort((a, b) => NumCast(a.y) - NumCast(b.y)).map((anno, index) => +                <Annotation {...this.props} anno={anno} key={`${anno[Id]}-annotation`} />)} +        </div>; +    } +    @computed get pdfViewerDiv() { +        trace(); +        return <div className="pdfViewer-text" ref={this._viewer} style={{ transformOrigin: "left top" }} />; +    } +    @computed get standinViews() { +        trace(); +        return <> +            {this._showCover ? this.getCoverImage() : (null)} +            {this._showWaiting ? <img className="pdfViewer-waiting" key="waiting" src={"/assets/loading.gif"} /> : (null)} +        </>; +    } +    marqueeWidth = () => this._marqueeWidth; +    marqueeHeight = () => this._marqueeHeight; +    marqueeX = () => this._marqueeX; +    marqueeY = () => this._marqueeY; +    marqueeing = () => this._marqueeing; +    render() { +        trace(); +        return (<div className={"pdfViewer-viewer" + (this._zoomed !== 1 ? "-zoomed" : "")} onScroll={this.onScroll} onWheel={this.onZoomWheel} onPointerDown={this.onPointerDown} onClick={this.onClick} ref={this._mainCont}> +            {this.pdfViewerDiv} +            <PdfViewerMarquee isMarqueeing={this.marqueeing} width={this.marqueeWidth} height={this.marqueeHeight} x={this.marqueeX} y={this.marqueeY} /> +            <div className="pdfViewer-overlay" style={{ transform: `scale(${this._zoomed})` }}> +                {this.annotationLayer} +                <CollectionFreeFormView {...this.props} +                    setPreviewCursor={this.setPreviewCursor} +                    PanelHeight={() => NumCast(this.props.Document.scrollHeight, NumCast(this.props.Document.nativeHeight))} +                    PanelWidth={() => this._pageSizes.length && this._pageSizes[0] ? this._pageSizes[0].width : NumCast(this.props.Document.nativeWidth)} +                    focus={emptyFunction} +                    isSelected={this.props.isSelected} +                    select={emptyFunction} +                    active={this.active} +                    ContentScaling={returnOne} +                    whenActiveChanged={this.whenActiveChanged} +                    removeDocument={this.removeDocument} +                    moveDocument={this.moveDocument} +                    addDocument={(doc: Doc, allow: boolean | undefined) => { Doc.AddDocToList(this.props.fieldExtensionDoc, this.props.fieldExt, doc); return true; }} +                    CollectionView={this.props.ContainingCollectionView} +                    ScreenToLocalTransform={this.scrollXf} +                    ruleProvider={undefined} +                    renderDepth={this.props.renderDepth + 1} +                    ContainingCollectionDoc={this.props.ContainingCollectionView && this.props.ContainingCollectionView.props.Document} +                    chromeCollapsed={true}> +                </CollectionFreeFormView> +            </div> +            {this.standinViews} +        </div >);      } +} +interface PdfViewerMarqueeProps { +    isMarqueeing: () => boolean; +    width: () => number; +    height: () => number; +    x: () => number; +    y: () => number; +} -    get rotation() { return 0; } -    set rotation(value: any) { } -}
\ No newline at end of file +@observer +class PdfViewerMarquee extends React.Component<PdfViewerMarqueeProps> { +    render() { +        return !this.props.isMarqueeing() ? (null) : <div className="pdfViewer-dragAnnotationBox" +            style={{ +                left: `${this.props.x()}px`, top: `${this.props.y()}px`, +                width: `${this.props.width()}px`, height: `${this.props.height()}px`, +                border: `${this.props.width() === 0 ? "" : "2px dashed black"}` +            }}> +        </div> +    } +} + + +export enum AnnotationTypes { Region } diff --git a/src/client/views/pdf/Page.scss b/src/client/views/pdf/Page.scss deleted file mode 100644 index d8034b4b4..000000000 --- a/src/client/views/pdf/Page.scss +++ /dev/null @@ -1,36 +0,0 @@ - -.pdfViewer-text { -    .page { -        position: relative; -    } -} -.pdfPage-cont { -    position: relative; - -    .pdfPage-canvasContainer { -        position: absolute; -    } - -    .pdfPage-dragAnnotationBox { -        position: absolute; -        background-color: transparent; -        opacity: 0.1; -    } - -    .pdfPage-textLayer { -        position: absolute; -        width: 100%; -        height: 100%; -        div { -            user-select: text; -        } -        span { -            color: transparent; -            position: absolute; -            white-space: pre; -            cursor: text; -            -webkit-transform-origin: 0% 0%; -            transform-origin: 0% 0%; -        } -    } -}
\ No newline at end of file diff --git a/src/client/views/pdf/Page.tsx b/src/client/views/pdf/Page.tsx deleted file mode 100644 index 533247170..000000000 --- a/src/client/views/pdf/Page.tsx +++ /dev/null @@ -1,293 +0,0 @@ -import { action, IReactionDisposer, observable } from "mobx"; -import { observer } from "mobx-react"; -import * as Pdfjs from "pdfjs-dist"; -import "pdfjs-dist/web/pdf_viewer.css"; -import { Doc, DocListCastAsync, Opt, WidthSym } from "../../../new_fields/Doc"; -import { BoolCast, Cast, NumCast, StrCast } from "../../../new_fields/Types"; -import { Docs, DocUtils } from "../../documents/Documents"; -import { DragManager } from "../../util/DragManager"; -import PDFMenu from "./PDFMenu"; -import { scale } from "./PDFViewer"; -import "./Page.scss"; -import React = require("react"); - - -interface IPageProps { -    size: { width: number, height: number }; -    pdf: Pdfjs.PDFDocumentProxy; -    name: string; -    numPages: number; -    page: number; -    pageLoaded: (page: Pdfjs.PDFPageViewport) => void; -    fieldExtensionDoc: Doc; -    Document: Doc; -    renderAnnotations: (annotations: Doc[], removeOld: boolean) => void; -    sendAnnotations: (annotations: HTMLDivElement[], page: number) => void; -    createAnnotation: (div: HTMLDivElement, page: number) => void; -    makeAnnotationDocuments: (doc: Doc | undefined, color: string, linkTo: boolean) => Doc; -    getScrollFromPage: (page: number) => number; -    setSelectionText: (text: string) => void; -} - -@observer -export default class Page extends React.Component<IPageProps> { -    @observable private _state: "N/A" | "rendering" = "N/A"; -    @observable private _width: number = this.props.size.width; -    @observable private _height: number = this.props.size.height; -    @observable private _page: Opt<Pdfjs.PDFPageProxy>; -    @observable private _currPage: number = this.props.page + 1; -    @observable private _marqueeX: number = 0; -    @observable private _marqueeY: number = 0; -    @observable private _marqueeWidth: number = 0; -    @observable private _marqueeHeight: number = 0; - -    private _canvas: React.RefObject<HTMLCanvasElement> = React.createRef(); -    private _textLayer: React.RefObject<HTMLDivElement> = React.createRef(); -    private _marquee: React.RefObject<HTMLDivElement> = React.createRef(); -    private _marqueeing: boolean = false; -    private _reactionDisposer?: IReactionDisposer; -    private _startY: number = 0; -    private _startX: number = 0; - -    componentDidMount = (): void => this.loadPage(this.props.pdf); - -    componentDidUpdate = (): void => this.loadPage(this.props.pdf); - -    componentWillUnmount = (): void => this._reactionDisposer && this._reactionDisposer(); - -    loadPage = (pdf: Pdfjs.PDFDocumentProxy): void => { -        pdf.getPage(this._currPage).then(page => this.renderPage(page)); -    } - -    @action -    renderPage = (page: Pdfjs.PDFPageProxy): void => { -        // lower scale = easier to read at small sizes, higher scale = easier to read at large sizes -        if (this._state !== "rendering" && !this._page && this._canvas.current && this._textLayer.current) { -            this._state = "rendering"; -            let viewport = page.getViewport(scale); -            this._canvas.current.width = this._width = viewport.width; -            this._canvas.current.height = this._height = viewport.height; -            this.props.pageLoaded(viewport); -            let ctx = this._canvas.current.getContext("2d"); -            if (ctx) { -                //@ts-ignore -                page.render({ canvasContext: ctx, viewport: viewport, enableWebGL: true }); // renders the page onto the canvas context -                page.getTextContent().then(res =>                   // renders text onto the text container -                    //@ts-ignore -                    Pdfjs.renderTextLayer({ -                        textContent: res, -                        container: this._textLayer.current, -                        viewport: viewport -                    })); - -                this._page = page; -            } -        } -    } - -    @action -    highlight = (targetDoc: Doc | undefined, color: string) => { -        // creates annotation documents for current highlights -        let annotationDoc = this.props.makeAnnotationDocuments(targetDoc, color, false); -        Doc.AddDocToList(this.props.fieldExtensionDoc, "annotations", annotationDoc); -        return annotationDoc; -    } - -    /** -     * This is temporary for creating annotations from highlights. It will -     * start a drag event and create or put the necessary info into the drag event. -     */ -    @action -    startDrag = (e: PointerEvent, ele: HTMLElement): void => { -        e.preventDefault(); -        e.stopPropagation(); -        if (this._textLayer.current) { -            let targetDoc = Docs.Create.TextDocument({ width: 200, height: 200, title: "New Annotation" }); -            targetDoc.targetPage = this.props.page; -            let annotationDoc = this.highlight(undefined, "red"); -            annotationDoc.linkedToDoc = false; -            let dragData = new DragManager.AnnotationDragData(this.props.Document, annotationDoc, targetDoc); -            DragManager.StartAnnotationDrag([ele], dragData, e.pageX, e.pageY, { -                handlers: { -                    dragComplete: async () => { -                        if (!BoolCast(annotationDoc.linkedToDoc)) { -                            let annotations = await DocListCastAsync(annotationDoc.annotations); -                            annotations && annotations.forEach(anno => anno.target = targetDoc); -                            DocUtils.MakeLink(annotationDoc, targetDoc, dragData.targetContext, `Annotation from ${StrCast(this.props.Document.title)}`); -                        } -                    } -                }, -                hideSource: false -            }); -        } -    } - -    // cleans up events and boolean -    endDrag = (e: PointerEvent): void => { -        e.stopPropagation(); -    } - -    createSnippet = (marquee: { left: number, top: number, width: number, height: number }): void => { -        let view = Doc.MakeAlias(this.props.Document); -        let data = Doc.MakeDelegate(Doc.GetProto(this.props.Document)); -        data.title = StrCast(data.title) + "_snippet"; -        view.proto = data; -        view.nativeHeight = marquee.height; -        view.height = (this.props.Document[WidthSym]() / NumCast(this.props.Document.nativeWidth)) * marquee.height; -        view.nativeWidth = this.props.Document.nativeWidth; -        view.startY = marquee.top + this.props.getScrollFromPage(this.props.page); -        view.width = this.props.Document[WidthSym](); -        DragManager.StartDocumentDrag([], new DragManager.DocumentDragData([view]), 0, 0); -    } - -    @action -    onPointerDown = (e: React.PointerEvent): void => { -        // if alt+left click, drag and annotate -        if (NumCast(this.props.Document.scale, 1) !== 1) return; -        if (!e.altKey && e.button === 0) { -            PDFMenu.Instance.StartDrag = this.startDrag; -            PDFMenu.Instance.Highlight = this.highlight; -            PDFMenu.Instance.Snippet = this.createSnippet; -            PDFMenu.Instance.Status = "pdf"; -            PDFMenu.Instance.fadeOut(true); -            if (e.target && (e.target as any).parentElement === this._textLayer.current) { -                e.stopPropagation(); -                if (!e.ctrlKey) { -                    this.props.sendAnnotations([], -1); -                } -            } -            else { -                // set marquee x and y positions to the spatially transformed position -                if (this._textLayer.current) { -                    let boundingRect = this._textLayer.current.getBoundingClientRect(); -                    this._startX = this._marqueeX = (e.clientX - boundingRect.left) * (this._textLayer.current.offsetWidth / boundingRect.width); -                    this._startY = this._marqueeY = (e.clientY - boundingRect.top) * (this._textLayer.current.offsetHeight / boundingRect.height); -                } -                this._marqueeing = true; -                this._marquee.current && (this._marquee.current.style.opacity = "0.2"); -                this.props.sendAnnotations([], -1); -            } -            document.removeEventListener("pointermove", this.onSelectStart); -            document.addEventListener("pointermove", this.onSelectStart); -            document.removeEventListener("pointerup", this.onSelectEnd); -            document.addEventListener("pointerup", this.onSelectEnd); -        } -    } - -    @action -    onSelectStart = (e: PointerEvent): void => { -        if (this._marqueeing && this._textLayer.current) { -            // transform positions and find the width and height to set the marquee to -            let boundingRect = this._textLayer.current.getBoundingClientRect(); -            this._marqueeWidth = ((e.clientX - boundingRect.left) * (this._textLayer.current.offsetWidth / boundingRect.width)) - this._startX; -            this._marqueeHeight = ((e.clientY - boundingRect.top) * (this._textLayer.current.offsetHeight / boundingRect.height)) - this._startY; -            this._marqueeX = Math.min(this._startX, this._startX + this._marqueeWidth); -            this._marqueeY = Math.min(this._startY, this._startY + this._marqueeHeight); -            this._marqueeWidth = Math.abs(this._marqueeWidth); -            e.stopPropagation(); -            e.preventDefault(); -        } -        else if (e.target && (e.target as any).parentElement === this._textLayer.current) { -            e.stopPropagation(); -        } -    } - -    @action -    onSelectEnd = (e: PointerEvent): void => { -        if (this._marqueeing) { -            this._marqueeing = false; -            if (this._marqueeWidth > 10 || this._marqueeHeight > 10) { -                if (this._marquee.current) { // make a copy of the marquee -                    let copy = document.createElement("div"); -                    let style = this._marquee.current.style; -                    copy.style.left = style.left; -                    copy.style.top = style.top; -                    copy.style.width = style.width; -                    copy.style.height = style.height; -                    copy.style.border = style.border; -                    copy.style.opacity = style.opacity; -                    copy.className = "pdfPage-annotationBox"; -                    this.props.createAnnotation(copy, this.props.page); -                    this._marquee.current.style.opacity = "0"; -                } - -                if (!e.ctrlKey) { -                    PDFMenu.Instance.Status = "snippet"; -                    PDFMenu.Instance.Marquee = { left: this._marqueeX, top: this._marqueeY, width: this._marqueeWidth, height: this._marqueeHeight }; -                } -                PDFMenu.Instance.jumpTo(e.clientX, e.clientY); -            } - -            this._marqueeHeight = this._marqueeWidth = 0; -        } -        else { -            let sel = window.getSelection(); -            if (sel && sel.type === "Range") { -                let selRange = sel.getRangeAt(0); -                this.createTextAnnotation(sel, selRange); -                PDFMenu.Instance.jumpTo(e.clientX, e.clientY); -            } -        } - -        if (PDFMenu.Instance.Highlighting) { -            this.highlight(undefined, "goldenrod"); -        } -        else { -            PDFMenu.Instance.StartDrag = this.startDrag; -            PDFMenu.Instance.Highlight = this.highlight; -        } -        document.removeEventListener("pointermove", this.onSelectStart); -        document.removeEventListener("pointerup", this.onSelectEnd); -    } - -    @action -    createTextAnnotation = (sel: Selection, selRange: Range) => { -        if (this._textLayer.current) { -            let boundingRect = this._textLayer.current.getBoundingClientRect(); -            let clientRects = selRange.getClientRects(); -            for (let i = 0; i < clientRects.length; i++) { -                let rect = clientRects.item(i); -                if (rect && rect.width !== this._textLayer.current.getBoundingClientRect().width && rect.height !== this._textLayer.current.getBoundingClientRect().height) { -                    let annoBox = document.createElement("div"); -                    annoBox.className = "pdfPage-annotationBox"; -                    // transforms the positions from screen onto the pdf div -                    annoBox.style.top = ((rect.top - boundingRect.top) * (this._textLayer.current.offsetHeight / boundingRect.height)).toString(); -                    annoBox.style.left = ((rect.left - boundingRect.left) * (this._textLayer.current.offsetWidth / boundingRect.width)).toString(); -                    annoBox.style.width = (rect.width * this._textLayer.current.offsetWidth / boundingRect.width).toString(); -                    annoBox.style.height = (rect.height * this._textLayer.current.offsetHeight / boundingRect.height).toString(); -                    this.props.createAnnotation(annoBox, this.props.page); -                } -            } -        } -        let text = selRange.cloneContents().textContent; -        text && this.props.setSelectionText(text); - -        // clear selection -        if (sel.empty) {  // Chrome -            sel.empty(); -        } else if (sel.removeAllRanges) {  // Firefox -            sel.removeAllRanges(); -        } -    } - -    doubleClick = (e: React.MouseEvent) => { -        if (e.target && (e.target as any).parentElement === this._textLayer.current) { -            // do something to select the paragraph ideally -        } -    } - -    render() { -        return ( -            <div className={"pdfPage-cont"} onPointerDown={this.onPointerDown} onDoubleClick={this.doubleClick} style={{ "width": this._width, "height": this._height }}> -                <canvas className="PdfPage-canvasContainer" ref={this._canvas} /> -                <div className="pdfPage-dragAnnotationBox" ref={this._marquee} -                    style={{ -                        left: `${this._marqueeX}px`, top: `${this._marqueeY}px`, -                        width: `${this._marqueeWidth}px`, height: `${this._marqueeHeight}px`, -                        border: `${this._marqueeWidth === 0 ? "" : "10px dashed black"}` -                    }}> -                </div> -                <div className="pdfPage-textlayer" ref={this._textLayer} /> -            </div>); -    } -}
\ No newline at end of file diff --git a/src/debug/Test.tsx b/src/debug/Test.tsx index 79f87f4ac..3baedce4b 100644 --- a/src/debug/Test.tsx +++ b/src/debug/Test.tsx @@ -2,39 +2,12 @@ import * as React from 'react';  import * as ReactDOM from 'react-dom';  import { DocServer } from '../client/DocServer';  import { Doc } from '../new_fields/Doc'; +import * as Pdfjs from "pdfjs-dist"; +import "pdfjs-dist/web/pdf_viewer.css"; +import { Utils } from '../Utils'; +const PDFJSViewer = require("pdfjs-dist/web/pdf_viewer");  const protoId = "protoDoc";  const delegateId = "delegateDoc";  class Test extends React.Component { -    onCreateClick = () => { -        const proto = new Doc(protoId, true); -        const delegate = Doc.MakeDelegate(proto, delegateId); -    } - -    onReadClick = async () => { -        console.log("reading"); -        const docs = await DocServer.GetRefFields([delegateId, protoId]); -        console.log("done"); -        console.log(docs); -    } - -    onDeleteClick = () => { -        DocServer.DeleteDocuments([protoId, delegateId]); -    } - -    render() { -        return ( -            <div> -                <button onClick={this.onCreateClick}>Create Docs</button> -                <button onClick={this.onReadClick}>Read Docs</button> -                <button onClick={this.onDeleteClick}>Delete Docs</button> -            </div> -        ); -    }  } - -DocServer.init(window.location.protocol, window.location.hostname, 4321, "test"); -ReactDOM.render( -    <Test />, -    document.getElementById('root') -);
\ No newline at end of file diff --git a/src/new_fields/Doc.ts b/src/new_fields/Doc.ts index fef3b8cc5..f95df0ccb 100644 --- a/src/new_fields/Doc.ts +++ b/src/new_fields/Doc.ts @@ -13,6 +13,7 @@ import { listSpec } from "./Schema";  import { ComputedField } from "./ScriptField";  import { BoolCast, Cast, FieldValue, NumCast, PromiseValue, StrCast, ToConstructor } from "./Types";  import { deleteProperty, getField, getter, makeEditable, makeReadOnly, setter, updateFunction } from "./util"; +import { intersectRect } from "../Utils";  export namespace Field {      export function toKeyValueString(doc: Doc, key: string): string { @@ -614,6 +615,18 @@ export namespace Doc {          }), 0);      } +    export function overlapping(doc: Doc, doc2: Doc, clusterDistance: number) { +        var x2 = NumCast(doc2.x) - clusterDistance; +        var y2 = NumCast(doc2.y) - clusterDistance; +        var w2 = NumCast(doc2.width) + clusterDistance; +        var h2 = NumCast(doc2.height) + clusterDistance; +        var x = NumCast(doc.x) - clusterDistance; +        var y = NumCast(doc.y) - clusterDistance; +        var w = NumCast(doc.width) + clusterDistance; +        var h = NumCast(doc.height) + clusterDistance; +        return doc.z === doc2.z && intersectRect({ left: x, top: y, width: w, height: h }, { left: x2, top: y2, width: w2, height: h2 }); +    } +      export function isBrushedHighlightedDegree(doc: Doc) {          if (Doc.IsHighlighted(doc)) {              return 3; | 
