diff options
| author | andrewdkim <adkim414@gmail.com> | 2019-06-20 11:09:03 -0400 | 
|---|---|---|
| committer | andrewdkim <adkim414@gmail.com> | 2019-06-20 11:09:03 -0400 | 
| commit | b856d4c0be0b08bf154d552226457b82d31cf81c (patch) | |
| tree | fbcf2ac0ebe6ea358bada7efb5689f92bf3d2bc9 /src/client/views/nodes/PDFBox.tsx | |
| parent | da2157616ca4e614dacbf26fb5fa6758b51a209e (diff) | |
| parent | a5478b2d4cc3b66c6b58471cbb05c623d0109724 (diff) | |
merge with master
Diffstat (limited to 'src/client/views/nodes/PDFBox.tsx')
| -rw-r--r-- | src/client/views/nodes/PDFBox.tsx | 426 | 
1 files changed, 69 insertions, 357 deletions
| diff --git a/src/client/views/nodes/PDFBox.tsx b/src/client/views/nodes/PDFBox.tsx index 5cf3d8607..d2de1cb1c 100644 --- a/src/client/views/nodes/PDFBox.tsx +++ b/src/client/views/nodes/PDFBox.tsx @@ -1,52 +1,22 @@ -import * as htmlToImage from "html-to-image"; -import { action, computed, IReactionDisposer, observable, reaction, runInAction, trace } from 'mobx'; +import { action, IReactionDisposer, observable, reaction, trace, untracked } from 'mobx';  import { observer } from "mobx-react";  import 'react-image-lightbox/style.css'; -import Measure from "react-measure"; -//@ts-ignore -import { Document, Page } from "react-pdf"; -import 'react-pdf/dist/Page/AnnotationLayer.css'; -import { Id } from "../../../new_fields/FieldSymbols"; +import { WidthSym } from "../../../new_fields/Doc";  import { makeInterface } from "../../../new_fields/Schema"; -import { Cast, FieldValue, NumCast } from "../../../new_fields/Types"; -import { ImageField, PdfField } from "../../../new_fields/URLField"; +import { Cast, NumCast } from "../../../new_fields/Types"; +import { PdfField } from "../../../new_fields/URLField"; +//@ts-ignore +// import { Document, Page } from "react-pdf"; +// import 'react-pdf/dist/Page/AnnotationLayer.css';  import { RouteStore } from "../../../server/RouteStore"; -import { Utils } from '../../../Utils'; -import { DocServer } from "../../DocServer";  import { DocComponent } from "../DocComponent";  import { InkingControl } from "../InkingControl"; -import { SearchBox } from "../SearchBox"; -import { Annotation } from './Annotation'; +import { PDFViewer } from "../pdf/PDFViewer";  import { positionSchema } from "./DocumentView";  import { FieldView, FieldViewProps } from './FieldView';  import { pageSchema } from "./ImageBox";  import "./PDFBox.scss"; -var path = require('path');  import React = require("react"); -import { ContextMenu } from "../ContextMenu"; - -/** ALSO LOOK AT: Annotation.tsx, Sticky.tsx - * This method renders PDF and puts all kinds of functionalities such as annotation, highlighting,  - * area selection (I call it stickies), embedded ink node for directly annotating using a pen or  - * mouse, and pagination.  - * - *  - * HOW TO USE:  - * AREA selection:  - *          1) Click on Area button.  - *          2) click on any part of the PDF, and drag to get desired sized area shape - *          3) You can write on the area (hence the reason why it's called sticky) - *          4) to make another area, you need to click on area button AGAIN.  - *  - * HIGHLIGHT: (Buggy. No multiline/multidiv text highlighting for now...) - *          1) just click and drag on a text - *          2) click highlight - *          3) for annotation, just pull your cursor over to that text - *          4) another method: click on highlight first and then drag on your desired text - *          5) To make another highlight, you need to reclick on the button  - *  - * written by: Andrew Kim  - */  type PdfDocument = makeInterface<[typeof positionSchema, typeof pageSchema]>;  const PdfDocument = makeInterface(positionSchema, pageSchema); @@ -55,351 +25,93 @@ const PdfDocument = makeInterface(positionSchema, pageSchema);  export class PDFBox extends DocComponent<FieldViewProps, PdfDocument>(PdfDocument) {      public static LayoutString() { return FieldView.LayoutString(PDFBox); } -    private _mainDiv = React.createRef<HTMLDivElement>(); -    private renderHeight = 2400; - -    @observable private _renderAsSvg = true;      @observable private _alt = false; - +    @observable private _scrollY: number = 0;      private _reactionDisposer?: IReactionDisposer; -    @observable private _perPageInfo: Object[] = []; //stores pageInfo -    @observable private _pageInfo: any = { area: [], divs: [], anno: [] }; //divs is array of objects linked to anno - -    @observable private _currAnno: any = []; -    @observable private _interactive: boolean = false; - -    @computed private get curPage() { return NumCast(this.Document.curPage, 1); } -    @computed private get thumbnailPage() { return NumCast(this.props.Document.thumbnailPage, -1); } -      componentDidMount() { -        let wasSelected = this.props.active(); -        this._reactionDisposer = reaction( -            () => [this.props.active(), this.curPage], -            () => { -                setTimeout(action(() => {  // this forces the active() check to happen after all changes in a transaction have occurred. -                    if (this.curPage > 0 && !this.props.isTopMost && this.curPage !== this.thumbnailPage && wasSelected && !this.props.active()) { -                        this.saveThumbnail(); -                    } -                    wasSelected = this._interactive = this.props.active(); -                }), 0); -            }, -            { fireImmediately: true }); - +        if (this.props.setPdfBox) this.props.setPdfBox(this);      } -    componentWillUnmount() { -        if (this._reactionDisposer) this._reactionDisposer(); +    public GetPage() { +        return Math.floor(NumCast(this.props.Document.scrollY) / NumCast(this.Document.pdfHeight)) + 1;      } - -    /** -     * highlighting helper function -     */ -    makeEditableAndHighlight = (colour: string) => { -        var range, sel = window.getSelection(); -        if (sel && sel.rangeCount && sel.getRangeAt) { -            range = sel.getRangeAt(0); -        } -        document.designMode = "on"; -        if (!document.execCommand("HiliteColor", false, colour)) { -            document.execCommand("HiliteColor", false, colour); -        } - -        if (range && sel) { -            sel.removeAllRanges(); -            sel.addRange(range); - -            let obj: Object = { parentDivs: [], spans: [] }; -            //@ts-ignore -            if (range.commonAncestorContainer.className === 'react-pdf__Page__textContent') { //multiline highlighting case -                obj = this.highlightNodes(range.commonAncestorContainer.childNodes); -            } else { //single line highlighting case -                let parentDiv = range.commonAncestorContainer.parentElement; -                if (parentDiv) { -                    if (parentDiv.className === 'react-pdf__Page__textContent') { //when highlight is overwritten -                        obj = this.highlightNodes(parentDiv.childNodes); -                    } else { -                        parentDiv.childNodes.forEach((child) => { -                            if (child.nodeName === 'SPAN') { -                                //@ts-ignore -                                obj.parentDivs.push(parentDiv); -                                //@ts-ignore -                                child.id = "highlighted"; -                                //@ts-ignore -                                obj.spans.push(child); -                                // child.addEventListener("mouseover", this.onEnter); //adds mouseover annotation handler -                            } -                        }); -                    } -                } -            } -            this._pageInfo.divs.push(obj); - +    public BackPage() { +        let cp = Math.ceil(NumCast(this.props.Document.scrollY) / NumCast(this.Document.pdfHeight)) + 1; +        cp = cp - 1; +        if (cp > 0) { +            this.props.Document.curPage = cp; +            this.props.Document.scrollY = (cp - 1) * NumCast(this.Document.pdfHeight);          } -        document.designMode = "off";      } - -    highlightNodes = (nodes: NodeListOf<ChildNode>) => { -        let temp = { parentDivs: [], spans: [] }; -        nodes.forEach((div) => { -            div.childNodes.forEach((child) => { -                if (child.nodeName === 'SPAN') { -                    //@ts-ignore -                    temp.parentDivs.push(div); -                    //@ts-ignore -                    child.id = "highlighted"; -                    //@ts-ignore -                    temp.spans.push(child); -                    // child.addEventListener("mouseover", this.onEnter); //adds mouseover annotation handler -                } -            }); - -        }); -        return temp; -    } - -    /** -     * when the cursor enters the highlight, it pops out annotation. ONLY WORKS FOR SINGLE DIV LINES -     */ -    @action -    onEnter = (e: any) => { -        let span: HTMLSpanElement = e.toElement; -        let index: any; -        this._pageInfo.divs.forEach((obj: any) => { -            obj.spans.forEach((element: any) => { -                if (element === span && !index) { -                    index = this._pageInfo.divs.indexOf(obj); -                } -            }); -        }); - -        if (this._pageInfo.anno.length >= index + 1) { -            if (this._currAnno.length === 0) { -                this._currAnno.push(this._pageInfo.anno[index]); -            } -        } else { -            if (this._currAnno.length === 0) { //if there are no current annotation -                let div = span.offsetParent; -                //@ts-ignore -                let divX = div.style.left; -                //@ts-ignore -                let divY = div.style.top; -                //slicing "px" from the end -                divX = divX.slice(0, divX.length - 2); //gets X of the DIV element (parent of Span) -                divY = divY.slice(0, divY.length - 2); //gets Y of the DIV element (parent of Span) -                let annotation = <Annotation key={Utils.GenerateGuid()} Span={span} X={divX} Y={divY - 300} Highlights={this._pageInfo.divs} Annotations={this._pageInfo.anno} CurrAnno={this._currAnno} />; -                this._pageInfo.anno.push(annotation); -                this._currAnno.push(annotation); -            } +    public GotoPage(p: number) { +        if (p > 0 && p <= NumCast(this.props.Document.numPages)) { +            this.props.Document.curPage = p; +            this.props.Document.scrollY = (p - 1) * NumCast(this.Document.pdfHeight);          } -      } -    /** -     * highlight function for highlighting actual text. This works fine.  -     */ -    highlight = (color: string) => { -        if (window.getSelection()) { -            try { -                if (!document.execCommand("hiliteColor", false, color)) { -                    this.makeEditableAndHighlight(color); -                } -            } catch (ex) { -                this.makeEditableAndHighlight(color); -            } +    public ForwardPage() { +        let cp = this.GetPage() + 1; +        if (cp <= NumCast(this.props.Document.numPages)) { +            this.props.Document.curPage = cp; +            this.props.Document.scrollY = (cp - 1) * NumCast(this.Document.pdfHeight);          }      } -    /** -     * controls the area highlighting (stickies) Kinda temporary -     */ -    onPointerDown = (e: React.PointerEvent) => { -        if (this.props.isSelected() && !InkingControl.Instance.selectedTool && e.buttons === 1) { -            if (e.altKey) { -                this._alt = true; -            } else { -                if (e.metaKey) { -                    e.stopPropagation(); -                } -            } -            document.removeEventListener("pointerup", this.onPointerUp); -            document.addEventListener("pointerup", this.onPointerUp); -        } -        if (this.props.isSelected() && e.buttons === 2) { -            runInAction(() => this._alt = true); -            document.removeEventListener("pointerup", this.onPointerUp); -            document.addEventListener("pointerup", this.onPointerUp); -        } -    } - -    /** -     * controls area highlighting and partially highlighting. Kinda temporary -     */ -    @action -    onPointerUp = (e: PointerEvent) => { -        this._alt = false; -        document.removeEventListener("pointerup", this.onPointerUp); -        if (this.props.isSelected()) { -            this.highlight("rgba(76, 175, 80, 0.3)"); //highlights to this default color.  -        } -        this._interactive = true; -    } - - -    @action -    saveThumbnail = () => { -        this.props.Document.thumbnailPage = FieldValue(this.Document.curPage, -1); -        this._renderAsSvg = false; -        setTimeout(() => { -            runInAction(() => this._smallRetryCount = this._mediumRetryCount = this._largeRetryCount = 0); -            let nwidth = FieldValue(this.Document.nativeWidth, 0); -            let nheight = FieldValue(this.Document.nativeHeight, 0); -            htmlToImage.toPng(this._mainDiv.current!, { width: nwidth, height: nheight, quality: 0.8 }) -                .then(action((dataUrl: string) => { -                    SearchBox.convertDataUri(dataUrl, "icon" + this.Document[Id] + "_" + this.curPage).then((returnedFilename) => { -                        if (returnedFilename) { -                            let url = DocServer.prepend(returnedFilename); -                            this.props.Document.thumbnail = new ImageField(new URL(url)); -                        } -                        runInAction(() => this._renderAsSvg = true); -                    }); -                })) -                .catch(function (error: any) { -                    console.error('oops, something went wrong!', error); -                }); -        }, 1250); +    createRef = (ele: HTMLDivElement | null) => { +        if (this._reactionDisposer) this._reactionDisposer(); +        this._reactionDisposer = reaction(() => this.props.Document.scrollY, () => { +            ele && ele.scrollTo({ top: NumCast(this.Document.scrollY), behavior: "auto" }); +        });      } -    @action -    onLoaded = (page: any) => { -        // bcz: the number of pages should really be set when the document is imported. -        this.props.Document.numPages = page._transport.numPages; -        if (this._perPageInfo.length === 0) { //Makes sure it only runs once -            this._perPageInfo = [...Array(page._transport.numPages)]; +    loaded = (nw: number, nh: number, np: number) => { +        if (this.props.Document) { +            let doc = this.props.Document.proto ? this.props.Document.proto : this.props.Document; +            doc.numPages = np; +            if (doc.nativeWidth && doc.nativeHeight) return; +            let oldaspect = NumCast(doc.nativeHeight) / NumCast(doc.nativeWidth, 1); +            doc.nativeWidth = nw; +            if (doc.nativeHeight) doc.nativeHeight = nw * oldaspect; +            else doc.nativeHeight = nh; +            let ccv = this.props.ContainingCollectionView; +            if (ccv) { +                ccv.props.Document.pdfHeight = nh; +            } +            doc.height = nh * (doc[WidthSym]() / nw);          }      }      @action -    setScaling = (r: any) => { -        // bcz: the nativeHeight should really be set when the document is imported. -        //      also, the native dimensions could be different for different pages of the canvas -        //      so this design is flawed. -        var nativeWidth = FieldValue(this.Document.nativeWidth, 0); -        if (!FieldValue(this.Document.nativeHeight, 0)) { -            var nativeHeight = nativeWidth * r.offset.height / r.offset.width; -            this.props.Document.height = nativeHeight / nativeWidth * FieldValue(this.Document.width, 0); -            this.props.Document.nativeHeight = nativeHeight; -        } -    } -    @computed -    get pdfPage() { -        return <Page height={this.renderHeight} renderTextLayer={false} pageNumber={this.curPage} onLoadSuccess={this.onLoaded} />; -    } -    @computed -    get pdfContent() { -        let pdfUrl = Cast(this.props.Document[this.props.fieldKey], PdfField); -        if (!pdfUrl) { -            return <p>No pdf url to render</p>; -        } -        let pdfpage = this.pdfPage; -        let body = this.Document.nativeHeight ? -            pdfpage : -            <Measure offset onResize={this.setScaling}> -                {({ measureRef }) => -                    <div className="pdfBox-page" ref={measureRef}> -                        {pdfpage} -                    </div> -                } -            </Measure>; -        let xf = (this.Document.nativeHeight || 0) / this.renderHeight; -        return <div className="pdfBox-contentContainer" key="container" style={{ transform: `scale(${xf}, ${xf})` }}> -            <Document file={window.origin + RouteStore.corsProxy + `/${pdfUrl.url}`} renderMode={this._renderAsSvg || this.props.isTopMost ? "svg" : "canvas"}> -                {body} -            </Document> -        </div >; -    } - -    @computed -    get pdfRenderer() { -        let pdfUrl = Cast(this.props.Document[this.props.fieldKey], PdfField); -        let proxy = this.imageProxyRenderer; -        if ((!this._interactive && proxy && (!this.props.ContainingCollectionView || !this.props.ContainingCollectionView.props.isTopMost)) || !pdfUrl) { -            return proxy; -        } -        return [ -            proxy, -            this._pageInfo.area.filter(() => this._pageInfo.area).map((element: any) => element), -            this._currAnno.map((element: any) => element), -            this.pdfContent -        ]; -    } - -    choosePath(url: URL) { -        if (url.protocol === "data" || url.href.indexOf(window.location.origin) === -1) { -            return url.href; -        } -        let ext = path.extname(url.href); -        return url.href.replace(ext, this._curSuffix + ext); -    } -    @observable _smallRetryCount = 1; -    @observable _mediumRetryCount = 1; -    @observable _largeRetryCount = 1; -    @action retryPath = () => { -        if (this._curSuffix === "_s") this._smallRetryCount++; -        if (this._curSuffix === "_m") this._mediumRetryCount++; -        if (this._curSuffix === "_l") this._largeRetryCount++; -    } -    @action onError = () => { -        let timeout = this._curSuffix === "_s" ? this._smallRetryCount : this._curSuffix === "_m" ? this._mediumRetryCount : this._largeRetryCount; -        if (timeout < 10) { -            setTimeout(this.retryPath, Math.min(10000, timeout * 5)); +    onScroll = (e: React.UIEvent<HTMLDivElement>) => { +        if (e.currentTarget) { +            this._scrollY = e.currentTarget.scrollTop; +            let ccv = this.props.ContainingCollectionView; +            if (ccv) { +                ccv.props.Document.scrollY = this._scrollY; +            }          }      } -    _curSuffix = "_m"; - -    @computed -    get imageProxyRenderer() { -        let thumbField = this.props.Document.thumbnail; -        if (thumbField && this._renderAsSvg && NumCast(this.props.Document.thumbnailPage, 0) === this.Document.curPage) { -            // let transform = this.props.ScreenToLocalTransform().inverse(); -            let pw = typeof this.props.PanelWidth === "function" ? this.props.PanelWidth() : typeof this.props.PanelWidth === "number" ? (this.props.PanelWidth as any) as number : 50; -            // var [sptX, sptY] = transform.transformPoint(0, 0); -            // let [bptX, bptY] = transform.transformPoint(pw, this.props.PanelHeight()); -            // let w = bptX - sptX; - -            let path = thumbField instanceof ImageField ? thumbField.url.href : "http://cs.brown.edu/people/bcz/prairie.jpg"; -            // this._curSuffix = ""; -            // if (w > 20) { -            let field = thumbField; -            // if (w < 100 && this._smallRetryCount < 10) this._curSuffix = "_s"; -            // else if (w < 400 && this._mediumRetryCount < 10) this._curSuffix = "_m"; -            // else if (this._largeRetryCount < 10) this._curSuffix = "_l"; -            if (field instanceof ImageField) path = this.choosePath(field.url); -            // } -            return <img className="pdfBox-thumbnail" key={path + (this._mediumRetryCount).toString()} src={path} onError={this.onError} />; -        } -        return (null); -    } -    @action onKeyDown = (e: React.KeyboardEvent) => e.key === "Alt" && (this._alt = true); -    @action onKeyUp = (e: React.KeyboardEvent) => e.key === "Alt" && (this._alt = false); -    onContextMenu = (e: React.MouseEvent): void => { -        let field = Cast(this.Document[this.props.fieldKey], PdfField); -        if (field) { -            let url = field.url.href; -            ContextMenu.Instance.addItem({ -                description: "Copy path", event: () => { -                    Utils.CopyText(url); -                }, icon: "expand-arrows-alt" -            }); -        } -    }      render() { -        let classname = "pdfBox-cont" + (this.props.isSelected() && !InkingControl.Instance.selectedTool && !this._alt ? "-interactive" : ""); +        // uses mozilla pdf as default +        const pdfUrl = Cast(this.props.Document.data, PdfField, new PdfField(window.origin + RouteStore.corsProxy + "/https://mozilla.github.io/pdf.js/web/compressed.tracemonkey-pldi-09.pdf")); +        let classname = "pdfBox-cont" + (this.props.active() && !InkingControl.Instance.selectedTool && !this._alt ? "-interactive" : "");          return ( -            <div className={classname} tabIndex={0} ref={this._mainDiv} onPointerDown={this.onPointerDown} onKeyDown={this.onKeyDown} onKeyUp={this.onKeyUp} onContextMenu={this.onContextMenu} > -                {this.pdfRenderer} -            </div > +            <div onScroll={this.onScroll} +                style={{ +                    height: "100%", +                    overflowY: "scroll", overflowX: "hidden", +                    marginTop: `${NumCast(this.props.ContainingCollectionView!.props.Document.panY)}px` +                }} +                ref={this.createRef} +                onWheel={(e: React.WheelEvent) => { +                    e.stopPropagation(); +                }} className={classname}> +                <PDFViewer url={pdfUrl.url.pathname} loaded={this.loaded} scrollY={this._scrollY} parent={this} /> +                {/* <div style={{ width: "100px", height: "300px" }}></div> */} +            </div>          );      } | 
