Merge pull request #8724 from Snuffleupagus/PDFSinglePageViewer
Implement a `PDFSinglePageViewer` class (issue 8188)
This commit is contained in:
		
						commit
						f3987bba23
					
				| @ -17,7 +17,7 @@ | |||||||
| import { | import { | ||||||
|   animationStarted, DEFAULT_SCALE_VALUE, getPDFFileNameFromURL, isValidRotation, |   animationStarted, DEFAULT_SCALE_VALUE, getPDFFileNameFromURL, isValidRotation, | ||||||
|   MAX_SCALE, MIN_SCALE, noContextMenuHandler, normalizeWheelEventDelta, |   MAX_SCALE, MIN_SCALE, noContextMenuHandler, normalizeWheelEventDelta, | ||||||
|   parseQueryString, ProgressBar, RendererType |   parseQueryString, PresentationModeState, ProgressBar, RendererType | ||||||
| } from './ui_utils'; | } from './ui_utils'; | ||||||
| import { | import { | ||||||
|   build, createBlob, getDocument, getFilenameFromUrl, InvalidPDFException, |   build, createBlob, getDocument, getFilenameFromUrl, InvalidPDFException, | ||||||
| @ -27,7 +27,6 @@ import { | |||||||
| import { CursorTool, PDFCursorTools } from './pdf_cursor_tools'; | import { CursorTool, PDFCursorTools } from './pdf_cursor_tools'; | ||||||
| import { PDFRenderingQueue, RenderingStates } from './pdf_rendering_queue'; | import { PDFRenderingQueue, RenderingStates } from './pdf_rendering_queue'; | ||||||
| import { PDFSidebar, SidebarView } from './pdf_sidebar'; | import { PDFSidebar, SidebarView } from './pdf_sidebar'; | ||||||
| import { PDFViewer, PresentationModeState } from './pdf_viewer'; |  | ||||||
| import { getGlobalEventBus } from './dom_events'; | import { getGlobalEventBus } from './dom_events'; | ||||||
| import { OverlayManager } from './overlay_manager'; | import { OverlayManager } from './overlay_manager'; | ||||||
| import { PasswordPrompt } from './password_prompt'; | import { PasswordPrompt } from './password_prompt'; | ||||||
| @ -40,6 +39,7 @@ import { PDFLinkService } from './pdf_link_service'; | |||||||
| import { PDFOutlineViewer } from './pdf_outline_viewer'; | import { PDFOutlineViewer } from './pdf_outline_viewer'; | ||||||
| import { PDFPresentationMode } from './pdf_presentation_mode'; | import { PDFPresentationMode } from './pdf_presentation_mode'; | ||||||
| import { PDFThumbnailViewer } from './pdf_thumbnail_viewer'; | import { PDFThumbnailViewer } from './pdf_thumbnail_viewer'; | ||||||
|  | import { PDFViewer } from './pdf_viewer'; | ||||||
| import { SecondaryToolbar } from './secondary_toolbar'; | import { SecondaryToolbar } from './secondary_toolbar'; | ||||||
| import { Toolbar } from './toolbar'; | import { Toolbar } from './toolbar'; | ||||||
| import { ViewHistory } from './view_history'; | import { ViewHistory } from './view_history'; | ||||||
|  | |||||||
							
								
								
									
										948
									
								
								web/base_viewer.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										948
									
								
								web/base_viewer.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,948 @@ | |||||||
|  | /* Copyright 2014 Mozilla Foundation | ||||||
|  |  * | ||||||
|  |  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
|  |  * you may not use this file except in compliance with the License. | ||||||
|  |  * You may obtain a copy of the License at | ||||||
|  |  * | ||||||
|  |  *     http://www.apache.org/licenses/LICENSE-2.0
 | ||||||
|  |  * | ||||||
|  |  * Unless required by applicable law or agreed to in writing, software | ||||||
|  |  * distributed under the License is distributed on an "AS IS" BASIS, | ||||||
|  |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||||
|  |  * See the License for the specific language governing permissions and | ||||||
|  |  * limitations under the License. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | import { createPromiseCapability, PDFJS } from 'pdfjs-lib'; | ||||||
|  | import { | ||||||
|  |   CSS_UNITS, DEFAULT_SCALE, DEFAULT_SCALE_VALUE, isValidRotation, | ||||||
|  |   MAX_AUTO_SCALE, NullL10n, PresentationModeState, RendererType, | ||||||
|  |   SCROLLBAR_PADDING, UNKNOWN_SCALE, VERTICAL_PADDING, watchScroll | ||||||
|  | } from './ui_utils'; | ||||||
|  | import { PDFRenderingQueue, RenderingStates } from './pdf_rendering_queue'; | ||||||
|  | import { AnnotationLayerBuilder } from './annotation_layer_builder'; | ||||||
|  | import { getGlobalEventBus } from './dom_events'; | ||||||
|  | import { PDFPageView } from './pdf_page_view'; | ||||||
|  | import { SimpleLinkService } from './pdf_link_service'; | ||||||
|  | import { TextLayerBuilder } from './text_layer_builder'; | ||||||
|  | 
 | ||||||
|  | const DEFAULT_CACHE_SIZE = 10; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * @typedef {Object} PDFViewerOptions | ||||||
|  |  * @property {HTMLDivElement} container - The container for the viewer element. | ||||||
|  |  * @property {HTMLDivElement} viewer - (optional) The viewer element. | ||||||
|  |  * @property {EventBus} eventBus - The application event bus. | ||||||
|  |  * @property {IPDFLinkService} linkService - The navigation/linking service. | ||||||
|  |  * @property {DownloadManager} downloadManager - (optional) The download | ||||||
|  |  *   manager component. | ||||||
|  |  * @property {PDFRenderingQueue} renderingQueue - (optional) The rendering | ||||||
|  |  *   queue object. | ||||||
|  |  * @property {boolean} removePageBorders - (optional) Removes the border shadow | ||||||
|  |  *   around the pages. The default is false. | ||||||
|  |  * @property {boolean} enhanceTextSelection - (optional) Enables the improved | ||||||
|  |  *   text selection behaviour. The default is `false`. | ||||||
|  |  * @property {boolean} renderInteractiveForms - (optional) Enables rendering of | ||||||
|  |  *   interactive form elements. The default is `false`. | ||||||
|  |  * @property {boolean} enablePrintAutoRotate - (optional) Enables automatic | ||||||
|  |  *   rotation of pages whose orientation differ from the first page upon | ||||||
|  |  *   printing. The default is `false`. | ||||||
|  |  * @property {string} renderer - 'canvas' or 'svg'. The default is 'canvas'. | ||||||
|  |  * @property {IL10n} l10n - Localization service. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | function PDFPageViewBuffer(size) { | ||||||
|  |   let data = []; | ||||||
|  |   this.push = function(view) { | ||||||
|  |     let i = data.indexOf(view); | ||||||
|  |     if (i >= 0) { | ||||||
|  |       data.splice(i, 1); | ||||||
|  |     } | ||||||
|  |     data.push(view); | ||||||
|  |     if (data.length > size) { | ||||||
|  |       data.shift().destroy(); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |   this.resize = function(newSize) { | ||||||
|  |     size = newSize; | ||||||
|  |     while (data.length > size) { | ||||||
|  |       data.shift().destroy(); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function isSameScale(oldScale, newScale) { | ||||||
|  |   if (newScale === oldScale) { | ||||||
|  |     return true; | ||||||
|  |   } | ||||||
|  |   if (Math.abs(newScale - oldScale) < 1e-15) { | ||||||
|  |     // Prevent unnecessary re-rendering of all pages when the scale
 | ||||||
|  |     // changes only because of limited numerical precision.
 | ||||||
|  |     return true; | ||||||
|  |   } | ||||||
|  |   return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function isPortraitOrientation(size) { | ||||||
|  |   return size.width <= size.height; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Simple viewer control to display PDF content/pages. | ||||||
|  |  * @implements {IRenderableView} | ||||||
|  |  */ | ||||||
|  | class BaseViewer { | ||||||
|  |   /** | ||||||
|  |    * @param {PDFViewerOptions} options | ||||||
|  |    */ | ||||||
|  |   constructor(options) { | ||||||
|  |     if (this.constructor === BaseViewer) { | ||||||
|  |       throw new Error('Cannot initialize BaseViewer.'); | ||||||
|  |     } | ||||||
|  |     this._name = this.constructor.name; | ||||||
|  | 
 | ||||||
|  |     this.container = options.container; | ||||||
|  |     this.viewer = options.viewer || options.container.firstElementChild; | ||||||
|  |     this.eventBus = options.eventBus || getGlobalEventBus(); | ||||||
|  |     this.linkService = options.linkService || new SimpleLinkService(); | ||||||
|  |     this.downloadManager = options.downloadManager || null; | ||||||
|  |     this.removePageBorders = options.removePageBorders || false; | ||||||
|  |     this.enhanceTextSelection = options.enhanceTextSelection || false; | ||||||
|  |     this.renderInteractiveForms = options.renderInteractiveForms || false; | ||||||
|  |     this.enablePrintAutoRotate = options.enablePrintAutoRotate || false; | ||||||
|  |     this.renderer = options.renderer || RendererType.CANVAS; | ||||||
|  |     this.l10n = options.l10n || NullL10n; | ||||||
|  | 
 | ||||||
|  |     this.defaultRenderingQueue = !options.renderingQueue; | ||||||
|  |     if (this.defaultRenderingQueue) { | ||||||
|  |       // Custom rendering queue is not specified, using default one
 | ||||||
|  |       this.renderingQueue = new PDFRenderingQueue(); | ||||||
|  |       this.renderingQueue.setViewer(this); | ||||||
|  |     } else { | ||||||
|  |       this.renderingQueue = options.renderingQueue; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     this.scroll = watchScroll(this.container, this._scrollUpdate.bind(this)); | ||||||
|  |     this.presentationModeState = PresentationModeState.UNKNOWN; | ||||||
|  |     this._resetView(); | ||||||
|  | 
 | ||||||
|  |     if (this.removePageBorders) { | ||||||
|  |       this.viewer.classList.add('removePageBorders'); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get pagesCount() { | ||||||
|  |     return this._pages.length; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   getPageView(index) { | ||||||
|  |     return this._pages[index]; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @returns {boolean} true if all {PDFPageView} objects are initialized. | ||||||
|  |    */ | ||||||
|  |   get pageViewsReady() { | ||||||
|  |     return this._pageViewsReady; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @returns {number} | ||||||
|  |    */ | ||||||
|  |   get currentPageNumber() { | ||||||
|  |     return this._currentPageNumber; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @param {number} val - The page number. | ||||||
|  |    */ | ||||||
|  |   set currentPageNumber(val) { | ||||||
|  |     if (!Number.isInteger(val)) { | ||||||
|  |       throw new Error('Invalid page number.'); | ||||||
|  |     } | ||||||
|  |     if (!this.pdfDocument) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     // The intent can be to just reset a scroll position and/or scale.
 | ||||||
|  |     this._setCurrentPageNumber(val, /* resetCurrentPageView = */ true); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @private | ||||||
|  |    */ | ||||||
|  |   _setCurrentPageNumber(val, resetCurrentPageView = false) { | ||||||
|  |     if (this._currentPageNumber === val) { | ||||||
|  |       if (resetCurrentPageView) { | ||||||
|  |         this._resetCurrentPageView(); | ||||||
|  |       } | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (!(0 < val && val <= this.pagesCount)) { | ||||||
|  |       console.error( | ||||||
|  |         `${this._name}._setCurrentPageNumber: "${val}" is out of bounds.`); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     let arg = { | ||||||
|  |       source: this, | ||||||
|  |       pageNumber: val, | ||||||
|  |       pageLabel: this._pageLabels && this._pageLabels[val - 1], | ||||||
|  |     }; | ||||||
|  |     this._currentPageNumber = val; | ||||||
|  |     this.eventBus.dispatch('pagechanging', arg); | ||||||
|  |     this.eventBus.dispatch('pagechange', arg); | ||||||
|  | 
 | ||||||
|  |     if (resetCurrentPageView) { | ||||||
|  |       this._resetCurrentPageView(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @returns {string|null} Returns the current page label, | ||||||
|  |    *                        or `null` if no page labels exist. | ||||||
|  |    */ | ||||||
|  |   get currentPageLabel() { | ||||||
|  |     return this._pageLabels && this._pageLabels[this._currentPageNumber - 1]; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @param {string} val - The page label. | ||||||
|  |    */ | ||||||
|  |   set currentPageLabel(val) { | ||||||
|  |     let pageNumber = val | 0; // Fallback page number.
 | ||||||
|  |     if (this._pageLabels) { | ||||||
|  |       let i = this._pageLabels.indexOf(val); | ||||||
|  |       if (i >= 0) { | ||||||
|  |         pageNumber = i + 1; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     this.currentPageNumber = pageNumber; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @returns {number} | ||||||
|  |    */ | ||||||
|  |   get currentScale() { | ||||||
|  |     return this._currentScale !== UNKNOWN_SCALE ? this._currentScale : | ||||||
|  |                                                   DEFAULT_SCALE; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @param {number} val - Scale of the pages in percents. | ||||||
|  |    */ | ||||||
|  |   set currentScale(val) { | ||||||
|  |     if (isNaN(val)) { | ||||||
|  |       throw new Error('Invalid numeric scale'); | ||||||
|  |     } | ||||||
|  |     if (!this.pdfDocument) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     this._setScale(val, false); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @returns {string} | ||||||
|  |    */ | ||||||
|  |   get currentScaleValue() { | ||||||
|  |     return this._currentScaleValue; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @param val - The scale of the pages (in percent or predefined value). | ||||||
|  |    */ | ||||||
|  |   set currentScaleValue(val) { | ||||||
|  |     if (!this.pdfDocument) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     this._setScale(val, false); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @returns {number} | ||||||
|  |    */ | ||||||
|  |   get pagesRotation() { | ||||||
|  |     return this._pagesRotation; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @param {number} rotation - The rotation of the pages (0, 90, 180, 270). | ||||||
|  |    */ | ||||||
|  |   set pagesRotation(rotation) { | ||||||
|  |     if (!isValidRotation(rotation)) { | ||||||
|  |       throw new Error('Invalid pages rotation angle.'); | ||||||
|  |     } | ||||||
|  |     if (!this.pdfDocument) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     if (this._pagesRotation === rotation) { | ||||||
|  |       return; // The rotation didn't change.
 | ||||||
|  |     } | ||||||
|  |     this._pagesRotation = rotation; | ||||||
|  | 
 | ||||||
|  |     let pageNumber = this._currentPageNumber; | ||||||
|  | 
 | ||||||
|  |     for (let i = 0, ii = this._pages.length; i < ii; i++) { | ||||||
|  |       let pageView = this._pages[i]; | ||||||
|  |       pageView.update(pageView.scale, rotation); | ||||||
|  |     } | ||||||
|  |     // Prevent errors in case the rotation changes *before* the scale has been
 | ||||||
|  |     // set to a non-default value.
 | ||||||
|  |     if (this._currentScaleValue) { | ||||||
|  |       this._setScale(this._currentScaleValue, true); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     this.eventBus.dispatch('rotationchanging', { | ||||||
|  |       source: this, | ||||||
|  |       pagesRotation: rotation, | ||||||
|  |       pageNumber, | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     if (this.defaultRenderingQueue) { | ||||||
|  |       this.update(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get _setDocumentViewerElement() { | ||||||
|  |     throw new Error('Not implemented: _setDocumentViewerElement'); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @param pdfDocument {PDFDocument} | ||||||
|  |    */ | ||||||
|  |   setDocument(pdfDocument) { | ||||||
|  |     if (this.pdfDocument) { | ||||||
|  |       this._cancelRendering(); | ||||||
|  |       this._resetView(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     this.pdfDocument = pdfDocument; | ||||||
|  |     if (!pdfDocument) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     let pagesCount = pdfDocument.numPages; | ||||||
|  | 
 | ||||||
|  |     let pagesCapability = createPromiseCapability(); | ||||||
|  |     this.pagesPromise = pagesCapability.promise; | ||||||
|  | 
 | ||||||
|  |     pagesCapability.promise.then(() => { | ||||||
|  |       this._pageViewsReady = true; | ||||||
|  |       this.eventBus.dispatch('pagesloaded', { | ||||||
|  |         source: this, | ||||||
|  |         pagesCount, | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     let isOnePageRenderedResolved = false; | ||||||
|  |     let onePageRenderedCapability = createPromiseCapability(); | ||||||
|  |     this.onePageRendered = onePageRenderedCapability.promise; | ||||||
|  | 
 | ||||||
|  |     let bindOnAfterAndBeforeDraw = (pageView) => { | ||||||
|  |       pageView.onBeforeDraw = () => { | ||||||
|  |         // Add the page to the buffer at the start of drawing. That way it can
 | ||||||
|  |         // be evicted from the buffer and destroyed even if we pause its
 | ||||||
|  |         // rendering.
 | ||||||
|  |         this._buffer.push(pageView); | ||||||
|  |       }; | ||||||
|  |       pageView.onAfterDraw = () => { | ||||||
|  |         if (!isOnePageRenderedResolved) { | ||||||
|  |           isOnePageRenderedResolved = true; | ||||||
|  |           onePageRenderedCapability.resolve(); | ||||||
|  |         } | ||||||
|  |       }; | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     let firstPagePromise = pdfDocument.getPage(1); | ||||||
|  |     this.firstPagePromise = firstPagePromise; | ||||||
|  | 
 | ||||||
|  |     // Fetch a single page so we can get a viewport that will be the default
 | ||||||
|  |     // viewport for all pages
 | ||||||
|  |     firstPagePromise.then((pdfPage) => { | ||||||
|  |       let scale = this.currentScale; | ||||||
|  |       let viewport = pdfPage.getViewport(scale * CSS_UNITS); | ||||||
|  |       for (let pageNum = 1; pageNum <= pagesCount; ++pageNum) { | ||||||
|  |         let textLayerFactory = null; | ||||||
|  |         if (!PDFJS.disableTextLayer) { | ||||||
|  |           textLayerFactory = this; | ||||||
|  |         } | ||||||
|  |         let pageView = new PDFPageView({ | ||||||
|  |           container: this._setDocumentViewerElement, | ||||||
|  |           eventBus: this.eventBus, | ||||||
|  |           id: pageNum, | ||||||
|  |           scale, | ||||||
|  |           defaultViewport: viewport.clone(), | ||||||
|  |           renderingQueue: this.renderingQueue, | ||||||
|  |           textLayerFactory, | ||||||
|  |           annotationLayerFactory: this, | ||||||
|  |           enhanceTextSelection: this.enhanceTextSelection, | ||||||
|  |           renderInteractiveForms: this.renderInteractiveForms, | ||||||
|  |           renderer: this.renderer, | ||||||
|  |           l10n: this.l10n, | ||||||
|  |         }); | ||||||
|  |         bindOnAfterAndBeforeDraw(pageView); | ||||||
|  |         this._pages.push(pageView); | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       // Fetch all the pages since the viewport is needed before printing
 | ||||||
|  |       // starts to create the correct size canvas. Wait until one page is
 | ||||||
|  |       // rendered so we don't tie up too many resources early on.
 | ||||||
|  |       onePageRenderedCapability.promise.then(() => { | ||||||
|  |         if (PDFJS.disableAutoFetch) { | ||||||
|  |           // XXX: Printing is semi-broken with auto fetch disabled.
 | ||||||
|  |           pagesCapability.resolve(); | ||||||
|  |           return; | ||||||
|  |         } | ||||||
|  |         let getPagesLeft = pagesCount; | ||||||
|  |         for (let pageNum = 1; pageNum <= pagesCount; ++pageNum) { | ||||||
|  |           pdfDocument.getPage(pageNum).then((pdfPage) => { | ||||||
|  |             let pageView = this._pages[pageNum - 1]; | ||||||
|  |             if (!pageView.pdfPage) { | ||||||
|  |               pageView.setPdfPage(pdfPage); | ||||||
|  |             } | ||||||
|  |             this.linkService.cachePageRef(pageNum, pdfPage.ref); | ||||||
|  |             if (--getPagesLeft === 0) { | ||||||
|  |               pagesCapability.resolve(); | ||||||
|  |             } | ||||||
|  |           }, (reason) => { | ||||||
|  |             console.error(`Unable to get page ${pageNum} to initialize viewer`, | ||||||
|  |                           reason); | ||||||
|  |             if (--getPagesLeft === 0) { | ||||||
|  |               pagesCapability.resolve(); | ||||||
|  |             } | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  | 
 | ||||||
|  |       this.eventBus.dispatch('pagesinit', { source: this, }); | ||||||
|  | 
 | ||||||
|  |       if (this.defaultRenderingQueue) { | ||||||
|  |         this.update(); | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       if (this.findController) { | ||||||
|  |         this.findController.resolveFirstPage(); | ||||||
|  |       } | ||||||
|  |     }).catch((reason) => { | ||||||
|  |       console.error('Unable to initialize viewer', reason); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @param {Array|null} labels | ||||||
|  |    */ | ||||||
|  |   setPageLabels(labels) { | ||||||
|  |     if (!this.pdfDocument) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     if (!labels) { | ||||||
|  |       this._pageLabels = null; | ||||||
|  |     } else if (!(labels instanceof Array && | ||||||
|  |                  this.pdfDocument.numPages === labels.length)) { | ||||||
|  |       this._pageLabels = null; | ||||||
|  |       console.error(`${this._name}.setPageLabels: Invalid page labels.`); | ||||||
|  |     } else { | ||||||
|  |       this._pageLabels = labels; | ||||||
|  |     } | ||||||
|  |     // Update all the `PDFPageView` instances.
 | ||||||
|  |     for (let i = 0, ii = this._pages.length; i < ii; i++) { | ||||||
|  |       let pageView = this._pages[i]; | ||||||
|  |       let label = this._pageLabels && this._pageLabels[i]; | ||||||
|  |       pageView.setPageLabel(label); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   _resetView() { | ||||||
|  |     this._pages = []; | ||||||
|  |     this._currentPageNumber = 1; | ||||||
|  |     this._currentScale = UNKNOWN_SCALE; | ||||||
|  |     this._currentScaleValue = null; | ||||||
|  |     this._pageLabels = null; | ||||||
|  |     this._buffer = new PDFPageViewBuffer(DEFAULT_CACHE_SIZE); | ||||||
|  |     this._location = null; | ||||||
|  |     this._pagesRotation = 0; | ||||||
|  |     this._pagesRequests = []; | ||||||
|  |     this._pageViewsReady = false; | ||||||
|  | 
 | ||||||
|  |     // Remove the pages from the DOM.
 | ||||||
|  |     this.viewer.textContent = ''; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   _scrollUpdate() { | ||||||
|  |     if (this.pagesCount === 0) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     this.update(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   _scrollIntoView({ pageDiv, pageSpot = null, pageNumber = null, }) { | ||||||
|  |     throw new Error('Not implemented: _scrollIntoView'); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   _setScaleDispatchEvent(newScale, newValue, preset = false) { | ||||||
|  |     let arg = { | ||||||
|  |       source: this, | ||||||
|  |       scale: newScale, | ||||||
|  |       presetValue: preset ? newValue : undefined, | ||||||
|  |     }; | ||||||
|  |     this.eventBus.dispatch('scalechanging', arg); | ||||||
|  |     this.eventBus.dispatch('scalechange', arg); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   _setScaleUpdatePages(newScale, newValue, noScroll = false, preset = false) { | ||||||
|  |     this._currentScaleValue = newValue.toString(); | ||||||
|  | 
 | ||||||
|  |     if (isSameScale(this._currentScale, newScale)) { | ||||||
|  |       if (preset) { | ||||||
|  |         this._setScaleDispatchEvent(newScale, newValue, true); | ||||||
|  |       } | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     for (let i = 0, ii = this._pages.length; i < ii; i++) { | ||||||
|  |       this._pages[i].update(newScale); | ||||||
|  |     } | ||||||
|  |     this._currentScale = newScale; | ||||||
|  | 
 | ||||||
|  |     if (!noScroll) { | ||||||
|  |       let page = this._currentPageNumber, dest; | ||||||
|  |       if (this._location && !PDFJS.ignoreCurrentPositionOnZoom && | ||||||
|  |           !(this.isInPresentationMode || this.isChangingPresentationMode)) { | ||||||
|  |         page = this._location.pageNumber; | ||||||
|  |         dest = [null, { name: 'XYZ', }, this._location.left, | ||||||
|  |                 this._location.top, null]; | ||||||
|  |       } | ||||||
|  |       this.scrollPageIntoView({ | ||||||
|  |         pageNumber: page, | ||||||
|  |         destArray: dest, | ||||||
|  |         allowNegativeOffset: true, | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     this._setScaleDispatchEvent(newScale, newValue, preset); | ||||||
|  | 
 | ||||||
|  |     if (this.defaultRenderingQueue) { | ||||||
|  |       this.update(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   _setScale(value, noScroll = false) { | ||||||
|  |     let scale = parseFloat(value); | ||||||
|  | 
 | ||||||
|  |     if (scale > 0) { | ||||||
|  |       this._setScaleUpdatePages(scale, value, noScroll, /* preset = */ false); | ||||||
|  |     } else { | ||||||
|  |       let currentPage = this._pages[this._currentPageNumber - 1]; | ||||||
|  |       if (!currentPage) { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |       let hPadding = (this.isInPresentationMode || this.removePageBorders) ? | ||||||
|  |         0 : SCROLLBAR_PADDING; | ||||||
|  |       let vPadding = (this.isInPresentationMode || this.removePageBorders) ? | ||||||
|  |         0 : VERTICAL_PADDING; | ||||||
|  |       let pageWidthScale = (this.container.clientWidth - hPadding) / | ||||||
|  |                            currentPage.width * currentPage.scale; | ||||||
|  |       let pageHeightScale = (this.container.clientHeight - vPadding) / | ||||||
|  |                             currentPage.height * currentPage.scale; | ||||||
|  |       switch (value) { | ||||||
|  |         case 'page-actual': | ||||||
|  |           scale = 1; | ||||||
|  |           break; | ||||||
|  |         case 'page-width': | ||||||
|  |           scale = pageWidthScale; | ||||||
|  |           break; | ||||||
|  |         case 'page-height': | ||||||
|  |           scale = pageHeightScale; | ||||||
|  |           break; | ||||||
|  |         case 'page-fit': | ||||||
|  |           scale = Math.min(pageWidthScale, pageHeightScale); | ||||||
|  |           break; | ||||||
|  |         case 'auto': | ||||||
|  |           let isLandscape = (currentPage.width > currentPage.height); | ||||||
|  |           // For pages in landscape mode, fit the page height to the viewer
 | ||||||
|  |           // *unless* the page would thus become too wide to fit horizontally.
 | ||||||
|  |           let horizontalScale = isLandscape ? | ||||||
|  |             Math.min(pageHeightScale, pageWidthScale) : pageWidthScale; | ||||||
|  |           scale = Math.min(MAX_AUTO_SCALE, horizontalScale); | ||||||
|  |           break; | ||||||
|  |         default: | ||||||
|  |           console.error( | ||||||
|  |             `${this._name}._setScale: "${value}" is an unknown zoom value.`); | ||||||
|  |           return; | ||||||
|  |       } | ||||||
|  |       this._setScaleUpdatePages(scale, value, noScroll, /* preset = */ true); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * Refreshes page view: scrolls to the current page and updates the scale. | ||||||
|  |    * @private | ||||||
|  |    */ | ||||||
|  |   _resetCurrentPageView() { | ||||||
|  |     if (this.isInPresentationMode) { | ||||||
|  |       // Fixes the case when PDF has different page sizes.
 | ||||||
|  |       this._setScale(this._currentScaleValue, true); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     let pageView = this._pages[this._currentPageNumber - 1]; | ||||||
|  |     this._scrollIntoView({ pageDiv: pageView.div, }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @typedef ScrollPageIntoViewParameters | ||||||
|  |    * @property {number} pageNumber - The page number. | ||||||
|  |    * @property {Array} destArray - (optional) The original PDF destination | ||||||
|  |    *   array, in the format: <page-ref> </XYZ|/FitXXX> <args..> | ||||||
|  |    * @property {boolean} allowNegativeOffset - (optional) Allow negative page | ||||||
|  |    *   offsets. The default value is `false`. | ||||||
|  |    */ | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * Scrolls page into view. | ||||||
|  |    * @param {ScrollPageIntoViewParameters} params | ||||||
|  |    */ | ||||||
|  |   scrollPageIntoView(params) { | ||||||
|  |     if ((typeof PDFJSDev === 'undefined' || PDFJSDev.test('GENERIC')) && | ||||||
|  |         (arguments.length > 1 || typeof params === 'number')) { | ||||||
|  |       console.error('Call of scrollPageIntoView() with obsolete signature.'); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     if (!this.pdfDocument) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     let pageNumber = params.pageNumber || 0; | ||||||
|  |     let dest = params.destArray || null; | ||||||
|  |     let allowNegativeOffset = params.allowNegativeOffset || false; | ||||||
|  | 
 | ||||||
|  |     if (this.isInPresentationMode || !dest) { | ||||||
|  |       this._setCurrentPageNumber(pageNumber, /* resetCurrentPageView = */ true); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     let pageView = this._pages[pageNumber - 1]; | ||||||
|  |     if (!pageView) { | ||||||
|  |       console.error( | ||||||
|  |         `${this._name}.scrollPageIntoView: Invalid "pageNumber" parameter.`); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     let x = 0, y = 0; | ||||||
|  |     let width = 0, height = 0, widthScale, heightScale; | ||||||
|  |     let changeOrientation = (pageView.rotation % 180 === 0 ? false : true); | ||||||
|  |     let pageWidth = (changeOrientation ? pageView.height : pageView.width) / | ||||||
|  |       pageView.scale / CSS_UNITS; | ||||||
|  |     let pageHeight = (changeOrientation ? pageView.width : pageView.height) / | ||||||
|  |       pageView.scale / CSS_UNITS; | ||||||
|  |     let scale = 0; | ||||||
|  |     switch (dest[1].name) { | ||||||
|  |       case 'XYZ': | ||||||
|  |         x = dest[2]; | ||||||
|  |         y = dest[3]; | ||||||
|  |         scale = dest[4]; | ||||||
|  |         // If x and/or y coordinates are not supplied, default to
 | ||||||
|  |         // _top_ left of the page (not the obvious bottom left,
 | ||||||
|  |         // since aligning the bottom of the intended page with the
 | ||||||
|  |         // top of the window is rarely helpful).
 | ||||||
|  |         x = x !== null ? x : 0; | ||||||
|  |         y = y !== null ? y : pageHeight; | ||||||
|  |         break; | ||||||
|  |       case 'Fit': | ||||||
|  |       case 'FitB': | ||||||
|  |         scale = 'page-fit'; | ||||||
|  |         break; | ||||||
|  |       case 'FitH': | ||||||
|  |       case 'FitBH': | ||||||
|  |         y = dest[2]; | ||||||
|  |         scale = 'page-width'; | ||||||
|  |         // According to the PDF spec, section 12.3.2.2, a `null` value in the
 | ||||||
|  |         // parameter should maintain the position relative to the new page.
 | ||||||
|  |         if (y === null && this._location) { | ||||||
|  |           x = this._location.left; | ||||||
|  |           y = this._location.top; | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |       case 'FitV': | ||||||
|  |       case 'FitBV': | ||||||
|  |         x = dest[2]; | ||||||
|  |         width = pageWidth; | ||||||
|  |         height = pageHeight; | ||||||
|  |         scale = 'page-height'; | ||||||
|  |         break; | ||||||
|  |       case 'FitR': | ||||||
|  |         x = dest[2]; | ||||||
|  |         y = dest[3]; | ||||||
|  |         width = dest[4] - x; | ||||||
|  |         height = dest[5] - y; | ||||||
|  |         let hPadding = this.removePageBorders ? 0 : SCROLLBAR_PADDING; | ||||||
|  |         let vPadding = this.removePageBorders ? 0 : VERTICAL_PADDING; | ||||||
|  | 
 | ||||||
|  |         widthScale = (this.container.clientWidth - hPadding) / | ||||||
|  |           width / CSS_UNITS; | ||||||
|  |         heightScale = (this.container.clientHeight - vPadding) / | ||||||
|  |           height / CSS_UNITS; | ||||||
|  |         scale = Math.min(Math.abs(widthScale), Math.abs(heightScale)); | ||||||
|  |         break; | ||||||
|  |       default: | ||||||
|  |         console.error(`${this._name}.scrollPageIntoView: "${dest[1].name}" ` + | ||||||
|  |                       'is not a valid destination type.'); | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (scale && scale !== this._currentScale) { | ||||||
|  |       this.currentScaleValue = scale; | ||||||
|  |     } else if (this._currentScale === UNKNOWN_SCALE) { | ||||||
|  |       this.currentScaleValue = DEFAULT_SCALE_VALUE; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (scale === 'page-fit' && !dest[4]) { | ||||||
|  |       this._scrollIntoView({ | ||||||
|  |         pageDiv: pageView.div, | ||||||
|  |         pageNumber, | ||||||
|  |       }); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     let boundingRect = [ | ||||||
|  |       pageView.viewport.convertToViewportPoint(x, y), | ||||||
|  |       pageView.viewport.convertToViewportPoint(x + width, y + height) | ||||||
|  |     ]; | ||||||
|  |     let left = Math.min(boundingRect[0][0], boundingRect[1][0]); | ||||||
|  |     let top = Math.min(boundingRect[0][1], boundingRect[1][1]); | ||||||
|  | 
 | ||||||
|  |     if (!allowNegativeOffset) { | ||||||
|  |       // Some bad PDF generators will create destinations with e.g. top values
 | ||||||
|  |       // that exceeds the page height. Ensure that offsets are not negative,
 | ||||||
|  |       // to prevent a previous page from becoming visible (fixes bug 874482).
 | ||||||
|  |       left = Math.max(left, 0); | ||||||
|  |       top = Math.max(top, 0); | ||||||
|  |     } | ||||||
|  |     this._scrollIntoView({ | ||||||
|  |       pageDiv: pageView.div, | ||||||
|  |       pageSpot: { left, top, }, | ||||||
|  |       pageNumber, | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   _resizeBuffer(numVisiblePages) { | ||||||
|  |     let suggestedCacheSize = Math.max(DEFAULT_CACHE_SIZE, | ||||||
|  |                                       2 * numVisiblePages + 1); | ||||||
|  |     this._buffer.resize(suggestedCacheSize); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   _updateLocation(firstPage) { | ||||||
|  |     let currentScale = this._currentScale; | ||||||
|  |     let currentScaleValue = this._currentScaleValue; | ||||||
|  |     let normalizedScaleValue = | ||||||
|  |       parseFloat(currentScaleValue) === currentScale ? | ||||||
|  |       Math.round(currentScale * 10000) / 100 : currentScaleValue; | ||||||
|  | 
 | ||||||
|  |     let pageNumber = firstPage.id; | ||||||
|  |     let pdfOpenParams = '#page=' + pageNumber; | ||||||
|  |     pdfOpenParams += '&zoom=' + normalizedScaleValue; | ||||||
|  |     let currentPageView = this._pages[pageNumber - 1]; | ||||||
|  |     let container = this.container; | ||||||
|  |     let topLeft = currentPageView.getPagePoint( | ||||||
|  |       (container.scrollLeft - firstPage.x), | ||||||
|  |       (container.scrollTop - firstPage.y)); | ||||||
|  |     let intLeft = Math.round(topLeft[0]); | ||||||
|  |     let intTop = Math.round(topLeft[1]); | ||||||
|  |     pdfOpenParams += ',' + intLeft + ',' + intTop; | ||||||
|  | 
 | ||||||
|  |     this._location = { | ||||||
|  |       pageNumber, | ||||||
|  |       scale: normalizedScaleValue, | ||||||
|  |       top: intTop, | ||||||
|  |       left: intLeft, | ||||||
|  |       rotation: this._pagesRotation, | ||||||
|  |       pdfOpenParams, | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   update() { | ||||||
|  |     throw new Error('Not implemented: update'); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   containsElement(element) { | ||||||
|  |     return this.container.contains(element); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   focus() { | ||||||
|  |     this.container.focus(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get isInPresentationMode() { | ||||||
|  |     return this.presentationModeState === PresentationModeState.FULLSCREEN; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get isChangingPresentationMode() { | ||||||
|  |     return this.presentationModeState === PresentationModeState.CHANGING; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get isHorizontalScrollbarEnabled() { | ||||||
|  |     return (this.isInPresentationMode ? | ||||||
|  |       false : (this.container.scrollWidth > this.container.clientWidth)); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   _getVisiblePages() { | ||||||
|  |     throw new Error('Not implemented: _getVisiblePages'); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   cleanup() { | ||||||
|  |     for (let i = 0, ii = this._pages.length; i < ii; i++) { | ||||||
|  |       if (this._pages[i] && | ||||||
|  |           this._pages[i].renderingState !== RenderingStates.FINISHED) { | ||||||
|  |         this._pages[i].reset(); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @private | ||||||
|  |    */ | ||||||
|  |   _cancelRendering() { | ||||||
|  |     for (let i = 0, ii = this._pages.length; i < ii; i++) { | ||||||
|  |       if (this._pages[i]) { | ||||||
|  |         this._pages[i].cancelRendering(); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @param {PDFPageView} pageView | ||||||
|  |    * @returns {Promise} Returns a promise containing a {PDFPageProxy} object. | ||||||
|  |    * @private | ||||||
|  |    */ | ||||||
|  |   _ensurePdfPageLoaded(pageView) { | ||||||
|  |     if (pageView.pdfPage) { | ||||||
|  |       return Promise.resolve(pageView.pdfPage); | ||||||
|  |     } | ||||||
|  |     let pageNumber = pageView.id; | ||||||
|  |     if (this._pagesRequests[pageNumber]) { | ||||||
|  |       return this._pagesRequests[pageNumber]; | ||||||
|  |     } | ||||||
|  |     let promise = this.pdfDocument.getPage(pageNumber).then((pdfPage) => { | ||||||
|  |       if (!pageView.pdfPage) { | ||||||
|  |         pageView.setPdfPage(pdfPage); | ||||||
|  |       } | ||||||
|  |       this._pagesRequests[pageNumber] = null; | ||||||
|  |       return pdfPage; | ||||||
|  |     }).catch((reason) => { | ||||||
|  |       console.error('Unable to get page for page view', reason); | ||||||
|  |       // Page error -- there is nothing can be done.
 | ||||||
|  |       this._pagesRequests[pageNumber] = null; | ||||||
|  |     }); | ||||||
|  |     this._pagesRequests[pageNumber] = promise; | ||||||
|  |     return promise; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   forceRendering(currentlyVisiblePages) { | ||||||
|  |     let visiblePages = currentlyVisiblePages || this._getVisiblePages(); | ||||||
|  |     let pageView = this.renderingQueue.getHighestPriority(visiblePages, | ||||||
|  |                                                           this._pages, | ||||||
|  |                                                           this.scroll.down); | ||||||
|  |     if (pageView) { | ||||||
|  |       this._ensurePdfPageLoaded(pageView).then(() => { | ||||||
|  |         this.renderingQueue.renderView(pageView); | ||||||
|  |       }); | ||||||
|  |       return true; | ||||||
|  |     } | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   getPageTextContent(pageIndex) { | ||||||
|  |     return this.pdfDocument.getPage(pageIndex + 1).then(function(page) { | ||||||
|  |       return page.getTextContent({ | ||||||
|  |         normalizeWhitespace: true, | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @param {HTMLDivElement} textLayerDiv | ||||||
|  |    * @param {number} pageIndex | ||||||
|  |    * @param {PageViewport} viewport | ||||||
|  |    * @returns {TextLayerBuilder} | ||||||
|  |    */ | ||||||
|  |   createTextLayerBuilder(textLayerDiv, pageIndex, viewport, | ||||||
|  |                          enhanceTextSelection = false) { | ||||||
|  |     return new TextLayerBuilder({ | ||||||
|  |       textLayerDiv, | ||||||
|  |       eventBus: this.eventBus, | ||||||
|  |       pageIndex, | ||||||
|  |       viewport, | ||||||
|  |       findController: this.isInPresentationMode ? null : this.findController, | ||||||
|  |       enhanceTextSelection: this.isInPresentationMode ? false : | ||||||
|  |                                                         enhanceTextSelection, | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @param {HTMLDivElement} pageDiv | ||||||
|  |    * @param {PDFPage} pdfPage | ||||||
|  |    * @param {boolean} renderInteractiveForms | ||||||
|  |    * @param {IL10n} l10n | ||||||
|  |    * @returns {AnnotationLayerBuilder} | ||||||
|  |    */ | ||||||
|  |   createAnnotationLayerBuilder(pageDiv, pdfPage, renderInteractiveForms = false, | ||||||
|  |                                l10n = NullL10n) { | ||||||
|  |     return new AnnotationLayerBuilder({ | ||||||
|  |       pageDiv, | ||||||
|  |       pdfPage, | ||||||
|  |       renderInteractiveForms, | ||||||
|  |       linkService: this.linkService, | ||||||
|  |       downloadManager: this.downloadManager, | ||||||
|  |       l10n, | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   setFindController(findController) { | ||||||
|  |     this.findController = findController; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * @returns {boolean} Whether all pages of the PDF document have identical | ||||||
|  |    *                    widths and heights. | ||||||
|  |    */ | ||||||
|  |   get hasEqualPageSizes() { | ||||||
|  |     let firstPageView = this._pages[0]; | ||||||
|  |     for (let i = 1, ii = this._pages.length; i < ii; ++i) { | ||||||
|  |       let pageView = this._pages[i]; | ||||||
|  |       if (pageView.width !== firstPageView.width || | ||||||
|  |           pageView.height !== firstPageView.height) { | ||||||
|  |         return false; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return true; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * Returns sizes of the pages. | ||||||
|  |    * @returns {Array} Array of objects with width/height/rotation fields. | ||||||
|  |    */ | ||||||
|  |   getPagesOverview() { | ||||||
|  |     let pagesOverview = this._pages.map(function(pageView) { | ||||||
|  |       let viewport = pageView.pdfPage.getViewport(1); | ||||||
|  |       return { | ||||||
|  |         width: viewport.width, | ||||||
|  |         height: viewport.height, | ||||||
|  |         rotation: viewport.rotation, | ||||||
|  |       }; | ||||||
|  |     }); | ||||||
|  |     if (!this.enablePrintAutoRotate) { | ||||||
|  |       return pagesOverview; | ||||||
|  |     } | ||||||
|  |     let isFirstPagePortrait = isPortraitOrientation(pagesOverview[0]); | ||||||
|  |     return pagesOverview.map(function (size) { | ||||||
|  |       if (isFirstPagePortrait === isPortraitOrientation(size)) { | ||||||
|  |         return size; | ||||||
|  |       } | ||||||
|  |       return { | ||||||
|  |         width: size.height, | ||||||
|  |         height: size.width, | ||||||
|  |         rotation: (size.rotation + 90) % 360, | ||||||
|  |       }; | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export { | ||||||
|  |   BaseViewer, | ||||||
|  | }; | ||||||
							
								
								
									
										149
									
								
								web/pdf_single_page_viewer.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										149
									
								
								web/pdf_single_page_viewer.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,149 @@ | |||||||
|  | /* Copyright 2017 Mozilla Foundation | ||||||
|  |  * | ||||||
|  |  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
|  |  * you may not use this file except in compliance with the License. | ||||||
|  |  * You may obtain a copy of the License at | ||||||
|  |  * | ||||||
|  |  *     http://www.apache.org/licenses/LICENSE-2.0
 | ||||||
|  |  * | ||||||
|  |  * Unless required by applicable law or agreed to in writing, software | ||||||
|  |  * distributed under the License is distributed on an "AS IS" BASIS, | ||||||
|  |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||||
|  |  * See the License for the specific language governing permissions and | ||||||
|  |  * limitations under the License. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | import { BaseViewer } from './base_viewer'; | ||||||
|  | import { scrollIntoView } from './ui_utils'; | ||||||
|  | import { shadow } from 'pdfjs-lib'; | ||||||
|  | 
 | ||||||
|  | class PDFSinglePageViewer extends BaseViewer { | ||||||
|  |   constructor(options) { | ||||||
|  |     super(options); | ||||||
|  | 
 | ||||||
|  |     this.eventBus.on('pagesinit', (evt) => { | ||||||
|  |       // Since the pages are placed in a `DocumentFragment`, make sure that
 | ||||||
|  |       // the current page becomes visible upon loading of the document.
 | ||||||
|  |       this._ensurePageViewVisible(); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get _setDocumentViewerElement() { | ||||||
|  |     // Since we only want to display *one* page at a time when using the
 | ||||||
|  |     // `PDFSinglePageViewer`, we cannot append them to the `viewer` DOM element.
 | ||||||
|  |     // Instead, they are placed in a `DocumentFragment`, and only the current
 | ||||||
|  |     // page is displayed in the viewer (refer to `this._ensurePageViewVisible`).
 | ||||||
|  |     return shadow(this, '_setDocumentViewerElement', this._shadowViewer); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   _resetView() { | ||||||
|  |     super._resetView(); | ||||||
|  |     this._previousPageNumber = 1; | ||||||
|  |     this._shadowViewer = document.createDocumentFragment(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   _ensurePageViewVisible() { | ||||||
|  |     let pageView = this._pages[this._currentPageNumber - 1]; | ||||||
|  |     let previousPageView = this._pages[this._previousPageNumber - 1]; | ||||||
|  | 
 | ||||||
|  |     let viewerNodes = this.viewer.childNodes; | ||||||
|  |     switch (viewerNodes.length) { | ||||||
|  |       case 0: // Should *only* occur on initial loading.
 | ||||||
|  |         this.viewer.appendChild(pageView.div); | ||||||
|  |         break; | ||||||
|  |       case 1: // The normal page-switching case.
 | ||||||
|  |         if (viewerNodes[0] !== previousPageView.div) { | ||||||
|  |           throw new Error( | ||||||
|  |             '_ensurePageViewVisible: Unexpected previously visible page.'); | ||||||
|  |         } | ||||||
|  |         if (pageView === previousPageView) { | ||||||
|  |           break; // The correct page is already visible.
 | ||||||
|  |         } | ||||||
|  |         // Switch visible pages, and reset the viewerContainer scroll position.
 | ||||||
|  |         this._shadowViewer.appendChild(previousPageView.div); | ||||||
|  |         this.viewer.appendChild(pageView.div); | ||||||
|  | 
 | ||||||
|  |         this.container.scrollTop = 0; | ||||||
|  |         break; | ||||||
|  |       default: | ||||||
|  |         throw new Error( | ||||||
|  |           '_ensurePageViewVisible: Only one page should be visible at a time.'); | ||||||
|  |     } | ||||||
|  |     this._previousPageNumber = this._currentPageNumber; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   _scrollUpdate() { | ||||||
|  |     if (this._updateScrollDown) { | ||||||
|  |       this._updateScrollDown(); | ||||||
|  |     } | ||||||
|  |     super._scrollUpdate(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   _scrollIntoView({ pageDiv, pageSpot = null, pageNumber = null, }) { | ||||||
|  |     if (pageNumber) { // Ensure that `this._currentPageNumber` is correct.
 | ||||||
|  |       this._setCurrentPageNumber(pageNumber); | ||||||
|  |     } | ||||||
|  |     let scrolledDown = this._currentPageNumber >= this._previousPageNumber; | ||||||
|  |     let previousLocation = this._location; | ||||||
|  |     this._ensurePageViewVisible(); | ||||||
|  | 
 | ||||||
|  |     scrollIntoView(pageDiv, pageSpot); | ||||||
|  | 
 | ||||||
|  |     // Since scrolling is tracked using `requestAnimationFrame`, update the
 | ||||||
|  |     // scroll direction during the next `this._scrollUpdate` invocation.
 | ||||||
|  |     this._updateScrollDown = () => { | ||||||
|  |       this.scroll.down = scrolledDown; | ||||||
|  |       delete this._updateScrollDown; | ||||||
|  |     }; | ||||||
|  |     // If the scroll position doesn't change as a result of the `scrollIntoView`
 | ||||||
|  |     // call, ensure that rendering always occurs to avoid showing a blank page.
 | ||||||
|  |     setTimeout(() => { | ||||||
|  |       if (this._location === previousLocation) { | ||||||
|  |         if (this._updateScrollDown) { | ||||||
|  |           this._updateScrollDown(); | ||||||
|  |         } | ||||||
|  |         this.update(); | ||||||
|  |       } | ||||||
|  |     }, 0); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   _getVisiblePages() { | ||||||
|  |     if (!this.pagesCount) { | ||||||
|  |       return { views: [], }; | ||||||
|  |     } | ||||||
|  |     let pageView = this._pages[this._currentPageNumber - 1]; | ||||||
|  |     // NOTE: Compute the `x` and `y` properties of the current view,
 | ||||||
|  |     // since `this._updateLocation` depends of them being available.
 | ||||||
|  |     let element = pageView.div; | ||||||
|  | 
 | ||||||
|  |     let view = { | ||||||
|  |       id: pageView.id, | ||||||
|  |       x: element.offsetLeft + element.clientLeft, | ||||||
|  |       y: element.offsetTop + element.clientTop, | ||||||
|  |       view: pageView, | ||||||
|  |     }; | ||||||
|  |     return { first: view, last: view, views: [view], }; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   update() { | ||||||
|  |     let visible = this._getVisiblePages(); | ||||||
|  |     let visiblePages = visible.views, numVisiblePages = visiblePages.length; | ||||||
|  | 
 | ||||||
|  |     if (numVisiblePages === 0) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     this._resizeBuffer(numVisiblePages); | ||||||
|  | 
 | ||||||
|  |     this.renderingQueue.renderHighestPriority(visible); | ||||||
|  | 
 | ||||||
|  |     this._updateLocation(visible.first); | ||||||
|  |     this.eventBus.dispatch('updateviewarea', { | ||||||
|  |       source: this, | ||||||
|  |       location: this._location, | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export { | ||||||
|  |   PDFSinglePageViewer, | ||||||
|  | }; | ||||||
| @ -17,6 +17,7 @@ | |||||||
| 
 | 
 | ||||||
| var pdfjsLib = require('./pdfjs.js'); | var pdfjsLib = require('./pdfjs.js'); | ||||||
| var pdfjsWebPDFViewer = require('./pdf_viewer.js'); | var pdfjsWebPDFViewer = require('./pdf_viewer.js'); | ||||||
|  | var pdfjsWebPDFSinglePageViewer = require('./pdf_single_page_viewer'); | ||||||
| var pdfjsWebPDFPageView = require('./pdf_page_view.js'); | var pdfjsWebPDFPageView = require('./pdf_page_view.js'); | ||||||
| var pdfjsWebPDFLinkService = require('./pdf_link_service.js'); | var pdfjsWebPDFLinkService = require('./pdf_link_service.js'); | ||||||
| var pdfjsWebTextLayerBuilder = require('./text_layer_builder.js'); | var pdfjsWebTextLayerBuilder = require('./text_layer_builder.js'); | ||||||
| @ -30,6 +31,7 @@ var pdfjsWebGenericL10n = require('./genericl10n.js'); | |||||||
| var PDFJS = pdfjsLib.PDFJS; | var PDFJS = pdfjsLib.PDFJS; | ||||||
| 
 | 
 | ||||||
| PDFJS.PDFViewer = pdfjsWebPDFViewer.PDFViewer; | PDFJS.PDFViewer = pdfjsWebPDFViewer.PDFViewer; | ||||||
|  | PDFJS.PDFSinglePageViewer = pdfjsWebPDFSinglePageViewer.PDFSinglePageViewer; | ||||||
| PDFJS.PDFPageView = pdfjsWebPDFPageView.PDFPageView; | PDFJS.PDFPageView = pdfjsWebPDFPageView.PDFPageView; | ||||||
| PDFJS.PDFLinkService = pdfjsWebPDFLinkService.PDFLinkService; | PDFJS.PDFLinkService = pdfjsWebPDFLinkService.PDFLinkService; | ||||||
| PDFJS.SimpleLinkService = pdfjsWebPDFLinkService.SimpleLinkService; | PDFJS.SimpleLinkService = pdfjsWebPDFLinkService.SimpleLinkService; | ||||||
|  | |||||||
| @ -13,748 +13,45 @@ | |||||||
|  * limitations under the License. |  * limitations under the License. | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| import { createPromiseCapability, PDFJS } from 'pdfjs-lib'; | import { getVisibleElements, scrollIntoView } from './ui_utils'; | ||||||
| import { | import { BaseViewer } from './base_viewer'; | ||||||
|   CSS_UNITS, DEFAULT_SCALE, DEFAULT_SCALE_VALUE, getVisibleElements, | import { shadow } from 'pdfjs-lib'; | ||||||
|   isValidRotation, MAX_AUTO_SCALE, NullL10n, RendererType, SCROLLBAR_PADDING, |  | ||||||
|   scrollIntoView, UNKNOWN_SCALE, VERTICAL_PADDING, watchScroll |  | ||||||
| } from './ui_utils'; |  | ||||||
| import { PDFRenderingQueue, RenderingStates } from './pdf_rendering_queue'; |  | ||||||
| import { AnnotationLayerBuilder } from './annotation_layer_builder'; |  | ||||||
| import { getGlobalEventBus } from './dom_events'; |  | ||||||
| import { PDFPageView } from './pdf_page_view'; |  | ||||||
| import { SimpleLinkService } from './pdf_link_service'; |  | ||||||
| import { TextLayerBuilder } from './text_layer_builder'; |  | ||||||
| 
 | 
 | ||||||
| const PresentationModeState = { | class PDFViewer extends BaseViewer { | ||||||
|   UNKNOWN: 0, |   get _setDocumentViewerElement() { | ||||||
|   NORMAL: 1, |     return shadow(this, '_setDocumentViewerElement', this.viewer); | ||||||
|   CHANGING: 2, |  | ||||||
|   FULLSCREEN: 3, |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| const DEFAULT_CACHE_SIZE = 10; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * @typedef {Object} PDFViewerOptions |  | ||||||
|  * @property {HTMLDivElement} container - The container for the viewer element. |  | ||||||
|  * @property {HTMLDivElement} viewer - (optional) The viewer element. |  | ||||||
|  * @property {EventBus} eventBus - The application event bus. |  | ||||||
|  * @property {IPDFLinkService} linkService - The navigation/linking service. |  | ||||||
|  * @property {DownloadManager} downloadManager - (optional) The download |  | ||||||
|  *   manager component. |  | ||||||
|  * @property {PDFRenderingQueue} renderingQueue - (optional) The rendering |  | ||||||
|  *   queue object. |  | ||||||
|  * @property {boolean} removePageBorders - (optional) Removes the border shadow |  | ||||||
|  *   around the pages. The default is false. |  | ||||||
|  * @property {boolean} enhanceTextSelection - (optional) Enables the improved |  | ||||||
|  *   text selection behaviour. The default is `false`. |  | ||||||
|  * @property {boolean} renderInteractiveForms - (optional) Enables rendering of |  | ||||||
|  *   interactive form elements. The default is `false`. |  | ||||||
|  * @property {boolean} enablePrintAutoRotate - (optional) Enables automatic |  | ||||||
|  *   rotation of pages whose orientation differ from the first page upon |  | ||||||
|  *   printing. The default is `false`. |  | ||||||
|  * @property {string} renderer - 'canvas' or 'svg'. The default is 'canvas'. |  | ||||||
|  * @property {IL10n} l10n - Localization service. |  | ||||||
|  */ |  | ||||||
| 
 |  | ||||||
| function PDFPageViewBuffer(size) { |  | ||||||
|   let data = []; |  | ||||||
|   this.push = function cachePush(view) { |  | ||||||
|     let i = data.indexOf(view); |  | ||||||
|     if (i >= 0) { |  | ||||||
|       data.splice(i, 1); |  | ||||||
|     } |  | ||||||
|     data.push(view); |  | ||||||
|     if (data.length > size) { |  | ||||||
|       data.shift().destroy(); |  | ||||||
|     } |  | ||||||
|   }; |  | ||||||
|   this.resize = function (newSize) { |  | ||||||
|     size = newSize; |  | ||||||
|     while (data.length > size) { |  | ||||||
|       data.shift().destroy(); |  | ||||||
|     } |  | ||||||
|   }; |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| function isSameScale(oldScale, newScale) { |   _scrollIntoView({ pageDiv, pageSpot = null, }) { | ||||||
|   if (newScale === oldScale) { |     scrollIntoView(pageDiv, pageSpot); | ||||||
|     return true; |  | ||||||
|   } |  | ||||||
|   if (Math.abs(newScale - oldScale) < 1e-15) { |  | ||||||
|     // Prevent unnecessary re-rendering of all pages when the scale
 |  | ||||||
|     // changes only because of limited numerical precision.
 |  | ||||||
|     return true; |  | ||||||
|   } |  | ||||||
|   return false; |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| function isPortraitOrientation(size) { |   _getVisiblePages() { | ||||||
|   return size.width <= size.height; |     if (!this.isInPresentationMode) { | ||||||
|  |       return getVisibleElements(this.container, this._pages, true); | ||||||
|     } |     } | ||||||
| 
 |     // The algorithm in getVisibleElements doesn't work in all browsers and
 | ||||||
| /** |     // configurations when presentation mode is active.
 | ||||||
|  * Simple viewer control to display PDF content/pages. |  | ||||||
|  * @implements {IRenderableView} |  | ||||||
|  */ |  | ||||||
| class PDFViewer { |  | ||||||
|   /** |  | ||||||
|    * @param {PDFViewerOptions} options |  | ||||||
|    */ |  | ||||||
|   constructor(options) { |  | ||||||
|     this.container = options.container; |  | ||||||
|     this.viewer = options.viewer || options.container.firstElementChild; |  | ||||||
|     this.eventBus = options.eventBus || getGlobalEventBus(); |  | ||||||
|     this.linkService = options.linkService || new SimpleLinkService(); |  | ||||||
|     this.downloadManager = options.downloadManager || null; |  | ||||||
|     this.removePageBorders = options.removePageBorders || false; |  | ||||||
|     this.enhanceTextSelection = options.enhanceTextSelection || false; |  | ||||||
|     this.renderInteractiveForms = options.renderInteractiveForms || false; |  | ||||||
|     this.enablePrintAutoRotate = options.enablePrintAutoRotate || false; |  | ||||||
|     this.renderer = options.renderer || RendererType.CANVAS; |  | ||||||
|     this.l10n = options.l10n || NullL10n; |  | ||||||
| 
 |  | ||||||
|     this.defaultRenderingQueue = !options.renderingQueue; |  | ||||||
|     if (this.defaultRenderingQueue) { |  | ||||||
|       // Custom rendering queue is not specified, using default one
 |  | ||||||
|       this.renderingQueue = new PDFRenderingQueue(); |  | ||||||
|       this.renderingQueue.setViewer(this); |  | ||||||
|     } else { |  | ||||||
|       this.renderingQueue = options.renderingQueue; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     this.scroll = watchScroll(this.container, this._scrollUpdate.bind(this)); |  | ||||||
|     this.presentationModeState = PresentationModeState.UNKNOWN; |  | ||||||
|     this._resetView(); |  | ||||||
| 
 |  | ||||||
|     if (this.removePageBorders) { |  | ||||||
|       this.viewer.classList.add('removePageBorders'); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   get pagesCount() { |  | ||||||
|     return this._pages.length; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   getPageView(index) { |  | ||||||
|     return this._pages[index]; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @returns {boolean} true if all {PDFPageView} objects are initialized. |  | ||||||
|    */ |  | ||||||
|   get pageViewsReady() { |  | ||||||
|     return this._pageViewsReady; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @returns {number} |  | ||||||
|    */ |  | ||||||
|   get currentPageNumber() { |  | ||||||
|     return this._currentPageNumber; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @param {number} val - The page number. |  | ||||||
|    */ |  | ||||||
|   set currentPageNumber(val) { |  | ||||||
|     if (!Number.isInteger(val)) { |  | ||||||
|       throw new Error('Invalid page number.'); |  | ||||||
|     } |  | ||||||
|     if (!this.pdfDocument) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     // The intent can be to just reset a scroll position and/or scale.
 |  | ||||||
|     this._setCurrentPageNumber(val, /* resetCurrentPageView = */ true); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @private |  | ||||||
|    */ |  | ||||||
|   _setCurrentPageNumber(val, resetCurrentPageView = false) { |  | ||||||
|     if (this._currentPageNumber === val) { |  | ||||||
|       if (resetCurrentPageView) { |  | ||||||
|         this._resetCurrentPageView(); |  | ||||||
|       } |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if (!(0 < val && val <= this.pagesCount)) { |  | ||||||
|       console.error( |  | ||||||
|         `PDFViewer._setCurrentPageNumber: "${val}" is out of bounds.`); |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     let arg = { |  | ||||||
|       source: this, |  | ||||||
|       pageNumber: val, |  | ||||||
|       pageLabel: this._pageLabels && this._pageLabels[val - 1], |  | ||||||
|     }; |  | ||||||
|     this._currentPageNumber = val; |  | ||||||
|     this.eventBus.dispatch('pagechanging', arg); |  | ||||||
|     this.eventBus.dispatch('pagechange', arg); |  | ||||||
| 
 |  | ||||||
|     if (resetCurrentPageView) { |  | ||||||
|       this._resetCurrentPageView(); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @returns {string|null} Returns the current page label, |  | ||||||
|    *                        or `null` if no page labels exist. |  | ||||||
|    */ |  | ||||||
|   get currentPageLabel() { |  | ||||||
|     return this._pageLabels && this._pageLabels[this._currentPageNumber - 1]; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @param {string} val - The page label. |  | ||||||
|    */ |  | ||||||
|   set currentPageLabel(val) { |  | ||||||
|     let pageNumber = val | 0; // Fallback page number.
 |  | ||||||
|     if (this._pageLabels) { |  | ||||||
|       let i = this._pageLabels.indexOf(val); |  | ||||||
|       if (i >= 0) { |  | ||||||
|         pageNumber = i + 1; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     this.currentPageNumber = pageNumber; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @returns {number} |  | ||||||
|    */ |  | ||||||
|   get currentScale() { |  | ||||||
|     return this._currentScale !== UNKNOWN_SCALE ? this._currentScale : |  | ||||||
|                                                   DEFAULT_SCALE; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @param {number} val - Scale of the pages in percents. |  | ||||||
|    */ |  | ||||||
|   set currentScale(val) { |  | ||||||
|     if (isNaN(val)) { |  | ||||||
|       throw new Error('Invalid numeric scale'); |  | ||||||
|     } |  | ||||||
|     if (!this.pdfDocument) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     this._setScale(val, false); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @returns {string} |  | ||||||
|    */ |  | ||||||
|   get currentScaleValue() { |  | ||||||
|     return this._currentScaleValue; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @param val - The scale of the pages (in percent or predefined value). |  | ||||||
|    */ |  | ||||||
|   set currentScaleValue(val) { |  | ||||||
|     if (!this.pdfDocument) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     this._setScale(val, false); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @returns {number} |  | ||||||
|    */ |  | ||||||
|   get pagesRotation() { |  | ||||||
|     return this._pagesRotation; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @param {number} rotation - The rotation of the pages (0, 90, 180, 270). |  | ||||||
|    */ |  | ||||||
|   set pagesRotation(rotation) { |  | ||||||
|     if (!isValidRotation(rotation)) { |  | ||||||
|       throw new Error('Invalid pages rotation angle.'); |  | ||||||
|     } |  | ||||||
|     if (!this.pdfDocument) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     if (this._pagesRotation === rotation) { |  | ||||||
|       return; // The rotation didn't change.
 |  | ||||||
|     } |  | ||||||
|     this._pagesRotation = rotation; |  | ||||||
| 
 |  | ||||||
|     let pageNumber = this._currentPageNumber; |  | ||||||
| 
 |  | ||||||
|     for (let i = 0, ii = this._pages.length; i < ii; i++) { |  | ||||||
|       let pageView = this._pages[i]; |  | ||||||
|       pageView.update(pageView.scale, rotation); |  | ||||||
|     } |  | ||||||
|     // Prevent errors in case the rotation changes *before* the scale has been
 |  | ||||||
|     // set to a non-default value.
 |  | ||||||
|     if (this._currentScaleValue) { |  | ||||||
|       this._setScale(this._currentScaleValue, true); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     this.eventBus.dispatch('rotationchanging', { |  | ||||||
|       source: this, |  | ||||||
|       pagesRotation: rotation, |  | ||||||
|       pageNumber, |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     if (this.defaultRenderingQueue) { |  | ||||||
|       this.update(); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @param pdfDocument {PDFDocument} |  | ||||||
|    */ |  | ||||||
|   setDocument(pdfDocument) { |  | ||||||
|     if (this.pdfDocument) { |  | ||||||
|       this._cancelRendering(); |  | ||||||
|       this._resetView(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     this.pdfDocument = pdfDocument; |  | ||||||
|     if (!pdfDocument) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     let pagesCount = pdfDocument.numPages; |  | ||||||
| 
 |  | ||||||
|     let pagesCapability = createPromiseCapability(); |  | ||||||
|     this.pagesPromise = pagesCapability.promise; |  | ||||||
| 
 |  | ||||||
|     pagesCapability.promise.then(() => { |  | ||||||
|       this._pageViewsReady = true; |  | ||||||
|       this.eventBus.dispatch('pagesloaded', { |  | ||||||
|         source: this, |  | ||||||
|         pagesCount, |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     let isOnePageRenderedResolved = false; |  | ||||||
|     let onePageRenderedCapability = createPromiseCapability(); |  | ||||||
|     this.onePageRendered = onePageRenderedCapability.promise; |  | ||||||
| 
 |  | ||||||
|     let bindOnAfterAndBeforeDraw = (pageView) => { |  | ||||||
|       pageView.onBeforeDraw = () => { |  | ||||||
|         // Add the page to the buffer at the start of drawing. That way it can
 |  | ||||||
|         // be evicted from the buffer and destroyed even if we pause its
 |  | ||||||
|         // rendering.
 |  | ||||||
|         this._buffer.push(pageView); |  | ||||||
|       }; |  | ||||||
|       pageView.onAfterDraw = () => { |  | ||||||
|         if (!isOnePageRenderedResolved) { |  | ||||||
|           isOnePageRenderedResolved = true; |  | ||||||
|           onePageRenderedCapability.resolve(); |  | ||||||
|         } |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     let firstPagePromise = pdfDocument.getPage(1); |  | ||||||
|     this.firstPagePromise = firstPagePromise; |  | ||||||
| 
 |  | ||||||
|     // Fetch a single page so we can get a viewport that will be the default
 |  | ||||||
|     // viewport for all pages
 |  | ||||||
|     firstPagePromise.then((pdfPage) => { |  | ||||||
|       let scale = this.currentScale; |  | ||||||
|       let viewport = pdfPage.getViewport(scale * CSS_UNITS); |  | ||||||
|       for (let pageNum = 1; pageNum <= pagesCount; ++pageNum) { |  | ||||||
|         let textLayerFactory = null; |  | ||||||
|         if (!PDFJS.disableTextLayer) { |  | ||||||
|           textLayerFactory = this; |  | ||||||
|         } |  | ||||||
|         let pageView = new PDFPageView({ |  | ||||||
|           container: this.viewer, |  | ||||||
|           eventBus: this.eventBus, |  | ||||||
|           id: pageNum, |  | ||||||
|           scale, |  | ||||||
|           defaultViewport: viewport.clone(), |  | ||||||
|           renderingQueue: this.renderingQueue, |  | ||||||
|           textLayerFactory, |  | ||||||
|           annotationLayerFactory: this, |  | ||||||
|           enhanceTextSelection: this.enhanceTextSelection, |  | ||||||
|           renderInteractiveForms: this.renderInteractiveForms, |  | ||||||
|           renderer: this.renderer, |  | ||||||
|           l10n: this.l10n, |  | ||||||
|         }); |  | ||||||
|         bindOnAfterAndBeforeDraw(pageView); |  | ||||||
|         this._pages.push(pageView); |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       // Fetch all the pages since the viewport is needed before printing
 |  | ||||||
|       // starts to create the correct size canvas. Wait until one page is
 |  | ||||||
|       // rendered so we don't tie up too many resources early on.
 |  | ||||||
|       onePageRenderedCapability.promise.then(() => { |  | ||||||
|         if (PDFJS.disableAutoFetch) { |  | ||||||
|           // XXX: Printing is semi-broken with auto fetch disabled.
 |  | ||||||
|           pagesCapability.resolve(); |  | ||||||
|           return; |  | ||||||
|         } |  | ||||||
|         let getPagesLeft = pagesCount; |  | ||||||
|         for (let pageNum = 1; pageNum <= pagesCount; ++pageNum) { |  | ||||||
|           pdfDocument.getPage(pageNum).then((pdfPage) => { |  | ||||||
|             let pageView = this._pages[pageNum - 1]; |  | ||||||
|             if (!pageView.pdfPage) { |  | ||||||
|               pageView.setPdfPage(pdfPage); |  | ||||||
|             } |  | ||||||
|             this.linkService.cachePageRef(pageNum, pdfPage.ref); |  | ||||||
|             if (--getPagesLeft === 0) { |  | ||||||
|               pagesCapability.resolve(); |  | ||||||
|             } |  | ||||||
|           }, (reason) => { |  | ||||||
|             console.error(`Unable to get page ${pageNum} to initialize viewer`, |  | ||||||
|                           reason); |  | ||||||
|             if (--getPagesLeft === 0) { |  | ||||||
|               pagesCapability.resolve(); |  | ||||||
|             } |  | ||||||
|           }); |  | ||||||
|         } |  | ||||||
|       }); |  | ||||||
| 
 |  | ||||||
|       this.eventBus.dispatch('pagesinit', { source: this, }); |  | ||||||
| 
 |  | ||||||
|       if (this.defaultRenderingQueue) { |  | ||||||
|         this.update(); |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       if (this.findController) { |  | ||||||
|         this.findController.resolveFirstPage(); |  | ||||||
|       } |  | ||||||
|     }).catch((reason) => { |  | ||||||
|       console.error('Unable to initialize viewer', reason); |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @param {Array|null} labels |  | ||||||
|    */ |  | ||||||
|   setPageLabels(labels) { |  | ||||||
|     if (!this.pdfDocument) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     if (!labels) { |  | ||||||
|       this._pageLabels = null; |  | ||||||
|     } else if (!(labels instanceof Array && |  | ||||||
|                  this.pdfDocument.numPages === labels.length)) { |  | ||||||
|       this._pageLabels = null; |  | ||||||
|       console.error('PDFViewer.setPageLabels: Invalid page labels.'); |  | ||||||
|     } else { |  | ||||||
|       this._pageLabels = labels; |  | ||||||
|     } |  | ||||||
|     // Update all the `PDFPageView` instances.
 |  | ||||||
|     for (let i = 0, ii = this._pages.length; i < ii; i++) { |  | ||||||
|       let pageView = this._pages[i]; |  | ||||||
|       let label = this._pageLabels && this._pageLabels[i]; |  | ||||||
|       pageView.setPageLabel(label); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   _resetView() { |  | ||||||
|     this._pages = []; |  | ||||||
|     this._currentPageNumber = 1; |  | ||||||
|     this._currentScale = UNKNOWN_SCALE; |  | ||||||
|     this._currentScaleValue = null; |  | ||||||
|     this._pageLabels = null; |  | ||||||
|     this._buffer = new PDFPageViewBuffer(DEFAULT_CACHE_SIZE); |  | ||||||
|     this._location = null; |  | ||||||
|     this._pagesRotation = 0; |  | ||||||
|     this._pagesRequests = []; |  | ||||||
|     this._pageViewsReady = false; |  | ||||||
| 
 |  | ||||||
|     // Remove the pages from the DOM.
 |  | ||||||
|     this.viewer.textContent = ''; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   _scrollUpdate() { |  | ||||||
|     if (this.pagesCount === 0) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     this.update(); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   _setScaleDispatchEvent(newScale, newValue, preset = false) { |  | ||||||
|     let arg = { |  | ||||||
|       source: this, |  | ||||||
|       scale: newScale, |  | ||||||
|       presetValue: preset ? newValue : undefined, |  | ||||||
|     }; |  | ||||||
|     this.eventBus.dispatch('scalechanging', arg); |  | ||||||
|     this.eventBus.dispatch('scalechange', arg); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   _setScaleUpdatePages(newScale, newValue, noScroll = false, preset = false) { |  | ||||||
|     this._currentScaleValue = newValue.toString(); |  | ||||||
| 
 |  | ||||||
|     if (isSameScale(this._currentScale, newScale)) { |  | ||||||
|       if (preset) { |  | ||||||
|         this._setScaleDispatchEvent(newScale, newValue, true); |  | ||||||
|       } |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     for (let i = 0, ii = this._pages.length; i < ii; i++) { |  | ||||||
|       this._pages[i].update(newScale); |  | ||||||
|     } |  | ||||||
|     this._currentScale = newScale; |  | ||||||
| 
 |  | ||||||
|     if (!noScroll) { |  | ||||||
|       let page = this._currentPageNumber, dest; |  | ||||||
|       if (this._location && !PDFJS.ignoreCurrentPositionOnZoom && |  | ||||||
|           !(this.isInPresentationMode || this.isChangingPresentationMode)) { |  | ||||||
|         page = this._location.pageNumber; |  | ||||||
|         dest = [null, { name: 'XYZ', }, this._location.left, |  | ||||||
|                 this._location.top, null]; |  | ||||||
|       } |  | ||||||
|       this.scrollPageIntoView({ |  | ||||||
|         pageNumber: page, |  | ||||||
|         destArray: dest, |  | ||||||
|         allowNegativeOffset: true, |  | ||||||
|       }); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     this._setScaleDispatchEvent(newScale, newValue, preset); |  | ||||||
| 
 |  | ||||||
|     if (this.defaultRenderingQueue) { |  | ||||||
|       this.update(); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   _setScale(value, noScroll = false) { |  | ||||||
|     let scale = parseFloat(value); |  | ||||||
| 
 |  | ||||||
|     if (scale > 0) { |  | ||||||
|       this._setScaleUpdatePages(scale, value, noScroll, /* preset = */ false); |  | ||||||
|     } else { |  | ||||||
|     let currentPage = this._pages[this._currentPageNumber - 1]; |     let currentPage = this._pages[this._currentPageNumber - 1]; | ||||||
|       if (!currentPage) { |     let visible = [{ id: currentPage.id, view: currentPage, }]; | ||||||
|         return; |     return { first: currentPage, last: currentPage, views: visible, }; | ||||||
|       } |  | ||||||
|       let hPadding = (this.isInPresentationMode || this.removePageBorders) ? |  | ||||||
|         0 : SCROLLBAR_PADDING; |  | ||||||
|       let vPadding = (this.isInPresentationMode || this.removePageBorders) ? |  | ||||||
|         0 : VERTICAL_PADDING; |  | ||||||
|       let pageWidthScale = (this.container.clientWidth - hPadding) / |  | ||||||
|                            currentPage.width * currentPage.scale; |  | ||||||
|       let pageHeightScale = (this.container.clientHeight - vPadding) / |  | ||||||
|                             currentPage.height * currentPage.scale; |  | ||||||
|       switch (value) { |  | ||||||
|         case 'page-actual': |  | ||||||
|           scale = 1; |  | ||||||
|           break; |  | ||||||
|         case 'page-width': |  | ||||||
|           scale = pageWidthScale; |  | ||||||
|           break; |  | ||||||
|         case 'page-height': |  | ||||||
|           scale = pageHeightScale; |  | ||||||
|           break; |  | ||||||
|         case 'page-fit': |  | ||||||
|           scale = Math.min(pageWidthScale, pageHeightScale); |  | ||||||
|           break; |  | ||||||
|         case 'auto': |  | ||||||
|           let isLandscape = (currentPage.width > currentPage.height); |  | ||||||
|           // For pages in landscape mode, fit the page height to the viewer
 |  | ||||||
|           // *unless* the page would thus become too wide to fit horizontally.
 |  | ||||||
|           let horizontalScale = isLandscape ? |  | ||||||
|             Math.min(pageHeightScale, pageWidthScale) : pageWidthScale; |  | ||||||
|           scale = Math.min(MAX_AUTO_SCALE, horizontalScale); |  | ||||||
|           break; |  | ||||||
|         default: |  | ||||||
|           console.error( |  | ||||||
|             `PDFViewer._setScale: "${value}" is an unknown zoom value.`); |  | ||||||
|           return; |  | ||||||
|       } |  | ||||||
|       this._setScaleUpdatePages(scale, value, noScroll, /* preset = */ true); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * Refreshes page view: scrolls to the current page and updates the scale. |  | ||||||
|    * @private |  | ||||||
|    */ |  | ||||||
|   _resetCurrentPageView() { |  | ||||||
|     if (this.isInPresentationMode) { |  | ||||||
|       // Fixes the case when PDF has different page sizes.
 |  | ||||||
|       this._setScale(this._currentScaleValue, true); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     let pageView = this._pages[this._currentPageNumber - 1]; |  | ||||||
|     scrollIntoView(pageView.div); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @typedef ScrollPageIntoViewParameters |  | ||||||
|    * @property {number} pageNumber - The page number. |  | ||||||
|    * @property {Array} destArray - (optional) The original PDF destination |  | ||||||
|    *   array, in the format: <page-ref> </XYZ|/FitXXX> <args..> |  | ||||||
|    * @property {boolean} allowNegativeOffset - (optional) Allow negative page |  | ||||||
|    *   offsets. The default value is `false`. |  | ||||||
|    */ |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * Scrolls page into view. |  | ||||||
|    * @param {ScrollPageIntoViewParameters} params |  | ||||||
|    */ |  | ||||||
|   scrollPageIntoView(params) { |  | ||||||
|     if ((typeof PDFJSDev === 'undefined' || PDFJSDev.test('GENERIC')) && |  | ||||||
|         (arguments.length > 1 || typeof params === 'number')) { |  | ||||||
|       console.error('Call of scrollPageIntoView() with obsolete signature.'); |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     if (!this.pdfDocument) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     let pageNumber = params.pageNumber || 0; |  | ||||||
|     let dest = params.destArray || null; |  | ||||||
|     let allowNegativeOffset = params.allowNegativeOffset || false; |  | ||||||
| 
 |  | ||||||
|     if (this.isInPresentationMode || !dest) { |  | ||||||
|       this._setCurrentPageNumber(pageNumber, /* resetCurrentPageView = */ true); |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     let pageView = this._pages[pageNumber - 1]; |  | ||||||
|     if (!pageView) { |  | ||||||
|       console.error( |  | ||||||
|         'PDFViewer.scrollPageIntoView: Invalid "pageNumber" parameter.'); |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     let x = 0, y = 0; |  | ||||||
|     let width = 0, height = 0, widthScale, heightScale; |  | ||||||
|     let changeOrientation = (pageView.rotation % 180 === 0 ? false : true); |  | ||||||
|     let pageWidth = (changeOrientation ? pageView.height : pageView.width) / |  | ||||||
|       pageView.scale / CSS_UNITS; |  | ||||||
|     let pageHeight = (changeOrientation ? pageView.width : pageView.height) / |  | ||||||
|       pageView.scale / CSS_UNITS; |  | ||||||
|     let scale = 0; |  | ||||||
|     switch (dest[1].name) { |  | ||||||
|       case 'XYZ': |  | ||||||
|         x = dest[2]; |  | ||||||
|         y = dest[3]; |  | ||||||
|         scale = dest[4]; |  | ||||||
|         // If x and/or y coordinates are not supplied, default to
 |  | ||||||
|         // _top_ left of the page (not the obvious bottom left,
 |  | ||||||
|         // since aligning the bottom of the intended page with the
 |  | ||||||
|         // top of the window is rarely helpful).
 |  | ||||||
|         x = x !== null ? x : 0; |  | ||||||
|         y = y !== null ? y : pageHeight; |  | ||||||
|         break; |  | ||||||
|       case 'Fit': |  | ||||||
|       case 'FitB': |  | ||||||
|         scale = 'page-fit'; |  | ||||||
|         break; |  | ||||||
|       case 'FitH': |  | ||||||
|       case 'FitBH': |  | ||||||
|         y = dest[2]; |  | ||||||
|         scale = 'page-width'; |  | ||||||
|         // According to the PDF spec, section 12.3.2.2, a `null` value in the
 |  | ||||||
|         // parameter should maintain the position relative to the new page.
 |  | ||||||
|         if (y === null && this._location) { |  | ||||||
|           x = this._location.left; |  | ||||||
|           y = this._location.top; |  | ||||||
|         } |  | ||||||
|         break; |  | ||||||
|       case 'FitV': |  | ||||||
|       case 'FitBV': |  | ||||||
|         x = dest[2]; |  | ||||||
|         width = pageWidth; |  | ||||||
|         height = pageHeight; |  | ||||||
|         scale = 'page-height'; |  | ||||||
|         break; |  | ||||||
|       case 'FitR': |  | ||||||
|         x = dest[2]; |  | ||||||
|         y = dest[3]; |  | ||||||
|         width = dest[4] - x; |  | ||||||
|         height = dest[5] - y; |  | ||||||
|         let hPadding = this.removePageBorders ? 0 : SCROLLBAR_PADDING; |  | ||||||
|         let vPadding = this.removePageBorders ? 0 : VERTICAL_PADDING; |  | ||||||
| 
 |  | ||||||
|         widthScale = (this.container.clientWidth - hPadding) / |  | ||||||
|           width / CSS_UNITS; |  | ||||||
|         heightScale = (this.container.clientHeight - vPadding) / |  | ||||||
|           height / CSS_UNITS; |  | ||||||
|         scale = Math.min(Math.abs(widthScale), Math.abs(heightScale)); |  | ||||||
|         break; |  | ||||||
|       default: |  | ||||||
|         console.error(`PDFViewer.scrollPageIntoView: "${dest[1].name}" ` + |  | ||||||
|                       'is not a valid destination type.'); |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if (scale && scale !== this._currentScale) { |  | ||||||
|       this.currentScaleValue = scale; |  | ||||||
|     } else if (this._currentScale === UNKNOWN_SCALE) { |  | ||||||
|       this.currentScaleValue = DEFAULT_SCALE_VALUE; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if (scale === 'page-fit' && !dest[4]) { |  | ||||||
|       scrollIntoView(pageView.div); |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     let boundingRect = [ |  | ||||||
|       pageView.viewport.convertToViewportPoint(x, y), |  | ||||||
|       pageView.viewport.convertToViewportPoint(x + width, y + height) |  | ||||||
|     ]; |  | ||||||
|     let left = Math.min(boundingRect[0][0], boundingRect[1][0]); |  | ||||||
|     let top = Math.min(boundingRect[0][1], boundingRect[1][1]); |  | ||||||
| 
 |  | ||||||
|     if (!allowNegativeOffset) { |  | ||||||
|       // Some bad PDF generators will create destinations with e.g. top values
 |  | ||||||
|       // that exceeds the page height. Ensure that offsets are not negative,
 |  | ||||||
|       // to prevent a previous page from becoming visible (fixes bug 874482).
 |  | ||||||
|       left = Math.max(left, 0); |  | ||||||
|       top = Math.max(top, 0); |  | ||||||
|     } |  | ||||||
|     scrollIntoView(pageView.div, { left, top, }); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   _updateLocation(firstPage) { |  | ||||||
|     let currentScale = this._currentScale; |  | ||||||
|     let currentScaleValue = this._currentScaleValue; |  | ||||||
|     let normalizedScaleValue = |  | ||||||
|       parseFloat(currentScaleValue) === currentScale ? |  | ||||||
|       Math.round(currentScale * 10000) / 100 : currentScaleValue; |  | ||||||
| 
 |  | ||||||
|     let pageNumber = firstPage.id; |  | ||||||
|     let pdfOpenParams = '#page=' + pageNumber; |  | ||||||
|     pdfOpenParams += '&zoom=' + normalizedScaleValue; |  | ||||||
|     let currentPageView = this._pages[pageNumber - 1]; |  | ||||||
|     let container = this.container; |  | ||||||
|     let topLeft = currentPageView.getPagePoint( |  | ||||||
|       (container.scrollLeft - firstPage.x), |  | ||||||
|       (container.scrollTop - firstPage.y)); |  | ||||||
|     let intLeft = Math.round(topLeft[0]); |  | ||||||
|     let intTop = Math.round(topLeft[1]); |  | ||||||
|     pdfOpenParams += ',' + intLeft + ',' + intTop; |  | ||||||
| 
 |  | ||||||
|     this._location = { |  | ||||||
|       pageNumber, |  | ||||||
|       scale: normalizedScaleValue, |  | ||||||
|       top: intTop, |  | ||||||
|       left: intLeft, |  | ||||||
|       rotation: this._pagesRotation, |  | ||||||
|       pdfOpenParams, |  | ||||||
|     }; |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   update() { |   update() { | ||||||
|     let visible = this._getVisiblePages(); |     let visible = this._getVisiblePages(); | ||||||
|     let visiblePages = visible.views; |     let visiblePages = visible.views, numVisiblePages = visiblePages.length; | ||||||
|     if (visiblePages.length === 0) { | 
 | ||||||
|  |     if (numVisiblePages === 0) { | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
| 
 |     this._resizeBuffer(numVisiblePages); | ||||||
|     let suggestedCacheSize = Math.max(DEFAULT_CACHE_SIZE, |  | ||||||
|                                       2 * visiblePages.length + 1); |  | ||||||
|     this._buffer.resize(suggestedCacheSize); |  | ||||||
| 
 | 
 | ||||||
|     this.renderingQueue.renderHighestPriority(visible); |     this.renderingQueue.renderHighestPriority(visible); | ||||||
| 
 | 
 | ||||||
|     let currentId = this._currentPageNumber; |     let currentId = this._currentPageNumber; | ||||||
|     let firstPage = visible.first; |  | ||||||
|     let stillFullyVisible = false; |     let stillFullyVisible = false; | ||||||
| 
 | 
 | ||||||
|     for (let i = 0, ii = visiblePages.length; i < ii; ++i) { |     for (let i = 0; i < numVisiblePages; ++i) { | ||||||
|       let page = visiblePages[i]; |       let page = visiblePages[i]; | ||||||
| 
 | 
 | ||||||
|       if (page.percent < 100) { |       if (page.percent < 100) { | ||||||
| @ -769,211 +66,18 @@ class PDFViewer { | |||||||
|     if (!stillFullyVisible) { |     if (!stillFullyVisible) { | ||||||
|       currentId = visiblePages[0].id; |       currentId = visiblePages[0].id; | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     if (!this.isInPresentationMode) { |     if (!this.isInPresentationMode) { | ||||||
|       this._setCurrentPageNumber(currentId); |       this._setCurrentPageNumber(currentId); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     this._updateLocation(firstPage); |     this._updateLocation(visible.first); | ||||||
| 
 |  | ||||||
|     this.eventBus.dispatch('updateviewarea', { |     this.eventBus.dispatch('updateviewarea', { | ||||||
|       source: this, |       source: this, | ||||||
|       location: this._location, |       location: this._location, | ||||||
|     }); |     }); | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   containsElement(element) { |  | ||||||
|     return this.container.contains(element); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   focus() { |  | ||||||
|     this.container.focus(); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   get isInPresentationMode() { |  | ||||||
|     return this.presentationModeState === PresentationModeState.FULLSCREEN; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   get isChangingPresentationMode() { |  | ||||||
|     return this.presentationModeState === PresentationModeState.CHANGING; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   get isHorizontalScrollbarEnabled() { |  | ||||||
|     return (this.isInPresentationMode ? |  | ||||||
|       false : (this.container.scrollWidth > this.container.clientWidth)); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   _getVisiblePages() { |  | ||||||
|     if (!this.isInPresentationMode) { |  | ||||||
|       return getVisibleElements(this.container, this._pages, true); |  | ||||||
|     } |  | ||||||
|     // The algorithm in getVisibleElements doesn't work in all browsers and
 |  | ||||||
|     // configurations when presentation mode is active.
 |  | ||||||
|     let visible = []; |  | ||||||
|     let currentPage = this._pages[this._currentPageNumber - 1]; |  | ||||||
|     visible.push({ id: currentPage.id, view: currentPage, }); |  | ||||||
|     return { first: currentPage, last: currentPage, views: visible, }; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   cleanup() { |  | ||||||
|     for (let i = 0, ii = this._pages.length; i < ii; i++) { |  | ||||||
|       if (this._pages[i] && |  | ||||||
|           this._pages[i].renderingState !== RenderingStates.FINISHED) { |  | ||||||
|         this._pages[i].reset(); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @private |  | ||||||
|    */ |  | ||||||
|   _cancelRendering() { |  | ||||||
|     for (let i = 0, ii = this._pages.length; i < ii; i++) { |  | ||||||
|       if (this._pages[i]) { |  | ||||||
|         this._pages[i].cancelRendering(); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @param {PDFPageView} pageView |  | ||||||
|    * @returns {Promise} Returns a promise containing a {PDFPageProxy} object. |  | ||||||
|    * @private |  | ||||||
|    */ |  | ||||||
|   _ensurePdfPageLoaded(pageView) { |  | ||||||
|     if (pageView.pdfPage) { |  | ||||||
|       return Promise.resolve(pageView.pdfPage); |  | ||||||
|     } |  | ||||||
|     let pageNumber = pageView.id; |  | ||||||
|     if (this._pagesRequests[pageNumber]) { |  | ||||||
|       return this._pagesRequests[pageNumber]; |  | ||||||
|     } |  | ||||||
|     let promise = this.pdfDocument.getPage(pageNumber).then((pdfPage) => { |  | ||||||
|       if (!pageView.pdfPage) { |  | ||||||
|         pageView.setPdfPage(pdfPage); |  | ||||||
|       } |  | ||||||
|       this._pagesRequests[pageNumber] = null; |  | ||||||
|       return pdfPage; |  | ||||||
|     }).catch((reason) => { |  | ||||||
|       console.error('Unable to get page for page view', reason); |  | ||||||
|       // Page error -- there is nothing can be done.
 |  | ||||||
|       this._pagesRequests[pageNumber] = null; |  | ||||||
|     }); |  | ||||||
|     this._pagesRequests[pageNumber] = promise; |  | ||||||
|     return promise; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   forceRendering(currentlyVisiblePages) { |  | ||||||
|     let visiblePages = currentlyVisiblePages || this._getVisiblePages(); |  | ||||||
|     let pageView = this.renderingQueue.getHighestPriority(visiblePages, |  | ||||||
|                                                           this._pages, |  | ||||||
|                                                           this.scroll.down); |  | ||||||
|     if (pageView) { |  | ||||||
|       this._ensurePdfPageLoaded(pageView).then(() => { |  | ||||||
|         this.renderingQueue.renderView(pageView); |  | ||||||
|       }); |  | ||||||
|       return true; |  | ||||||
|     } |  | ||||||
|     return false; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   getPageTextContent(pageIndex) { |  | ||||||
|     return this.pdfDocument.getPage(pageIndex + 1).then(function(page) { |  | ||||||
|       return page.getTextContent({ |  | ||||||
|         normalizeWhitespace: true, |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @param {HTMLDivElement} textLayerDiv |  | ||||||
|    * @param {number} pageIndex |  | ||||||
|    * @param {PageViewport} viewport |  | ||||||
|    * @returns {TextLayerBuilder} |  | ||||||
|    */ |  | ||||||
|   createTextLayerBuilder(textLayerDiv, pageIndex, viewport, |  | ||||||
|                          enhanceTextSelection = false) { |  | ||||||
|     return new TextLayerBuilder({ |  | ||||||
|       textLayerDiv, |  | ||||||
|       eventBus: this.eventBus, |  | ||||||
|       pageIndex, |  | ||||||
|       viewport, |  | ||||||
|       findController: this.isInPresentationMode ? null : this.findController, |  | ||||||
|       enhanceTextSelection: this.isInPresentationMode ? false : |  | ||||||
|                                                         enhanceTextSelection, |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @param {HTMLDivElement} pageDiv |  | ||||||
|    * @param {PDFPage} pdfPage |  | ||||||
|    * @param {boolean} renderInteractiveForms |  | ||||||
|    * @param {IL10n} l10n |  | ||||||
|    * @returns {AnnotationLayerBuilder} |  | ||||||
|    */ |  | ||||||
|   createAnnotationLayerBuilder(pageDiv, pdfPage, renderInteractiveForms = false, |  | ||||||
|                                l10n = NullL10n) { |  | ||||||
|     return new AnnotationLayerBuilder({ |  | ||||||
|       pageDiv, |  | ||||||
|       pdfPage, |  | ||||||
|       renderInteractiveForms, |  | ||||||
|       linkService: this.linkService, |  | ||||||
|       downloadManager: this.downloadManager, |  | ||||||
|       l10n, |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   setFindController(findController) { |  | ||||||
|     this.findController = findController; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * @returns {boolean} Whether all pages of the PDF document have identical |  | ||||||
|    *                    widths and heights. |  | ||||||
|    */ |  | ||||||
|   get hasEqualPageSizes() { |  | ||||||
|     let firstPageView = this._pages[0]; |  | ||||||
|     for (let i = 1, ii = this._pages.length; i < ii; ++i) { |  | ||||||
|       let pageView = this._pages[i]; |  | ||||||
|       if (pageView.width !== firstPageView.width || |  | ||||||
|           pageView.height !== firstPageView.height) { |  | ||||||
|         return false; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     return true; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * Returns sizes of the pages. |  | ||||||
|    * @returns {Array} Array of objects with width/height/rotation fields. |  | ||||||
|    */ |  | ||||||
|   getPagesOverview() { |  | ||||||
|     let pagesOverview = this._pages.map(function(pageView) { |  | ||||||
|       let viewport = pageView.pdfPage.getViewport(1); |  | ||||||
|       return { |  | ||||||
|         width: viewport.width, |  | ||||||
|         height: viewport.height, |  | ||||||
|         rotation: viewport.rotation, |  | ||||||
|       }; |  | ||||||
|     }); |  | ||||||
|     if (!this.enablePrintAutoRotate) { |  | ||||||
|       return pagesOverview; |  | ||||||
|     } |  | ||||||
|     let isFirstPagePortrait = isPortraitOrientation(pagesOverview[0]); |  | ||||||
|     return pagesOverview.map(function (size) { |  | ||||||
|       if (isFirstPagePortrait === isPortraitOrientation(size)) { |  | ||||||
|         return size; |  | ||||||
|       } |  | ||||||
|       return { |  | ||||||
|         width: size.height, |  | ||||||
|         height: size.width, |  | ||||||
|         rotation: (size.rotation + 90) % 360, |  | ||||||
|       }; |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export { | export { | ||||||
|   PresentationModeState, |  | ||||||
|   PDFViewer, |   PDFViewer, | ||||||
| }; | }; | ||||||
|  | |||||||
| @ -25,6 +25,13 @@ const MAX_AUTO_SCALE = 1.25; | |||||||
| const SCROLLBAR_PADDING = 40; | const SCROLLBAR_PADDING = 40; | ||||||
| const VERTICAL_PADDING = 5; | const VERTICAL_PADDING = 5; | ||||||
| 
 | 
 | ||||||
|  | const PresentationModeState = { | ||||||
|  |   UNKNOWN: 0, | ||||||
|  |   NORMAL: 1, | ||||||
|  |   CHANGING: 2, | ||||||
|  |   FULLSCREEN: 3, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| const RendererType = { | const RendererType = { | ||||||
|   CANVAS: 'canvas', |   CANVAS: 'canvas', | ||||||
|   SVG: 'svg', |   SVG: 'svg', | ||||||
| @ -661,6 +668,7 @@ export { | |||||||
|   VERTICAL_PADDING, |   VERTICAL_PADDING, | ||||||
|   isValidRotation, |   isValidRotation, | ||||||
|   cloneObj, |   cloneObj, | ||||||
|  |   PresentationModeState, | ||||||
|   RendererType, |   RendererType, | ||||||
|   mozL10n, |   mozL10n, | ||||||
|   NullL10n, |   NullL10n, | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user