mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-26 12:01:08 +00:00 
			
		
		
		
	* Cleaning up public/ and documenting js/css libs. This commit mostly addresses #1484 by moving vendor'ed plugins into a vendor/ directory and documenting their upstream source and license in vendor/librejs.html. This also proves gitea is using only open source js/css libraries which helps toward reaching #1524. * Removing unused css file. The version of this file in use is located at: vendor/plugins/highlight/github.css * Cleaned up librejs.html and added javascript header A SafeJS function was added to templates/helper.go to allow keeping comments inside of javascript. A javascript comment was added in the header of templates/base/head.tmpl to mark all non-inline source as free. The librejs.html file was updated to meet the current librejs spec. I have now verified that the librejs plugin detects most of the scripts included in gitea and suspect the non-free detections are the result of a bug in the plugin. I believe this commit is enough to meet the C0.0 requirement of #1534. * Updating SafeJS function per lint suggestion * Added VERSIONS file, per request
		
			
				
	
	
		
			7952 lines
		
	
	
	
		
			254 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			7952 lines
		
	
	
	
		
			254 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /* Copyright 2012 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.
 | |
|  */
 | |
| /* globals PDFJS, PDFBug, FirefoxCom, Stats, ProgressBar,
 | |
|            DownloadManager, getFileName, getPDFFileNameFromURL,
 | |
|            PDFHistory, Preferences, SidebarView, ViewHistory, Stats,
 | |
|            PDFThumbnailViewer, URL, noContextMenuHandler, SecondaryToolbar,
 | |
|            PasswordPrompt, PDFPresentationMode, PDFDocumentProperties, HandTool,
 | |
|            Promise, PDFLinkService, PDFOutlineView, PDFAttachmentView,
 | |
|            OverlayManager, PDFFindController, PDFFindBar, PDFViewer,
 | |
|            PDFRenderingQueue, PresentationModeState, parseQueryString,
 | |
|            RenderingStates, UNKNOWN_SCALE, DEFAULT_SCALE_VALUE,
 | |
|            IGNORE_CURRENT_POSITION_ON_ZOOM: true */
 | |
| 
 | |
| 'use strict';
 | |
| 
 | |
| var DEFAULT_URL = 'compressed.tracemonkey-pldi-09.pdf';
 | |
| var DEFAULT_SCALE_DELTA = 1.1;
 | |
| var MIN_SCALE = 0.25;
 | |
| var MAX_SCALE = 10.0;
 | |
| var SCALE_SELECT_CONTAINER_PADDING = 8;
 | |
| var SCALE_SELECT_PADDING = 22;
 | |
| var PAGE_NUMBER_LOADING_INDICATOR = 'visiblePageIsLoading';
 | |
| var DISABLE_AUTO_FETCH_LOADING_BAR_TIMEOUT = 5000;
 | |
| 
 | |
| function configure(PDFJS) {
 | |
|   PDFJS.imageResourcesPath = './images/';
 | |
|   PDFJS.workerSrc = '../build/pdf.worker.js';
 | |
| }
 | |
| 
 | |
| var CSS_UNITS = 96.0 / 72.0;
 | |
| var DEFAULT_SCALE_VALUE = 'auto';
 | |
| var DEFAULT_SCALE = 1.0;
 | |
| var UNKNOWN_SCALE = 0;
 | |
| var MAX_AUTO_SCALE = 1.25;
 | |
| var SCROLLBAR_PADDING = 40;
 | |
| var VERTICAL_PADDING = 5;
 | |
| 
 | |
| function getFileName(url) {
 | |
|   var anchor = url.indexOf('#');
 | |
|   var query = url.indexOf('?');
 | |
|   var end = Math.min(
 | |
|     anchor > 0 ? anchor : url.length,
 | |
|     query > 0 ? query : url.length);
 | |
|   return url.substring(url.lastIndexOf('/', end) + 1, end);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Returns scale factor for the canvas. It makes sense for the HiDPI displays.
 | |
|  * @return {Object} The object with horizontal (sx) and vertical (sy)
 | |
|                     scales. The scaled property is set to false if scaling is
 | |
|                     not required, true otherwise.
 | |
|  */
 | |
| function getOutputScale(ctx) {
 | |
|   var devicePixelRatio = window.devicePixelRatio || 1;
 | |
|   var backingStoreRatio = ctx.webkitBackingStorePixelRatio ||
 | |
|                           ctx.mozBackingStorePixelRatio ||
 | |
|                           ctx.msBackingStorePixelRatio ||
 | |
|                           ctx.oBackingStorePixelRatio ||
 | |
|                           ctx.backingStorePixelRatio || 1;
 | |
|   var pixelRatio = devicePixelRatio / backingStoreRatio;
 | |
|   return {
 | |
|     sx: pixelRatio,
 | |
|     sy: pixelRatio,
 | |
|     scaled: pixelRatio !== 1
 | |
|   };
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Scrolls specified element into view of its parent.
 | |
|  * @param {Object} element - The element to be visible.
 | |
|  * @param {Object} spot - An object with optional top and left properties,
 | |
|  *   specifying the offset from the top left edge.
 | |
|  * @param {boolean} skipOverflowHiddenElements - Ignore elements that have
 | |
|  *   the CSS rule `overflow: hidden;` set. The default is false.
 | |
|  */
 | |
| function scrollIntoView(element, spot, skipOverflowHiddenElements) {
 | |
|   // Assuming offsetParent is available (it's not available when viewer is in
 | |
|   // hidden iframe or object). We have to scroll: if the offsetParent is not set
 | |
|   // producing the error. See also animationStartedClosure.
 | |
|   var parent = element.offsetParent;
 | |
|   if (!parent) {
 | |
|     console.error('offsetParent is not set -- cannot scroll');
 | |
|     return;
 | |
|   }
 | |
|   var checkOverflow = skipOverflowHiddenElements || false;
 | |
|   var offsetY = element.offsetTop + element.clientTop;
 | |
|   var offsetX = element.offsetLeft + element.clientLeft;
 | |
|   while (parent.clientHeight === parent.scrollHeight ||
 | |
|          (checkOverflow && getComputedStyle(parent).overflow === 'hidden')) {
 | |
|     if (parent.dataset._scaleY) {
 | |
|       offsetY /= parent.dataset._scaleY;
 | |
|       offsetX /= parent.dataset._scaleX;
 | |
|     }
 | |
|     offsetY += parent.offsetTop;
 | |
|     offsetX += parent.offsetLeft;
 | |
|     parent = parent.offsetParent;
 | |
|     if (!parent) {
 | |
|       return; // no need to scroll
 | |
|     }
 | |
|   }
 | |
|   if (spot) {
 | |
|     if (spot.top !== undefined) {
 | |
|       offsetY += spot.top;
 | |
|     }
 | |
|     if (spot.left !== undefined) {
 | |
|       offsetX += spot.left;
 | |
|       parent.scrollLeft = offsetX;
 | |
|     }
 | |
|   }
 | |
|   parent.scrollTop = offsetY;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Helper function to start monitoring the scroll event and converting them into
 | |
|  * PDF.js friendly one: with scroll debounce and scroll direction.
 | |
|  */
 | |
| function watchScroll(viewAreaElement, callback) {
 | |
|   var debounceScroll = function debounceScroll(evt) {
 | |
|     if (rAF) {
 | |
|       return;
 | |
|     }
 | |
|     // schedule an invocation of scroll for next animation frame.
 | |
|     rAF = window.requestAnimationFrame(function viewAreaElementScrolled() {
 | |
|       rAF = null;
 | |
| 
 | |
|       var currentY = viewAreaElement.scrollTop;
 | |
|       var lastY = state.lastY;
 | |
|       if (currentY !== lastY) {
 | |
|         state.down = currentY > lastY;
 | |
|       }
 | |
|       state.lastY = currentY;
 | |
|       callback(state);
 | |
|     });
 | |
|   };
 | |
| 
 | |
|   var state = {
 | |
|     down: true,
 | |
|     lastY: viewAreaElement.scrollTop,
 | |
|     _eventHandler: debounceScroll
 | |
|   };
 | |
| 
 | |
|   var rAF = null;
 | |
|   viewAreaElement.addEventListener('scroll', debounceScroll, true);
 | |
|   return state;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Helper function to parse query string (e.g. ?param1=value&parm2=...).
 | |
|  */
 | |
| function parseQueryString(query) {
 | |
|   var parts = query.split('&');
 | |
|   var params = {};
 | |
|   for (var i = 0, ii = parts.length; i < ii; ++i) {
 | |
|     var param = parts[i].split('=');
 | |
|     var key = param[0].toLowerCase();
 | |
|     var value = param.length > 1 ? param[1] : null;
 | |
|     params[decodeURIComponent(key)] = decodeURIComponent(value);
 | |
|   }
 | |
|   return params;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Use binary search to find the index of the first item in a given array which
 | |
|  * passes a given condition. The items are expected to be sorted in the sense
 | |
|  * that if the condition is true for one item in the array, then it is also true
 | |
|  * for all following items.
 | |
|  *
 | |
|  * @returns {Number} Index of the first array element to pass the test,
 | |
|  *                   or |items.length| if no such element exists.
 | |
|  */
 | |
| function binarySearchFirstItem(items, condition) {
 | |
|   var minIndex = 0;
 | |
|   var maxIndex = items.length - 1;
 | |
| 
 | |
|   if (items.length === 0 || !condition(items[maxIndex])) {
 | |
|     return items.length;
 | |
|   }
 | |
|   if (condition(items[minIndex])) {
 | |
|     return minIndex;
 | |
|   }
 | |
| 
 | |
|   while (minIndex < maxIndex) {
 | |
|     var currentIndex = (minIndex + maxIndex) >> 1;
 | |
|     var currentItem = items[currentIndex];
 | |
|     if (condition(currentItem)) {
 | |
|       maxIndex = currentIndex;
 | |
|     } else {
 | |
|       minIndex = currentIndex + 1;
 | |
|     }
 | |
|   }
 | |
|   return minIndex; /* === maxIndex */
 | |
| }
 | |
| 
 | |
| /**
 | |
|  *  Approximates float number as a fraction using Farey sequence (max order
 | |
|  *  of 8).
 | |
|  *  @param {number} x - Positive float number.
 | |
|  *  @returns {Array} Estimated fraction: the first array item is a numerator,
 | |
|  *                   the second one is a denominator.
 | |
|  */
 | |
| function approximateFraction(x) {
 | |
|   // Fast paths for int numbers or their inversions.
 | |
|   if (Math.floor(x) === x) {
 | |
|     return [x, 1];
 | |
|   }
 | |
|   var xinv = 1 / x;
 | |
|   var limit = 8;
 | |
|   if (xinv > limit) {
 | |
|     return [1, limit];
 | |
|   } else  if (Math.floor(xinv) === xinv) {
 | |
|     return [1, xinv];
 | |
|   }
 | |
| 
 | |
|   var x_ = x > 1 ? xinv : x;
 | |
|   // a/b and c/d are neighbours in Farey sequence.
 | |
|   var a = 0, b = 1, c = 1, d = 1;
 | |
|   // Limiting search to order 8.
 | |
|   while (true) {
 | |
|     // Generating next term in sequence (order of q).
 | |
|     var p = a + c, q = b + d;
 | |
|     if (q > limit) {
 | |
|       break;
 | |
|     }
 | |
|     if (x_ <= p / q) {
 | |
|       c = p; d = q;
 | |
|     } else {
 | |
|       a = p; b = q;
 | |
|     }
 | |
|   }
 | |
|   // Select closest of the neighbours to x.
 | |
|   if (x_ - a / b < c / d - x_) {
 | |
|     return x_ === x ? [a, b] : [b, a];
 | |
|   } else {
 | |
|     return x_ === x ? [c, d] : [d, c];
 | |
|   }
 | |
| }
 | |
| 
 | |
| function roundToDivide(x, div) {
 | |
|   var r = x % div;
 | |
|   return r === 0 ? x : Math.round(x - r + div);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Generic helper to find out what elements are visible within a scroll pane.
 | |
|  */
 | |
| function getVisibleElements(scrollEl, views, sortByVisibility) {
 | |
|   var top = scrollEl.scrollTop, bottom = top + scrollEl.clientHeight;
 | |
|   var left = scrollEl.scrollLeft, right = left + scrollEl.clientWidth;
 | |
| 
 | |
|   function isElementBottomBelowViewTop(view) {
 | |
|     var element = view.div;
 | |
|     var elementBottom =
 | |
|       element.offsetTop + element.clientTop + element.clientHeight;
 | |
|     return elementBottom > top;
 | |
|   }
 | |
| 
 | |
|   var visible = [], view, element;
 | |
|   var currentHeight, viewHeight, hiddenHeight, percentHeight;
 | |
|   var currentWidth, viewWidth;
 | |
|   var firstVisibleElementInd = (views.length === 0) ? 0 :
 | |
|     binarySearchFirstItem(views, isElementBottomBelowViewTop);
 | |
| 
 | |
|   for (var i = firstVisibleElementInd, ii = views.length; i < ii; i++) {
 | |
|     view = views[i];
 | |
|     element = view.div;
 | |
|     currentHeight = element.offsetTop + element.clientTop;
 | |
|     viewHeight = element.clientHeight;
 | |
| 
 | |
|     if (currentHeight > bottom) {
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     currentWidth = element.offsetLeft + element.clientLeft;
 | |
|     viewWidth = element.clientWidth;
 | |
|     if (currentWidth + viewWidth < left || currentWidth > right) {
 | |
|       continue;
 | |
|     }
 | |
|     hiddenHeight = Math.max(0, top - currentHeight) +
 | |
|       Math.max(0, currentHeight + viewHeight - bottom);
 | |
|     percentHeight = ((viewHeight - hiddenHeight) * 100 / viewHeight) | 0;
 | |
| 
 | |
|     visible.push({
 | |
|       id: view.id,
 | |
|       x: currentWidth,
 | |
|       y: currentHeight,
 | |
|       view: view,
 | |
|       percent: percentHeight
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   var first = visible[0];
 | |
|   var last = visible[visible.length - 1];
 | |
| 
 | |
|   if (sortByVisibility) {
 | |
|     visible.sort(function(a, b) {
 | |
|       var pc = a.percent - b.percent;
 | |
|       if (Math.abs(pc) > 0.001) {
 | |
|         return -pc;
 | |
|       }
 | |
|       return a.id - b.id; // ensure stability
 | |
|     });
 | |
|   }
 | |
|   return {first: first, last: last, views: visible};
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Event handler to suppress context menu.
 | |
|  */
 | |
| function noContextMenuHandler(e) {
 | |
|   e.preventDefault();
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Returns the filename or guessed filename from the url (see issue 3455).
 | |
|  * url {String} The original PDF location.
 | |
|  * @return {String} Guessed PDF file name.
 | |
|  */
 | |
| function getPDFFileNameFromURL(url) {
 | |
|   var reURI = /^(?:([^:]+:)?\/\/[^\/]+)?([^?#]*)(\?[^#]*)?(#.*)?$/;
 | |
|   //            SCHEME      HOST         1.PATH  2.QUERY   3.REF
 | |
|   // Pattern to get last matching NAME.pdf
 | |
|   var reFilename = /[^\/?#=]+\.pdf\b(?!.*\.pdf\b)/i;
 | |
|   var splitURI = reURI.exec(url);
 | |
|   var suggestedFilename = reFilename.exec(splitURI[1]) ||
 | |
|                            reFilename.exec(splitURI[2]) ||
 | |
|                            reFilename.exec(splitURI[3]);
 | |
|   if (suggestedFilename) {
 | |
|     suggestedFilename = suggestedFilename[0];
 | |
|     if (suggestedFilename.indexOf('%') !== -1) {
 | |
|       // URL-encoded %2Fpath%2Fto%2Ffile.pdf should be file.pdf
 | |
|       try {
 | |
|         suggestedFilename =
 | |
|           reFilename.exec(decodeURIComponent(suggestedFilename))[0];
 | |
|       } catch(e) { // Possible (extremely rare) errors:
 | |
|         // URIError "Malformed URI", e.g. for "%AA.pdf"
 | |
|         // TypeError "null has no properties", e.g. for "%2F.pdf"
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   return suggestedFilename || 'document.pdf';
 | |
| }
 | |
| 
 | |
| var ProgressBar = (function ProgressBarClosure() {
 | |
| 
 | |
|   function clamp(v, min, max) {
 | |
|     return Math.min(Math.max(v, min), max);
 | |
|   }
 | |
| 
 | |
|   function ProgressBar(id, opts) {
 | |
|     this.visible = true;
 | |
| 
 | |
|     // Fetch the sub-elements for later.
 | |
|     this.div = document.querySelector(id + ' .progress');
 | |
| 
 | |
|     // Get the loading bar element, so it can be resized to fit the viewer.
 | |
|     this.bar = this.div.parentNode;
 | |
| 
 | |
|     // Get options, with sensible defaults.
 | |
|     this.height = opts.height || 100;
 | |
|     this.width = opts.width || 100;
 | |
|     this.units = opts.units || '%';
 | |
| 
 | |
|     // Initialize heights.
 | |
|     this.div.style.height = this.height + this.units;
 | |
|     this.percent = 0;
 | |
|   }
 | |
| 
 | |
|   ProgressBar.prototype = {
 | |
| 
 | |
|     updateBar: function ProgressBar_updateBar() {
 | |
|       if (this._indeterminate) {
 | |
|         this.div.classList.add('indeterminate');
 | |
|         this.div.style.width = this.width + this.units;
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       this.div.classList.remove('indeterminate');
 | |
|       var progressSize = this.width * this._percent / 100;
 | |
|       this.div.style.width = progressSize + this.units;
 | |
|     },
 | |
| 
 | |
|     get percent() {
 | |
|       return this._percent;
 | |
|     },
 | |
| 
 | |
|     set percent(val) {
 | |
|       this._indeterminate = isNaN(val);
 | |
|       this._percent = clamp(val, 0, 100);
 | |
|       this.updateBar();
 | |
|     },
 | |
| 
 | |
|     setWidth: function ProgressBar_setWidth(viewer) {
 | |
|       if (viewer) {
 | |
|         var container = viewer.parentNode;
 | |
|         var scrollbarWidth = container.offsetWidth - viewer.offsetWidth;
 | |
|         if (scrollbarWidth > 0) {
 | |
|           this.bar.setAttribute('style', 'width: calc(100% - ' +
 | |
|                                          scrollbarWidth + 'px);');
 | |
|         }
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     hide: function ProgressBar_hide() {
 | |
|       if (!this.visible) {
 | |
|         return;
 | |
|       }
 | |
|       this.visible = false;
 | |
|       this.bar.classList.add('hidden');
 | |
|       document.body.classList.remove('loadingInProgress');
 | |
|     },
 | |
| 
 | |
|     show: function ProgressBar_show() {
 | |
|       if (this.visible) {
 | |
|         return;
 | |
|       }
 | |
|       this.visible = true;
 | |
|       document.body.classList.add('loadingInProgress');
 | |
|       this.bar.classList.remove('hidden');
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   return ProgressBar;
 | |
| })();
 | |
| 
 | |
| 
 | |
| 
 | |
| var DEFAULT_PREFERENCES = {
 | |
|   showPreviousViewOnLoad: true,
 | |
|   defaultZoomValue: '',
 | |
|   sidebarViewOnLoad: 0,
 | |
|   enableHandToolOnLoad: false,
 | |
|   enableWebGL: false,
 | |
|   pdfBugEnabled: false,
 | |
|   disableRange: false,
 | |
|   disableStream: false,
 | |
|   disableAutoFetch: false,
 | |
|   disableFontFace: false,
 | |
|   disableTextLayer: false,
 | |
|   useOnlyCssZoom: false,
 | |
|   externalLinkTarget: 0,
 | |
| };
 | |
| 
 | |
| 
 | |
| var SidebarView = {
 | |
|   NONE: 0,
 | |
|   THUMBS: 1,
 | |
|   OUTLINE: 2,
 | |
|   ATTACHMENTS: 3
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * Preferences - Utility for storing persistent settings.
 | |
|  *   Used for settings that should be applied to all opened documents,
 | |
|  *   or every time the viewer is loaded.
 | |
|  */
 | |
| var Preferences = {
 | |
|   prefs: Object.create(DEFAULT_PREFERENCES),
 | |
|   isInitializedPromiseResolved: false,
 | |
|   initializedPromise: null,
 | |
| 
 | |
|   /**
 | |
|    * Initialize and fetch the current preference values from storage.
 | |
|    * @return {Promise} A promise that is resolved when the preferences
 | |
|    *                   have been initialized.
 | |
|    */
 | |
|   initialize: function preferencesInitialize() {
 | |
|     return this.initializedPromise =
 | |
|         this._readFromStorage(DEFAULT_PREFERENCES).then(function(prefObj) {
 | |
|       this.isInitializedPromiseResolved = true;
 | |
|       if (prefObj) {
 | |
|         this.prefs = prefObj;
 | |
|       }
 | |
|     }.bind(this));
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Stub function for writing preferences to storage.
 | |
|    * NOTE: This should be overridden by a build-specific function defined below.
 | |
|    * @param {Object} prefObj The preferences that should be written to storage.
 | |
|    * @return {Promise} A promise that is resolved when the preference values
 | |
|    *                   have been written.
 | |
|    */
 | |
|   _writeToStorage: function preferences_writeToStorage(prefObj) {
 | |
|     return Promise.resolve();
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Stub function for reading preferences from storage.
 | |
|    * NOTE: This should be overridden by a build-specific function defined below.
 | |
|    * @param {Object} prefObj The preferences that should be read from storage.
 | |
|    * @return {Promise} A promise that is resolved with an {Object} containing
 | |
|    *                   the preferences that have been read.
 | |
|    */
 | |
|   _readFromStorage: function preferences_readFromStorage(prefObj) {
 | |
|     return Promise.resolve();
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Reset the preferences to their default values and update storage.
 | |
|    * @return {Promise} A promise that is resolved when the preference values
 | |
|    *                   have been reset.
 | |
|    */
 | |
|   reset: function preferencesReset() {
 | |
|     return this.initializedPromise.then(function() {
 | |
|       this.prefs = Object.create(DEFAULT_PREFERENCES);
 | |
|       return this._writeToStorage(DEFAULT_PREFERENCES);
 | |
|     }.bind(this));
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Replace the current preference values with the ones from storage.
 | |
|    * @return {Promise} A promise that is resolved when the preference values
 | |
|    *                   have been updated.
 | |
|    */
 | |
|   reload: function preferencesReload() {
 | |
|     return this.initializedPromise.then(function () {
 | |
|       this._readFromStorage(DEFAULT_PREFERENCES).then(function(prefObj) {
 | |
|         if (prefObj) {
 | |
|           this.prefs = prefObj;
 | |
|         }
 | |
|       }.bind(this));
 | |
|     }.bind(this));
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Set the value of a preference.
 | |
|    * @param {string} name The name of the preference that should be changed.
 | |
|    * @param {boolean|number|string} value The new value of the preference.
 | |
|    * @return {Promise} A promise that is resolved when the value has been set,
 | |
|    *                   provided that the preference exists and the types match.
 | |
|    */
 | |
|   set: function preferencesSet(name, value) {
 | |
|     return this.initializedPromise.then(function () {
 | |
|       if (DEFAULT_PREFERENCES[name] === undefined) {
 | |
|         throw new Error('preferencesSet: \'' + name + '\' is undefined.');
 | |
|       } else if (value === undefined) {
 | |
|         throw new Error('preferencesSet: no value is specified.');
 | |
|       }
 | |
|       var valueType = typeof value;
 | |
|       var defaultType = typeof DEFAULT_PREFERENCES[name];
 | |
| 
 | |
|       if (valueType !== defaultType) {
 | |
|         if (valueType === 'number' && defaultType === 'string') {
 | |
|           value = value.toString();
 | |
|         } else {
 | |
|           throw new Error('Preferences_set: \'' + value + '\' is a \"' +
 | |
|                           valueType + '\", expected \"' + defaultType + '\".');
 | |
|         }
 | |
|       } else {
 | |
|         if (valueType === 'number' && (value | 0) !== value) {
 | |
|           throw new Error('Preferences_set: \'' + value +
 | |
|                           '\' must be an \"integer\".');
 | |
|         }
 | |
|       }
 | |
|       this.prefs[name] = value;
 | |
|       return this._writeToStorage(this.prefs);
 | |
|     }.bind(this));
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Get the value of a preference.
 | |
|    * @param {string} name The name of the preference whose value is requested.
 | |
|    * @return {Promise} A promise that is resolved with a {boolean|number|string}
 | |
|    *                   containing the value of the preference.
 | |
|    */
 | |
|   get: function preferencesGet(name) {
 | |
|     return this.initializedPromise.then(function () {
 | |
|       var defaultValue = DEFAULT_PREFERENCES[name];
 | |
| 
 | |
|       if (defaultValue === undefined) {
 | |
|         throw new Error('preferencesGet: \'' + name + '\' is undefined.');
 | |
|       } else {
 | |
|         var prefValue = this.prefs[name];
 | |
| 
 | |
|         if (prefValue !== undefined) {
 | |
|           return prefValue;
 | |
|         }
 | |
|       }
 | |
|       return defaultValue;
 | |
|     }.bind(this));
 | |
|   }
 | |
| };
 | |
| 
 | |
| 
 | |
| Preferences._writeToStorage = function (prefObj) {
 | |
|   return new Promise(function (resolve) {
 | |
|     localStorage.setItem('pdfjs.preferences', JSON.stringify(prefObj));
 | |
|     resolve();
 | |
|   });
 | |
| };
 | |
| 
 | |
| Preferences._readFromStorage = function (prefObj) {
 | |
|   return new Promise(function (resolve) {
 | |
|     var readPrefs = JSON.parse(localStorage.getItem('pdfjs.preferences'));
 | |
|     resolve(readPrefs);
 | |
|   });
 | |
| };
 | |
| 
 | |
| 
 | |
| (function mozPrintCallbackPolyfillClosure() {
 | |
|   if ('mozPrintCallback' in document.createElement('canvas')) {
 | |
|     return;
 | |
|   }
 | |
|   // Cause positive result on feature-detection:
 | |
|   HTMLCanvasElement.prototype.mozPrintCallback = undefined;
 | |
| 
 | |
|   var canvases;   // During print task: non-live NodeList of <canvas> elements
 | |
|   var index;      // Index of <canvas> element that is being processed
 | |
| 
 | |
|   var print = window.print;
 | |
|   window.print = function print() {
 | |
|     if (canvases) {
 | |
|       console.warn('Ignored window.print() because of a pending print job.');
 | |
|       return;
 | |
|     }
 | |
|     try {
 | |
|       dispatchEvent('beforeprint');
 | |
|     } finally {
 | |
|       canvases = document.querySelectorAll('canvas');
 | |
|       index = -1;
 | |
|       next();
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   function dispatchEvent(eventType) {
 | |
|     var event = document.createEvent('CustomEvent');
 | |
|     event.initCustomEvent(eventType, false, false, 'custom');
 | |
|     window.dispatchEvent(event);
 | |
|   }
 | |
| 
 | |
|   function next() {
 | |
|     if (!canvases) {
 | |
|       return; // Print task cancelled by user (state reset in abort())
 | |
|     }
 | |
| 
 | |
|     renderProgress();
 | |
|     if (++index < canvases.length) {
 | |
|       var canvas = canvases[index];
 | |
|       if (typeof canvas.mozPrintCallback === 'function') {
 | |
|         canvas.mozPrintCallback({
 | |
|           context: canvas.getContext('2d'),
 | |
|           abort: abort,
 | |
|           done: next
 | |
|         });
 | |
|       } else {
 | |
|         next();
 | |
|       }
 | |
|     } else {
 | |
|       renderProgress();
 | |
|       print.call(window);
 | |
|       setTimeout(abort, 20); // Tidy-up
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   function abort() {
 | |
|     if (canvases) {
 | |
|       canvases = null;
 | |
|       renderProgress();
 | |
|       dispatchEvent('afterprint');
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   function renderProgress() {
 | |
|     var progressContainer = document.getElementById('mozPrintCallback-shim');
 | |
|     if (canvases && canvases.length) {
 | |
|       var progress = Math.round(100 * index / canvases.length);
 | |
|       var progressBar = progressContainer.querySelector('progress');
 | |
|       var progressPerc = progressContainer.querySelector('.relative-progress');
 | |
|       progressBar.value = progress;
 | |
|       progressPerc.textContent = progress + '%';
 | |
|       progressContainer.removeAttribute('hidden');
 | |
|       progressContainer.onclick = abort;
 | |
|     } else {
 | |
|       progressContainer.setAttribute('hidden', '');
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   var hasAttachEvent = !!document.attachEvent;
 | |
| 
 | |
|   window.addEventListener('keydown', function(event) {
 | |
|     // Intercept Cmd/Ctrl + P in all browsers.
 | |
|     // Also intercept Cmd/Ctrl + Shift + P in Chrome and Opera
 | |
|     if (event.keyCode === 80/*P*/ && (event.ctrlKey || event.metaKey) &&
 | |
|         !event.altKey && (!event.shiftKey || window.chrome || window.opera)) {
 | |
|       window.print();
 | |
|       if (hasAttachEvent) {
 | |
|         // Only attachEvent can cancel Ctrl + P dialog in IE <=10
 | |
|         // attachEvent is gone in IE11, so the dialog will re-appear in IE11.
 | |
|         return;
 | |
|       }
 | |
|       event.preventDefault();
 | |
|       if (event.stopImmediatePropagation) {
 | |
|         event.stopImmediatePropagation();
 | |
|       } else {
 | |
|         event.stopPropagation();
 | |
|       }
 | |
|       return;
 | |
|     }
 | |
|     if (event.keyCode === 27 && canvases) { // Esc
 | |
|       abort();
 | |
|     }
 | |
|   }, true);
 | |
|   if (hasAttachEvent) {
 | |
|     document.attachEvent('onkeydown', function(event) {
 | |
|       event = event || window.event;
 | |
|       if (event.keyCode === 80/*P*/ && event.ctrlKey) {
 | |
|         event.keyCode = 0;
 | |
|         return false;
 | |
|       }
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   if ('onbeforeprint' in window) {
 | |
|     // Do not propagate before/afterprint events when they are not triggered
 | |
|     // from within this polyfill. (FF/IE).
 | |
|     var stopPropagationIfNeeded = function(event) {
 | |
|       if (event.detail !== 'custom' && event.stopImmediatePropagation) {
 | |
|         event.stopImmediatePropagation();
 | |
|       }
 | |
|     };
 | |
|     window.addEventListener('beforeprint', stopPropagationIfNeeded, false);
 | |
|     window.addEventListener('afterprint', stopPropagationIfNeeded, false);
 | |
|   }
 | |
| })();
 | |
| 
 | |
| 
 | |
| 
 | |
| var DownloadManager = (function DownloadManagerClosure() {
 | |
| 
 | |
|   function download(blobUrl, filename) {
 | |
|     var a = document.createElement('a');
 | |
|     if (a.click) {
 | |
|       // Use a.click() if available. Otherwise, Chrome might show
 | |
|       // "Unsafe JavaScript attempt to initiate a navigation change
 | |
|       //  for frame with URL" and not open the PDF at all.
 | |
|       // Supported by (not mentioned = untested):
 | |
|       // - Firefox 6 - 19 (4- does not support a.click, 5 ignores a.click)
 | |
|       // - Chrome 19 - 26 (18- does not support a.click)
 | |
|       // - Opera 9 - 12.15
 | |
|       // - Internet Explorer 6 - 10
 | |
|       // - Safari 6 (5.1- does not support a.click)
 | |
|       a.href = blobUrl;
 | |
|       a.target = '_parent';
 | |
|       // Use a.download if available. This increases the likelihood that
 | |
|       // the file is downloaded instead of opened by another PDF plugin.
 | |
|       if ('download' in a) {
 | |
|         a.download = filename;
 | |
|       }
 | |
|       // <a> must be in the document for IE and recent Firefox versions.
 | |
|       // (otherwise .click() is ignored)
 | |
|       (document.body || document.documentElement).appendChild(a);
 | |
|       a.click();
 | |
|       a.parentNode.removeChild(a);
 | |
|     } else {
 | |
|       if (window.top === window &&
 | |
|           blobUrl.split('#')[0] === window.location.href.split('#')[0]) {
 | |
|         // If _parent == self, then opening an identical URL with different
 | |
|         // location hash will only cause a navigation, not a download.
 | |
|         var padCharacter = blobUrl.indexOf('?') === -1 ? '?' : '&';
 | |
|         blobUrl = blobUrl.replace(/#|$/, padCharacter + '$&');
 | |
|       }
 | |
|       window.open(blobUrl, '_parent');
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   function DownloadManager() {}
 | |
| 
 | |
|   DownloadManager.prototype = {
 | |
|     downloadUrl: function DownloadManager_downloadUrl(url, filename) {
 | |
|       if (!PDFJS.isValidUrl(url, true)) {
 | |
|         return; // restricted/invalid URL
 | |
|       }
 | |
| 
 | |
|       download(url + '#pdfjs.action=download', filename);
 | |
|     },
 | |
| 
 | |
|     downloadData: function DownloadManager_downloadData(data, filename,
 | |
|                                                         contentType) {
 | |
|       if (navigator.msSaveBlob) { // IE10 and above
 | |
|         return navigator.msSaveBlob(new Blob([data], { type: contentType }),
 | |
|                                     filename);
 | |
|       }
 | |
| 
 | |
|       var blobUrl = PDFJS.createObjectURL(data, contentType);
 | |
|       download(blobUrl, filename);
 | |
|     },
 | |
| 
 | |
|     download: function DownloadManager_download(blob, url, filename) {
 | |
|       if (!URL) {
 | |
|         // URL.createObjectURL is not supported
 | |
|         this.downloadUrl(url, filename);
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       if (navigator.msSaveBlob) {
 | |
|         // IE10 / IE11
 | |
|         if (!navigator.msSaveBlob(blob, filename)) {
 | |
|           this.downloadUrl(url, filename);
 | |
|         }
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       var blobUrl = URL.createObjectURL(blob);
 | |
|       download(blobUrl, filename);
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   return DownloadManager;
 | |
| })();
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| var DEFAULT_VIEW_HISTORY_CACHE_SIZE = 20;
 | |
| 
 | |
| /**
 | |
|  * View History - This is a utility for saving various view parameters for
 | |
|  *                recently opened files.
 | |
|  *
 | |
|  * The way that the view parameters are stored depends on how PDF.js is built,
 | |
|  * for 'node make <flag>' the following cases exist:
 | |
|  *  - FIREFOX or MOZCENTRAL - uses sessionStorage.
 | |
|  *  - GENERIC or CHROME     - uses localStorage, if it is available.
 | |
|  */
 | |
| var ViewHistory = (function ViewHistoryClosure() {
 | |
|   function ViewHistory(fingerprint, cacheSize) {
 | |
|     this.fingerprint = fingerprint;
 | |
|     this.cacheSize = cacheSize || DEFAULT_VIEW_HISTORY_CACHE_SIZE;
 | |
|     this.isInitializedPromiseResolved = false;
 | |
|     this.initializedPromise =
 | |
|         this._readFromStorage().then(function (databaseStr) {
 | |
|       this.isInitializedPromiseResolved = true;
 | |
| 
 | |
|       var database = JSON.parse(databaseStr || '{}');
 | |
|       if (!('files' in database)) {
 | |
|         database.files = [];
 | |
|       }
 | |
|       if (database.files.length >= this.cacheSize) {
 | |
|         database.files.shift();
 | |
|       }
 | |
|       var index;
 | |
|       for (var i = 0, length = database.files.length; i < length; i++) {
 | |
|         var branch = database.files[i];
 | |
|         if (branch.fingerprint === this.fingerprint) {
 | |
|           index = i;
 | |
|           break;
 | |
|         }
 | |
|       }
 | |
|       if (typeof index !== 'number') {
 | |
|         index = database.files.push({fingerprint: this.fingerprint}) - 1;
 | |
|       }
 | |
|       this.file = database.files[index];
 | |
|       this.database = database;
 | |
|     }.bind(this));
 | |
|   }
 | |
| 
 | |
|   ViewHistory.prototype = {
 | |
|     _writeToStorage: function ViewHistory_writeToStorage() {
 | |
|       return new Promise(function (resolve) {
 | |
|         var databaseStr = JSON.stringify(this.database);
 | |
| 
 | |
| 
 | |
|         localStorage.setItem('database', databaseStr);
 | |
|         resolve();
 | |
|       }.bind(this));
 | |
|     },
 | |
| 
 | |
|     _readFromStorage: function ViewHistory_readFromStorage() {
 | |
|       return new Promise(function (resolve) {
 | |
| 
 | |
|         resolve(localStorage.getItem('database'));
 | |
|       });
 | |
|     },
 | |
| 
 | |
|     set: function ViewHistory_set(name, val) {
 | |
|       if (!this.isInitializedPromiseResolved) {
 | |
|         return;
 | |
|       }
 | |
|       this.file[name] = val;
 | |
|       return this._writeToStorage();
 | |
|     },
 | |
| 
 | |
|     setMultiple: function ViewHistory_setMultiple(properties) {
 | |
|       if (!this.isInitializedPromiseResolved) {
 | |
|         return;
 | |
|       }
 | |
|       for (var name in properties) {
 | |
|         this.file[name] = properties[name];
 | |
|       }
 | |
|       return this._writeToStorage();
 | |
|     },
 | |
| 
 | |
|     get: function ViewHistory_get(name, defaultValue) {
 | |
|       if (!this.isInitializedPromiseResolved) {
 | |
|         return defaultValue;
 | |
|       }
 | |
|       return this.file[name] || defaultValue;
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   return ViewHistory;
 | |
| })();
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Creates a "search bar" given a set of DOM elements that act as controls
 | |
|  * for searching or for setting search preferences in the UI. This object
 | |
|  * also sets up the appropriate events for the controls. Actual searching
 | |
|  * is done by PDFFindController.
 | |
|  */
 | |
| var PDFFindBar = (function PDFFindBarClosure() {
 | |
|   function PDFFindBar(options) {
 | |
|     this.opened = false;
 | |
|     this.bar = options.bar || null;
 | |
|     this.toggleButton = options.toggleButton || null;
 | |
|     this.findField = options.findField || null;
 | |
|     this.highlightAll = options.highlightAllCheckbox || null;
 | |
|     this.caseSensitive = options.caseSensitiveCheckbox || null;
 | |
|     this.findMsg = options.findMsg || null;
 | |
|     this.findResultsCount = options.findResultsCount || null;
 | |
|     this.findStatusIcon = options.findStatusIcon || null;
 | |
|     this.findPreviousButton = options.findPreviousButton || null;
 | |
|     this.findNextButton = options.findNextButton || null;
 | |
|     this.findController = options.findController || null;
 | |
| 
 | |
|     if (this.findController === null) {
 | |
|       throw new Error('PDFFindBar cannot be used without a ' +
 | |
|                       'PDFFindController instance.');
 | |
|     }
 | |
| 
 | |
|     // Add event listeners to the DOM elements.
 | |
|     var self = this;
 | |
|     this.toggleButton.addEventListener('click', function() {
 | |
|       self.toggle();
 | |
|     });
 | |
| 
 | |
|     this.findField.addEventListener('input', function() {
 | |
|       self.dispatchEvent('');
 | |
|     });
 | |
| 
 | |
|     this.bar.addEventListener('keydown', function(evt) {
 | |
|       switch (evt.keyCode) {
 | |
|         case 13: // Enter
 | |
|           if (evt.target === self.findField) {
 | |
|             self.dispatchEvent('again', evt.shiftKey);
 | |
|           }
 | |
|           break;
 | |
|         case 27: // Escape
 | |
|           self.close();
 | |
|           break;
 | |
|       }
 | |
|     });
 | |
| 
 | |
|     this.findPreviousButton.addEventListener('click', function() {
 | |
|       self.dispatchEvent('again', true);
 | |
|     });
 | |
| 
 | |
|     this.findNextButton.addEventListener('click', function() {
 | |
|       self.dispatchEvent('again', false);
 | |
|     });
 | |
| 
 | |
|     this.highlightAll.addEventListener('click', function() {
 | |
|       self.dispatchEvent('highlightallchange');
 | |
|     });
 | |
| 
 | |
|     this.caseSensitive.addEventListener('click', function() {
 | |
|       self.dispatchEvent('casesensitivitychange');
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   PDFFindBar.prototype = {
 | |
|     dispatchEvent: function PDFFindBar_dispatchEvent(type, findPrev) {
 | |
|       var event = document.createEvent('CustomEvent');
 | |
|       event.initCustomEvent('find' + type, true, true, {
 | |
|         query: this.findField.value,
 | |
|         caseSensitive: this.caseSensitive.checked,
 | |
|         highlightAll: this.highlightAll.checked,
 | |
|         findPrevious: findPrev
 | |
|       });
 | |
|       return window.dispatchEvent(event);
 | |
|     },
 | |
| 
 | |
|     updateUIState:
 | |
|         function PDFFindBar_updateUIState(state, previous, matchCount) {
 | |
|       var notFound = false;
 | |
|       var findMsg = '';
 | |
|       var status = '';
 | |
| 
 | |
|       switch (state) {
 | |
|         case FindStates.FIND_FOUND:
 | |
|           break;
 | |
| 
 | |
|         case FindStates.FIND_PENDING:
 | |
|           status = 'pending';
 | |
|           break;
 | |
| 
 | |
|         case FindStates.FIND_NOTFOUND:
 | |
|           findMsg = 'Phrase not found';
 | |
|           notFound = true;
 | |
|           break;
 | |
| 
 | |
|         case FindStates.FIND_WRAPPED:
 | |
|           if (previous) {
 | |
|             findMsg = 'Reached top of document, continued from bottom';
 | |
|           } else {
 | |
|             findMsg = 'Reached end of document, continued from top';
 | |
|           }
 | |
|           break;
 | |
|       }
 | |
| 
 | |
|       if (notFound) {
 | |
|         this.findField.classList.add('notFound');
 | |
|       } else {
 | |
|         this.findField.classList.remove('notFound');
 | |
|       }
 | |
| 
 | |
|       this.findField.setAttribute('data-status', status);
 | |
|       this.findMsg.textContent = findMsg;
 | |
| 
 | |
|       this.updateResultsCount(matchCount);
 | |
|     },
 | |
| 
 | |
|     updateResultsCount: function(matchCount) {
 | |
|       if (!this.findResultsCount) {
 | |
|         return; // no UI control is provided
 | |
|       }
 | |
| 
 | |
|       // If there are no matches, hide the counter
 | |
|       if (!matchCount) {
 | |
|         this.findResultsCount.classList.add('hidden');
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       // Create the match counter
 | |
|       this.findResultsCount.textContent = matchCount.toLocaleString();
 | |
| 
 | |
|       // Show the counter
 | |
|       this.findResultsCount.classList.remove('hidden');
 | |
|     },
 | |
| 
 | |
|     open: function PDFFindBar_open() {
 | |
|       if (!this.opened) {
 | |
|         this.opened = true;
 | |
|         this.toggleButton.classList.add('toggled');
 | |
|         this.bar.classList.remove('hidden');
 | |
|       }
 | |
|       this.findField.select();
 | |
|       this.findField.focus();
 | |
|     },
 | |
| 
 | |
|     close: function PDFFindBar_close() {
 | |
|       if (!this.opened) {
 | |
|         return;
 | |
|       }
 | |
|       this.opened = false;
 | |
|       this.toggleButton.classList.remove('toggled');
 | |
|       this.bar.classList.add('hidden');
 | |
|       this.findController.active = false;
 | |
|     },
 | |
| 
 | |
|     toggle: function PDFFindBar_toggle() {
 | |
|       if (this.opened) {
 | |
|         this.close();
 | |
|       } else {
 | |
|         this.open();
 | |
|       }
 | |
|     }
 | |
|   };
 | |
|   return PDFFindBar;
 | |
| })();
 | |
| 
 | |
| 
 | |
| var FindStates = {
 | |
|   FIND_FOUND: 0,
 | |
|   FIND_NOTFOUND: 1,
 | |
|   FIND_WRAPPED: 2,
 | |
|   FIND_PENDING: 3
 | |
| };
 | |
| 
 | |
| var FIND_SCROLL_OFFSET_TOP = -50;
 | |
| var FIND_SCROLL_OFFSET_LEFT = -400;
 | |
| 
 | |
| /**
 | |
|  * Provides "search" or "find" functionality for the PDF.
 | |
|  * This object actually performs the search for a given string.
 | |
|  */
 | |
| var PDFFindController = (function PDFFindControllerClosure() {
 | |
|   function PDFFindController(options) {
 | |
|     this.startedTextExtraction = false;
 | |
|     this.extractTextPromises = [];
 | |
|     this.pendingFindMatches = {};
 | |
|     this.active = false; // If active, find results will be highlighted.
 | |
|     this.pageContents = []; // Stores the text for each page.
 | |
|     this.pageMatches = [];
 | |
|     this.matchCount = 0;
 | |
|     this.selected = { // Currently selected match.
 | |
|       pageIdx: -1,
 | |
|       matchIdx: -1
 | |
|     };
 | |
|     this.offset = { // Where the find algorithm currently is in the document.
 | |
|       pageIdx: null,
 | |
|       matchIdx: null
 | |
|     };
 | |
|     this.pagesToSearch = null;
 | |
|     this.resumePageIdx = null;
 | |
|     this.state = null;
 | |
|     this.dirtyMatch = false;
 | |
|     this.findTimeout = null;
 | |
|     this.pdfViewer = options.pdfViewer || null;
 | |
|     this.integratedFind = options.integratedFind || false;
 | |
|     this.charactersToNormalize = {
 | |
|       '\u2018': '\'', // Left single quotation mark
 | |
|       '\u2019': '\'', // Right single quotation mark
 | |
|       '\u201A': '\'', // Single low-9 quotation mark
 | |
|       '\u201B': '\'', // Single high-reversed-9 quotation mark
 | |
|       '\u201C': '"', // Left double quotation mark
 | |
|       '\u201D': '"', // Right double quotation mark
 | |
|       '\u201E': '"', // Double low-9 quotation mark
 | |
|       '\u201F': '"', // Double high-reversed-9 quotation mark
 | |
|       '\u00BC': '1/4', // Vulgar fraction one quarter
 | |
|       '\u00BD': '1/2', // Vulgar fraction one half
 | |
|       '\u00BE': '3/4', // Vulgar fraction three quarters
 | |
|     };
 | |
|     this.findBar = options.findBar || null;
 | |
| 
 | |
|     // Compile the regular expression for text normalization once
 | |
|     var replace = Object.keys(this.charactersToNormalize).join('');
 | |
|     this.normalizationRegex = new RegExp('[' + replace + ']', 'g');
 | |
| 
 | |
|     var events = [
 | |
|       'find',
 | |
|       'findagain',
 | |
|       'findhighlightallchange',
 | |
|       'findcasesensitivitychange'
 | |
|     ];
 | |
| 
 | |
|     this.firstPagePromise = new Promise(function (resolve) {
 | |
|       this.resolveFirstPage = resolve;
 | |
|     }.bind(this));
 | |
|     this.handleEvent = this.handleEvent.bind(this);
 | |
| 
 | |
|     for (var i = 0, len = events.length; i < len; i++) {
 | |
|       window.addEventListener(events[i], this.handleEvent);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   PDFFindController.prototype = {
 | |
|     setFindBar: function PDFFindController_setFindBar(findBar) {
 | |
|       this.findBar = findBar;
 | |
|     },
 | |
| 
 | |
|     reset: function PDFFindController_reset() {
 | |
|       this.startedTextExtraction = false;
 | |
|       this.extractTextPromises = [];
 | |
|       this.active = false;
 | |
|     },
 | |
| 
 | |
|     normalize: function PDFFindController_normalize(text) {
 | |
|       var self = this;
 | |
|       return text.replace(this.normalizationRegex, function (ch) {
 | |
|         return self.charactersToNormalize[ch];
 | |
|       });
 | |
|     },
 | |
| 
 | |
|     calcFindMatch: function PDFFindController_calcFindMatch(pageIndex) {
 | |
|       var pageContent = this.normalize(this.pageContents[pageIndex]);
 | |
|       var query = this.normalize(this.state.query);
 | |
|       var caseSensitive = this.state.caseSensitive;
 | |
|       var queryLen = query.length;
 | |
| 
 | |
|       if (queryLen === 0) {
 | |
|         // Do nothing: the matches should be wiped out already.
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       if (!caseSensitive) {
 | |
|         pageContent = pageContent.toLowerCase();
 | |
|         query = query.toLowerCase();
 | |
|       }
 | |
| 
 | |
|       var matches = [];
 | |
|       var matchIdx = -queryLen;
 | |
|       while (true) {
 | |
|         matchIdx = pageContent.indexOf(query, matchIdx + queryLen);
 | |
|         if (matchIdx === -1) {
 | |
|           break;
 | |
|         }
 | |
|         matches.push(matchIdx);
 | |
|       }
 | |
|       this.pageMatches[pageIndex] = matches;
 | |
|       this.updatePage(pageIndex);
 | |
|       if (this.resumePageIdx === pageIndex) {
 | |
|         this.resumePageIdx = null;
 | |
|         this.nextPageMatch();
 | |
|       }
 | |
| 
 | |
|       // Update the matches count
 | |
|       if (matches.length > 0) {
 | |
|         this.matchCount += matches.length;
 | |
|         this.updateUIResultsCount();
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     extractText: function PDFFindController_extractText() {
 | |
|       if (this.startedTextExtraction) {
 | |
|         return;
 | |
|       }
 | |
|       this.startedTextExtraction = true;
 | |
| 
 | |
|       this.pageContents = [];
 | |
|       var extractTextPromisesResolves = [];
 | |
|       var numPages = this.pdfViewer.pagesCount;
 | |
|       for (var i = 0; i < numPages; i++) {
 | |
|         this.extractTextPromises.push(new Promise(function (resolve) {
 | |
|           extractTextPromisesResolves.push(resolve);
 | |
|         }));
 | |
|       }
 | |
| 
 | |
|       var self = this;
 | |
|       function extractPageText(pageIndex) {
 | |
|         self.pdfViewer.getPageTextContent(pageIndex).then(
 | |
|           function textContentResolved(textContent) {
 | |
|             var textItems = textContent.items;
 | |
|             var str = [];
 | |
| 
 | |
|             for (var i = 0, len = textItems.length; i < len; i++) {
 | |
|               str.push(textItems[i].str);
 | |
|             }
 | |
| 
 | |
|             // Store the pageContent as a string.
 | |
|             self.pageContents.push(str.join(''));
 | |
| 
 | |
|             extractTextPromisesResolves[pageIndex](pageIndex);
 | |
|             if ((pageIndex + 1) < self.pdfViewer.pagesCount) {
 | |
|               extractPageText(pageIndex + 1);
 | |
|             }
 | |
|           }
 | |
|         );
 | |
|       }
 | |
|       extractPageText(0);
 | |
|     },
 | |
| 
 | |
|     handleEvent: function PDFFindController_handleEvent(e) {
 | |
|       if (this.state === null || e.type !== 'findagain') {
 | |
|         this.dirtyMatch = true;
 | |
|       }
 | |
|       this.state = e.detail;
 | |
|       this.updateUIState(FindStates.FIND_PENDING);
 | |
| 
 | |
|       this.firstPagePromise.then(function() {
 | |
|         this.extractText();
 | |
| 
 | |
|         clearTimeout(this.findTimeout);
 | |
|         if (e.type === 'find') {
 | |
|           // Only trigger the find action after 250ms of silence.
 | |
|           this.findTimeout = setTimeout(this.nextMatch.bind(this), 250);
 | |
|         } else {
 | |
|           this.nextMatch();
 | |
|         }
 | |
|       }.bind(this));
 | |
|     },
 | |
| 
 | |
|     updatePage: function PDFFindController_updatePage(index) {
 | |
|       if (this.selected.pageIdx === index) {
 | |
|         // If the page is selected, scroll the page into view, which triggers
 | |
|         // rendering the page, which adds the textLayer. Once the textLayer is
 | |
|         // build, it will scroll onto the selected match.
 | |
|         this.pdfViewer.scrollPageIntoView(index + 1);
 | |
|       }
 | |
| 
 | |
|       var page = this.pdfViewer.getPageView(index);
 | |
|       if (page.textLayer) {
 | |
|         page.textLayer.updateMatches();
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     nextMatch: function PDFFindController_nextMatch() {
 | |
|       var previous = this.state.findPrevious;
 | |
|       var currentPageIndex = this.pdfViewer.currentPageNumber - 1;
 | |
|       var numPages = this.pdfViewer.pagesCount;
 | |
| 
 | |
|       this.active = true;
 | |
| 
 | |
|       if (this.dirtyMatch) {
 | |
|         // Need to recalculate the matches, reset everything.
 | |
|         this.dirtyMatch = false;
 | |
|         this.selected.pageIdx = this.selected.matchIdx = -1;
 | |
|         this.offset.pageIdx = currentPageIndex;
 | |
|         this.offset.matchIdx = null;
 | |
|         this.hadMatch = false;
 | |
|         this.resumePageIdx = null;
 | |
|         this.pageMatches = [];
 | |
|         this.matchCount = 0;
 | |
|         var self = this;
 | |
| 
 | |
|         for (var i = 0; i < numPages; i++) {
 | |
|           // Wipe out any previous highlighted matches.
 | |
|           this.updatePage(i);
 | |
| 
 | |
|           // As soon as the text is extracted start finding the matches.
 | |
|           if (!(i in this.pendingFindMatches)) {
 | |
|             this.pendingFindMatches[i] = true;
 | |
|             this.extractTextPromises[i].then(function(pageIdx) {
 | |
|               delete self.pendingFindMatches[pageIdx];
 | |
|               self.calcFindMatch(pageIdx);
 | |
|             });
 | |
|           }
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       // If there's no query there's no point in searching.
 | |
|       if (this.state.query === '') {
 | |
|         this.updateUIState(FindStates.FIND_FOUND);
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       // If we're waiting on a page, we return since we can't do anything else.
 | |
|       if (this.resumePageIdx) {
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       var offset = this.offset;
 | |
|       // Keep track of how many pages we should maximally iterate through.
 | |
|       this.pagesToSearch = numPages;
 | |
|       // If there's already a matchIdx that means we are iterating through a
 | |
|       // page's matches.
 | |
|       if (offset.matchIdx !== null) {
 | |
|         var numPageMatches = this.pageMatches[offset.pageIdx].length;
 | |
|         if ((!previous && offset.matchIdx + 1 < numPageMatches) ||
 | |
|             (previous && offset.matchIdx > 0)) {
 | |
|           // The simple case; we just have advance the matchIdx to select
 | |
|           // the next match on the page.
 | |
|           this.hadMatch = true;
 | |
|           offset.matchIdx = (previous ? offset.matchIdx - 1 :
 | |
|                                         offset.matchIdx + 1);
 | |
|           this.updateMatch(true);
 | |
|           return;
 | |
|         }
 | |
|         // We went beyond the current page's matches, so we advance to
 | |
|         // the next page.
 | |
|         this.advanceOffsetPage(previous);
 | |
|       }
 | |
|       // Start searching through the page.
 | |
|       this.nextPageMatch();
 | |
|     },
 | |
| 
 | |
|     matchesReady: function PDFFindController_matchesReady(matches) {
 | |
|       var offset = this.offset;
 | |
|       var numMatches = matches.length;
 | |
|       var previous = this.state.findPrevious;
 | |
| 
 | |
|       if (numMatches) {
 | |
|         // There were matches for the page, so initialize the matchIdx.
 | |
|         this.hadMatch = true;
 | |
|         offset.matchIdx = (previous ? numMatches - 1 : 0);
 | |
|         this.updateMatch(true);
 | |
|         return true;
 | |
|       } else {
 | |
|         // No matches, so attempt to search the next page.
 | |
|         this.advanceOffsetPage(previous);
 | |
|         if (offset.wrapped) {
 | |
|           offset.matchIdx = null;
 | |
|           if (this.pagesToSearch < 0) {
 | |
|             // No point in wrapping again, there were no matches.
 | |
|             this.updateMatch(false);
 | |
|             // while matches were not found, searching for a page
 | |
|             // with matches should nevertheless halt.
 | |
|             return true;
 | |
|           }
 | |
|         }
 | |
|         // Matches were not found (and searching is not done).
 | |
|         return false;
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * The method is called back from the text layer when match presentation
 | |
|      * is updated.
 | |
|      * @param {number} pageIndex - page index.
 | |
|      * @param {number} index - match index.
 | |
|      * @param {Array} elements - text layer div elements array.
 | |
|      * @param {number} beginIdx - start index of the div array for the match.
 | |
|      * @param {number} endIdx - end index of the div array for the match.
 | |
|      */
 | |
|     updateMatchPosition: function PDFFindController_updateMatchPosition(
 | |
|         pageIndex, index, elements, beginIdx, endIdx) {
 | |
|       if (this.selected.matchIdx === index &&
 | |
|           this.selected.pageIdx === pageIndex) {
 | |
|         var spot = {
 | |
|           top: FIND_SCROLL_OFFSET_TOP,
 | |
|           left: FIND_SCROLL_OFFSET_LEFT
 | |
|         };
 | |
|         scrollIntoView(elements[beginIdx], spot,
 | |
|                        /* skipOverflowHiddenElements = */ true);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     nextPageMatch: function PDFFindController_nextPageMatch() {
 | |
|       if (this.resumePageIdx !== null) {
 | |
|         console.error('There can only be one pending page.');
 | |
|       }
 | |
|       do {
 | |
|         var pageIdx = this.offset.pageIdx;
 | |
|         var matches = this.pageMatches[pageIdx];
 | |
|         if (!matches) {
 | |
|           // The matches don't exist yet for processing by "matchesReady",
 | |
|           // so set a resume point for when they do exist.
 | |
|           this.resumePageIdx = pageIdx;
 | |
|           break;
 | |
|         }
 | |
|       } while (!this.matchesReady(matches));
 | |
|     },
 | |
| 
 | |
|     advanceOffsetPage: function PDFFindController_advanceOffsetPage(previous) {
 | |
|       var offset = this.offset;
 | |
|       var numPages = this.extractTextPromises.length;
 | |
|       offset.pageIdx = (previous ? offset.pageIdx - 1 : offset.pageIdx + 1);
 | |
|       offset.matchIdx = null;
 | |
| 
 | |
|       this.pagesToSearch--;
 | |
| 
 | |
|       if (offset.pageIdx >= numPages || offset.pageIdx < 0) {
 | |
|         offset.pageIdx = (previous ? numPages - 1 : 0);
 | |
|         offset.wrapped = true;
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     updateMatch: function PDFFindController_updateMatch(found) {
 | |
|       var state = FindStates.FIND_NOTFOUND;
 | |
|       var wrapped = this.offset.wrapped;
 | |
|       this.offset.wrapped = false;
 | |
| 
 | |
|       if (found) {
 | |
|         var previousPage = this.selected.pageIdx;
 | |
|         this.selected.pageIdx = this.offset.pageIdx;
 | |
|         this.selected.matchIdx = this.offset.matchIdx;
 | |
|         state = (wrapped ? FindStates.FIND_WRAPPED : FindStates.FIND_FOUND);
 | |
|         // Update the currently selected page to wipe out any selected matches.
 | |
|         if (previousPage !== -1 && previousPage !== this.selected.pageIdx) {
 | |
|           this.updatePage(previousPage);
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       this.updateUIState(state, this.state.findPrevious);
 | |
|       if (this.selected.pageIdx !== -1) {
 | |
|         this.updatePage(this.selected.pageIdx);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     updateUIResultsCount:
 | |
|         function PDFFindController_updateUIResultsCount() {
 | |
|       if (this.findBar === null) {
 | |
|         throw new Error('PDFFindController is not initialized with a ' +
 | |
|           'PDFFindBar instance.');
 | |
|       }
 | |
|       this.findBar.updateResultsCount(this.matchCount);
 | |
|     },
 | |
| 
 | |
|     updateUIState: function PDFFindController_updateUIState(state, previous) {
 | |
|       if (this.integratedFind) {
 | |
|         FirefoxCom.request('updateFindControlState',
 | |
|                            { result: state, findPrevious: previous });
 | |
|         return;
 | |
|       }
 | |
|       if (this.findBar === null) {
 | |
|         throw new Error('PDFFindController is not initialized with a ' +
 | |
|                         'PDFFindBar instance.');
 | |
|       }
 | |
|       this.findBar.updateUIState(state, previous, this.matchCount);
 | |
|     }
 | |
|   };
 | |
|   return PDFFindController;
 | |
| })();
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Performs navigation functions inside PDF, such as opening specified page,
 | |
|  * or destination.
 | |
|  * @class
 | |
|  * @implements {IPDFLinkService}
 | |
|  */
 | |
| var PDFLinkService = (function () {
 | |
|   /**
 | |
|    * @constructs PDFLinkService
 | |
|    */
 | |
|   function PDFLinkService() {
 | |
|     this.baseUrl = null;
 | |
|     this.pdfDocument = null;
 | |
|     this.pdfViewer = null;
 | |
|     this.pdfHistory = null;
 | |
| 
 | |
|     this._pagesRefCache = null;
 | |
|   }
 | |
| 
 | |
|   PDFLinkService.prototype = {
 | |
|     setDocument: function PDFLinkService_setDocument(pdfDocument, baseUrl) {
 | |
|       this.baseUrl = baseUrl;
 | |
|       this.pdfDocument = pdfDocument;
 | |
|       this._pagesRefCache = Object.create(null);
 | |
|     },
 | |
| 
 | |
|     setViewer: function PDFLinkService_setViewer(pdfViewer) {
 | |
|       this.pdfViewer = pdfViewer;
 | |
|     },
 | |
| 
 | |
|     setHistory: function PDFLinkService_setHistory(pdfHistory) {
 | |
|       this.pdfHistory = pdfHistory;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @returns {number}
 | |
|      */
 | |
|     get pagesCount() {
 | |
|       return this.pdfDocument.numPages;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @returns {number}
 | |
|      */
 | |
|     get page() {
 | |
|       return this.pdfViewer.currentPageNumber;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @param {number} value
 | |
|      */
 | |
|     set page(value) {
 | |
|       this.pdfViewer.currentPageNumber = value;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @param dest - The PDF destination object.
 | |
|      */
 | |
|     navigateTo: function PDFLinkService_navigateTo(dest) {
 | |
|       var destString = '';
 | |
|       var self = this;
 | |
| 
 | |
|       var goToDestination = function(destRef) {
 | |
|         // dest array looks like that: <page-ref> </XYZ|FitXXX> <args..>
 | |
|         var pageNumber = destRef instanceof Object ?
 | |
|           self._pagesRefCache[destRef.num + ' ' + destRef.gen + ' R'] :
 | |
|           (destRef + 1);
 | |
|         if (pageNumber) {
 | |
|           if (pageNumber > self.pagesCount) {
 | |
|             pageNumber = self.pagesCount;
 | |
|           }
 | |
|           self.pdfViewer.scrollPageIntoView(pageNumber, dest);
 | |
| 
 | |
|           if (self.pdfHistory) {
 | |
|             // Update the browsing history.
 | |
|             self.pdfHistory.push({
 | |
|               dest: dest,
 | |
|               hash: destString,
 | |
|               page: pageNumber
 | |
|             });
 | |
|           }
 | |
|         } else {
 | |
|           self.pdfDocument.getPageIndex(destRef).then(function (pageIndex) {
 | |
|             var pageNum = pageIndex + 1;
 | |
|             var cacheKey = destRef.num + ' ' + destRef.gen + ' R';
 | |
|             self._pagesRefCache[cacheKey] = pageNum;
 | |
|             goToDestination(destRef);
 | |
|           });
 | |
|         }
 | |
|       };
 | |
| 
 | |
|       var destinationPromise;
 | |
|       if (typeof dest === 'string') {
 | |
|         destString = dest;
 | |
|         destinationPromise = this.pdfDocument.getDestination(dest);
 | |
|       } else {
 | |
|         destinationPromise = Promise.resolve(dest);
 | |
|       }
 | |
|       destinationPromise.then(function(destination) {
 | |
|         dest = destination;
 | |
|         if (!(destination instanceof Array)) {
 | |
|           return; // invalid destination
 | |
|         }
 | |
|         goToDestination(destination[0]);
 | |
|       });
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @param dest - The PDF destination object.
 | |
|      * @returns {string} The hyperlink to the PDF object.
 | |
|      */
 | |
|     getDestinationHash: function PDFLinkService_getDestinationHash(dest) {
 | |
|       if (typeof dest === 'string') {
 | |
|         return this.getAnchorUrl('#' + escape(dest));
 | |
|       }
 | |
|       if (dest instanceof Array) {
 | |
|         var destRef = dest[0]; // see navigateTo method for dest format
 | |
|         var pageNumber = destRef instanceof Object ?
 | |
|           this._pagesRefCache[destRef.num + ' ' + destRef.gen + ' R'] :
 | |
|           (destRef + 1);
 | |
|         if (pageNumber) {
 | |
|           var pdfOpenParams = this.getAnchorUrl('#page=' + pageNumber);
 | |
|           var destKind = dest[1];
 | |
|           if (typeof destKind === 'object' && 'name' in destKind &&
 | |
|               destKind.name === 'XYZ') {
 | |
|             var scale = (dest[4] || this.pdfViewer.currentScaleValue);
 | |
|             var scaleNumber = parseFloat(scale);
 | |
|             if (scaleNumber) {
 | |
|               scale = scaleNumber * 100;
 | |
|             }
 | |
|             pdfOpenParams += '&zoom=' + scale;
 | |
|             if (dest[2] || dest[3]) {
 | |
|               pdfOpenParams += ',' + (dest[2] || 0) + ',' + (dest[3] || 0);
 | |
|             }
 | |
|           }
 | |
|           return pdfOpenParams;
 | |
|         }
 | |
|       }
 | |
|       return this.getAnchorUrl('');
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Prefix the full url on anchor links to make sure that links are resolved
 | |
|      * relative to the current URL instead of the one defined in <base href>.
 | |
|      * @param {String} anchor The anchor hash, including the #.
 | |
|      * @returns {string} The hyperlink to the PDF object.
 | |
|      */
 | |
|     getAnchorUrl: function PDFLinkService_getAnchorUrl(anchor) {
 | |
|       return (this.baseUrl || '') + anchor;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @param {string} hash
 | |
|      */
 | |
|     setHash: function PDFLinkService_setHash(hash) {
 | |
|       if (hash.indexOf('=') >= 0) {
 | |
|         var params = parseQueryString(hash);
 | |
|         // borrowing syntax from "Parameters for Opening PDF Files"
 | |
|         if ('nameddest' in params) {
 | |
|           if (this.pdfHistory) {
 | |
|             this.pdfHistory.updateNextHashParam(params.nameddest);
 | |
|           }
 | |
|           this.navigateTo(params.nameddest);
 | |
|           return;
 | |
|         }
 | |
|         var pageNumber, dest;
 | |
|         if ('page' in params) {
 | |
|           pageNumber = (params.page | 0) || 1;
 | |
|         }
 | |
|         if ('zoom' in params) {
 | |
|           // Build the destination array.
 | |
|           var zoomArgs = params.zoom.split(','); // scale,left,top
 | |
|           var zoomArg = zoomArgs[0];
 | |
|           var zoomArgNumber = parseFloat(zoomArg);
 | |
| 
 | |
|           if (zoomArg.indexOf('Fit') === -1) {
 | |
|             // If the zoomArg is a number, it has to get divided by 100. If it's
 | |
|             // a string, it should stay as it is.
 | |
|             dest = [null, { name: 'XYZ' },
 | |
|                     zoomArgs.length > 1 ? (zoomArgs[1] | 0) : null,
 | |
|                     zoomArgs.length > 2 ? (zoomArgs[2] | 0) : null,
 | |
|                     (zoomArgNumber ? zoomArgNumber / 100 : zoomArg)];
 | |
|           } else {
 | |
|             if (zoomArg === 'Fit' || zoomArg === 'FitB') {
 | |
|               dest = [null, { name: zoomArg }];
 | |
|             } else if ((zoomArg === 'FitH' || zoomArg === 'FitBH') ||
 | |
|                        (zoomArg === 'FitV' || zoomArg === 'FitBV')) {
 | |
|               dest = [null, { name: zoomArg },
 | |
|                       zoomArgs.length > 1 ? (zoomArgs[1] | 0) : null];
 | |
|             } else if (zoomArg === 'FitR') {
 | |
|               if (zoomArgs.length !== 5) {
 | |
|                 console.error('PDFLinkService_setHash: ' +
 | |
|                               'Not enough parameters for \'FitR\'.');
 | |
|               } else {
 | |
|                 dest = [null, { name: zoomArg },
 | |
|                         (zoomArgs[1] | 0), (zoomArgs[2] | 0),
 | |
|                         (zoomArgs[3] | 0), (zoomArgs[4] | 0)];
 | |
|               }
 | |
|             } else {
 | |
|               console.error('PDFLinkService_setHash: \'' + zoomArg +
 | |
|                             '\' is not a valid zoom value.');
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|         if (dest) {
 | |
|           this.pdfViewer.scrollPageIntoView(pageNumber || this.page, dest);
 | |
|         } else if (pageNumber) {
 | |
|           this.page = pageNumber; // simple page
 | |
|         }
 | |
|         if ('pagemode' in params) {
 | |
|           var event = document.createEvent('CustomEvent');
 | |
|           event.initCustomEvent('pagemode', true, true, {
 | |
|             mode: params.pagemode,
 | |
|           });
 | |
|           this.pdfViewer.container.dispatchEvent(event);
 | |
|         }
 | |
|       } else if (/^\d+$/.test(hash)) { // page number
 | |
|         this.page = hash;
 | |
|       } else { // named destination
 | |
|         if (this.pdfHistory) {
 | |
|           this.pdfHistory.updateNextHashParam(unescape(hash));
 | |
|         }
 | |
|         this.navigateTo(unescape(hash));
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @param {string} action
 | |
|      */
 | |
|     executeNamedAction: function PDFLinkService_executeNamedAction(action) {
 | |
|       // See PDF reference, table 8.45 - Named action
 | |
|       switch (action) {
 | |
|         case 'GoBack':
 | |
|           if (this.pdfHistory) {
 | |
|             this.pdfHistory.back();
 | |
|           }
 | |
|           break;
 | |
| 
 | |
|         case 'GoForward':
 | |
|           if (this.pdfHistory) {
 | |
|             this.pdfHistory.forward();
 | |
|           }
 | |
|           break;
 | |
| 
 | |
|         case 'NextPage':
 | |
|           this.page++;
 | |
|           break;
 | |
| 
 | |
|         case 'PrevPage':
 | |
|           this.page--;
 | |
|           break;
 | |
| 
 | |
|         case 'LastPage':
 | |
|           this.page = this.pagesCount;
 | |
|           break;
 | |
| 
 | |
|         case 'FirstPage':
 | |
|           this.page = 1;
 | |
|           break;
 | |
| 
 | |
|         default:
 | |
|           break; // No action according to spec
 | |
|       }
 | |
| 
 | |
|       var event = document.createEvent('CustomEvent');
 | |
|       event.initCustomEvent('namedaction', true, true, {
 | |
|         action: action
 | |
|       });
 | |
|       this.pdfViewer.container.dispatchEvent(event);
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @param {number} pageNum - page number.
 | |
|      * @param {Object} pageRef - reference to the page.
 | |
|      */
 | |
|     cachePageRef: function PDFLinkService_cachePageRef(pageNum, pageRef) {
 | |
|       var refStr = pageRef.num + ' ' + pageRef.gen + ' R';
 | |
|       this._pagesRefCache[refStr] = pageNum;
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   return PDFLinkService;
 | |
| })();
 | |
| 
 | |
| 
 | |
| var PDFHistory = (function () {
 | |
|   function PDFHistory(options) {
 | |
|     this.linkService = options.linkService;
 | |
| 
 | |
|     this.initialized = false;
 | |
|     this.initialDestination = null;
 | |
|     this.initialBookmark = null;
 | |
|   }
 | |
| 
 | |
|   PDFHistory.prototype = {
 | |
|     /**
 | |
|      * @param {string} fingerprint
 | |
|      * @param {IPDFLinkService} linkService
 | |
|      */
 | |
|     initialize: function pdfHistoryInitialize(fingerprint) {
 | |
|       this.initialized = true;
 | |
|       this.reInitialized = false;
 | |
|       this.allowHashChange = true;
 | |
|       this.historyUnlocked = true;
 | |
|       this.isViewerInPresentationMode = false;
 | |
| 
 | |
|       this.previousHash = window.location.hash.substring(1);
 | |
|       this.currentBookmark = '';
 | |
|       this.currentPage = 0;
 | |
|       this.updatePreviousBookmark = false;
 | |
|       this.previousBookmark = '';
 | |
|       this.previousPage = 0;
 | |
|       this.nextHashParam = '';
 | |
| 
 | |
|       this.fingerprint = fingerprint;
 | |
|       this.currentUid = this.uid = 0;
 | |
|       this.current = {};
 | |
| 
 | |
|       var state = window.history.state;
 | |
|       if (this._isStateObjectDefined(state)) {
 | |
|         // This corresponds to navigating back to the document
 | |
|         // from another page in the browser history.
 | |
|         if (state.target.dest) {
 | |
|           this.initialDestination = state.target.dest;
 | |
|         } else {
 | |
|           this.initialBookmark = state.target.hash;
 | |
|         }
 | |
|         this.currentUid = state.uid;
 | |
|         this.uid = state.uid + 1;
 | |
|         this.current = state.target;
 | |
|       } else {
 | |
|         // This corresponds to the loading of a new document.
 | |
|         if (state && state.fingerprint &&
 | |
|           this.fingerprint !== state.fingerprint) {
 | |
|           // Reinitialize the browsing history when a new document
 | |
|           // is opened in the web viewer.
 | |
|           this.reInitialized = true;
 | |
|         }
 | |
|         this._pushOrReplaceState({fingerprint: this.fingerprint}, true);
 | |
|       }
 | |
| 
 | |
|       var self = this;
 | |
|       window.addEventListener('popstate', function pdfHistoryPopstate(evt) {
 | |
|         if (!self.historyUnlocked) {
 | |
|           return;
 | |
|         }
 | |
|         if (evt.state) {
 | |
|           // Move back/forward in the history.
 | |
|           self._goTo(evt.state);
 | |
|           return;
 | |
|         }
 | |
| 
 | |
|         // If the state is not set, then the user tried to navigate to a
 | |
|         // different hash by manually editing the URL and pressing Enter, or by
 | |
|         // clicking on an in-page link (e.g. the "current view" link).
 | |
|         // Save the current view state to the browser history.
 | |
| 
 | |
|         // Note: In Firefox, history.null could also be null after an in-page
 | |
|         // navigation to the same URL, and without dispatching the popstate
 | |
|         // event: https://bugzilla.mozilla.org/show_bug.cgi?id=1183881
 | |
| 
 | |
|         if (self.uid === 0) {
 | |
|           // Replace the previous state if it was not explicitly set.
 | |
|           var previousParams = (self.previousHash && self.currentBookmark &&
 | |
|             self.previousHash !== self.currentBookmark) ?
 | |
|             {hash: self.currentBookmark, page: self.currentPage} :
 | |
|             {page: 1};
 | |
|           replacePreviousHistoryState(previousParams, function() {
 | |
|             updateHistoryWithCurrentHash();
 | |
|           });
 | |
|         } else {
 | |
|           updateHistoryWithCurrentHash();
 | |
|         }
 | |
|       }, false);
 | |
| 
 | |
| 
 | |
|       function updateHistoryWithCurrentHash() {
 | |
|         self.previousHash = window.location.hash.slice(1);
 | |
|         self._pushToHistory({hash: self.previousHash}, false, true);
 | |
|         self._updatePreviousBookmark();
 | |
|       }
 | |
| 
 | |
|       function replacePreviousHistoryState(params, callback) {
 | |
|         // To modify the previous history entry, the following happens:
 | |
|         // 1. history.back()
 | |
|         // 2. _pushToHistory, which calls history.replaceState( ... )
 | |
|         // 3. history.forward()
 | |
|         // Because a navigation via the history API does not immediately update
 | |
|         // the history state, the popstate event is used for synchronization.
 | |
|         self.historyUnlocked = false;
 | |
| 
 | |
|         // Suppress the hashchange event to avoid side effects caused by
 | |
|         // navigating back and forward.
 | |
|         self.allowHashChange = false;
 | |
|         window.addEventListener('popstate', rewriteHistoryAfterBack);
 | |
|         history.back();
 | |
| 
 | |
|         function rewriteHistoryAfterBack() {
 | |
|           window.removeEventListener('popstate', rewriteHistoryAfterBack);
 | |
|           window.addEventListener('popstate', rewriteHistoryAfterForward);
 | |
|           self._pushToHistory(params, false, true);
 | |
|           history.forward();
 | |
|         }
 | |
|         function rewriteHistoryAfterForward() {
 | |
|           window.removeEventListener('popstate', rewriteHistoryAfterForward);
 | |
|           self.allowHashChange = true;
 | |
|           self.historyUnlocked = true;
 | |
|           callback();
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       function pdfHistoryBeforeUnload() {
 | |
|         var previousParams = self._getPreviousParams(null, true);
 | |
|         if (previousParams) {
 | |
|           var replacePrevious = (!self.current.dest &&
 | |
|           self.current.hash !== self.previousHash);
 | |
|           self._pushToHistory(previousParams, false, replacePrevious);
 | |
|           self._updatePreviousBookmark();
 | |
|         }
 | |
|         // Remove the event listener when navigating away from the document,
 | |
|         // since 'beforeunload' prevents Firefox from caching the document.
 | |
|         window.removeEventListener('beforeunload', pdfHistoryBeforeUnload,
 | |
|                                    false);
 | |
|       }
 | |
| 
 | |
|       window.addEventListener('beforeunload', pdfHistoryBeforeUnload, false);
 | |
| 
 | |
|       window.addEventListener('pageshow', function pdfHistoryPageShow(evt) {
 | |
|         // If the entire viewer (including the PDF file) is cached in
 | |
|         // the browser, we need to reattach the 'beforeunload' event listener
 | |
|         // since the 'DOMContentLoaded' event is not fired on 'pageshow'.
 | |
|         window.addEventListener('beforeunload', pdfHistoryBeforeUnload, false);
 | |
|       }, false);
 | |
| 
 | |
|       window.addEventListener('presentationmodechanged', function(e) {
 | |
|         self.isViewerInPresentationMode = !!e.detail.active;
 | |
|       });
 | |
|     },
 | |
| 
 | |
|     clearHistoryState: function pdfHistory_clearHistoryState() {
 | |
|       this._pushOrReplaceState(null, true);
 | |
|     },
 | |
| 
 | |
|     _isStateObjectDefined: function pdfHistory_isStateObjectDefined(state) {
 | |
|       return (state && state.uid >= 0 &&
 | |
|       state.fingerprint && this.fingerprint === state.fingerprint &&
 | |
|       state.target && state.target.hash) ? true : false;
 | |
|     },
 | |
| 
 | |
|     _pushOrReplaceState: function pdfHistory_pushOrReplaceState(stateObj,
 | |
|                                                                 replace) {
 | |
|       if (replace) {
 | |
|         window.history.replaceState(stateObj, '', document.URL);
 | |
|       } else {
 | |
|         window.history.pushState(stateObj, '', document.URL);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     get isHashChangeUnlocked() {
 | |
|       if (!this.initialized) {
 | |
|         return true;
 | |
|       }
 | |
|       return this.allowHashChange;
 | |
|     },
 | |
| 
 | |
|     _updatePreviousBookmark: function pdfHistory_updatePreviousBookmark() {
 | |
|       if (this.updatePreviousBookmark &&
 | |
|         this.currentBookmark && this.currentPage) {
 | |
|         this.previousBookmark = this.currentBookmark;
 | |
|         this.previousPage = this.currentPage;
 | |
|         this.updatePreviousBookmark = false;
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     updateCurrentBookmark: function pdfHistoryUpdateCurrentBookmark(bookmark,
 | |
|                                                                     pageNum) {
 | |
|       if (this.initialized) {
 | |
|         this.currentBookmark = bookmark.substring(1);
 | |
|         this.currentPage = pageNum | 0;
 | |
|         this._updatePreviousBookmark();
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     updateNextHashParam: function pdfHistoryUpdateNextHashParam(param) {
 | |
|       if (this.initialized) {
 | |
|         this.nextHashParam = param;
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     push: function pdfHistoryPush(params, isInitialBookmark) {
 | |
|       if (!(this.initialized && this.historyUnlocked)) {
 | |
|         return;
 | |
|       }
 | |
|       if (params.dest && !params.hash) {
 | |
|         params.hash = (this.current.hash && this.current.dest &&
 | |
|         this.current.dest === params.dest) ?
 | |
|           this.current.hash :
 | |
|           this.linkService.getDestinationHash(params.dest).split('#')[1];
 | |
|       }
 | |
|       if (params.page) {
 | |
|         params.page |= 0;
 | |
|       }
 | |
|       if (isInitialBookmark) {
 | |
|         var target = window.history.state.target;
 | |
|         if (!target) {
 | |
|           // Invoked when the user specifies an initial bookmark,
 | |
|           // thus setting initialBookmark, when the document is loaded.
 | |
|           this._pushToHistory(params, false);
 | |
|           this.previousHash = window.location.hash.substring(1);
 | |
|         }
 | |
|         this.updatePreviousBookmark = this.nextHashParam ? false : true;
 | |
|         if (target) {
 | |
|           // If the current document is reloaded,
 | |
|           // avoid creating duplicate entries in the history.
 | |
|           this._updatePreviousBookmark();
 | |
|         }
 | |
|         return;
 | |
|       }
 | |
|       if (this.nextHashParam) {
 | |
|         if (this.nextHashParam === params.hash) {
 | |
|           this.nextHashParam = null;
 | |
|           this.updatePreviousBookmark = true;
 | |
|           return;
 | |
|         } else {
 | |
|           this.nextHashParam = null;
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       if (params.hash) {
 | |
|         if (this.current.hash) {
 | |
|           if (this.current.hash !== params.hash) {
 | |
|             this._pushToHistory(params, true);
 | |
|           } else {
 | |
|             if (!this.current.page && params.page) {
 | |
|               this._pushToHistory(params, false, true);
 | |
|             }
 | |
|             this.updatePreviousBookmark = true;
 | |
|           }
 | |
|         } else {
 | |
|           this._pushToHistory(params, true);
 | |
|         }
 | |
|       } else if (this.current.page && params.page &&
 | |
|         this.current.page !== params.page) {
 | |
|         this._pushToHistory(params, true);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     _getPreviousParams: function pdfHistory_getPreviousParams(onlyCheckPage,
 | |
|                                                               beforeUnload) {
 | |
|       if (!(this.currentBookmark && this.currentPage)) {
 | |
|         return null;
 | |
|       } else if (this.updatePreviousBookmark) {
 | |
|         this.updatePreviousBookmark = false;
 | |
|       }
 | |
|       if (this.uid > 0 && !(this.previousBookmark && this.previousPage)) {
 | |
|         // Prevent the history from getting stuck in the current state,
 | |
|         // effectively preventing the user from going back/forward in
 | |
|         // the history.
 | |
|         //
 | |
|         // This happens if the current position in the document didn't change
 | |
|         // when the history was previously updated. The reasons for this are
 | |
|         // either:
 | |
|         // 1. The current zoom value is such that the document does not need to,
 | |
|         //    or cannot, be scrolled to display the destination.
 | |
|         // 2. The previous destination is broken, and doesn't actally point to a
 | |
|         //    position within the document.
 | |
|         //    (This is either due to a bad PDF generator, or the user making a
 | |
|         //     mistake when entering a destination in the hash parameters.)
 | |
|         return null;
 | |
|       }
 | |
|       if ((!this.current.dest && !onlyCheckPage) || beforeUnload) {
 | |
|         if (this.previousBookmark === this.currentBookmark) {
 | |
|           return null;
 | |
|         }
 | |
|       } else if (this.current.page || onlyCheckPage) {
 | |
|         if (this.previousPage === this.currentPage) {
 | |
|           return null;
 | |
|         }
 | |
|       } else {
 | |
|         return null;
 | |
|       }
 | |
|       var params = {hash: this.currentBookmark, page: this.currentPage};
 | |
|       if (this.isViewerInPresentationMode) {
 | |
|         params.hash = null;
 | |
|       }
 | |
|       return params;
 | |
|     },
 | |
| 
 | |
|     _stateObj: function pdfHistory_stateObj(params) {
 | |
|       return {fingerprint: this.fingerprint, uid: this.uid, target: params};
 | |
|     },
 | |
| 
 | |
|     _pushToHistory: function pdfHistory_pushToHistory(params,
 | |
|                                                       addPrevious, overwrite) {
 | |
|       if (!this.initialized) {
 | |
|         return;
 | |
|       }
 | |
|       if (!params.hash && params.page) {
 | |
|         params.hash = ('page=' + params.page);
 | |
|       }
 | |
|       if (addPrevious && !overwrite) {
 | |
|         var previousParams = this._getPreviousParams();
 | |
|         if (previousParams) {
 | |
|           var replacePrevious = (!this.current.dest &&
 | |
|           this.current.hash !== this.previousHash);
 | |
|           this._pushToHistory(previousParams, false, replacePrevious);
 | |
|         }
 | |
|       }
 | |
|       this._pushOrReplaceState(this._stateObj(params),
 | |
|         (overwrite || this.uid === 0));
 | |
|       this.currentUid = this.uid++;
 | |
|       this.current = params;
 | |
|       this.updatePreviousBookmark = true;
 | |
|     },
 | |
| 
 | |
|     _goTo: function pdfHistory_goTo(state) {
 | |
|       if (!(this.initialized && this.historyUnlocked &&
 | |
|         this._isStateObjectDefined(state))) {
 | |
|         return;
 | |
|       }
 | |
|       if (!this.reInitialized && state.uid < this.currentUid) {
 | |
|         var previousParams = this._getPreviousParams(true);
 | |
|         if (previousParams) {
 | |
|           this._pushToHistory(this.current, false);
 | |
|           this._pushToHistory(previousParams, false);
 | |
|           this.currentUid = state.uid;
 | |
|           window.history.back();
 | |
|           return;
 | |
|         }
 | |
|       }
 | |
|       this.historyUnlocked = false;
 | |
| 
 | |
|       if (state.target.dest) {
 | |
|         this.linkService.navigateTo(state.target.dest);
 | |
|       } else {
 | |
|         this.linkService.setHash(state.target.hash);
 | |
|       }
 | |
|       this.currentUid = state.uid;
 | |
|       if (state.uid > this.uid) {
 | |
|         this.uid = state.uid;
 | |
|       }
 | |
|       this.current = state.target;
 | |
|       this.updatePreviousBookmark = true;
 | |
| 
 | |
|       var currentHash = window.location.hash.substring(1);
 | |
|       if (this.previousHash !== currentHash) {
 | |
|         this.allowHashChange = false;
 | |
|       }
 | |
|       this.previousHash = currentHash;
 | |
| 
 | |
|       this.historyUnlocked = true;
 | |
|     },
 | |
| 
 | |
|     back: function pdfHistoryBack() {
 | |
|       this.go(-1);
 | |
|     },
 | |
| 
 | |
|     forward: function pdfHistoryForward() {
 | |
|       this.go(1);
 | |
|     },
 | |
| 
 | |
|     go: function pdfHistoryGo(direction) {
 | |
|       if (this.initialized && this.historyUnlocked) {
 | |
|         var state = window.history.state;
 | |
|         if (direction === -1 && state && state.uid > 0) {
 | |
|           window.history.back();
 | |
|         } else if (direction === 1 && state && state.uid < (this.uid - 1)) {
 | |
|           window.history.forward();
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   return PDFHistory;
 | |
| })();
 | |
| 
 | |
| 
 | |
| var SecondaryToolbar = {
 | |
|   opened: false,
 | |
|   previousContainerHeight: null,
 | |
|   newContainerHeight: null,
 | |
| 
 | |
|   initialize: function secondaryToolbarInitialize(options) {
 | |
|     this.toolbar = options.toolbar;
 | |
|     this.buttonContainer = this.toolbar.firstElementChild;
 | |
| 
 | |
|     // Define the toolbar buttons.
 | |
|     this.toggleButton = options.toggleButton;
 | |
|     this.presentationModeButton = options.presentationModeButton;
 | |
|     this.openFile = options.openFile;
 | |
|     this.print = options.print;
 | |
|     this.download = options.download;
 | |
|     this.viewBookmark = options.viewBookmark;
 | |
|     this.firstPage = options.firstPage;
 | |
|     this.lastPage = options.lastPage;
 | |
|     this.pageRotateCw = options.pageRotateCw;
 | |
|     this.pageRotateCcw = options.pageRotateCcw;
 | |
|     this.documentPropertiesButton = options.documentPropertiesButton;
 | |
| 
 | |
|     // Attach the event listeners.
 | |
|     var elements = [
 | |
|       // Button to toggle the visibility of the secondary toolbar:
 | |
|       { element: this.toggleButton, handler: this.toggle },
 | |
|       // All items within the secondary toolbar
 | |
|       // (except for toggleHandTool, hand_tool.js is responsible for it):
 | |
|       { element: this.presentationModeButton,
 | |
|         handler: this.presentationModeClick },
 | |
|       { element: this.openFile, handler: this.openFileClick },
 | |
|       { element: this.print, handler: this.printClick },
 | |
|       { element: this.download, handler: this.downloadClick },
 | |
|       { element: this.viewBookmark, handler: this.viewBookmarkClick },
 | |
|       { element: this.firstPage, handler: this.firstPageClick },
 | |
|       { element: this.lastPage, handler: this.lastPageClick },
 | |
|       { element: this.pageRotateCw, handler: this.pageRotateCwClick },
 | |
|       { element: this.pageRotateCcw, handler: this.pageRotateCcwClick },
 | |
|       { element: this.documentPropertiesButton,
 | |
|         handler: this.documentPropertiesClick }
 | |
|     ];
 | |
| 
 | |
|     for (var item in elements) {
 | |
|       var element = elements[item].element;
 | |
|       if (element) {
 | |
|         element.addEventListener('click', elements[item].handler.bind(this));
 | |
|       }
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   // Event handling functions.
 | |
|   presentationModeClick: function secondaryToolbarPresentationModeClick(evt) {
 | |
|     PDFViewerApplication.requestPresentationMode();
 | |
|     this.close();
 | |
|   },
 | |
| 
 | |
|   openFileClick: function secondaryToolbarOpenFileClick(evt) {
 | |
|     document.getElementById('fileInput').click();
 | |
|     this.close();
 | |
|   },
 | |
| 
 | |
|   printClick: function secondaryToolbarPrintClick(evt) {
 | |
|     window.print();
 | |
|     this.close();
 | |
|   },
 | |
| 
 | |
|   downloadClick: function secondaryToolbarDownloadClick(evt) {
 | |
|     PDFViewerApplication.download();
 | |
|     this.close();
 | |
|   },
 | |
| 
 | |
|   viewBookmarkClick: function secondaryToolbarViewBookmarkClick(evt) {
 | |
|     this.close();
 | |
|   },
 | |
| 
 | |
|   firstPageClick: function secondaryToolbarFirstPageClick(evt) {
 | |
|     PDFViewerApplication.page = 1;
 | |
|     this.close();
 | |
|   },
 | |
| 
 | |
|   lastPageClick: function secondaryToolbarLastPageClick(evt) {
 | |
|     if (PDFViewerApplication.pdfDocument) {
 | |
|       PDFViewerApplication.page = PDFViewerApplication.pagesCount;
 | |
|     }
 | |
|     this.close();
 | |
|   },
 | |
| 
 | |
|   pageRotateCwClick: function secondaryToolbarPageRotateCwClick(evt) {
 | |
|     PDFViewerApplication.rotatePages(90);
 | |
|   },
 | |
| 
 | |
|   pageRotateCcwClick: function secondaryToolbarPageRotateCcwClick(evt) {
 | |
|     PDFViewerApplication.rotatePages(-90);
 | |
|   },
 | |
| 
 | |
|   documentPropertiesClick: function secondaryToolbarDocumentPropsClick(evt) {
 | |
|     PDFViewerApplication.pdfDocumentProperties.open();
 | |
|     this.close();
 | |
|   },
 | |
| 
 | |
|   // Misc. functions for interacting with the toolbar.
 | |
|   setMaxHeight: function secondaryToolbarSetMaxHeight(container) {
 | |
|     if (!container || !this.buttonContainer) {
 | |
|       return;
 | |
|     }
 | |
|     this.newContainerHeight = container.clientHeight;
 | |
|     if (this.previousContainerHeight === this.newContainerHeight) {
 | |
|       return;
 | |
|     }
 | |
|     this.buttonContainer.setAttribute('style',
 | |
|       'max-height: ' + (this.newContainerHeight - SCROLLBAR_PADDING) + 'px;');
 | |
|     this.previousContainerHeight = this.newContainerHeight;
 | |
|   },
 | |
| 
 | |
|   open: function secondaryToolbarOpen() {
 | |
|     if (this.opened) {
 | |
|       return;
 | |
|     }
 | |
|     this.opened = true;
 | |
|     this.toggleButton.classList.add('toggled');
 | |
|     this.toolbar.classList.remove('hidden');
 | |
|   },
 | |
| 
 | |
|   close: function secondaryToolbarClose(target) {
 | |
|     if (!this.opened) {
 | |
|       return;
 | |
|     } else if (target && !this.toolbar.contains(target)) {
 | |
|       return;
 | |
|     }
 | |
|     this.opened = false;
 | |
|     this.toolbar.classList.add('hidden');
 | |
|     this.toggleButton.classList.remove('toggled');
 | |
|   },
 | |
| 
 | |
|   toggle: function secondaryToolbarToggle() {
 | |
|     if (this.opened) {
 | |
|       this.close();
 | |
|     } else {
 | |
|       this.open();
 | |
|     }
 | |
|   }
 | |
| };
 | |
| 
 | |
| 
 | |
| var DELAY_BEFORE_RESETTING_SWITCH_IN_PROGRESS = 1500; // in ms
 | |
| var DELAY_BEFORE_HIDING_CONTROLS = 3000; // in ms
 | |
| var ACTIVE_SELECTOR = 'pdfPresentationMode';
 | |
| var CONTROLS_SELECTOR = 'pdfPresentationModeControls';
 | |
| 
 | |
| /**
 | |
|  * @typedef {Object} PDFPresentationModeOptions
 | |
|  * @property {HTMLDivElement} container - The container for the viewer element.
 | |
|  * @property {HTMLDivElement} viewer - (optional) The viewer element.
 | |
|  * @property {PDFViewer} pdfViewer - The document viewer.
 | |
|  * @property {PDFThumbnailViewer} pdfThumbnailViewer - (optional) The thumbnail
 | |
|  *   viewer.
 | |
|  * @property {Array} contextMenuItems - (optional) The menuitems that are added
 | |
|  *   to the context menu in Presentation Mode.
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * @class
 | |
|  */
 | |
| var PDFPresentationMode = (function PDFPresentationModeClosure() {
 | |
|   /**
 | |
|    * @constructs PDFPresentationMode
 | |
|    * @param {PDFPresentationModeOptions} options
 | |
|    */
 | |
|   function PDFPresentationMode(options) {
 | |
|     this.container = options.container;
 | |
|     this.viewer = options.viewer || options.container.firstElementChild;
 | |
|     this.pdfViewer = options.pdfViewer;
 | |
|     this.pdfThumbnailViewer = options.pdfThumbnailViewer || null;
 | |
|     var contextMenuItems = options.contextMenuItems || null;
 | |
| 
 | |
|     this.active = false;
 | |
|     this.args = null;
 | |
|     this.contextMenuOpen = false;
 | |
|     this.mouseScrollTimeStamp = 0;
 | |
|     this.mouseScrollDelta = 0;
 | |
| 
 | |
|     if (contextMenuItems) {
 | |
|       for (var i = 0, ii = contextMenuItems.length; i < ii; i++) {
 | |
|         var item = contextMenuItems[i];
 | |
|         item.element.addEventListener('click', function (handler) {
 | |
|           this.contextMenuOpen = false;
 | |
|           handler();
 | |
|         }.bind(this, item.handler));
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   PDFPresentationMode.prototype = {
 | |
|     /**
 | |
|      * Request the browser to enter fullscreen mode.
 | |
|      * @returns {boolean} Indicating if the request was successful.
 | |
|      */
 | |
|     request: function PDFPresentationMode_request() {
 | |
|       if (this.switchInProgress || this.active ||
 | |
|           !this.viewer.hasChildNodes()) {
 | |
|         return false;
 | |
|       }
 | |
|       this._addFullscreenChangeListeners();
 | |
|       this._setSwitchInProgress();
 | |
|       this._notifyStateChange();
 | |
| 
 | |
|       if (this.container.requestFullscreen) {
 | |
|         this.container.requestFullscreen();
 | |
|       } else if (this.container.mozRequestFullScreen) {
 | |
|         this.container.mozRequestFullScreen();
 | |
|       } else if (this.container.webkitRequestFullscreen) {
 | |
|         this.container.webkitRequestFullscreen(Element.ALLOW_KEYBOARD_INPUT);
 | |
|       } else if (this.container.msRequestFullscreen) {
 | |
|         this.container.msRequestFullscreen();
 | |
|       } else {
 | |
|         return false;
 | |
|       }
 | |
| 
 | |
|       this.args = {
 | |
|         page: this.pdfViewer.currentPageNumber,
 | |
|         previousScale: this.pdfViewer.currentScaleValue,
 | |
|       };
 | |
| 
 | |
|       return true;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Switches page when the user scrolls (using a scroll wheel or a touchpad)
 | |
|      * with large enough motion, to prevent accidental page switches.
 | |
|      * @param {number} delta - The delta value from the mouse event.
 | |
|      */
 | |
|     mouseScroll: function PDFPresentationMode_mouseScroll(delta) {
 | |
|       if (!this.active) {
 | |
|         return;
 | |
|       }
 | |
|       var MOUSE_SCROLL_COOLDOWN_TIME = 50;
 | |
|       var PAGE_SWITCH_THRESHOLD = 120;
 | |
|       var PageSwitchDirection = {
 | |
|         UP: -1,
 | |
|         DOWN: 1
 | |
|       };
 | |
| 
 | |
|       var currentTime = (new Date()).getTime();
 | |
|       var storedTime = this.mouseScrollTimeStamp;
 | |
| 
 | |
|       // If we've already switched page, avoid accidentally switching again.
 | |
|       if (currentTime > storedTime &&
 | |
|           currentTime - storedTime < MOUSE_SCROLL_COOLDOWN_TIME) {
 | |
|         return;
 | |
|       }
 | |
|       // If the scroll direction changed, reset the accumulated scroll delta.
 | |
|       if ((this.mouseScrollDelta > 0 && delta < 0) ||
 | |
|           (this.mouseScrollDelta < 0 && delta > 0)) {
 | |
|         this._resetMouseScrollState();
 | |
|       }
 | |
|       this.mouseScrollDelta += delta;
 | |
| 
 | |
|       if (Math.abs(this.mouseScrollDelta) >= PAGE_SWITCH_THRESHOLD) {
 | |
|         var pageSwitchDirection = (this.mouseScrollDelta > 0) ?
 | |
|           PageSwitchDirection.UP : PageSwitchDirection.DOWN;
 | |
|         var page = this.pdfViewer.currentPageNumber;
 | |
|         this._resetMouseScrollState();
 | |
| 
 | |
|         // If we're at the first/last page, we don't need to do anything.
 | |
|         if ((page === 1 && pageSwitchDirection === PageSwitchDirection.UP) ||
 | |
|             (page === this.pdfViewer.pagesCount &&
 | |
|              pageSwitchDirection === PageSwitchDirection.DOWN)) {
 | |
|           return;
 | |
|         }
 | |
|         this.pdfViewer.currentPageNumber = (page + pageSwitchDirection);
 | |
|         this.mouseScrollTimeStamp = currentTime;
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     get isFullscreen() {
 | |
|       return !!(document.fullscreenElement ||
 | |
|                 document.mozFullScreen ||
 | |
|                 document.webkitIsFullScreen ||
 | |
|                 document.msFullscreenElement);
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _notifyStateChange: function PDFPresentationMode_notifyStateChange() {
 | |
|       var event = document.createEvent('CustomEvent');
 | |
|       event.initCustomEvent('presentationmodechanged', true, true, {
 | |
|         active: this.active,
 | |
|         switchInProgress: !!this.switchInProgress
 | |
|       });
 | |
|       window.dispatchEvent(event);
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Used to initialize a timeout when requesting Presentation Mode,
 | |
|      * i.e. when the browser is requested to enter fullscreen mode.
 | |
|      * This timeout is used to prevent the current page from being scrolled
 | |
|      * partially, or completely, out of view when entering Presentation Mode.
 | |
|      * NOTE: This issue seems limited to certain zoom levels (e.g. page-width).
 | |
|      * @private
 | |
|      */
 | |
|     _setSwitchInProgress: function PDFPresentationMode_setSwitchInProgress() {
 | |
|       if (this.switchInProgress) {
 | |
|         clearTimeout(this.switchInProgress);
 | |
|       }
 | |
|       this.switchInProgress = setTimeout(function switchInProgressTimeout() {
 | |
|         this._removeFullscreenChangeListeners();
 | |
|         delete this.switchInProgress;
 | |
|         this._notifyStateChange();
 | |
|       }.bind(this), DELAY_BEFORE_RESETTING_SWITCH_IN_PROGRESS);
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _resetSwitchInProgress:
 | |
|         function PDFPresentationMode_resetSwitchInProgress() {
 | |
|       if (this.switchInProgress) {
 | |
|         clearTimeout(this.switchInProgress);
 | |
|         delete this.switchInProgress;
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _enter: function PDFPresentationMode_enter() {
 | |
|       this.active = true;
 | |
|       this._resetSwitchInProgress();
 | |
|       this._notifyStateChange();
 | |
|       this.container.classList.add(ACTIVE_SELECTOR);
 | |
| 
 | |
|       // Ensure that the correct page is scrolled into view when entering
 | |
|       // Presentation Mode, by waiting until fullscreen mode in enabled.
 | |
|       setTimeout(function enterPresentationModeTimeout() {
 | |
|         this.pdfViewer.currentPageNumber = this.args.page;
 | |
|         this.pdfViewer.currentScaleValue = 'page-fit';
 | |
|       }.bind(this), 0);
 | |
| 
 | |
|       this._addWindowListeners();
 | |
|       this._showControls();
 | |
|       this.contextMenuOpen = false;
 | |
|       this.container.setAttribute('contextmenu', 'viewerContextMenu');
 | |
| 
 | |
|       // Text selection is disabled in Presentation Mode, thus it's not possible
 | |
|       // for the user to deselect text that is selected (e.g. with "Select all")
 | |
|       // when entering Presentation Mode, hence we remove any active selection.
 | |
|       window.getSelection().removeAllRanges();
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _exit: function PDFPresentationMode_exit() {
 | |
|       var page = this.pdfViewer.currentPageNumber;
 | |
|       this.container.classList.remove(ACTIVE_SELECTOR);
 | |
| 
 | |
|       // Ensure that the correct page is scrolled into view when exiting
 | |
|       // Presentation Mode, by waiting until fullscreen mode is disabled.
 | |
|       setTimeout(function exitPresentationModeTimeout() {
 | |
|         this.active = false;
 | |
|         this._removeFullscreenChangeListeners();
 | |
|         this._notifyStateChange();
 | |
| 
 | |
|         this.pdfViewer.currentScaleValue = this.args.previousScale;
 | |
|         this.pdfViewer.currentPageNumber = page;
 | |
|         this.args = null;
 | |
|       }.bind(this), 0);
 | |
| 
 | |
|       this._removeWindowListeners();
 | |
|       this._hideControls();
 | |
|       this._resetMouseScrollState();
 | |
|       this.container.removeAttribute('contextmenu');
 | |
|       this.contextMenuOpen = false;
 | |
| 
 | |
|       if (this.pdfThumbnailViewer) {
 | |
|         this.pdfThumbnailViewer.ensureThumbnailVisible(page);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _mouseDown: function PDFPresentationMode_mouseDown(evt) {
 | |
|       if (this.contextMenuOpen) {
 | |
|         this.contextMenuOpen = false;
 | |
|         evt.preventDefault();
 | |
|         return;
 | |
|       }
 | |
|       if (evt.button === 0) {
 | |
|         // Enable clicking of links in presentation mode. Please note:
 | |
|         // Only links pointing to destinations in the current PDF document work.
 | |
|         var isInternalLink = (evt.target.href &&
 | |
|                               evt.target.classList.contains('internalLink'));
 | |
|         if (!isInternalLink) {
 | |
|           // Unless an internal link was clicked, advance one page.
 | |
|           evt.preventDefault();
 | |
|           this.pdfViewer.currentPageNumber += (evt.shiftKey ? -1 : 1);
 | |
|         }
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _contextMenu: function PDFPresentationMode_contextMenu() {
 | |
|       this.contextMenuOpen = true;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _showControls: function PDFPresentationMode_showControls() {
 | |
|       if (this.controlsTimeout) {
 | |
|         clearTimeout(this.controlsTimeout);
 | |
|       } else {
 | |
|         this.container.classList.add(CONTROLS_SELECTOR);
 | |
|       }
 | |
|       this.controlsTimeout = setTimeout(function showControlsTimeout() {
 | |
|         this.container.classList.remove(CONTROLS_SELECTOR);
 | |
|         delete this.controlsTimeout;
 | |
|       }.bind(this), DELAY_BEFORE_HIDING_CONTROLS);
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _hideControls: function PDFPresentationMode_hideControls() {
 | |
|       if (!this.controlsTimeout) {
 | |
|         return;
 | |
|       }
 | |
|       clearTimeout(this.controlsTimeout);
 | |
|       this.container.classList.remove(CONTROLS_SELECTOR);
 | |
|       delete this.controlsTimeout;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Resets the properties used for tracking mouse scrolling events.
 | |
|      * @private
 | |
|      */
 | |
|     _resetMouseScrollState:
 | |
|         function PDFPresentationMode_resetMouseScrollState() {
 | |
|       this.mouseScrollTimeStamp = 0;
 | |
|       this.mouseScrollDelta = 0;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _addWindowListeners: function PDFPresentationMode_addWindowListeners() {
 | |
|       this.showControlsBind = this._showControls.bind(this);
 | |
|       this.mouseDownBind = this._mouseDown.bind(this);
 | |
|       this.resetMouseScrollStateBind = this._resetMouseScrollState.bind(this);
 | |
|       this.contextMenuBind = this._contextMenu.bind(this);
 | |
| 
 | |
|       window.addEventListener('mousemove', this.showControlsBind);
 | |
|       window.addEventListener('mousedown', this.mouseDownBind);
 | |
|       window.addEventListener('keydown', this.resetMouseScrollStateBind);
 | |
|       window.addEventListener('contextmenu', this.contextMenuBind);
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _removeWindowListeners:
 | |
|         function PDFPresentationMode_removeWindowListeners() {
 | |
|       window.removeEventListener('mousemove', this.showControlsBind);
 | |
|       window.removeEventListener('mousedown', this.mouseDownBind);
 | |
|       window.removeEventListener('keydown', this.resetMouseScrollStateBind);
 | |
|       window.removeEventListener('contextmenu', this.contextMenuBind);
 | |
| 
 | |
|       delete this.showControlsBind;
 | |
|       delete this.mouseDownBind;
 | |
|       delete this.resetMouseScrollStateBind;
 | |
|       delete this.contextMenuBind;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _fullscreenChange: function PDFPresentationMode_fullscreenChange() {
 | |
|       if (this.isFullscreen) {
 | |
|         this._enter();
 | |
|       } else {
 | |
|         this._exit();
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _addFullscreenChangeListeners:
 | |
|         function PDFPresentationMode_addFullscreenChangeListeners() {
 | |
|       this.fullscreenChangeBind = this._fullscreenChange.bind(this);
 | |
| 
 | |
|       window.addEventListener('fullscreenchange', this.fullscreenChangeBind);
 | |
|       window.addEventListener('mozfullscreenchange', this.fullscreenChangeBind);
 | |
|       window.addEventListener('webkitfullscreenchange',
 | |
|                               this.fullscreenChangeBind);
 | |
|       window.addEventListener('MSFullscreenChange', this.fullscreenChangeBind);
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _removeFullscreenChangeListeners:
 | |
|         function PDFPresentationMode_removeFullscreenChangeListeners() {
 | |
|       window.removeEventListener('fullscreenchange', this.fullscreenChangeBind);
 | |
|       window.removeEventListener('mozfullscreenchange',
 | |
|                                  this.fullscreenChangeBind);
 | |
|       window.removeEventListener('webkitfullscreenchange',
 | |
|                               this.fullscreenChangeBind);
 | |
|       window.removeEventListener('MSFullscreenChange',
 | |
|                                  this.fullscreenChangeBind);
 | |
| 
 | |
|       delete this.fullscreenChangeBind;
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   return PDFPresentationMode;
 | |
| })();
 | |
| 
 | |
| 
 | |
| 
 | |
| var GrabToPan = (function GrabToPanClosure() {
 | |
|   /**
 | |
|    * Construct a GrabToPan instance for a given HTML element.
 | |
|    * @param options.element {Element}
 | |
|    * @param options.ignoreTarget {function} optional. See `ignoreTarget(node)`
 | |
|    * @param options.onActiveChanged {function(boolean)} optional. Called
 | |
|    *  when grab-to-pan is (de)activated. The first argument is a boolean that
 | |
|    *  shows whether grab-to-pan is activated.
 | |
|    */
 | |
|   function GrabToPan(options) {
 | |
|     this.element = options.element;
 | |
|     this.document = options.element.ownerDocument;
 | |
|     if (typeof options.ignoreTarget === 'function') {
 | |
|       this.ignoreTarget = options.ignoreTarget;
 | |
|     }
 | |
|     this.onActiveChanged = options.onActiveChanged;
 | |
| 
 | |
|     // Bind the contexts to ensure that `this` always points to
 | |
|     // the GrabToPan instance.
 | |
|     this.activate = this.activate.bind(this);
 | |
|     this.deactivate = this.deactivate.bind(this);
 | |
|     this.toggle = this.toggle.bind(this);
 | |
|     this._onmousedown = this._onmousedown.bind(this);
 | |
|     this._onmousemove = this._onmousemove.bind(this);
 | |
|     this._endPan = this._endPan.bind(this);
 | |
| 
 | |
|     // This overlay will be inserted in the document when the mouse moves during
 | |
|     // a grab operation, to ensure that the cursor has the desired appearance.
 | |
|     var overlay = this.overlay = document.createElement('div');
 | |
|     overlay.className = 'grab-to-pan-grabbing';
 | |
|   }
 | |
|   GrabToPan.prototype = {
 | |
|     /**
 | |
|      * Class name of element which can be grabbed
 | |
|      */
 | |
|     CSS_CLASS_GRAB: 'grab-to-pan-grab',
 | |
| 
 | |
|     /**
 | |
|      * Bind a mousedown event to the element to enable grab-detection.
 | |
|      */
 | |
|     activate: function GrabToPan_activate() {
 | |
|       if (!this.active) {
 | |
|         this.active = true;
 | |
|         this.element.addEventListener('mousedown', this._onmousedown, true);
 | |
|         this.element.classList.add(this.CSS_CLASS_GRAB);
 | |
|         if (this.onActiveChanged) {
 | |
|           this.onActiveChanged(true);
 | |
|         }
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Removes all events. Any pending pan session is immediately stopped.
 | |
|      */
 | |
|     deactivate: function GrabToPan_deactivate() {
 | |
|       if (this.active) {
 | |
|         this.active = false;
 | |
|         this.element.removeEventListener('mousedown', this._onmousedown, true);
 | |
|         this._endPan();
 | |
|         this.element.classList.remove(this.CSS_CLASS_GRAB);
 | |
|         if (this.onActiveChanged) {
 | |
|           this.onActiveChanged(false);
 | |
|         }
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     toggle: function GrabToPan_toggle() {
 | |
|       if (this.active) {
 | |
|         this.deactivate();
 | |
|       } else {
 | |
|         this.activate();
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Whether to not pan if the target element is clicked.
 | |
|      * Override this method to change the default behaviour.
 | |
|      *
 | |
|      * @param node {Element} The target of the event
 | |
|      * @return {boolean} Whether to not react to the click event.
 | |
|      */
 | |
|     ignoreTarget: function GrabToPan_ignoreTarget(node) {
 | |
|       // Use matchesSelector to check whether the clicked element
 | |
|       // is (a child of) an input element / link
 | |
|       return node[matchesSelector](
 | |
|         'a[href], a[href] *, input, textarea, button, button *, select, option'
 | |
|       );
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _onmousedown: function GrabToPan__onmousedown(event) {
 | |
|       if (event.button !== 0 || this.ignoreTarget(event.target)) {
 | |
|         return;
 | |
|       }
 | |
|       if (event.originalTarget) {
 | |
|         try {
 | |
|           /* jshint expr:true */
 | |
|           event.originalTarget.tagName;
 | |
|         } catch (e) {
 | |
|           // Mozilla-specific: element is a scrollbar (XUL element)
 | |
|           return;
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       this.scrollLeftStart = this.element.scrollLeft;
 | |
|       this.scrollTopStart = this.element.scrollTop;
 | |
|       this.clientXStart = event.clientX;
 | |
|       this.clientYStart = event.clientY;
 | |
|       this.document.addEventListener('mousemove', this._onmousemove, true);
 | |
|       this.document.addEventListener('mouseup', this._endPan, true);
 | |
|       // When a scroll event occurs before a mousemove, assume that the user
 | |
|       // dragged a scrollbar (necessary for Opera Presto, Safari and IE)
 | |
|       // (not needed for Chrome/Firefox)
 | |
|       this.element.addEventListener('scroll', this._endPan, true);
 | |
|       event.preventDefault();
 | |
|       event.stopPropagation();
 | |
|       this.document.documentElement.classList.add(this.CSS_CLASS_GRABBING);
 | |
| 
 | |
|       var focusedElement = document.activeElement;
 | |
|       if (focusedElement && !focusedElement.contains(event.target)) {
 | |
|         focusedElement.blur();
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _onmousemove: function GrabToPan__onmousemove(event) {
 | |
|       this.element.removeEventListener('scroll', this._endPan, true);
 | |
|       if (isLeftMouseReleased(event)) {
 | |
|         this._endPan();
 | |
|         return;
 | |
|       }
 | |
|       var xDiff = event.clientX - this.clientXStart;
 | |
|       var yDiff = event.clientY - this.clientYStart;
 | |
|       this.element.scrollTop = this.scrollTopStart - yDiff;
 | |
|       this.element.scrollLeft = this.scrollLeftStart - xDiff;
 | |
|       if (!this.overlay.parentNode) {
 | |
|         document.body.appendChild(this.overlay);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _endPan: function GrabToPan__endPan() {
 | |
|       this.element.removeEventListener('scroll', this._endPan, true);
 | |
|       this.document.removeEventListener('mousemove', this._onmousemove, true);
 | |
|       this.document.removeEventListener('mouseup', this._endPan, true);
 | |
|       if (this.overlay.parentNode) {
 | |
|         this.overlay.parentNode.removeChild(this.overlay);
 | |
|       }
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   // Get the correct (vendor-prefixed) name of the matches method.
 | |
|   var matchesSelector;
 | |
|   ['webkitM', 'mozM', 'msM', 'oM', 'm'].some(function(prefix) {
 | |
|     var name = prefix + 'atches';
 | |
|     if (name in document.documentElement) {
 | |
|       matchesSelector = name;
 | |
|     }
 | |
|     name += 'Selector';
 | |
|     if (name in document.documentElement) {
 | |
|       matchesSelector = name;
 | |
|     }
 | |
|     return matchesSelector; // If found, then truthy, and [].some() ends.
 | |
|   });
 | |
| 
 | |
|   // Browser sniffing because it's impossible to feature-detect
 | |
|   // whether event.which for onmousemove is reliable
 | |
|   var isNotIEorIsIE10plus = !document.documentMode || document.documentMode > 9;
 | |
|   var chrome = window.chrome;
 | |
|   var isChrome15OrOpera15plus = chrome && (chrome.webstore || chrome.app);
 | |
|   //                                       ^ Chrome 15+       ^ Opera 15+
 | |
|   var isSafari6plus = /Apple/.test(navigator.vendor) &&
 | |
|                       /Version\/([6-9]\d*|[1-5]\d+)/.test(navigator.userAgent);
 | |
| 
 | |
|   /**
 | |
|    * Whether the left mouse is not pressed.
 | |
|    * @param event {MouseEvent}
 | |
|    * @return {boolean} True if the left mouse button is not pressed.
 | |
|    *                   False if unsure or if the left mouse button is pressed.
 | |
|    */
 | |
|   function isLeftMouseReleased(event) {
 | |
|     if ('buttons' in event && isNotIEorIsIE10plus) {
 | |
|       // http://www.w3.org/TR/DOM-Level-3-Events/#events-MouseEvent-buttons
 | |
|       // Firefox 15+
 | |
|       // Internet Explorer 10+
 | |
|       return !(event.buttons | 1);
 | |
|     }
 | |
|     if (isChrome15OrOpera15plus || isSafari6plus) {
 | |
|       // Chrome 14+
 | |
|       // Opera 15+
 | |
|       // Safari 6.0+
 | |
|       return event.which === 0;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return GrabToPan;
 | |
| })();
 | |
| 
 | |
| var HandTool = {
 | |
|   initialize: function handToolInitialize(options) {
 | |
|     var toggleHandTool = options.toggleHandTool;
 | |
|     this.handTool = new GrabToPan({
 | |
|       element: options.container,
 | |
|       onActiveChanged: function(isActive) {
 | |
|         if (!toggleHandTool) {
 | |
|           return;
 | |
|         }
 | |
|         if (isActive) {
 | |
|           toggleHandTool.title = 'Disable hand tool';
 | |
|           toggleHandTool.firstElementChild.textContent = 'Disable hand tool';
 | |
|         } else {
 | |
|           toggleHandTool.title = 'Enable hand tool';
 | |
|           toggleHandTool.firstElementChild.textContent = 'Enable hand tool';
 | |
|         }
 | |
|       }
 | |
|     });
 | |
|     if (toggleHandTool) {
 | |
|       toggleHandTool.addEventListener('click', this.toggle.bind(this), false);
 | |
| 
 | |
|       window.addEventListener('localized', function (evt) {
 | |
|         Preferences.get('enableHandToolOnLoad').then(function resolved(value) {
 | |
|           if (value) {
 | |
|             this.handTool.activate();
 | |
|           }
 | |
|         }.bind(this), function rejected(reason) {});
 | |
|       }.bind(this));
 | |
| 
 | |
|       window.addEventListener('presentationmodechanged', function (evt) {
 | |
|         if (evt.detail.switchInProgress) {
 | |
|           return;
 | |
|         }
 | |
|         if (evt.detail.active) {
 | |
|           this.enterPresentationMode();
 | |
|         } else {
 | |
|           this.exitPresentationMode();
 | |
|         }
 | |
|       }.bind(this));
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   toggle: function handToolToggle() {
 | |
|     this.handTool.toggle();
 | |
|     SecondaryToolbar.close();
 | |
|   },
 | |
| 
 | |
|   enterPresentationMode: function handToolEnterPresentationMode() {
 | |
|     if (this.handTool.active) {
 | |
|       this.wasActive = true;
 | |
|       this.handTool.deactivate();
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   exitPresentationMode: function handToolExitPresentationMode() {
 | |
|     if (this.wasActive) {
 | |
|       this.wasActive = null;
 | |
|       this.handTool.activate();
 | |
|     }
 | |
|   }
 | |
| };
 | |
| 
 | |
| 
 | |
| var OverlayManager = {
 | |
|   overlays: {},
 | |
|   active: null,
 | |
| 
 | |
|   /**
 | |
|    * @param {string} name The name of the overlay that is registered. This must
 | |
|    *                 be equal to the ID of the overlay's DOM element.
 | |
|    * @param {function} callerCloseMethod (optional) The method that, if present,
 | |
|    *                   will call OverlayManager.close from the Object
 | |
|    *                   registering the overlay. Access to this method is
 | |
|    *                   necessary in order to run cleanup code when e.g.
 | |
|    *                   the overlay is force closed. The default is null.
 | |
|    * @param {boolean} canForceClose (optional) Indicates if opening the overlay
 | |
|    *                  will close an active overlay. The default is false.
 | |
|    * @returns {Promise} A promise that is resolved when the overlay has been
 | |
|    *                    registered.
 | |
|    */
 | |
|   register: function overlayManagerRegister(name,
 | |
|                                             callerCloseMethod, canForceClose) {
 | |
|     return new Promise(function (resolve) {
 | |
|       var element, container;
 | |
|       if (!name || !(element = document.getElementById(name)) ||
 | |
|           !(container = element.parentNode)) {
 | |
|         throw new Error('Not enough parameters.');
 | |
|       } else if (this.overlays[name]) {
 | |
|         throw new Error('The overlay is already registered.');
 | |
|       }
 | |
|       this.overlays[name] = { element: element,
 | |
|                               container: container,
 | |
|                               callerCloseMethod: (callerCloseMethod || null),
 | |
|                               canForceClose: (canForceClose || false) };
 | |
|       resolve();
 | |
|     }.bind(this));
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * @param {string} name The name of the overlay that is unregistered.
 | |
|    * @returns {Promise} A promise that is resolved when the overlay has been
 | |
|    *                    unregistered.
 | |
|    */
 | |
|   unregister: function overlayManagerUnregister(name) {
 | |
|     return new Promise(function (resolve) {
 | |
|       if (!this.overlays[name]) {
 | |
|         throw new Error('The overlay does not exist.');
 | |
|       } else if (this.active === name) {
 | |
|         throw new Error('The overlay cannot be removed while it is active.');
 | |
|       }
 | |
|       delete this.overlays[name];
 | |
| 
 | |
|       resolve();
 | |
|     }.bind(this));
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * @param {string} name The name of the overlay that should be opened.
 | |
|    * @returns {Promise} A promise that is resolved when the overlay has been
 | |
|    *                    opened.
 | |
|    */
 | |
|   open: function overlayManagerOpen(name) {
 | |
|     return new Promise(function (resolve) {
 | |
|       if (!this.overlays[name]) {
 | |
|         throw new Error('The overlay does not exist.');
 | |
|       } else if (this.active) {
 | |
|         if (this.overlays[name].canForceClose) {
 | |
|           this._closeThroughCaller();
 | |
|         } else if (this.active === name) {
 | |
|           throw new Error('The overlay is already active.');
 | |
|         } else {
 | |
|           throw new Error('Another overlay is currently active.');
 | |
|         }
 | |
|       }
 | |
|       this.active = name;
 | |
|       this.overlays[this.active].element.classList.remove('hidden');
 | |
|       this.overlays[this.active].container.classList.remove('hidden');
 | |
| 
 | |
|       window.addEventListener('keydown', this._keyDown);
 | |
|       resolve();
 | |
|     }.bind(this));
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * @param {string} name The name of the overlay that should be closed.
 | |
|    * @returns {Promise} A promise that is resolved when the overlay has been
 | |
|    *                    closed.
 | |
|    */
 | |
|   close: function overlayManagerClose(name) {
 | |
|     return new Promise(function (resolve) {
 | |
|       if (!this.overlays[name]) {
 | |
|         throw new Error('The overlay does not exist.');
 | |
|       } else if (!this.active) {
 | |
|         throw new Error('The overlay is currently not active.');
 | |
|       } else if (this.active !== name) {
 | |
|         throw new Error('Another overlay is currently active.');
 | |
|       }
 | |
|       this.overlays[this.active].container.classList.add('hidden');
 | |
|       this.overlays[this.active].element.classList.add('hidden');
 | |
|       this.active = null;
 | |
| 
 | |
|       window.removeEventListener('keydown', this._keyDown);
 | |
|       resolve();
 | |
|     }.bind(this));
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * @private
 | |
|    */
 | |
|   _keyDown: function overlayManager_keyDown(evt) {
 | |
|     var self = OverlayManager;
 | |
|     if (self.active && evt.keyCode === 27) { // Esc key.
 | |
|       self._closeThroughCaller();
 | |
|       evt.preventDefault();
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * @private
 | |
|    */
 | |
|   _closeThroughCaller: function overlayManager_closeThroughCaller() {
 | |
|     if (this.overlays[this.active].callerCloseMethod) {
 | |
|       this.overlays[this.active].callerCloseMethod();
 | |
|     }
 | |
|     if (this.active) {
 | |
|       this.close(this.active);
 | |
|     }
 | |
|   }
 | |
| };
 | |
| 
 | |
| 
 | |
| var PasswordPrompt = {
 | |
|   overlayName: null,
 | |
|   updatePassword: null,
 | |
|   reason: null,
 | |
|   passwordField: null,
 | |
|   passwordText: null,
 | |
|   passwordSubmit: null,
 | |
|   passwordCancel: null,
 | |
| 
 | |
|   initialize: function secondaryToolbarInitialize(options) {
 | |
|     this.overlayName = options.overlayName;
 | |
|     this.passwordField = options.passwordField;
 | |
|     this.passwordText = options.passwordText;
 | |
|     this.passwordSubmit = options.passwordSubmit;
 | |
|     this.passwordCancel = options.passwordCancel;
 | |
| 
 | |
|     // Attach the event listeners.
 | |
|     this.passwordSubmit.addEventListener('click',
 | |
|       this.verifyPassword.bind(this));
 | |
| 
 | |
|     this.passwordCancel.addEventListener('click', this.close.bind(this));
 | |
| 
 | |
|     this.passwordField.addEventListener('keydown', function (e) {
 | |
|       if (e.keyCode === 13) { // Enter key
 | |
|         this.verifyPassword();
 | |
|       }
 | |
|     }.bind(this));
 | |
| 
 | |
|     OverlayManager.register(this.overlayName, this.close.bind(this), true);
 | |
|   },
 | |
| 
 | |
|   open: function passwordPromptOpen() {
 | |
|     OverlayManager.open(this.overlayName).then(function () {
 | |
|       this.passwordField.type = 'password';
 | |
|       this.passwordField.focus();
 | |
| 
 | |
|       var promptString = 'Enter the password to open this PDF file.';
 | |
| 
 | |
|       if (this.reason === PDFJS.PasswordResponses.INCORRECT_PASSWORD) {
 | |
|         promptString = 'Invalid password. Please try again.';
 | |
|       }
 | |
| 
 | |
|       this.passwordText.textContent = promptString;
 | |
|     }.bind(this));
 | |
|   },
 | |
| 
 | |
|   close: function passwordPromptClose() {
 | |
|     OverlayManager.close(this.overlayName).then(function () {
 | |
|       this.passwordField.value = '';
 | |
|       this.passwordField.type = '';
 | |
|     }.bind(this));
 | |
|   },
 | |
| 
 | |
|   verifyPassword: function passwordPromptVerifyPassword() {
 | |
|     var password = this.passwordField.value;
 | |
|     if (password && password.length > 0) {
 | |
|       this.close();
 | |
|       return this.updatePassword(password);
 | |
|     }
 | |
|   }
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * @typedef {Object} PDFDocumentPropertiesOptions
 | |
|  * @property {string} overlayName - Name/identifier for the overlay.
 | |
|  * @property {Object} fields - Names and elements of the overlay's fields.
 | |
|  * @property {HTMLButtonElement} closeButton - Button for closing the overlay.
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * @class
 | |
|  */
 | |
| var PDFDocumentProperties = (function PDFDocumentPropertiesClosure() {
 | |
|   /**
 | |
|    * @constructs PDFDocumentProperties
 | |
|    * @param {PDFDocumentPropertiesOptions} options
 | |
|    */
 | |
|   function PDFDocumentProperties(options) {
 | |
|     this.fields = options.fields;
 | |
|     this.overlayName = options.overlayName;
 | |
| 
 | |
|     this.rawFileSize = 0;
 | |
|     this.url = null;
 | |
|     this.pdfDocument = null;
 | |
| 
 | |
|     // Bind the event listener for the Close button.
 | |
|     if (options.closeButton) {
 | |
|       options.closeButton.addEventListener('click', this.close.bind(this));
 | |
|     }
 | |
| 
 | |
|     this.dataAvailablePromise = new Promise(function (resolve) {
 | |
|       this.resolveDataAvailable = resolve;
 | |
|     }.bind(this));
 | |
| 
 | |
|     OverlayManager.register(this.overlayName, this.close.bind(this));
 | |
|   }
 | |
| 
 | |
|   PDFDocumentProperties.prototype = {
 | |
|     /**
 | |
|      * Open the document properties overlay.
 | |
|      */
 | |
|     open: function PDFDocumentProperties_open() {
 | |
|       Promise.all([OverlayManager.open(this.overlayName),
 | |
|                    this.dataAvailablePromise]).then(function () {
 | |
|         this._getProperties();
 | |
|       }.bind(this));
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Close the document properties overlay.
 | |
|      */
 | |
|     close: function PDFDocumentProperties_close() {
 | |
|       OverlayManager.close(this.overlayName);
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Set the file size of the PDF document. This method is used to
 | |
|      * update the file size in the document properties overlay once it
 | |
|      * is known so we do not have to wait until the entire file is loaded.
 | |
|      *
 | |
|      * @param {number} fileSize - The file size of the PDF document.
 | |
|      */
 | |
|     setFileSize: function PDFDocumentProperties_setFileSize(fileSize) {
 | |
|       if (fileSize > 0) {
 | |
|         this.rawFileSize = fileSize;
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Set a reference to the PDF document and the URL in order
 | |
|      * to populate the overlay fields with the document properties.
 | |
|      * Note that the overlay will contain no information if this method
 | |
|      * is not called.
 | |
|      *
 | |
|      * @param {Object} pdfDocument - A reference to the PDF document.
 | |
|      * @param {string} url - The URL of the document.
 | |
|      */
 | |
|     setDocumentAndUrl:
 | |
|         function PDFDocumentProperties_setDocumentAndUrl(pdfDocument, url) {
 | |
|       this.pdfDocument = pdfDocument;
 | |
|       this.url = url;
 | |
|       this.resolveDataAvailable();
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _getProperties: function PDFDocumentProperties_getProperties() {
 | |
|       if (!OverlayManager.active) {
 | |
|         // If the dialog was closed before dataAvailablePromise was resolved,
 | |
|         // don't bother updating the properties.
 | |
|         return;
 | |
|       }
 | |
|       // Get the file size (if it hasn't already been set).
 | |
|       this.pdfDocument.getDownloadInfo().then(function(data) {
 | |
|         if (data.length === this.rawFileSize) {
 | |
|           return;
 | |
|         }
 | |
|         this.setFileSize(data.length);
 | |
|         this._updateUI(this.fields['fileSize'], this._parseFileSize());
 | |
|       }.bind(this));
 | |
| 
 | |
|       // Get the document properties.
 | |
|       this.pdfDocument.getMetadata().then(function(data) {
 | |
|         var content = {
 | |
|           'fileName': getPDFFileNameFromURL(this.url),
 | |
|           'fileSize': this._parseFileSize(),
 | |
|           'title': data.info.Title,
 | |
|           'author': data.info.Author,
 | |
|           'subject': data.info.Subject,
 | |
|           'keywords': data.info.Keywords,
 | |
|           'creationDate': this._parseDate(data.info.CreationDate),
 | |
|           'modificationDate': this._parseDate(data.info.ModDate),
 | |
|           'creator': data.info.Creator,
 | |
|           'producer': data.info.Producer,
 | |
|           'version': data.info.PDFFormatVersion,
 | |
|           'pageCount': this.pdfDocument.numPages
 | |
|         };
 | |
| 
 | |
|         // Show the properties in the dialog.
 | |
|         for (var identifier in content) {
 | |
|           this._updateUI(this.fields[identifier], content[identifier]);
 | |
|         }
 | |
|       }.bind(this));
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _updateUI: function PDFDocumentProperties_updateUI(field, content) {
 | |
|       if (field && content !== undefined && content !== '') {
 | |
|         field.textContent = content;
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _parseFileSize: function PDFDocumentProperties_parseFileSize() {
 | |
|       var fileSize = this.rawFileSize, kb = fileSize / 1024;
 | |
|       if (!kb) {
 | |
|         return;
 | |
|       } else if (kb < 1024) {
 | |
|         return kb + ' KB ' + size_b + ' bytes';
 | |
|       } else {
 | |
|         return (kb / 1024) + ' MB ' + size_b + ' bytes';
 | |
|        }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _parseDate: function PDFDocumentProperties_parseDate(inputDate) {
 | |
|       // This is implemented according to the PDF specification, but note that
 | |
|       // Adobe Reader doesn't handle changing the date to universal time
 | |
|       // and doesn't use the user's time zone (they're effectively ignoring
 | |
|       // the HH' and mm' parts of the date string).
 | |
|       var dateToParse = inputDate;
 | |
|       if (dateToParse === undefined) {
 | |
|         return '';
 | |
|       }
 | |
| 
 | |
|       // Remove the D: prefix if it is available.
 | |
|       if (dateToParse.substring(0,2) === 'D:') {
 | |
|         dateToParse = dateToParse.substring(2);
 | |
|       }
 | |
| 
 | |
|       // Get all elements from the PDF date string.
 | |
|       // JavaScript's Date object expects the month to be between
 | |
|       // 0 and 11 instead of 1 and 12, so we're correcting for this.
 | |
|       var year = parseInt(dateToParse.substring(0,4), 10);
 | |
|       var month = parseInt(dateToParse.substring(4,6), 10) - 1;
 | |
|       var day = parseInt(dateToParse.substring(6,8), 10);
 | |
|       var hours = parseInt(dateToParse.substring(8,10), 10);
 | |
|       var minutes = parseInt(dateToParse.substring(10,12), 10);
 | |
|       var seconds = parseInt(dateToParse.substring(12,14), 10);
 | |
|       var utRel = dateToParse.substring(14,15);
 | |
|       var offsetHours = parseInt(dateToParse.substring(15,17), 10);
 | |
|       var offsetMinutes = parseInt(dateToParse.substring(18,20), 10);
 | |
| 
 | |
|       // As per spec, utRel = 'Z' means equal to universal time.
 | |
|       // The other cases ('-' and '+') have to be handled here.
 | |
|       if (utRel === '-') {
 | |
|         hours += offsetHours;
 | |
|         minutes += offsetMinutes;
 | |
|       } else if (utRel === '+') {
 | |
|         hours -= offsetHours;
 | |
|         minutes -= offsetMinutes;
 | |
|       }
 | |
| 
 | |
|       // Return the new date format from the user's locale.
 | |
|       var date = new Date(Date.UTC(year, month, day, hours, minutes, seconds));
 | |
|       var dateString = date.toLocaleDateString();
 | |
|       var timeString = date.toLocaleTimeString();
 | |
|       return dateString + ', ' + timeString;
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   return PDFDocumentProperties;
 | |
| })();
 | |
| 
 | |
| 
 | |
| var PresentationModeState = {
 | |
|   UNKNOWN: 0,
 | |
|   NORMAL: 1,
 | |
|   CHANGING: 2,
 | |
|   FULLSCREEN: 3,
 | |
| };
 | |
| 
 | |
| var IGNORE_CURRENT_POSITION_ON_ZOOM = false;
 | |
| var DEFAULT_CACHE_SIZE = 10;
 | |
| 
 | |
| 
 | |
| var CLEANUP_TIMEOUT = 30000;
 | |
| 
 | |
| var RenderingStates = {
 | |
|   INITIAL: 0,
 | |
|   RUNNING: 1,
 | |
|   PAUSED: 2,
 | |
|   FINISHED: 3
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * Controls rendering of the views for pages and thumbnails.
 | |
|  * @class
 | |
|  */
 | |
| var PDFRenderingQueue = (function PDFRenderingQueueClosure() {
 | |
|   /**
 | |
|    * @constructs
 | |
|    */
 | |
|   function PDFRenderingQueue() {
 | |
|     this.pdfViewer = null;
 | |
|     this.pdfThumbnailViewer = null;
 | |
|     this.onIdle = null;
 | |
| 
 | |
|     this.highestPriorityPage = null;
 | |
|     this.idleTimeout = null;
 | |
|     this.printing = false;
 | |
|     this.isThumbnailViewEnabled = false;
 | |
|   }
 | |
| 
 | |
|   PDFRenderingQueue.prototype = /** @lends PDFRenderingQueue.prototype */ {
 | |
|     /**
 | |
|      * @param {PDFViewer} pdfViewer
 | |
|      */
 | |
|     setViewer: function PDFRenderingQueue_setViewer(pdfViewer) {
 | |
|       this.pdfViewer = pdfViewer;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @param {PDFThumbnailViewer} pdfThumbnailViewer
 | |
|      */
 | |
|     setThumbnailViewer:
 | |
|         function PDFRenderingQueue_setThumbnailViewer(pdfThumbnailViewer) {
 | |
|       this.pdfThumbnailViewer = pdfThumbnailViewer;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @param {IRenderableView} view
 | |
|      * @returns {boolean}
 | |
|      */
 | |
|     isHighestPriority: function PDFRenderingQueue_isHighestPriority(view) {
 | |
|       return this.highestPriorityPage === view.renderingId;
 | |
|     },
 | |
| 
 | |
|     renderHighestPriority: function
 | |
|         PDFRenderingQueue_renderHighestPriority(currentlyVisiblePages) {
 | |
|       if (this.idleTimeout) {
 | |
|         clearTimeout(this.idleTimeout);
 | |
|         this.idleTimeout = null;
 | |
|       }
 | |
| 
 | |
|       // Pages have a higher priority than thumbnails, so check them first.
 | |
|       if (this.pdfViewer.forceRendering(currentlyVisiblePages)) {
 | |
|         return;
 | |
|       }
 | |
|       // No pages needed rendering so check thumbnails.
 | |
|       if (this.pdfThumbnailViewer && this.isThumbnailViewEnabled) {
 | |
|         if (this.pdfThumbnailViewer.forceRendering()) {
 | |
|           return;
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       if (this.printing) {
 | |
|         // If printing is currently ongoing do not reschedule cleanup.
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       if (this.onIdle) {
 | |
|         this.idleTimeout = setTimeout(this.onIdle.bind(this), CLEANUP_TIMEOUT);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     getHighestPriority: function
 | |
|         PDFRenderingQueue_getHighestPriority(visible, views, scrolledDown) {
 | |
|       // The state has changed figure out which page has the highest priority to
 | |
|       // render next (if any).
 | |
|       // Priority:
 | |
|       // 1 visible pages
 | |
|       // 2 if last scrolled down page after the visible pages
 | |
|       // 2 if last scrolled up page before the visible pages
 | |
|       var visibleViews = visible.views;
 | |
| 
 | |
|       var numVisible = visibleViews.length;
 | |
|       if (numVisible === 0) {
 | |
|         return false;
 | |
|       }
 | |
|       for (var i = 0; i < numVisible; ++i) {
 | |
|         var view = visibleViews[i].view;
 | |
|         if (!this.isViewFinished(view)) {
 | |
|           return view;
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       // All the visible views have rendered, try to render next/previous pages.
 | |
|       if (scrolledDown) {
 | |
|         var nextPageIndex = visible.last.id;
 | |
|         // ID's start at 1 so no need to add 1.
 | |
|         if (views[nextPageIndex] &&
 | |
|             !this.isViewFinished(views[nextPageIndex])) {
 | |
|           return views[nextPageIndex];
 | |
|         }
 | |
|       } else {
 | |
|         var previousPageIndex = visible.first.id - 2;
 | |
|         if (views[previousPageIndex] &&
 | |
|           !this.isViewFinished(views[previousPageIndex])) {
 | |
|           return views[previousPageIndex];
 | |
|         }
 | |
|       }
 | |
|       // Everything that needs to be rendered has been.
 | |
|       return null;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @param {IRenderableView} view
 | |
|      * @returns {boolean}
 | |
|      */
 | |
|     isViewFinished: function PDFRenderingQueue_isViewFinished(view) {
 | |
|       return view.renderingState === RenderingStates.FINISHED;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Render a page or thumbnail view. This calls the appropriate function
 | |
|      * based on the views state. If the view is already rendered it will return
 | |
|      * false.
 | |
|      * @param {IRenderableView} view
 | |
|      */
 | |
|     renderView: function PDFRenderingQueue_renderView(view) {
 | |
|       var state = view.renderingState;
 | |
|       switch (state) {
 | |
|         case RenderingStates.FINISHED:
 | |
|           return false;
 | |
|         case RenderingStates.PAUSED:
 | |
|           this.highestPriorityPage = view.renderingId;
 | |
|           view.resume();
 | |
|           break;
 | |
|         case RenderingStates.RUNNING:
 | |
|           this.highestPriorityPage = view.renderingId;
 | |
|           break;
 | |
|         case RenderingStates.INITIAL:
 | |
|           this.highestPriorityPage = view.renderingId;
 | |
|           var continueRendering = function () {
 | |
|             this.renderHighestPriority();
 | |
|           }.bind(this);
 | |
|           view.draw().then(continueRendering, continueRendering);
 | |
|           break;
 | |
|       }
 | |
|       return true;
 | |
|     },
 | |
|   };
 | |
| 
 | |
|   return PDFRenderingQueue;
 | |
| })();
 | |
| 
 | |
| 
 | |
| var TEXT_LAYER_RENDER_DELAY = 200; // ms
 | |
| 
 | |
| /**
 | |
|  * @typedef {Object} PDFPageViewOptions
 | |
|  * @property {HTMLDivElement} container - The viewer element.
 | |
|  * @property {number} id - The page unique ID (normally its number).
 | |
|  * @property {number} scale - The page scale display.
 | |
|  * @property {PageViewport} defaultViewport - The page viewport.
 | |
|  * @property {PDFRenderingQueue} renderingQueue - The rendering queue object.
 | |
|  * @property {IPDFTextLayerFactory} textLayerFactory
 | |
|  * @property {IPDFAnnotationLayerFactory} annotationLayerFactory
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * @class
 | |
|  * @implements {IRenderableView}
 | |
|  */
 | |
| var PDFPageView = (function PDFPageViewClosure() {
 | |
|   /**
 | |
|    * @constructs PDFPageView
 | |
|    * @param {PDFPageViewOptions} options
 | |
|    */
 | |
|   function PDFPageView(options) {
 | |
|     var container = options.container;
 | |
|     var id = options.id;
 | |
|     var scale = options.scale;
 | |
|     var defaultViewport = options.defaultViewport;
 | |
|     var renderingQueue = options.renderingQueue;
 | |
|     var textLayerFactory = options.textLayerFactory;
 | |
|     var annotationLayerFactory = options.annotationLayerFactory;
 | |
| 
 | |
|     this.id = id;
 | |
|     this.renderingId = 'page' + id;
 | |
| 
 | |
|     this.rotation = 0;
 | |
|     this.scale = scale || DEFAULT_SCALE;
 | |
|     this.viewport = defaultViewport;
 | |
|     this.pdfPageRotate = defaultViewport.rotation;
 | |
|     this.hasRestrictedScaling = false;
 | |
| 
 | |
|     this.renderingQueue = renderingQueue;
 | |
|     this.textLayerFactory = textLayerFactory;
 | |
|     this.annotationLayerFactory = annotationLayerFactory;
 | |
| 
 | |
|     this.renderingState = RenderingStates.INITIAL;
 | |
|     this.resume = null;
 | |
| 
 | |
|     this.onBeforeDraw = null;
 | |
|     this.onAfterDraw = null;
 | |
| 
 | |
|     this.textLayer = null;
 | |
| 
 | |
|     this.zoomLayer = null;
 | |
| 
 | |
|     this.annotationLayer = null;
 | |
| 
 | |
|     var div = document.createElement('div');
 | |
|     div.id = 'pageContainer' + this.id;
 | |
|     div.className = 'page';
 | |
|     div.style.width = Math.floor(this.viewport.width) + 'px';
 | |
|     div.style.height = Math.floor(this.viewport.height) + 'px';
 | |
|     div.setAttribute('data-page-number', this.id);
 | |
|     this.div = div;
 | |
| 
 | |
|     container.appendChild(div);
 | |
|   }
 | |
| 
 | |
|   PDFPageView.prototype = {
 | |
|     setPdfPage: function PDFPageView_setPdfPage(pdfPage) {
 | |
|       this.pdfPage = pdfPage;
 | |
|       this.pdfPageRotate = pdfPage.rotate;
 | |
|       var totalRotation = (this.rotation + this.pdfPageRotate) % 360;
 | |
|       this.viewport = pdfPage.getViewport(this.scale * CSS_UNITS,
 | |
|                                           totalRotation);
 | |
|       this.stats = pdfPage.stats;
 | |
|       this.reset();
 | |
|     },
 | |
| 
 | |
|     destroy: function PDFPageView_destroy() {
 | |
|       this.zoomLayer = null;
 | |
|       this.reset();
 | |
|       if (this.pdfPage) {
 | |
|         this.pdfPage.cleanup();
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     reset: function PDFPageView_reset(keepZoomLayer, keepAnnotations) {
 | |
|       if (this.renderTask) {
 | |
|         this.renderTask.cancel();
 | |
|       }
 | |
|       this.resume = null;
 | |
|       this.renderingState = RenderingStates.INITIAL;
 | |
| 
 | |
|       var div = this.div;
 | |
|       div.style.width = Math.floor(this.viewport.width) + 'px';
 | |
|       div.style.height = Math.floor(this.viewport.height) + 'px';
 | |
| 
 | |
|       var childNodes = div.childNodes;
 | |
|       var currentZoomLayerNode = (keepZoomLayer && this.zoomLayer) || null;
 | |
|       var currentAnnotationNode = (keepAnnotations && this.annotationLayer &&
 | |
|                                    this.annotationLayer.div) || null;
 | |
|       for (var i = childNodes.length - 1; i >= 0; i--) {
 | |
|         var node = childNodes[i];
 | |
|         if (currentZoomLayerNode === node || currentAnnotationNode === node) {
 | |
|           continue;
 | |
|         }
 | |
|         div.removeChild(node);
 | |
|       }
 | |
|       div.removeAttribute('data-loaded');
 | |
| 
 | |
|       if (currentAnnotationNode) {
 | |
|         // Hide annotationLayer until all elements are resized
 | |
|         // so they are not displayed on the already-resized page
 | |
|         this.annotationLayer.hide();
 | |
|       } else {
 | |
|         this.annotationLayer = null;
 | |
|       }
 | |
| 
 | |
|       if (this.canvas && !currentZoomLayerNode) {
 | |
|         // Zeroing the width and height causes Firefox to release graphics
 | |
|         // resources immediately, which can greatly reduce memory consumption.
 | |
|         this.canvas.width = 0;
 | |
|         this.canvas.height = 0;
 | |
|         delete this.canvas;
 | |
|       }
 | |
| 
 | |
|       this.loadingIconDiv = document.createElement('div');
 | |
|       this.loadingIconDiv.className = 'loadingIcon';
 | |
|       div.appendChild(this.loadingIconDiv);
 | |
|     },
 | |
| 
 | |
|     update: function PDFPageView_update(scale, rotation) {
 | |
|       this.scale = scale || this.scale;
 | |
| 
 | |
|       if (typeof rotation !== 'undefined') {
 | |
|         this.rotation = rotation;
 | |
|       }
 | |
| 
 | |
|       var totalRotation = (this.rotation + this.pdfPageRotate) % 360;
 | |
|       this.viewport = this.viewport.clone({
 | |
|         scale: this.scale * CSS_UNITS,
 | |
|         rotation: totalRotation
 | |
|       });
 | |
| 
 | |
|       var isScalingRestricted = false;
 | |
|       if (this.canvas && PDFJS.maxCanvasPixels > 0) {
 | |
|         var outputScale = this.outputScale;
 | |
|         var pixelsInViewport = this.viewport.width * this.viewport.height;
 | |
|         var maxScale = Math.sqrt(PDFJS.maxCanvasPixels / pixelsInViewport);
 | |
|         if (((Math.floor(this.viewport.width) * outputScale.sx) | 0) *
 | |
|             ((Math.floor(this.viewport.height) * outputScale.sy) | 0) >
 | |
|             PDFJS.maxCanvasPixels) {
 | |
|           isScalingRestricted = true;
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       if (this.canvas) {
 | |
|         if (PDFJS.useOnlyCssZoom ||
 | |
|             (this.hasRestrictedScaling && isScalingRestricted)) {
 | |
|           this.cssTransform(this.canvas, true);
 | |
| 
 | |
|           var event = document.createEvent('CustomEvent');
 | |
|           event.initCustomEvent('pagerendered', true, true, {
 | |
|             pageNumber: this.id,
 | |
|             cssTransform: true,
 | |
|           });
 | |
|           this.div.dispatchEvent(event);
 | |
| 
 | |
|           return;
 | |
|         }
 | |
|         if (!this.zoomLayer) {
 | |
|           this.zoomLayer = this.canvas.parentNode;
 | |
|           this.zoomLayer.style.position = 'absolute';
 | |
|         }
 | |
|       }
 | |
|       if (this.zoomLayer) {
 | |
|         this.cssTransform(this.zoomLayer.firstChild);
 | |
|       }
 | |
|       this.reset(/* keepZoomLayer = */ true, /* keepAnnotations = */ true);
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Called when moved in the parent's container.
 | |
|      */
 | |
|     updatePosition: function PDFPageView_updatePosition() {
 | |
|       if (this.textLayer) {
 | |
|         this.textLayer.render(TEXT_LAYER_RENDER_DELAY);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     cssTransform: function PDFPageView_transform(canvas, redrawAnnotations) {
 | |
|       var CustomStyle = PDFJS.CustomStyle;
 | |
| 
 | |
|       // Scale canvas, canvas wrapper, and page container.
 | |
|       var width = this.viewport.width;
 | |
|       var height = this.viewport.height;
 | |
|       var div = this.div;
 | |
|       canvas.style.width = canvas.parentNode.style.width = div.style.width =
 | |
|         Math.floor(width) + 'px';
 | |
|       canvas.style.height = canvas.parentNode.style.height = div.style.height =
 | |
|         Math.floor(height) + 'px';
 | |
|       // The canvas may have been originally rotated, rotate relative to that.
 | |
|       var relativeRotation = this.viewport.rotation - canvas._viewport.rotation;
 | |
|       var absRotation = Math.abs(relativeRotation);
 | |
|       var scaleX = 1, scaleY = 1;
 | |
|       if (absRotation === 90 || absRotation === 270) {
 | |
|         // Scale x and y because of the rotation.
 | |
|         scaleX = height / width;
 | |
|         scaleY = width / height;
 | |
|       }
 | |
|       var cssTransform = 'rotate(' + relativeRotation + 'deg) ' +
 | |
|         'scale(' + scaleX + ',' + scaleY + ')';
 | |
|       CustomStyle.setProp('transform', canvas, cssTransform);
 | |
| 
 | |
|       if (this.textLayer) {
 | |
|         // Rotating the text layer is more complicated since the divs inside the
 | |
|         // the text layer are rotated.
 | |
|         // TODO: This could probably be simplified by drawing the text layer in
 | |
|         // one orientation then rotating overall.
 | |
|         var textLayerViewport = this.textLayer.viewport;
 | |
|         var textRelativeRotation = this.viewport.rotation -
 | |
|           textLayerViewport.rotation;
 | |
|         var textAbsRotation = Math.abs(textRelativeRotation);
 | |
|         var scale = width / textLayerViewport.width;
 | |
|         if (textAbsRotation === 90 || textAbsRotation === 270) {
 | |
|           scale = width / textLayerViewport.height;
 | |
|         }
 | |
|         var textLayerDiv = this.textLayer.textLayerDiv;
 | |
|         var transX, transY;
 | |
|         switch (textAbsRotation) {
 | |
|           case 0:
 | |
|             transX = transY = 0;
 | |
|             break;
 | |
|           case 90:
 | |
|             transX = 0;
 | |
|             transY = '-' + textLayerDiv.style.height;
 | |
|             break;
 | |
|           case 180:
 | |
|             transX = '-' + textLayerDiv.style.width;
 | |
|             transY = '-' + textLayerDiv.style.height;
 | |
|             break;
 | |
|           case 270:
 | |
|             transX = '-' + textLayerDiv.style.width;
 | |
|             transY = 0;
 | |
|             break;
 | |
|           default:
 | |
|             console.error('Bad rotation value.');
 | |
|             break;
 | |
|         }
 | |
|         CustomStyle.setProp('transform', textLayerDiv,
 | |
|             'rotate(' + textAbsRotation + 'deg) ' +
 | |
|             'scale(' + scale + ', ' + scale + ') ' +
 | |
|             'translate(' + transX + ', ' + transY + ')');
 | |
|         CustomStyle.setProp('transformOrigin', textLayerDiv, '0% 0%');
 | |
|       }
 | |
| 
 | |
|       if (redrawAnnotations && this.annotationLayer) {
 | |
|         this.annotationLayer.render(this.viewport, 'display');
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     get width() {
 | |
|       return this.viewport.width;
 | |
|     },
 | |
| 
 | |
|     get height() {
 | |
|       return this.viewport.height;
 | |
|     },
 | |
| 
 | |
|     getPagePoint: function PDFPageView_getPagePoint(x, y) {
 | |
|       return this.viewport.convertToPdfPoint(x, y);
 | |
|     },
 | |
| 
 | |
|     draw: function PDFPageView_draw() {
 | |
|       if (this.renderingState !== RenderingStates.INITIAL) {
 | |
|         console.error('Must be in new state before drawing');
 | |
|       }
 | |
| 
 | |
|       this.renderingState = RenderingStates.RUNNING;
 | |
| 
 | |
|       var pdfPage = this.pdfPage;
 | |
|       var viewport = this.viewport;
 | |
|       var div = this.div;
 | |
|       // Wrap the canvas so if it has a css transform for highdpi the overflow
 | |
|       // will be hidden in FF.
 | |
|       var canvasWrapper = document.createElement('div');
 | |
|       canvasWrapper.style.width = div.style.width;
 | |
|       canvasWrapper.style.height = div.style.height;
 | |
|       canvasWrapper.classList.add('canvasWrapper');
 | |
| 
 | |
|       var canvas = document.createElement('canvas');
 | |
|       canvas.id = 'page' + this.id;
 | |
|       // Keep the canvas hidden until the first draw callback, or until drawing
 | |
|       // is complete when `!this.renderingQueue`, to prevent black flickering.
 | |
|       canvas.setAttribute('hidden', 'hidden');
 | |
|       var isCanvasHidden = true;
 | |
| 
 | |
|       canvasWrapper.appendChild(canvas);
 | |
|       if (this.annotationLayer && this.annotationLayer.div) {
 | |
|         // annotationLayer needs to stay on top
 | |
|         div.insertBefore(canvasWrapper, this.annotationLayer.div);
 | |
|       } else {
 | |
|         div.appendChild(canvasWrapper);
 | |
|       }
 | |
|       this.canvas = canvas;
 | |
| 
 | |
|       canvas.mozOpaque = true;
 | |
|       var ctx = canvas.getContext('2d', {alpha: false});
 | |
|       var outputScale = getOutputScale(ctx);
 | |
|       this.outputScale = outputScale;
 | |
| 
 | |
|       if (PDFJS.useOnlyCssZoom) {
 | |
|         var actualSizeViewport = viewport.clone({scale: CSS_UNITS});
 | |
|         // Use a scale that will make the canvas be the original intended size
 | |
|         // of the page.
 | |
|         outputScale.sx *= actualSizeViewport.width / viewport.width;
 | |
|         outputScale.sy *= actualSizeViewport.height / viewport.height;
 | |
|         outputScale.scaled = true;
 | |
|       }
 | |
| 
 | |
|       if (PDFJS.maxCanvasPixels > 0) {
 | |
|         var pixelsInViewport = viewport.width * viewport.height;
 | |
|         var maxScale = Math.sqrt(PDFJS.maxCanvasPixels / pixelsInViewport);
 | |
|         if (outputScale.sx > maxScale || outputScale.sy > maxScale) {
 | |
|           outputScale.sx = maxScale;
 | |
|           outputScale.sy = maxScale;
 | |
|           outputScale.scaled = true;
 | |
|           this.hasRestrictedScaling = true;
 | |
|         } else {
 | |
|           this.hasRestrictedScaling = false;
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       var sfx = approximateFraction(outputScale.sx);
 | |
|       var sfy = approximateFraction(outputScale.sy);
 | |
|       canvas.width = roundToDivide(viewport.width * outputScale.sx, sfx[0]);
 | |
|       canvas.height = roundToDivide(viewport.height * outputScale.sy, sfy[0]);
 | |
|       canvas.style.width = roundToDivide(viewport.width, sfx[1]) + 'px';
 | |
|       canvas.style.height = roundToDivide(viewport.height, sfy[1]) + 'px';
 | |
|       // Add the viewport so it's known what it was originally drawn with.
 | |
|       canvas._viewport = viewport;
 | |
| 
 | |
|       var textLayerDiv = null;
 | |
|       var textLayer = null;
 | |
|       if (this.textLayerFactory) {
 | |
|         textLayerDiv = document.createElement('div');
 | |
|         textLayerDiv.className = 'textLayer';
 | |
|         textLayerDiv.style.width = canvasWrapper.style.width;
 | |
|         textLayerDiv.style.height = canvasWrapper.style.height;
 | |
|         if (this.annotationLayer && this.annotationLayer.div) {
 | |
|           // annotationLayer needs to stay on top
 | |
|           div.insertBefore(textLayerDiv, this.annotationLayer.div);
 | |
|         } else {
 | |
|           div.appendChild(textLayerDiv);
 | |
|         }
 | |
| 
 | |
|         textLayer = this.textLayerFactory.createTextLayerBuilder(textLayerDiv,
 | |
|                                                                  this.id - 1,
 | |
|                                                                  this.viewport);
 | |
|       }
 | |
|       this.textLayer = textLayer;
 | |
| 
 | |
|       var resolveRenderPromise, rejectRenderPromise;
 | |
|       var promise = new Promise(function (resolve, reject) {
 | |
|         resolveRenderPromise = resolve;
 | |
|         rejectRenderPromise = reject;
 | |
|       });
 | |
| 
 | |
|       // Rendering area
 | |
| 
 | |
|       var self = this;
 | |
|       function pageViewDrawCallback(error) {
 | |
|         // The renderTask may have been replaced by a new one, so only remove
 | |
|         // the reference to the renderTask if it matches the one that is
 | |
|         // triggering this callback.
 | |
|         if (renderTask === self.renderTask) {
 | |
|           self.renderTask = null;
 | |
|         }
 | |
| 
 | |
|         if (error === 'cancelled') {
 | |
|           rejectRenderPromise(error);
 | |
|           return;
 | |
|         }
 | |
| 
 | |
|         self.renderingState = RenderingStates.FINISHED;
 | |
| 
 | |
|         if (isCanvasHidden) {
 | |
|           self.canvas.removeAttribute('hidden');
 | |
|           isCanvasHidden = false;
 | |
|         }
 | |
| 
 | |
|         if (self.loadingIconDiv) {
 | |
|           div.removeChild(self.loadingIconDiv);
 | |
|           delete self.loadingIconDiv;
 | |
|         }
 | |
| 
 | |
|         if (self.zoomLayer) {
 | |
|           // Zeroing the width and height causes Firefox to release graphics
 | |
|           // resources immediately, which can greatly reduce memory consumption.
 | |
|           var zoomLayerCanvas = self.zoomLayer.firstChild;
 | |
|           zoomLayerCanvas.width = 0;
 | |
|           zoomLayerCanvas.height = 0;
 | |
| 
 | |
|           div.removeChild(self.zoomLayer);
 | |
|           self.zoomLayer = null;
 | |
|         }
 | |
| 
 | |
|         self.error = error;
 | |
|         self.stats = pdfPage.stats;
 | |
|         if (self.onAfterDraw) {
 | |
|           self.onAfterDraw();
 | |
|         }
 | |
|         var event = document.createEvent('CustomEvent');
 | |
|         event.initCustomEvent('pagerendered', true, true, {
 | |
|           pageNumber: self.id,
 | |
|           cssTransform: false,
 | |
|         });
 | |
|         div.dispatchEvent(event);
 | |
|         // This custom event is deprecated, and will be removed in the future,
 | |
|         // please use the |pagerendered| event instead.
 | |
|         var deprecatedEvent = document.createEvent('CustomEvent');
 | |
|         deprecatedEvent.initCustomEvent('pagerender', true, true, {
 | |
|           pageNumber: pdfPage.pageNumber
 | |
|         });
 | |
|         div.dispatchEvent(deprecatedEvent);
 | |
| 
 | |
|         if (!error) {
 | |
|           resolveRenderPromise(undefined);
 | |
|         } else {
 | |
|           rejectRenderPromise(error);
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       var renderContinueCallback = null;
 | |
|       if (this.renderingQueue) {
 | |
|         renderContinueCallback = function renderContinueCallback(cont) {
 | |
|           if (!self.renderingQueue.isHighestPriority(self)) {
 | |
|             self.renderingState = RenderingStates.PAUSED;
 | |
|             self.resume = function resumeCallback() {
 | |
|               self.renderingState = RenderingStates.RUNNING;
 | |
|               cont();
 | |
|             };
 | |
|             return;
 | |
|           }
 | |
|           if (isCanvasHidden) {
 | |
|             self.canvas.removeAttribute('hidden');
 | |
|             isCanvasHidden = false;
 | |
|           }
 | |
|           cont();
 | |
|         };
 | |
|       }
 | |
| 
 | |
|       var transform = !outputScale.scaled ? null :
 | |
|         [outputScale.sx, 0, 0, outputScale.sy, 0, 0];
 | |
|       var renderContext = {
 | |
|         canvasContext: ctx,
 | |
|         transform: transform,
 | |
|         viewport: this.viewport,
 | |
|         // intent: 'default', // === 'display'
 | |
|       };
 | |
|       var renderTask = this.renderTask = this.pdfPage.render(renderContext);
 | |
|       renderTask.onContinue = renderContinueCallback;
 | |
| 
 | |
|       this.renderTask.promise.then(
 | |
|         function pdfPageRenderCallback() {
 | |
|           pageViewDrawCallback(null);
 | |
|           if (textLayer) {
 | |
|             self.pdfPage.getTextContent({ normalizeWhitespace: true }).then(
 | |
|               function textContentResolved(textContent) {
 | |
|                 textLayer.setTextContent(textContent);
 | |
|                 textLayer.render(TEXT_LAYER_RENDER_DELAY);
 | |
|               }
 | |
|             );
 | |
|           }
 | |
|         },
 | |
|         function pdfPageRenderError(error) {
 | |
|           pageViewDrawCallback(error);
 | |
|         }
 | |
|       );
 | |
| 
 | |
|       if (this.annotationLayerFactory) {
 | |
|         if (!this.annotationLayer) {
 | |
|           this.annotationLayer = this.annotationLayerFactory.
 | |
|             createAnnotationLayerBuilder(div, this.pdfPage);
 | |
|         }
 | |
|         this.annotationLayer.render(this.viewport, 'display');
 | |
|       }
 | |
|       div.setAttribute('data-loaded', true);
 | |
| 
 | |
|       if (self.onBeforeDraw) {
 | |
|         self.onBeforeDraw();
 | |
|       }
 | |
|       return promise;
 | |
|     },
 | |
| 
 | |
|     beforePrint: function PDFPageView_beforePrint() {
 | |
|       var CustomStyle = PDFJS.CustomStyle;
 | |
|       var pdfPage = this.pdfPage;
 | |
| 
 | |
|       var viewport = pdfPage.getViewport(1);
 | |
|       // Use the same hack we use for high dpi displays for printing to get
 | |
|       // better output until bug 811002 is fixed in FF.
 | |
|       var PRINT_OUTPUT_SCALE = 2;
 | |
|       var canvas = document.createElement('canvas');
 | |
| 
 | |
|       // The logical size of the canvas.
 | |
|       canvas.width = Math.floor(viewport.width) * PRINT_OUTPUT_SCALE;
 | |
|       canvas.height = Math.floor(viewport.height) * PRINT_OUTPUT_SCALE;
 | |
| 
 | |
|       // The rendered size of the canvas, relative to the size of canvasWrapper.
 | |
|       canvas.style.width = (PRINT_OUTPUT_SCALE * 100) + '%';
 | |
|       canvas.style.height = (PRINT_OUTPUT_SCALE * 100) + '%';
 | |
| 
 | |
|       var cssScale = 'scale(' + (1 / PRINT_OUTPUT_SCALE) + ', ' +
 | |
|                                 (1 / PRINT_OUTPUT_SCALE) + ')';
 | |
|       CustomStyle.setProp('transform' , canvas, cssScale);
 | |
|       CustomStyle.setProp('transformOrigin' , canvas, '0% 0%');
 | |
| 
 | |
|       var printContainer = document.getElementById('printContainer');
 | |
|       var canvasWrapper = document.createElement('div');
 | |
|       canvasWrapper.style.width = viewport.width + 'pt';
 | |
|       canvasWrapper.style.height = viewport.height + 'pt';
 | |
|       canvasWrapper.appendChild(canvas);
 | |
|       printContainer.appendChild(canvasWrapper);
 | |
| 
 | |
|       canvas.mozPrintCallback = function(obj) {
 | |
|         var ctx = obj.context;
 | |
| 
 | |
|         ctx.save();
 | |
|         ctx.fillStyle = 'rgb(255, 255, 255)';
 | |
|         ctx.fillRect(0, 0, canvas.width, canvas.height);
 | |
|         ctx.restore();
 | |
|         // Used by the mozCurrentTransform polyfill in src/display/canvas.js.
 | |
|         ctx._transformMatrix =
 | |
|           [PRINT_OUTPUT_SCALE, 0, 0, PRINT_OUTPUT_SCALE, 0, 0];
 | |
|         ctx.scale(PRINT_OUTPUT_SCALE, PRINT_OUTPUT_SCALE);
 | |
| 
 | |
|         var renderContext = {
 | |
|           canvasContext: ctx,
 | |
|           viewport: viewport,
 | |
|           intent: 'print'
 | |
|         };
 | |
| 
 | |
|         pdfPage.render(renderContext).promise.then(function() {
 | |
|           // Tell the printEngine that rendering this canvas/page has finished.
 | |
|           obj.done();
 | |
|         }, function(error) {
 | |
|           console.error(error);
 | |
|           // Tell the printEngine that rendering this canvas/page has failed.
 | |
|           // This will make the print proces stop.
 | |
|           if ('abort' in obj) {
 | |
|             obj.abort();
 | |
|           } else {
 | |
|             obj.done();
 | |
|           }
 | |
|         });
 | |
|       };
 | |
|     },
 | |
|   };
 | |
| 
 | |
|   return PDFPageView;
 | |
| })();
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * @typedef {Object} TextLayerBuilderOptions
 | |
|  * @property {HTMLDivElement} textLayerDiv - The text layer container.
 | |
|  * @property {number} pageIndex - The page index.
 | |
|  * @property {PageViewport} viewport - The viewport of the text layer.
 | |
|  * @property {PDFFindController} findController
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * TextLayerBuilder provides text-selection functionality for the PDF.
 | |
|  * It does this by creating overlay divs over the PDF text. These divs
 | |
|  * contain text that matches the PDF text they are overlaying. This object
 | |
|  * also provides a way to highlight text that is being searched for.
 | |
|  * @class
 | |
|  */
 | |
| var TextLayerBuilder = (function TextLayerBuilderClosure() {
 | |
|   function TextLayerBuilder(options) {
 | |
|     this.textLayerDiv = options.textLayerDiv;
 | |
|     this.renderingDone = false;
 | |
|     this.divContentDone = false;
 | |
|     this.pageIdx = options.pageIndex;
 | |
|     this.pageNumber = this.pageIdx + 1;
 | |
|     this.matches = [];
 | |
|     this.viewport = options.viewport;
 | |
|     this.textDivs = [];
 | |
|     this.findController = options.findController || null;
 | |
|     this.textLayerRenderTask = null;
 | |
|     this._bindMouse();
 | |
|   }
 | |
| 
 | |
|   TextLayerBuilder.prototype = {
 | |
|     _finishRendering: function TextLayerBuilder_finishRendering() {
 | |
|       this.renderingDone = true;
 | |
| 
 | |
|       var endOfContent = document.createElement('div');
 | |
|       endOfContent.className = 'endOfContent';
 | |
|       this.textLayerDiv.appendChild(endOfContent);
 | |
| 
 | |
|       var event = document.createEvent('CustomEvent');
 | |
|       event.initCustomEvent('textlayerrendered', true, true, {
 | |
|         pageNumber: this.pageNumber
 | |
|       });
 | |
|       this.textLayerDiv.dispatchEvent(event);
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Renders the text layer.
 | |
|      * @param {number} timeout (optional) if specified, the rendering waits
 | |
|      *   for specified amount of ms.
 | |
|      */
 | |
|     render: function TextLayerBuilder_render(timeout) {
 | |
|       if (!this.divContentDone || this.renderingDone) {
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       if (this.textLayerRenderTask) {
 | |
|         this.textLayerRenderTask.cancel();
 | |
|         this.textLayerRenderTask = null;
 | |
|       }
 | |
| 
 | |
|       this.textDivs = [];
 | |
|       var textLayerFrag = document.createDocumentFragment();
 | |
|       this.textLayerRenderTask = PDFJS.renderTextLayer({
 | |
|         textContent: this.textContent,
 | |
|         container: textLayerFrag,
 | |
|         viewport: this.viewport,
 | |
|         textDivs: this.textDivs,
 | |
|         timeout: timeout
 | |
|       });
 | |
|       this.textLayerRenderTask.promise.then(function () {
 | |
|         this.textLayerDiv.appendChild(textLayerFrag);
 | |
|         this._finishRendering();
 | |
|         this.updateMatches();
 | |
|       }.bind(this), function (reason) {
 | |
|         // canceled or failed to render text layer -- skipping errors
 | |
|       });
 | |
|     },
 | |
| 
 | |
|     setTextContent: function TextLayerBuilder_setTextContent(textContent) {
 | |
|       if (this.textLayerRenderTask) {
 | |
|         this.textLayerRenderTask.cancel();
 | |
|         this.textLayerRenderTask = null;
 | |
|       }
 | |
|       this.textContent = textContent;
 | |
|       this.divContentDone = true;
 | |
|     },
 | |
| 
 | |
|     convertMatches: function TextLayerBuilder_convertMatches(matches) {
 | |
|       var i = 0;
 | |
|       var iIndex = 0;
 | |
|       var bidiTexts = this.textContent.items;
 | |
|       var end = bidiTexts.length - 1;
 | |
|       var queryLen = (this.findController === null ?
 | |
|                       0 : this.findController.state.query.length);
 | |
|       var ret = [];
 | |
| 
 | |
|       for (var m = 0, len = matches.length; m < len; m++) {
 | |
|         // Calculate the start position.
 | |
|         var matchIdx = matches[m];
 | |
| 
 | |
|         // Loop over the divIdxs.
 | |
|         while (i !== end && matchIdx >= (iIndex + bidiTexts[i].str.length)) {
 | |
|           iIndex += bidiTexts[i].str.length;
 | |
|           i++;
 | |
|         }
 | |
| 
 | |
|         if (i === bidiTexts.length) {
 | |
|           console.error('Could not find a matching mapping');
 | |
|         }
 | |
| 
 | |
|         var match = {
 | |
|           begin: {
 | |
|             divIdx: i,
 | |
|             offset: matchIdx - iIndex
 | |
|           }
 | |
|         };
 | |
| 
 | |
|         // Calculate the end position.
 | |
|         matchIdx += queryLen;
 | |
| 
 | |
|         // Somewhat the same array as above, but use > instead of >= to get
 | |
|         // the end position right.
 | |
|         while (i !== end && matchIdx > (iIndex + bidiTexts[i].str.length)) {
 | |
|           iIndex += bidiTexts[i].str.length;
 | |
|           i++;
 | |
|         }
 | |
| 
 | |
|         match.end = {
 | |
|           divIdx: i,
 | |
|           offset: matchIdx - iIndex
 | |
|         };
 | |
|         ret.push(match);
 | |
|       }
 | |
| 
 | |
|       return ret;
 | |
|     },
 | |
| 
 | |
|     renderMatches: function TextLayerBuilder_renderMatches(matches) {
 | |
|       // Early exit if there is nothing to render.
 | |
|       if (matches.length === 0) {
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       var bidiTexts = this.textContent.items;
 | |
|       var textDivs = this.textDivs;
 | |
|       var prevEnd = null;
 | |
|       var pageIdx = this.pageIdx;
 | |
|       var isSelectedPage = (this.findController === null ?
 | |
|         false : (pageIdx === this.findController.selected.pageIdx));
 | |
|       var selectedMatchIdx = (this.findController === null ?
 | |
|                               -1 : this.findController.selected.matchIdx);
 | |
|       var highlightAll = (this.findController === null ?
 | |
|                           false : this.findController.state.highlightAll);
 | |
|       var infinity = {
 | |
|         divIdx: -1,
 | |
|         offset: undefined
 | |
|       };
 | |
| 
 | |
|       function beginText(begin, className) {
 | |
|         var divIdx = begin.divIdx;
 | |
|         textDivs[divIdx].textContent = '';
 | |
|         appendTextToDiv(divIdx, 0, begin.offset, className);
 | |
|       }
 | |
| 
 | |
|       function appendTextToDiv(divIdx, fromOffset, toOffset, className) {
 | |
|         var div = textDivs[divIdx];
 | |
|         var content = bidiTexts[divIdx].str.substring(fromOffset, toOffset);
 | |
|         var node = document.createTextNode(content);
 | |
|         if (className) {
 | |
|           var span = document.createElement('span');
 | |
|           span.className = className;
 | |
|           span.appendChild(node);
 | |
|           div.appendChild(span);
 | |
|           return;
 | |
|         }
 | |
|         div.appendChild(node);
 | |
|       }
 | |
| 
 | |
|       var i0 = selectedMatchIdx, i1 = i0 + 1;
 | |
|       if (highlightAll) {
 | |
|         i0 = 0;
 | |
|         i1 = matches.length;
 | |
|       } else if (!isSelectedPage) {
 | |
|         // Not highlighting all and this isn't the selected page, so do nothing.
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       for (var i = i0; i < i1; i++) {
 | |
|         var match = matches[i];
 | |
|         var begin = match.begin;
 | |
|         var end = match.end;
 | |
|         var isSelected = (isSelectedPage && i === selectedMatchIdx);
 | |
|         var highlightSuffix = (isSelected ? ' selected' : '');
 | |
| 
 | |
|         if (this.findController) {
 | |
|           this.findController.updateMatchPosition(pageIdx, i, textDivs,
 | |
|                                                   begin.divIdx, end.divIdx);
 | |
|         }
 | |
| 
 | |
|         // Match inside new div.
 | |
|         if (!prevEnd || begin.divIdx !== prevEnd.divIdx) {
 | |
|           // If there was a previous div, then add the text at the end.
 | |
|           if (prevEnd !== null) {
 | |
|             appendTextToDiv(prevEnd.divIdx, prevEnd.offset, infinity.offset);
 | |
|           }
 | |
|           // Clear the divs and set the content until the starting point.
 | |
|           beginText(begin);
 | |
|         } else {
 | |
|           appendTextToDiv(prevEnd.divIdx, prevEnd.offset, begin.offset);
 | |
|         }
 | |
| 
 | |
|         if (begin.divIdx === end.divIdx) {
 | |
|           appendTextToDiv(begin.divIdx, begin.offset, end.offset,
 | |
|                           'highlight' + highlightSuffix);
 | |
|         } else {
 | |
|           appendTextToDiv(begin.divIdx, begin.offset, infinity.offset,
 | |
|                           'highlight begin' + highlightSuffix);
 | |
|           for (var n0 = begin.divIdx + 1, n1 = end.divIdx; n0 < n1; n0++) {
 | |
|             textDivs[n0].className = 'highlight middle' + highlightSuffix;
 | |
|           }
 | |
|           beginText(end, 'highlight end' + highlightSuffix);
 | |
|         }
 | |
|         prevEnd = end;
 | |
|       }
 | |
| 
 | |
|       if (prevEnd) {
 | |
|         appendTextToDiv(prevEnd.divIdx, prevEnd.offset, infinity.offset);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     updateMatches: function TextLayerBuilder_updateMatches() {
 | |
|       // Only show matches when all rendering is done.
 | |
|       if (!this.renderingDone) {
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       // Clear all matches.
 | |
|       var matches = this.matches;
 | |
|       var textDivs = this.textDivs;
 | |
|       var bidiTexts = this.textContent.items;
 | |
|       var clearedUntilDivIdx = -1;
 | |
| 
 | |
|       // Clear all current matches.
 | |
|       for (var i = 0, len = matches.length; i < len; i++) {
 | |
|         var match = matches[i];
 | |
|         var begin = Math.max(clearedUntilDivIdx, match.begin.divIdx);
 | |
|         for (var n = begin, end = match.end.divIdx; n <= end; n++) {
 | |
|           var div = textDivs[n];
 | |
|           div.textContent = bidiTexts[n].str;
 | |
|           div.className = '';
 | |
|         }
 | |
|         clearedUntilDivIdx = match.end.divIdx + 1;
 | |
|       }
 | |
| 
 | |
|       if (this.findController === null || !this.findController.active) {
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       // Convert the matches on the page controller into the match format
 | |
|       // used for the textLayer.
 | |
|       this.matches = this.convertMatches(this.findController === null ?
 | |
|         [] : (this.findController.pageMatches[this.pageIdx] || []));
 | |
|       this.renderMatches(this.matches);
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Fixes text selection: adds additional div where mouse was clicked.
 | |
|      * This reduces flickering of the content if mouse slowly dragged down/up.
 | |
|      * @private
 | |
|      */
 | |
|     _bindMouse: function TextLayerBuilder_bindMouse() {
 | |
|       var div = this.textLayerDiv;
 | |
|       div.addEventListener('mousedown', function (e) {
 | |
|         var end = div.querySelector('.endOfContent');
 | |
|         if (!end) {
 | |
|           return;
 | |
|         }
 | |
|         // On non-Firefox browsers, the selection will feel better if the height
 | |
|         // of the endOfContent div will be adjusted to start at mouse click
 | |
|         // location -- this will avoid flickering when selections moves up.
 | |
|         // However it does not work when selection started on empty space.
 | |
|         var adjustTop = e.target !== div;
 | |
|         adjustTop = adjustTop && window.getComputedStyle(end).
 | |
|           getPropertyValue('-moz-user-select') !== 'none';
 | |
|         if (adjustTop) {
 | |
|           var divBounds = div.getBoundingClientRect();
 | |
|           var r = Math.max(0, (e.pageY - divBounds.top) / divBounds.height);
 | |
|           end.style.top = (r * 100).toFixed(2) + '%';
 | |
|         }
 | |
|         end.classList.add('active');
 | |
|       });
 | |
|       div.addEventListener('mouseup', function (e) {
 | |
|         var end = div.querySelector('.endOfContent');
 | |
|         if (!end) {
 | |
|           return;
 | |
|         }
 | |
|         end.style.top = '';
 | |
|         end.classList.remove('active');
 | |
|       });
 | |
|     },
 | |
|   };
 | |
|   return TextLayerBuilder;
 | |
| })();
 | |
| 
 | |
| /**
 | |
|  * @constructor
 | |
|  * @implements IPDFTextLayerFactory
 | |
|  */
 | |
| function DefaultTextLayerFactory() {}
 | |
| DefaultTextLayerFactory.prototype = {
 | |
|   /**
 | |
|    * @param {HTMLDivElement} textLayerDiv
 | |
|    * @param {number} pageIndex
 | |
|    * @param {PageViewport} viewport
 | |
|    * @returns {TextLayerBuilder}
 | |
|    */
 | |
|   createTextLayerBuilder: function (textLayerDiv, pageIndex, viewport) {
 | |
|     return new TextLayerBuilder({
 | |
|       textLayerDiv: textLayerDiv,
 | |
|       pageIndex: pageIndex,
 | |
|       viewport: viewport
 | |
|     });
 | |
|   }
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * @typedef {Object} AnnotationLayerBuilderOptions
 | |
|  * @property {HTMLDivElement} pageDiv
 | |
|  * @property {PDFPage} pdfPage
 | |
|  * @property {IPDFLinkService} linkService
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * @class
 | |
|  */
 | |
| var AnnotationLayerBuilder = (function AnnotationLayerBuilderClosure() {
 | |
|   /**
 | |
|    * @param {AnnotationLayerBuilderOptions} options
 | |
|    * @constructs AnnotationLayerBuilder
 | |
|    */
 | |
|   function AnnotationLayerBuilder(options) {
 | |
|     this.pageDiv = options.pageDiv;
 | |
|     this.pdfPage = options.pdfPage;
 | |
|     this.linkService = options.linkService;
 | |
| 
 | |
|     this.div = null;
 | |
|   }
 | |
| 
 | |
|   AnnotationLayerBuilder.prototype =
 | |
|       /** @lends AnnotationLayerBuilder.prototype */ {
 | |
| 
 | |
|     /**
 | |
|      * @param {PageViewport} viewport
 | |
|      * @param {string} intent (default value is 'display')
 | |
|      */
 | |
|     render: function AnnotationLayerBuilder_render(viewport, intent) {
 | |
|       var self = this;
 | |
|       var parameters = {
 | |
|         intent: (intent === undefined ? 'display' : intent),
 | |
|       };
 | |
| 
 | |
|       this.pdfPage.getAnnotations(parameters).then(function (annotations) {
 | |
|         viewport = viewport.clone({ dontFlip: true });
 | |
|         parameters = {
 | |
|           viewport: viewport,
 | |
|           div: self.div,
 | |
|           annotations: annotations,
 | |
|           page: self.pdfPage,
 | |
|           linkService: self.linkService
 | |
|         };
 | |
| 
 | |
|         if (self.div) {
 | |
|           // If an annotationLayer already exists, refresh its children's
 | |
|           // transformation matrices.
 | |
|           PDFJS.AnnotationLayer.update(parameters);
 | |
|         } else {
 | |
|           // Create an annotation layer div and render the annotations
 | |
|           // if there is at least one annotation.
 | |
|           if (annotations.length === 0) {
 | |
|             return;
 | |
|           }
 | |
| 
 | |
|           self.div = document.createElement('div');
 | |
|           self.div.className = 'annotationLayer';
 | |
|           self.pageDiv.appendChild(self.div);
 | |
|           parameters.div = self.div;
 | |
| 
 | |
|           PDFJS.AnnotationLayer.render(parameters);
 | |
|         }
 | |
|       });
 | |
|     },
 | |
| 
 | |
|     hide: function AnnotationLayerBuilder_hide() {
 | |
|       if (!this.div) {
 | |
|         return;
 | |
|       }
 | |
|       this.div.setAttribute('hidden', 'true');
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   return AnnotationLayerBuilder;
 | |
| })();
 | |
| 
 | |
| /**
 | |
|  * @constructor
 | |
|  * @implements IPDFAnnotationLayerFactory
 | |
|  */
 | |
| function DefaultAnnotationLayerFactory() {}
 | |
| DefaultAnnotationLayerFactory.prototype = {
 | |
|   /**
 | |
|    * @param {HTMLDivElement} pageDiv
 | |
|    * @param {PDFPage} pdfPage
 | |
|    * @returns {AnnotationLayerBuilder}
 | |
|    */
 | |
|   createAnnotationLayerBuilder: function (pageDiv, pdfPage) {
 | |
|     return new AnnotationLayerBuilder({
 | |
|       pageDiv: pageDiv,
 | |
|       pdfPage: pdfPage,
 | |
|       linkService: new SimpleLinkService(),
 | |
|     });
 | |
|   }
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * @typedef {Object} PDFViewerOptions
 | |
|  * @property {HTMLDivElement} container - The container for the viewer element.
 | |
|  * @property {HTMLDivElement} viewer - (optional) The viewer element.
 | |
|  * @property {IPDFLinkService} linkService - The navigation/linking service.
 | |
|  * @property {PDFRenderingQueue} renderingQueue - (optional) The rendering
 | |
|  *   queue object.
 | |
|  * @property {boolean} removePageBorders - (optional) Removes the border shadow
 | |
|  *   around the pages. The default is false.
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * Simple viewer control to display PDF content/pages.
 | |
|  * @class
 | |
|  * @implements {IRenderableView}
 | |
|  */
 | |
| var PDFViewer = (function pdfViewer() {
 | |
|   function PDFPageViewBuffer(size) {
 | |
|     var data = [];
 | |
|     this.push = function cachePush(view) {
 | |
|       var 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;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * @constructs PDFViewer
 | |
|    * @param {PDFViewerOptions} options
 | |
|    */
 | |
|   function PDFViewer(options) {
 | |
|     this.container = options.container;
 | |
|     this.viewer = options.viewer || options.container.firstElementChild;
 | |
|     this.linkService = options.linkService || new SimpleLinkService();
 | |
|     this.removePageBorders = options.removePageBorders || false;
 | |
| 
 | |
|     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.updateInProgress = false;
 | |
|     this.presentationModeState = PresentationModeState.UNKNOWN;
 | |
|     this._resetView();
 | |
| 
 | |
|     if (this.removePageBorders) {
 | |
|       this.viewer.classList.add('removePageBorders');
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   PDFViewer.prototype = /** @lends PDFViewer.prototype */{
 | |
|     get pagesCount() {
 | |
|       return this._pages.length;
 | |
|     },
 | |
| 
 | |
|     getPageView: function (index) {
 | |
|       return this._pages[index];
 | |
|     },
 | |
| 
 | |
|     get currentPageNumber() {
 | |
|       return this._currentPageNumber;
 | |
|     },
 | |
| 
 | |
|     set currentPageNumber(val) {
 | |
|       if (!this.pdfDocument) {
 | |
|         this._currentPageNumber = val;
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       var event = document.createEvent('UIEvents');
 | |
|       event.initUIEvent('pagechange', true, true, window, 0);
 | |
|       event.updateInProgress = this.updateInProgress;
 | |
| 
 | |
|       if (!(0 < val && val <= this.pagesCount)) {
 | |
|         event.pageNumber = this._currentPageNumber;
 | |
|         event.previousPageNumber = val;
 | |
|         this.container.dispatchEvent(event);
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       event.previousPageNumber = this._currentPageNumber;
 | |
|       this._currentPageNumber = val;
 | |
|       event.pageNumber = val;
 | |
|       this.container.dispatchEvent(event);
 | |
| 
 | |
|       // Check if the caller is `PDFViewer_update`, to avoid breaking scrolling.
 | |
|       if (this.updateInProgress) {
 | |
|         return;
 | |
|       }
 | |
|       this.scrollPageIntoView(val);
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @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) {
 | |
|         this._currentScale = val;
 | |
|         this._currentScaleValue = val !== UNKNOWN_SCALE ? val.toString() : null;
 | |
|         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) {
 | |
|         this._currentScale = isNaN(val) ? UNKNOWN_SCALE : val;
 | |
|         this._currentScaleValue = val;
 | |
|         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) {
 | |
|       this._pagesRotation = rotation;
 | |
| 
 | |
|       for (var i = 0, l = this._pages.length; i < l; i++) {
 | |
|         var pageView = this._pages[i];
 | |
|         pageView.update(pageView.scale, rotation);
 | |
|       }
 | |
| 
 | |
|       this._setScale(this._currentScaleValue, true);
 | |
| 
 | |
|       if (this.defaultRenderingQueue) {
 | |
|         this.update();
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @param pdfDocument {PDFDocument}
 | |
|      */
 | |
|     setDocument: function (pdfDocument) {
 | |
|       if (this.pdfDocument) {
 | |
|         this._resetView();
 | |
|       }
 | |
| 
 | |
|       this.pdfDocument = pdfDocument;
 | |
|       if (!pdfDocument) {
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       var pagesCount = pdfDocument.numPages;
 | |
|       var self = this;
 | |
| 
 | |
|       var resolvePagesPromise;
 | |
|       var pagesPromise = new Promise(function (resolve) {
 | |
|         resolvePagesPromise = resolve;
 | |
|       });
 | |
|       this.pagesPromise = pagesPromise;
 | |
|       pagesPromise.then(function () {
 | |
|         var event = document.createEvent('CustomEvent');
 | |
|         event.initCustomEvent('pagesloaded', true, true, {
 | |
|           pagesCount: pagesCount
 | |
|         });
 | |
|         self.container.dispatchEvent(event);
 | |
|       });
 | |
| 
 | |
|       var isOnePageRenderedResolved = false;
 | |
|       var resolveOnePageRendered = null;
 | |
|       var onePageRendered = new Promise(function (resolve) {
 | |
|         resolveOnePageRendered = resolve;
 | |
|       });
 | |
|       this.onePageRendered = onePageRendered;
 | |
| 
 | |
|       var bindOnAfterAndBeforeDraw = function (pageView) {
 | |
|         pageView.onBeforeDraw = function pdfViewLoadOnBeforeDraw() {
 | |
|           // 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.
 | |
|           self._buffer.push(this);
 | |
|         };
 | |
|         // when page is painted, using the image as thumbnail base
 | |
|         pageView.onAfterDraw = function pdfViewLoadOnAfterDraw() {
 | |
|           if (!isOnePageRenderedResolved) {
 | |
|             isOnePageRenderedResolved = true;
 | |
|             resolveOnePageRendered();
 | |
|           }
 | |
|         };
 | |
|       };
 | |
| 
 | |
|       var 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
 | |
|       return firstPagePromise.then(function(pdfPage) {
 | |
|         var scale = this.currentScale;
 | |
|         var viewport = pdfPage.getViewport(scale * CSS_UNITS);
 | |
|         for (var pageNum = 1; pageNum <= pagesCount; ++pageNum) {
 | |
|           var textLayerFactory = null;
 | |
|           if (!PDFJS.disableTextLayer) {
 | |
|             textLayerFactory = this;
 | |
|           }
 | |
|           var pageView = new PDFPageView({
 | |
|             container: this.viewer,
 | |
|             id: pageNum,
 | |
|             scale: scale,
 | |
|             defaultViewport: viewport.clone(),
 | |
|             renderingQueue: this.renderingQueue,
 | |
|             textLayerFactory: textLayerFactory,
 | |
|             annotationLayerFactory: this
 | |
|           });
 | |
|           bindOnAfterAndBeforeDraw(pageView);
 | |
|           this._pages.push(pageView);
 | |
|         }
 | |
| 
 | |
|         var linkService = this.linkService;
 | |
| 
 | |
|         // 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.
 | |
|         onePageRendered.then(function () {
 | |
|           if (!PDFJS.disableAutoFetch) {
 | |
|             var getPagesLeft = pagesCount;
 | |
|             for (var pageNum = 1; pageNum <= pagesCount; ++pageNum) {
 | |
|               pdfDocument.getPage(pageNum).then(function (pageNum, pdfPage) {
 | |
|                 var pageView = self._pages[pageNum - 1];
 | |
|                 if (!pageView.pdfPage) {
 | |
|                   pageView.setPdfPage(pdfPage);
 | |
|                 }
 | |
|                 linkService.cachePageRef(pageNum, pdfPage.ref);
 | |
|                 getPagesLeft--;
 | |
|                 if (!getPagesLeft) {
 | |
|                   resolvePagesPromise();
 | |
|                 }
 | |
|               }.bind(null, pageNum));
 | |
|             }
 | |
|           } else {
 | |
|             // XXX: Printing is semi-broken with auto fetch disabled.
 | |
|             resolvePagesPromise();
 | |
|           }
 | |
|         });
 | |
| 
 | |
|         var event = document.createEvent('CustomEvent');
 | |
|         event.initCustomEvent('pagesinit', true, true, null);
 | |
|         self.container.dispatchEvent(event);
 | |
| 
 | |
|         if (this.defaultRenderingQueue) {
 | |
|           this.update();
 | |
|         }
 | |
| 
 | |
|         if (this.findController) {
 | |
|           this.findController.resolveFirstPage();
 | |
|         }
 | |
|       }.bind(this));
 | |
|     },
 | |
| 
 | |
|     _resetView: function () {
 | |
|       this._pages = [];
 | |
|       this._currentPageNumber = 1;
 | |
|       this._currentScale = UNKNOWN_SCALE;
 | |
|       this._currentScaleValue = null;
 | |
|       this._buffer = new PDFPageViewBuffer(DEFAULT_CACHE_SIZE);
 | |
|       this._location = null;
 | |
|       this._pagesRotation = 0;
 | |
|       this._pagesRequests = [];
 | |
| 
 | |
|       var container = this.viewer;
 | |
|       while (container.hasChildNodes()) {
 | |
|         container.removeChild(container.lastChild);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     _scrollUpdate: function PDFViewer_scrollUpdate() {
 | |
|       if (this.pagesCount === 0) {
 | |
|         return;
 | |
|       }
 | |
|       this.update();
 | |
|       for (var i = 0, ii = this._pages.length; i < ii; i++) {
 | |
|         this._pages[i].updatePosition();
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     _setScaleDispatchEvent: function pdfViewer_setScaleDispatchEvent(
 | |
|         newScale, newValue, preset) {
 | |
|       var event = document.createEvent('UIEvents');
 | |
|       event.initUIEvent('scalechange', true, true, window, 0);
 | |
|       event.scale = newScale;
 | |
|       if (preset) {
 | |
|         event.presetValue = newValue;
 | |
|       }
 | |
|       this.container.dispatchEvent(event);
 | |
|     },
 | |
| 
 | |
|     _setScaleUpdatePages: function pdfViewer_setScaleUpdatePages(
 | |
|         newScale, newValue, noScroll, preset) {
 | |
|       this._currentScaleValue = newValue;
 | |
| 
 | |
|       if (isSameScale(this._currentScale, newScale)) {
 | |
|         if (preset) {
 | |
|           this._setScaleDispatchEvent(newScale, newValue, true);
 | |
|         }
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       for (var i = 0, ii = this._pages.length; i < ii; i++) {
 | |
|         this._pages[i].update(newScale);
 | |
|       }
 | |
|       this._currentScale = newScale;
 | |
| 
 | |
|       if (!noScroll) {
 | |
|         var page = this._currentPageNumber, dest;
 | |
|         if (this._location && !IGNORE_CURRENT_POSITION_ON_ZOOM &&
 | |
|             !(this.isInPresentationMode || this.isChangingPresentationMode)) {
 | |
|           page = this._location.pageNumber;
 | |
|           dest = [null, { name: 'XYZ' }, this._location.left,
 | |
|                   this._location.top, null];
 | |
|         }
 | |
|         this.scrollPageIntoView(page, dest);
 | |
|       }
 | |
| 
 | |
|       this._setScaleDispatchEvent(newScale, newValue, preset);
 | |
| 
 | |
|       if (this.defaultRenderingQueue) {
 | |
|         this.update();
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     _setScale: function pdfViewer_setScale(value, noScroll) {
 | |
|       var scale = parseFloat(value);
 | |
| 
 | |
|       if (scale > 0) {
 | |
|         this._setScaleUpdatePages(scale, value, noScroll, false);
 | |
|       } else {
 | |
|         var currentPage = this._pages[this._currentPageNumber - 1];
 | |
|         if (!currentPage) {
 | |
|           return;
 | |
|         }
 | |
|         var hPadding = (this.isInPresentationMode || this.removePageBorders) ?
 | |
|           0 : SCROLLBAR_PADDING;
 | |
|         var vPadding = (this.isInPresentationMode || this.removePageBorders) ?
 | |
|           0 : VERTICAL_PADDING;
 | |
|         var pageWidthScale = (this.container.clientWidth - hPadding) /
 | |
|                              currentPage.width * currentPage.scale;
 | |
|         var 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':
 | |
|             var 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.
 | |
|             var horizontalScale = isLandscape ?
 | |
|               Math.min(pageHeightScale, pageWidthScale) : pageWidthScale;
 | |
|             scale = Math.min(MAX_AUTO_SCALE, horizontalScale);
 | |
|             break;
 | |
|           default:
 | |
|             console.error('pdfViewSetScale: \'' + value +
 | |
|               '\' is an unknown zoom value.');
 | |
|             return;
 | |
|         }
 | |
|         this._setScaleUpdatePages(scale, value, noScroll, true);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Scrolls page into view.
 | |
|      * @param {number} pageNumber
 | |
|      * @param {Array} dest - (optional) original PDF destination array:
 | |
|      *   <page-ref> </XYZ|FitXXX> <args..>
 | |
|      */
 | |
|     scrollPageIntoView: function PDFViewer_scrollPageIntoView(pageNumber,
 | |
|                                                               dest) {
 | |
|       if (!this.pdfDocument) {
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       var pageView = this._pages[pageNumber - 1];
 | |
| 
 | |
|       if (this.isInPresentationMode) {
 | |
|         if (this._currentPageNumber !== pageView.id) {
 | |
|           // Avoid breaking getVisiblePages in presentation mode.
 | |
|           this.currentPageNumber = pageView.id;
 | |
|           return;
 | |
|         }
 | |
|         dest = null;
 | |
|         // Fixes the case when PDF has different page sizes.
 | |
|         this._setScale(this._currentScaleValue, true);
 | |
|       }
 | |
|       if (!dest) {
 | |
|         scrollIntoView(pageView.div);
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       var x = 0, y = 0;
 | |
|       var width = 0, height = 0, widthScale, heightScale;
 | |
|       var changeOrientation = (pageView.rotation % 180 === 0 ? false : true);
 | |
|       var pageWidth = (changeOrientation ? pageView.height : pageView.width) /
 | |
|         pageView.scale / CSS_UNITS;
 | |
|       var pageHeight = (changeOrientation ? pageView.width : pageView.height) /
 | |
|         pageView.scale / CSS_UNITS;
 | |
|       var 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;
 | |
|           var hPadding = this.removePageBorders ? 0 : SCROLLBAR_PADDING;
 | |
|           var 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:
 | |
|           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;
 | |
|       }
 | |
| 
 | |
|       var boundingRect = [
 | |
|         pageView.viewport.convertToViewportPoint(x, y),
 | |
|         pageView.viewport.convertToViewportPoint(x + width, y + height)
 | |
|       ];
 | |
|       var left = Math.min(boundingRect[0][0], boundingRect[1][0]);
 | |
|       var top = Math.min(boundingRect[0][1], boundingRect[1][1]);
 | |
| 
 | |
|       scrollIntoView(pageView.div, { left: left, top: top });
 | |
|     },
 | |
| 
 | |
|     _updateLocation: function (firstPage) {
 | |
|       var currentScale = this._currentScale;
 | |
|       var currentScaleValue = this._currentScaleValue;
 | |
|       var normalizedScaleValue =
 | |
|         parseFloat(currentScaleValue) === currentScale ?
 | |
|         Math.round(currentScale * 10000) / 100 : currentScaleValue;
 | |
| 
 | |
|       var pageNumber = firstPage.id;
 | |
|       var pdfOpenParams = '#page=' + pageNumber;
 | |
|       pdfOpenParams += '&zoom=' + normalizedScaleValue;
 | |
|       var currentPageView = this._pages[pageNumber - 1];
 | |
|       var container = this.container;
 | |
|       var topLeft = currentPageView.getPagePoint(
 | |
|         (container.scrollLeft - firstPage.x),
 | |
|         (container.scrollTop - firstPage.y));
 | |
|       var intLeft = Math.round(topLeft[0]);
 | |
|       var intTop = Math.round(topLeft[1]);
 | |
|       pdfOpenParams += ',' + intLeft + ',' + intTop;
 | |
| 
 | |
|       this._location = {
 | |
|         pageNumber: pageNumber,
 | |
|         scale: normalizedScaleValue,
 | |
|         top: intTop,
 | |
|         left: intLeft,
 | |
|         pdfOpenParams: pdfOpenParams
 | |
|       };
 | |
|     },
 | |
| 
 | |
|     update: function PDFViewer_update() {
 | |
|       var visible = this._getVisiblePages();
 | |
|       var visiblePages = visible.views;
 | |
|       if (visiblePages.length === 0) {
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       this.updateInProgress = true;
 | |
| 
 | |
|       var suggestedCacheSize = Math.max(DEFAULT_CACHE_SIZE,
 | |
|           2 * visiblePages.length + 1);
 | |
|       this._buffer.resize(suggestedCacheSize);
 | |
| 
 | |
|       this.renderingQueue.renderHighestPriority(visible);
 | |
| 
 | |
|       var currentId = this._currentPageNumber;
 | |
|       var firstPage = visible.first;
 | |
| 
 | |
|       for (var i = 0, ii = visiblePages.length, stillFullyVisible = false;
 | |
|            i < ii; ++i) {
 | |
|         var page = visiblePages[i];
 | |
| 
 | |
|         if (page.percent < 100) {
 | |
|           break;
 | |
|         }
 | |
|         if (page.id === currentId) {
 | |
|           stillFullyVisible = true;
 | |
|           break;
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       if (!stillFullyVisible) {
 | |
|         currentId = visiblePages[0].id;
 | |
|       }
 | |
| 
 | |
|       if (!this.isInPresentationMode) {
 | |
|         this.currentPageNumber = currentId;
 | |
|       }
 | |
| 
 | |
|       this._updateLocation(firstPage);
 | |
| 
 | |
|       this.updateInProgress = false;
 | |
| 
 | |
|       var event = document.createEvent('UIEvents');
 | |
|       event.initUIEvent('updateviewarea', true, true, window, 0);
 | |
|       event.location = this._location;
 | |
|       this.container.dispatchEvent(event);
 | |
|     },
 | |
| 
 | |
|     containsElement: function (element) {
 | |
|       return this.container.contains(element);
 | |
|     },
 | |
| 
 | |
|     focus: function () {
 | |
|       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: function () {
 | |
|       if (!this.isInPresentationMode) {
 | |
|         return getVisibleElements(this.container, this._pages, true);
 | |
|       } else {
 | |
|         // The algorithm in getVisibleElements doesn't work in all browsers and
 | |
|         // configurations when presentation mode is active.
 | |
|         var visible = [];
 | |
|         var currentPage = this._pages[this._currentPageNumber - 1];
 | |
|         visible.push({ id: currentPage.id, view: currentPage });
 | |
|         return { first: currentPage, last: currentPage, views: visible };
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     cleanup: function () {
 | |
|       for (var i = 0, ii = this._pages.length; i < ii; i++) {
 | |
|         if (this._pages[i] &&
 | |
|             this._pages[i].renderingState !== RenderingStates.FINISHED) {
 | |
|           this._pages[i].reset();
 | |
|         }
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @param {PDFPageView} pageView
 | |
|      * @returns {PDFPage}
 | |
|      * @private
 | |
|      */
 | |
|     _ensurePdfPageLoaded: function (pageView) {
 | |
|       if (pageView.pdfPage) {
 | |
|         return Promise.resolve(pageView.pdfPage);
 | |
|       }
 | |
|       var pageNumber = pageView.id;
 | |
|       if (this._pagesRequests[pageNumber]) {
 | |
|         return this._pagesRequests[pageNumber];
 | |
|       }
 | |
|       var promise = this.pdfDocument.getPage(pageNumber).then(
 | |
|           function (pdfPage) {
 | |
|         pageView.setPdfPage(pdfPage);
 | |
|         this._pagesRequests[pageNumber] = null;
 | |
|         return pdfPage;
 | |
|       }.bind(this));
 | |
|       this._pagesRequests[pageNumber] = promise;
 | |
|       return promise;
 | |
|     },
 | |
| 
 | |
|     forceRendering: function (currentlyVisiblePages) {
 | |
|       var visiblePages = currentlyVisiblePages || this._getVisiblePages();
 | |
|       var pageView = this.renderingQueue.getHighestPriority(visiblePages,
 | |
|                                                             this._pages,
 | |
|                                                             this.scroll.down);
 | |
|       if (pageView) {
 | |
|         this._ensurePdfPageLoaded(pageView).then(function () {
 | |
|           this.renderingQueue.renderView(pageView);
 | |
|         }.bind(this));
 | |
|         return true;
 | |
|       }
 | |
|       return false;
 | |
|     },
 | |
| 
 | |
|     getPageTextContent: function (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: function (textLayerDiv, pageIndex, viewport) {
 | |
|       return new TextLayerBuilder({
 | |
|         textLayerDiv: textLayerDiv,
 | |
|         pageIndex: pageIndex,
 | |
|         viewport: viewport,
 | |
|         findController: this.isInPresentationMode ? null : this.findController
 | |
|       });
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @param {HTMLDivElement} pageDiv
 | |
|      * @param {PDFPage} pdfPage
 | |
|      * @returns {AnnotationLayerBuilder}
 | |
|      */
 | |
|     createAnnotationLayerBuilder: function (pageDiv, pdfPage) {
 | |
|       return new AnnotationLayerBuilder({
 | |
|         pageDiv: pageDiv,
 | |
|         pdfPage: pdfPage,
 | |
|         linkService: this.linkService
 | |
|       });
 | |
|     },
 | |
| 
 | |
|     setFindController: function (findController) {
 | |
|       this.findController = findController;
 | |
|     },
 | |
|   };
 | |
| 
 | |
|   return PDFViewer;
 | |
| })();
 | |
| 
 | |
| var SimpleLinkService = (function SimpleLinkServiceClosure() {
 | |
|   function SimpleLinkService() {}
 | |
| 
 | |
|   SimpleLinkService.prototype = {
 | |
|     /**
 | |
|      * @returns {number}
 | |
|      */
 | |
|     get page() {
 | |
|       return 0;
 | |
|     },
 | |
|     /**
 | |
|      * @param {number} value
 | |
|      */
 | |
|     set page(value) {},
 | |
|     /**
 | |
|      * @param dest - The PDF destination object.
 | |
|      */
 | |
|     navigateTo: function (dest) {},
 | |
|     /**
 | |
|      * @param dest - The PDF destination object.
 | |
|      * @returns {string} The hyperlink to the PDF object.
 | |
|      */
 | |
|     getDestinationHash: function (dest) {
 | |
|       return '#';
 | |
|     },
 | |
|     /**
 | |
|      * @param hash - The PDF parameters/hash.
 | |
|      * @returns {string} The hyperlink to the PDF object.
 | |
|      */
 | |
|     getAnchorUrl: function (hash) {
 | |
|       return '#';
 | |
|     },
 | |
|     /**
 | |
|      * @param {string} hash
 | |
|      */
 | |
|     setHash: function (hash) {},
 | |
|     /**
 | |
|      * @param {string} action
 | |
|      */
 | |
|     executeNamedAction: function (action) {},
 | |
|     /**
 | |
|      * @param {number} pageNum - page number.
 | |
|      * @param {Object} pageRef - reference to the page.
 | |
|      */
 | |
|     cachePageRef: function (pageNum, pageRef) {}
 | |
|   };
 | |
|   return SimpleLinkService;
 | |
| })();
 | |
| 
 | |
| 
 | |
| var THUMBNAIL_SCROLL_MARGIN = -19;
 | |
| 
 | |
| 
 | |
| var THUMBNAIL_WIDTH = 98; // px
 | |
| var THUMBNAIL_CANVAS_BORDER_WIDTH = 1; // px
 | |
| 
 | |
| /**
 | |
|  * @typedef {Object} PDFThumbnailViewOptions
 | |
|  * @property {HTMLDivElement} container - The viewer element.
 | |
|  * @property {number} id - The thumbnail's unique ID (normally its number).
 | |
|  * @property {PageViewport} defaultViewport - The page viewport.
 | |
|  * @property {IPDFLinkService} linkService - The navigation/linking service.
 | |
|  * @property {PDFRenderingQueue} renderingQueue - The rendering queue object.
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * @class
 | |
|  * @implements {IRenderableView}
 | |
|  */
 | |
| var PDFThumbnailView = (function PDFThumbnailViewClosure() {
 | |
|   function getTempCanvas(width, height) {
 | |
|     var tempCanvas = PDFThumbnailView.tempImageCache;
 | |
|     if (!tempCanvas) {
 | |
|       tempCanvas = document.createElement('canvas');
 | |
|       PDFThumbnailView.tempImageCache = tempCanvas;
 | |
|     }
 | |
|     tempCanvas.width = width;
 | |
|     tempCanvas.height = height;
 | |
| 
 | |
|     // Since this is a temporary canvas, we need to fill the canvas with a white
 | |
|     // background ourselves. |_getPageDrawContext| uses CSS rules for this.
 | |
|     tempCanvas.mozOpaque = true;
 | |
|     var ctx = tempCanvas.getContext('2d', {alpha: false});
 | |
|     ctx.save();
 | |
|     ctx.fillStyle = 'rgb(255, 255, 255)';
 | |
|     ctx.fillRect(0, 0, width, height);
 | |
|     ctx.restore();
 | |
|     return tempCanvas;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * @constructs PDFThumbnailView
 | |
|    * @param {PDFThumbnailViewOptions} options
 | |
|    */
 | |
|   function PDFThumbnailView(options) {
 | |
|     var container = options.container;
 | |
|     var id = options.id;
 | |
|     var defaultViewport = options.defaultViewport;
 | |
|     var linkService = options.linkService;
 | |
|     var renderingQueue = options.renderingQueue;
 | |
| 
 | |
|     this.id = id;
 | |
|     this.renderingId = 'thumbnail' + id;
 | |
| 
 | |
|     this.pdfPage = null;
 | |
|     this.rotation = 0;
 | |
|     this.viewport = defaultViewport;
 | |
|     this.pdfPageRotate = defaultViewport.rotation;
 | |
| 
 | |
|     this.linkService = linkService;
 | |
|     this.renderingQueue = renderingQueue;
 | |
| 
 | |
|     this.hasImage = false;
 | |
|     this.resume = null;
 | |
|     this.renderingState = RenderingStates.INITIAL;
 | |
| 
 | |
|     this.pageWidth = this.viewport.width;
 | |
|     this.pageHeight = this.viewport.height;
 | |
|     this.pageRatio = this.pageWidth / this.pageHeight;
 | |
| 
 | |
|     this.canvasWidth = THUMBNAIL_WIDTH;
 | |
|     this.canvasHeight = (this.canvasWidth / this.pageRatio) | 0;
 | |
|     this.scale = this.canvasWidth / this.pageWidth;
 | |
| 
 | |
|     var anchor = document.createElement('a');
 | |
|     anchor.href = linkService.getAnchorUrl('#page=' + id);
 | |
|     anchor.title = 'Page ' + id;
 | |
|     anchor.onclick = function stopNavigation() {
 | |
|       linkService.page = id;
 | |
|       return false;
 | |
|     };
 | |
| 
 | |
|     var div = document.createElement('div');
 | |
|     div.id = 'thumbnailContainer' + id;
 | |
|     div.className = 'thumbnail';
 | |
|     this.div = div;
 | |
| 
 | |
|     if (id === 1) {
 | |
|       // Highlight the thumbnail of the first page when no page number is
 | |
|       // specified (or exists in cache) when the document is loaded.
 | |
|       div.classList.add('selected');
 | |
|     }
 | |
| 
 | |
|     var ring = document.createElement('div');
 | |
|     ring.className = 'thumbnailSelectionRing';
 | |
|     var borderAdjustment = 2 * THUMBNAIL_CANVAS_BORDER_WIDTH;
 | |
|     ring.style.width = this.canvasWidth + borderAdjustment + 'px';
 | |
|     ring.style.height = this.canvasHeight + borderAdjustment + 'px';
 | |
|     this.ring = ring;
 | |
| 
 | |
|     div.appendChild(ring);
 | |
|     anchor.appendChild(div);
 | |
|     container.appendChild(anchor);
 | |
|   }
 | |
| 
 | |
|   PDFThumbnailView.prototype = {
 | |
|     setPdfPage: function PDFThumbnailView_setPdfPage(pdfPage) {
 | |
|       this.pdfPage = pdfPage;
 | |
|       this.pdfPageRotate = pdfPage.rotate;
 | |
|       var totalRotation = (this.rotation + this.pdfPageRotate) % 360;
 | |
|       this.viewport = pdfPage.getViewport(1, totalRotation);
 | |
|       this.reset();
 | |
|     },
 | |
| 
 | |
|     reset: function PDFThumbnailView_reset() {
 | |
|       if (this.renderTask) {
 | |
|         this.renderTask.cancel();
 | |
|       }
 | |
|       this.hasImage = false;
 | |
|       this.resume = null;
 | |
|       this.renderingState = RenderingStates.INITIAL;
 | |
| 
 | |
|       this.pageWidth = this.viewport.width;
 | |
|       this.pageHeight = this.viewport.height;
 | |
|       this.pageRatio = this.pageWidth / this.pageHeight;
 | |
| 
 | |
|       this.canvasHeight = (this.canvasWidth / this.pageRatio) | 0;
 | |
|       this.scale = (this.canvasWidth / this.pageWidth);
 | |
| 
 | |
|       this.div.removeAttribute('data-loaded');
 | |
|       var ring = this.ring;
 | |
|       var childNodes = ring.childNodes;
 | |
|       for (var i = childNodes.length - 1; i >= 0; i--) {
 | |
|         ring.removeChild(childNodes[i]);
 | |
|       }
 | |
|       var borderAdjustment = 2 * THUMBNAIL_CANVAS_BORDER_WIDTH;
 | |
|       ring.style.width = this.canvasWidth + borderAdjustment + 'px';
 | |
|       ring.style.height = this.canvasHeight + borderAdjustment + 'px';
 | |
| 
 | |
|       if (this.canvas) {
 | |
|         // Zeroing the width and height causes Firefox to release graphics
 | |
|         // resources immediately, which can greatly reduce memory consumption.
 | |
|         this.canvas.width = 0;
 | |
|         this.canvas.height = 0;
 | |
|         delete this.canvas;
 | |
|       }
 | |
|       if (this.image) {
 | |
|         this.image.removeAttribute('src');
 | |
|         delete this.image;
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     update: function PDFThumbnailView_update(rotation) {
 | |
|       if (typeof rotation !== 'undefined') {
 | |
|         this.rotation = rotation;
 | |
|       }
 | |
|       var totalRotation = (this.rotation + this.pdfPageRotate) % 360;
 | |
|       this.viewport = this.viewport.clone({
 | |
|         scale: 1,
 | |
|         rotation: totalRotation
 | |
|       });
 | |
|       this.reset();
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _getPageDrawContext:
 | |
|         function PDFThumbnailView_getPageDrawContext(noCtxScale) {
 | |
|       var canvas = document.createElement('canvas');
 | |
|       this.canvas = canvas;
 | |
| 
 | |
|       canvas.mozOpaque = true;
 | |
|       var ctx = canvas.getContext('2d', {alpha: false});
 | |
|       var outputScale = getOutputScale(ctx);
 | |
| 
 | |
|       canvas.width = (this.canvasWidth * outputScale.sx) | 0;
 | |
|       canvas.height = (this.canvasHeight * outputScale.sy) | 0;
 | |
|       canvas.style.width = this.canvasWidth + 'px';
 | |
|       canvas.style.height = this.canvasHeight + 'px';
 | |
| 
 | |
|       if (!noCtxScale && outputScale.scaled) {
 | |
|         ctx.scale(outputScale.sx, outputScale.sy);
 | |
|       }
 | |
| 
 | |
|       var image = document.createElement('img');
 | |
|       this.image = image;
 | |
| 
 | |
|       image.id = this.renderingId;
 | |
|       image.className = 'thumbnailImage';
 | |
|       image.setAttribute('aria-label', 'Thumbnail of Page ' + this.id);
 | |
| 
 | |
|       image.style.width = canvas.style.width;
 | |
|       image.style.height = canvas.style.height;
 | |
| 
 | |
|       return ctx;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _convertCanvasToImage: function PDFThumbnailView_convertCanvasToImage() {
 | |
|       if (!this.canvas) {
 | |
|         return;
 | |
|       }
 | |
|       this.image.src = this.canvas.toDataURL();
 | |
| 
 | |
|       this.div.setAttribute('data-loaded', true);
 | |
|       this.ring.appendChild(this.image);
 | |
| 
 | |
|       // Zeroing the width and height causes Firefox to release graphics
 | |
|       // resources immediately, which can greatly reduce memory consumption.
 | |
|       this.canvas.width = 0;
 | |
|       this.canvas.height = 0;
 | |
|       delete this.canvas;
 | |
|     },
 | |
| 
 | |
|     draw: function PDFThumbnailView_draw() {
 | |
|       if (this.renderingState !== RenderingStates.INITIAL) {
 | |
|         console.error('Must be in new state before drawing');
 | |
|       }
 | |
|       if (this.hasImage) {
 | |
|         return Promise.resolve(undefined);
 | |
|       }
 | |
|       this.hasImage = true;
 | |
|       this.renderingState = RenderingStates.RUNNING;
 | |
| 
 | |
|       var resolveRenderPromise, rejectRenderPromise;
 | |
|       var promise = new Promise(function (resolve, reject) {
 | |
|         resolveRenderPromise = resolve;
 | |
|         rejectRenderPromise = reject;
 | |
|       });
 | |
| 
 | |
|       var self = this;
 | |
|       function thumbnailDrawCallback(error) {
 | |
|         // The renderTask may have been replaced by a new one, so only remove
 | |
|         // the reference to the renderTask if it matches the one that is
 | |
|         // triggering this callback.
 | |
|         if (renderTask === self.renderTask) {
 | |
|           self.renderTask = null;
 | |
|         }
 | |
|         if (error === 'cancelled') {
 | |
|           rejectRenderPromise(error);
 | |
|           return;
 | |
|         }
 | |
|         self.renderingState = RenderingStates.FINISHED;
 | |
|         self._convertCanvasToImage();
 | |
| 
 | |
|         if (!error) {
 | |
|           resolveRenderPromise(undefined);
 | |
|         } else {
 | |
|           rejectRenderPromise(error);
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       var ctx = this._getPageDrawContext();
 | |
|       var drawViewport = this.viewport.clone({ scale: this.scale });
 | |
|       var renderContinueCallback = function renderContinueCallback(cont) {
 | |
|         if (!self.renderingQueue.isHighestPriority(self)) {
 | |
|           self.renderingState = RenderingStates.PAUSED;
 | |
|           self.resume = function resumeCallback() {
 | |
|             self.renderingState = RenderingStates.RUNNING;
 | |
|             cont();
 | |
|           };
 | |
|           return;
 | |
|         }
 | |
|         cont();
 | |
|       };
 | |
| 
 | |
|       var renderContext = {
 | |
|         canvasContext: ctx,
 | |
|         viewport: drawViewport
 | |
|       };
 | |
|       var renderTask = this.renderTask = this.pdfPage.render(renderContext);
 | |
|       renderTask.onContinue = renderContinueCallback;
 | |
| 
 | |
|       renderTask.promise.then(
 | |
|         function pdfPageRenderCallback() {
 | |
|           thumbnailDrawCallback(null);
 | |
|         },
 | |
|         function pdfPageRenderError(error) {
 | |
|           thumbnailDrawCallback(error);
 | |
|         }
 | |
|       );
 | |
|       return promise;
 | |
|     },
 | |
| 
 | |
|     setImage: function PDFThumbnailView_setImage(pageView) {
 | |
|       var img = pageView.canvas;
 | |
|       if (this.hasImage || !img) {
 | |
|         return;
 | |
|       }
 | |
|       if (!this.pdfPage) {
 | |
|         this.setPdfPage(pageView.pdfPage);
 | |
|       }
 | |
|       this.hasImage = true;
 | |
|       this.renderingState = RenderingStates.FINISHED;
 | |
| 
 | |
|       var ctx = this._getPageDrawContext(true);
 | |
|       var canvas = ctx.canvas;
 | |
| 
 | |
|       if (img.width <= 2 * canvas.width) {
 | |
|         ctx.drawImage(img, 0, 0, img.width, img.height,
 | |
|                       0, 0, canvas.width, canvas.height);
 | |
|         this._convertCanvasToImage();
 | |
|         return;
 | |
|       }
 | |
|       // drawImage does an awful job of rescaling the image, doing it gradually.
 | |
|       var MAX_NUM_SCALING_STEPS = 3;
 | |
|       var reducedWidth = canvas.width << MAX_NUM_SCALING_STEPS;
 | |
|       var reducedHeight = canvas.height << MAX_NUM_SCALING_STEPS;
 | |
|       var reducedImage = getTempCanvas(reducedWidth, reducedHeight);
 | |
|       var reducedImageCtx = reducedImage.getContext('2d');
 | |
| 
 | |
|       while (reducedWidth > img.width || reducedHeight > img.height) {
 | |
|         reducedWidth >>= 1;
 | |
|         reducedHeight >>= 1;
 | |
|       }
 | |
|       reducedImageCtx.drawImage(img, 0, 0, img.width, img.height,
 | |
|                                 0, 0, reducedWidth, reducedHeight);
 | |
|       while (reducedWidth > 2 * canvas.width) {
 | |
|         reducedImageCtx.drawImage(reducedImage,
 | |
|                                   0, 0, reducedWidth, reducedHeight,
 | |
|                                   0, 0, reducedWidth >> 1, reducedHeight >> 1);
 | |
|         reducedWidth >>= 1;
 | |
|         reducedHeight >>= 1;
 | |
|       }
 | |
|       ctx.drawImage(reducedImage, 0, 0, reducedWidth, reducedHeight,
 | |
|                     0, 0, canvas.width, canvas.height);
 | |
|       this._convertCanvasToImage();
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   return PDFThumbnailView;
 | |
| })();
 | |
| 
 | |
| PDFThumbnailView.tempImageCache = null;
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * @typedef {Object} PDFThumbnailViewerOptions
 | |
|  * @property {HTMLDivElement} container - The container for the thumbnail
 | |
|  *   elements.
 | |
|  * @property {IPDFLinkService} linkService - The navigation/linking service.
 | |
|  * @property {PDFRenderingQueue} renderingQueue - The rendering queue object.
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * Simple viewer control to display thumbnails for pages.
 | |
|  * @class
 | |
|  * @implements {IRenderableView}
 | |
|  */
 | |
| var PDFThumbnailViewer = (function PDFThumbnailViewerClosure() {
 | |
|   /**
 | |
|    * @constructs PDFThumbnailViewer
 | |
|    * @param {PDFThumbnailViewerOptions} options
 | |
|    */
 | |
|   function PDFThumbnailViewer(options) {
 | |
|     this.container = options.container;
 | |
|     this.renderingQueue = options.renderingQueue;
 | |
|     this.linkService = options.linkService;
 | |
| 
 | |
|     this.scroll = watchScroll(this.container, this._scrollUpdated.bind(this));
 | |
|     this._resetView();
 | |
|   }
 | |
| 
 | |
|   PDFThumbnailViewer.prototype = {
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _scrollUpdated: function PDFThumbnailViewer_scrollUpdated() {
 | |
|       this.renderingQueue.renderHighestPriority();
 | |
|     },
 | |
| 
 | |
|     getThumbnail: function PDFThumbnailViewer_getThumbnail(index) {
 | |
|       return this.thumbnails[index];
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _getVisibleThumbs: function PDFThumbnailViewer_getVisibleThumbs() {
 | |
|       return getVisibleElements(this.container, this.thumbnails);
 | |
|     },
 | |
| 
 | |
|     scrollThumbnailIntoView:
 | |
|         function PDFThumbnailViewer_scrollThumbnailIntoView(page) {
 | |
|       var selected = document.querySelector('.thumbnail.selected');
 | |
|       if (selected) {
 | |
|         selected.classList.remove('selected');
 | |
|       }
 | |
|       var thumbnail = document.getElementById('thumbnailContainer' + page);
 | |
|       if (thumbnail) {
 | |
|         thumbnail.classList.add('selected');
 | |
|       }
 | |
|       var visibleThumbs = this._getVisibleThumbs();
 | |
|       var numVisibleThumbs = visibleThumbs.views.length;
 | |
| 
 | |
|       // If the thumbnail isn't currently visible, scroll it into view.
 | |
|       if (numVisibleThumbs > 0) {
 | |
|         var first = visibleThumbs.first.id;
 | |
|         // Account for only one thumbnail being visible.
 | |
|         var last = (numVisibleThumbs > 1 ? visibleThumbs.last.id : first);
 | |
|         if (page <= first || page >= last) {
 | |
|           scrollIntoView(thumbnail, { top: THUMBNAIL_SCROLL_MARGIN });
 | |
|         }
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     get pagesRotation() {
 | |
|       return this._pagesRotation;
 | |
|     },
 | |
| 
 | |
|     set pagesRotation(rotation) {
 | |
|       this._pagesRotation = rotation;
 | |
|       for (var i = 0, l = this.thumbnails.length; i < l; i++) {
 | |
|         var thumb = this.thumbnails[i];
 | |
|         thumb.update(rotation);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     cleanup: function PDFThumbnailViewer_cleanup() {
 | |
|       var tempCanvas = PDFThumbnailView.tempImageCache;
 | |
|       if (tempCanvas) {
 | |
|         // Zeroing the width and height causes Firefox to release graphics
 | |
|         // resources immediately, which can greatly reduce memory consumption.
 | |
|         tempCanvas.width = 0;
 | |
|         tempCanvas.height = 0;
 | |
|       }
 | |
|       PDFThumbnailView.tempImageCache = null;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _resetView: function PDFThumbnailViewer_resetView() {
 | |
|       this.thumbnails = [];
 | |
|       this._pagesRotation = 0;
 | |
|       this._pagesRequests = [];
 | |
|     },
 | |
| 
 | |
|     setDocument: function PDFThumbnailViewer_setDocument(pdfDocument) {
 | |
|       if (this.pdfDocument) {
 | |
|         // cleanup of the elements and views
 | |
|         var thumbsView = this.container;
 | |
|         while (thumbsView.hasChildNodes()) {
 | |
|           thumbsView.removeChild(thumbsView.lastChild);
 | |
|         }
 | |
|         this._resetView();
 | |
|       }
 | |
| 
 | |
|       this.pdfDocument = pdfDocument;
 | |
|       if (!pdfDocument) {
 | |
|         return Promise.resolve();
 | |
|       }
 | |
| 
 | |
|       return pdfDocument.getPage(1).then(function (firstPage) {
 | |
|         var pagesCount = pdfDocument.numPages;
 | |
|         var viewport = firstPage.getViewport(1.0);
 | |
|         for (var pageNum = 1; pageNum <= pagesCount; ++pageNum) {
 | |
|           var thumbnail = new PDFThumbnailView({
 | |
|             container: this.container,
 | |
|             id: pageNum,
 | |
|             defaultViewport: viewport.clone(),
 | |
|             linkService: this.linkService,
 | |
|             renderingQueue: this.renderingQueue
 | |
|           });
 | |
|           this.thumbnails.push(thumbnail);
 | |
|         }
 | |
|       }.bind(this));
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @param {PDFPageView} pageView
 | |
|      * @returns {PDFPage}
 | |
|      * @private
 | |
|      */
 | |
|     _ensurePdfPageLoaded:
 | |
|         function PDFThumbnailViewer_ensurePdfPageLoaded(thumbView) {
 | |
|       if (thumbView.pdfPage) {
 | |
|         return Promise.resolve(thumbView.pdfPage);
 | |
|       }
 | |
|       var pageNumber = thumbView.id;
 | |
|       if (this._pagesRequests[pageNumber]) {
 | |
|         return this._pagesRequests[pageNumber];
 | |
|       }
 | |
|       var promise = this.pdfDocument.getPage(pageNumber).then(
 | |
|         function (pdfPage) {
 | |
|           thumbView.setPdfPage(pdfPage);
 | |
|           this._pagesRequests[pageNumber] = null;
 | |
|           return pdfPage;
 | |
|         }.bind(this));
 | |
|       this._pagesRequests[pageNumber] = promise;
 | |
|       return promise;
 | |
|     },
 | |
| 
 | |
|     ensureThumbnailVisible:
 | |
|         function PDFThumbnailViewer_ensureThumbnailVisible(page) {
 | |
|       // Ensure that the thumbnail of the current page is visible
 | |
|       // when switching from another view.
 | |
|       scrollIntoView(document.getElementById('thumbnailContainer' + page));
 | |
|     },
 | |
| 
 | |
|     forceRendering: function () {
 | |
|       var visibleThumbs = this._getVisibleThumbs();
 | |
|       var thumbView = this.renderingQueue.getHighestPriority(visibleThumbs,
 | |
|                                                              this.thumbnails,
 | |
|                                                              this.scroll.down);
 | |
|       if (thumbView) {
 | |
|         this._ensurePdfPageLoaded(thumbView).then(function () {
 | |
|           this.renderingQueue.renderView(thumbView);
 | |
|         }.bind(this));
 | |
|         return true;
 | |
|       }
 | |
|       return false;
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   return PDFThumbnailViewer;
 | |
| })();
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * @typedef {Object} PDFOutlineViewOptions
 | |
|  * @property {HTMLDivElement} container - The viewer element.
 | |
|  * @property {Array} outline - An array of outline objects.
 | |
|  * @property {IPDFLinkService} linkService - The navigation/linking service.
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * @class
 | |
|  */
 | |
| var PDFOutlineView = (function PDFOutlineViewClosure() {
 | |
|   /**
 | |
|    * @constructs PDFOutlineView
 | |
|    * @param {PDFOutlineViewOptions} options
 | |
|    */
 | |
|   function PDFOutlineView(options) {
 | |
|     this.container = options.container;
 | |
|     this.outline = options.outline;
 | |
|     this.linkService = options.linkService;
 | |
|     this.lastToggleIsShow = true;
 | |
|   }
 | |
| 
 | |
|   PDFOutlineView.prototype = {
 | |
|     reset: function PDFOutlineView_reset() {
 | |
|       var container = this.container;
 | |
|       while (container.firstChild) {
 | |
|         container.removeChild(container.firstChild);
 | |
|       }
 | |
|       this.lastToggleIsShow = true;
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _dispatchEvent: function PDFOutlineView_dispatchEvent(outlineCount) {
 | |
|       var event = document.createEvent('CustomEvent');
 | |
|       event.initCustomEvent('outlineloaded', true, true, {
 | |
|         outlineCount: outlineCount
 | |
|       });
 | |
|       this.container.dispatchEvent(event);
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _bindLink: function PDFOutlineView_bindLink(element, item) {
 | |
|       if (item.url) {
 | |
|         PDFJS.addLinkAttributes(element, { url: item.url });
 | |
|         return;
 | |
|       }
 | |
|       var linkService = this.linkService;
 | |
|       element.href = linkService.getDestinationHash(item.dest);
 | |
|       element.onclick = function goToDestination(e) {
 | |
|         linkService.navigateTo(item.dest);
 | |
|         return false;
 | |
|       };
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Prepend a button before an outline item which allows the user to toggle
 | |
|      * the visibility of all outline items at that level.
 | |
|      *
 | |
|      * @private
 | |
|      */
 | |
|     _addToggleButton: function PDFOutlineView_addToggleButton(div) {
 | |
|       var toggler = document.createElement('div');
 | |
|       toggler.className = 'outlineItemToggler';
 | |
|       toggler.onclick = function(event) {
 | |
|         event.stopPropagation();
 | |
|         toggler.classList.toggle('outlineItemsHidden');
 | |
| 
 | |
|         if (event.shiftKey) {
 | |
|           var shouldShowAll = !toggler.classList.contains('outlineItemsHidden');
 | |
|           this._toggleOutlineItem(div, shouldShowAll);
 | |
|         }
 | |
|       }.bind(this);
 | |
|       div.insertBefore(toggler, div.firstChild);
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Toggle the visibility of the subtree of an outline item.
 | |
|      *
 | |
|      * @param {Element} root - the root of the outline (sub)tree.
 | |
|      * @param {boolean} state - whether to show the outline (sub)tree. If false,
 | |
|      *   the outline subtree rooted at |root| will be collapsed.
 | |
|      *
 | |
|      * @private
 | |
|      */
 | |
|     _toggleOutlineItem: function PDFOutlineView_toggleOutlineItem(root, show) {
 | |
|       this.lastToggleIsShow = show;
 | |
|       var togglers = root.querySelectorAll('.outlineItemToggler');
 | |
|       for (var i = 0, ii = togglers.length; i < ii; ++i) {
 | |
|         togglers[i].classList[show ? 'remove' : 'add']('outlineItemsHidden');
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * Collapse or expand all subtrees of the outline.
 | |
|      */
 | |
|     toggleOutlineTree: function PDFOutlineView_toggleOutlineTree() {
 | |
|       this._toggleOutlineItem(this.container, !this.lastToggleIsShow);
 | |
|     },
 | |
| 
 | |
|     render: function PDFOutlineView_render() {
 | |
|       var outline = this.outline;
 | |
|       var outlineCount = 0;
 | |
| 
 | |
|       this.reset();
 | |
| 
 | |
|       if (!outline) {
 | |
|         this._dispatchEvent(outlineCount);
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       var fragment = document.createDocumentFragment();
 | |
|       var queue = [{ parent: fragment, items: this.outline }];
 | |
|       var hasAnyNesting = false;
 | |
|       while (queue.length > 0) {
 | |
|         var levelData = queue.shift();
 | |
|         for (var i = 0, len = levelData.items.length; i < len; i++) {
 | |
|           var item = levelData.items[i];
 | |
|           var div = document.createElement('div');
 | |
|           div.className = 'outlineItem';
 | |
|           var element = document.createElement('a');
 | |
|           this._bindLink(element, item);
 | |
|           element.textContent = PDFJS.removeNullCharacters(item.title);
 | |
|           div.appendChild(element);
 | |
| 
 | |
|           if (item.items.length > 0) {
 | |
|             hasAnyNesting = true;
 | |
|             this._addToggleButton(div);
 | |
| 
 | |
|             var itemsDiv = document.createElement('div');
 | |
|             itemsDiv.className = 'outlineItems';
 | |
|             div.appendChild(itemsDiv);
 | |
|             queue.push({ parent: itemsDiv, items: item.items });
 | |
|           }
 | |
| 
 | |
|           levelData.parent.appendChild(div);
 | |
|           outlineCount++;
 | |
|         }
 | |
|       }
 | |
|       if (hasAnyNesting) {
 | |
|         this.container.classList.add('outlineWithDeepNesting');
 | |
|       }
 | |
| 
 | |
|       this.container.appendChild(fragment);
 | |
| 
 | |
|       this._dispatchEvent(outlineCount);
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   return PDFOutlineView;
 | |
| })();
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * @typedef {Object} PDFAttachmentViewOptions
 | |
|  * @property {HTMLDivElement} container - The viewer element.
 | |
|  * @property {Array} attachments - An array of attachment objects.
 | |
|  * @property {DownloadManager} downloadManager - The download manager.
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * @class
 | |
|  */
 | |
| var PDFAttachmentView = (function PDFAttachmentViewClosure() {
 | |
|   /**
 | |
|    * @constructs PDFAttachmentView
 | |
|    * @param {PDFAttachmentViewOptions} options
 | |
|    */
 | |
|   function PDFAttachmentView(options) {
 | |
|     this.container = options.container;
 | |
|     this.attachments = options.attachments;
 | |
|     this.downloadManager = options.downloadManager;
 | |
|   }
 | |
| 
 | |
|   PDFAttachmentView.prototype = {
 | |
|     reset: function PDFAttachmentView_reset() {
 | |
|       var container = this.container;
 | |
|       while (container.firstChild) {
 | |
|         container.removeChild(container.firstChild);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _dispatchEvent: function PDFAttachmentView_dispatchEvent(attachmentsCount) {
 | |
|       var event = document.createEvent('CustomEvent');
 | |
|       event.initCustomEvent('attachmentsloaded', true, true, {
 | |
|         attachmentsCount: attachmentsCount
 | |
|       });
 | |
|       this.container.dispatchEvent(event);
 | |
|     },
 | |
| 
 | |
|     /**
 | |
|      * @private
 | |
|      */
 | |
|     _bindLink: function PDFAttachmentView_bindLink(button, content, filename) {
 | |
|       button.onclick = function downloadFile(e) {
 | |
|         this.downloadManager.downloadData(content, filename, '');
 | |
|         return false;
 | |
|       }.bind(this);
 | |
|     },
 | |
| 
 | |
|     render: function PDFAttachmentView_render() {
 | |
|       var attachments = this.attachments;
 | |
|       var attachmentsCount = 0;
 | |
| 
 | |
|       this.reset();
 | |
| 
 | |
|       if (!attachments) {
 | |
|         this._dispatchEvent(attachmentsCount);
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       var names = Object.keys(attachments).sort(function(a, b) {
 | |
|         return a.toLowerCase().localeCompare(b.toLowerCase());
 | |
|       });
 | |
|       attachmentsCount = names.length;
 | |
| 
 | |
|       for (var i = 0; i < attachmentsCount; i++) {
 | |
|         var item = attachments[names[i]];
 | |
|         var filename = getFileName(item.filename);
 | |
|         var div = document.createElement('div');
 | |
|         div.className = 'attachmentsItem';
 | |
|         var button = document.createElement('button');
 | |
|         this._bindLink(button, item.content, filename);
 | |
|         button.textContent = PDFJS.removeNullCharacters(filename);
 | |
|         div.appendChild(button);
 | |
|         this.container.appendChild(div);
 | |
|       }
 | |
| 
 | |
|       this._dispatchEvent(attachmentsCount);
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   return PDFAttachmentView;
 | |
| })();
 | |
| 
 | |
| 
 | |
| var PDFViewerApplication = {
 | |
|   initialBookmark: document.location.hash.substring(1),
 | |
|   initialDestination: null,
 | |
|   initialized: false,
 | |
|   fellback: false,
 | |
|   pdfDocument: null,
 | |
|   pdfLoadingTask: null,
 | |
|   sidebarOpen: false,
 | |
|   printing: false,
 | |
|   /** @type {PDFViewer} */
 | |
|   pdfViewer: null,
 | |
|   /** @type {PDFThumbnailViewer} */
 | |
|   pdfThumbnailViewer: null,
 | |
|   /** @type {PDFRenderingQueue} */
 | |
|   pdfRenderingQueue: null,
 | |
|   /** @type {PDFPresentationMode} */
 | |
|   pdfPresentationMode: null,
 | |
|   /** @type {PDFDocumentProperties} */
 | |
|   pdfDocumentProperties: null,
 | |
|   /** @type {PDFLinkService} */
 | |
|   pdfLinkService: null,
 | |
|   /** @type {PDFHistory} */
 | |
|   pdfHistory: null,
 | |
|   pageRotation: 0,
 | |
|   isInitialViewSet: false,
 | |
|   animationStartedPromise: null,
 | |
|   preferenceSidebarViewOnLoad: SidebarView.NONE,
 | |
|   preferencePdfBugEnabled: false,
 | |
|   preferenceShowPreviousViewOnLoad: true,
 | |
|   preferenceDefaultZoomValue: '',
 | |
|   isViewerEmbedded: (window.parent !== window),
 | |
|   url: '',
 | |
| 
 | |
|   // called once when the document is loaded
 | |
|   initialize: function pdfViewInitialize() {
 | |
|     var pdfRenderingQueue = new PDFRenderingQueue();
 | |
|     pdfRenderingQueue.onIdle = this.cleanup.bind(this);
 | |
|     this.pdfRenderingQueue = pdfRenderingQueue;
 | |
| 
 | |
|     var pdfLinkService = new PDFLinkService();
 | |
|     this.pdfLinkService = pdfLinkService;
 | |
| 
 | |
|     var container = document.getElementById('viewerContainer');
 | |
|     var viewer = document.getElementById('viewer');
 | |
|     this.pdfViewer = new PDFViewer({
 | |
|       container: container,
 | |
|       viewer: viewer,
 | |
|       renderingQueue: pdfRenderingQueue,
 | |
|       linkService: pdfLinkService
 | |
|     });
 | |
|     pdfRenderingQueue.setViewer(this.pdfViewer);
 | |
|     pdfLinkService.setViewer(this.pdfViewer);
 | |
| 
 | |
|     var thumbnailContainer = document.getElementById('thumbnailView');
 | |
|     this.pdfThumbnailViewer = new PDFThumbnailViewer({
 | |
|       container: thumbnailContainer,
 | |
|       renderingQueue: pdfRenderingQueue,
 | |
|       linkService: pdfLinkService
 | |
|     });
 | |
|     pdfRenderingQueue.setThumbnailViewer(this.pdfThumbnailViewer);
 | |
| 
 | |
|     Preferences.initialize();
 | |
| 
 | |
|     this.pdfHistory = new PDFHistory({
 | |
|       linkService: pdfLinkService
 | |
|     });
 | |
|     pdfLinkService.setHistory(this.pdfHistory);
 | |
| 
 | |
|     this.findController = new PDFFindController({
 | |
|       pdfViewer: this.pdfViewer,
 | |
|       integratedFind: this.supportsIntegratedFind
 | |
|     });
 | |
|     this.pdfViewer.setFindController(this.findController);
 | |
| 
 | |
|     this.findBar = new PDFFindBar({
 | |
|       bar: document.getElementById('findbar'),
 | |
|       toggleButton: document.getElementById('viewFind'),
 | |
|       findField: document.getElementById('findInput'),
 | |
|       highlightAllCheckbox: document.getElementById('findHighlightAll'),
 | |
|       caseSensitiveCheckbox: document.getElementById('findMatchCase'),
 | |
|       findMsg: document.getElementById('findMsg'),
 | |
|       findResultsCount: document.getElementById('findResultsCount'),
 | |
|       findStatusIcon: document.getElementById('findStatusIcon'),
 | |
|       findPreviousButton: document.getElementById('findPrevious'),
 | |
|       findNextButton: document.getElementById('findNext'),
 | |
|       findController: this.findController
 | |
|     });
 | |
| 
 | |
|     this.findController.setFindBar(this.findBar);
 | |
| 
 | |
|     HandTool.initialize({
 | |
|       container: container,
 | |
|       toggleHandTool: document.getElementById('toggleHandTool')
 | |
|     });
 | |
| 
 | |
|     this.pdfDocumentProperties = new PDFDocumentProperties({
 | |
|       overlayName: 'documentPropertiesOverlay',
 | |
|       closeButton: document.getElementById('documentPropertiesClose'),
 | |
|       fields: {
 | |
|         'fileName': document.getElementById('fileNameField'),
 | |
|         'fileSize': document.getElementById('fileSizeField'),
 | |
|         'title': document.getElementById('titleField'),
 | |
|         'author': document.getElementById('authorField'),
 | |
|         'subject': document.getElementById('subjectField'),
 | |
|         'keywords': document.getElementById('keywordsField'),
 | |
|         'creationDate': document.getElementById('creationDateField'),
 | |
|         'modificationDate': document.getElementById('modificationDateField'),
 | |
|         'creator': document.getElementById('creatorField'),
 | |
|         'producer': document.getElementById('producerField'),
 | |
|         'version': document.getElementById('versionField'),
 | |
|         'pageCount': document.getElementById('pageCountField')
 | |
|       }
 | |
|     });
 | |
| 
 | |
|     SecondaryToolbar.initialize({
 | |
|       toolbar: document.getElementById('secondaryToolbar'),
 | |
|       toggleButton: document.getElementById('secondaryToolbarToggle'),
 | |
|       presentationModeButton:
 | |
|         document.getElementById('secondaryPresentationMode'),
 | |
|       openFile: document.getElementById('secondaryOpenFile'),
 | |
|       print: document.getElementById('secondaryPrint'),
 | |
|       download: document.getElementById('secondaryDownload'),
 | |
|       viewBookmark: document.getElementById('secondaryViewBookmark'),
 | |
|       firstPage: document.getElementById('firstPage'),
 | |
|       lastPage: document.getElementById('lastPage'),
 | |
|       pageRotateCw: document.getElementById('pageRotateCw'),
 | |
|       pageRotateCcw: document.getElementById('pageRotateCcw'),
 | |
|       documentPropertiesButton: document.getElementById('documentProperties')
 | |
|     });
 | |
| 
 | |
|     if (this.supportsFullscreen) {
 | |
|       var toolbar = SecondaryToolbar;
 | |
|       this.pdfPresentationMode = new PDFPresentationMode({
 | |
|         container: container,
 | |
|         viewer: viewer,
 | |
|         pdfViewer: this.pdfViewer,
 | |
|         pdfThumbnailViewer: this.pdfThumbnailViewer,
 | |
|         contextMenuItems: [
 | |
|           { element: document.getElementById('contextFirstPage'),
 | |
|             handler: toolbar.firstPageClick.bind(toolbar) },
 | |
|           { element: document.getElementById('contextLastPage'),
 | |
|             handler: toolbar.lastPageClick.bind(toolbar) },
 | |
|           { element: document.getElementById('contextPageRotateCw'),
 | |
|             handler: toolbar.pageRotateCwClick.bind(toolbar) },
 | |
|           { element: document.getElementById('contextPageRotateCcw'),
 | |
|             handler: toolbar.pageRotateCcwClick.bind(toolbar) }
 | |
|         ]
 | |
|       });
 | |
|     }
 | |
| 
 | |
|     PasswordPrompt.initialize({
 | |
|       overlayName: 'passwordOverlay',
 | |
|       passwordField: document.getElementById('password'),
 | |
|       passwordText: document.getElementById('passwordText'),
 | |
|       passwordSubmit: document.getElementById('passwordSubmit'),
 | |
|       passwordCancel: document.getElementById('passwordCancel')
 | |
|     });
 | |
| 
 | |
|     var self = this;
 | |
|     var initializedPromise = Promise.all([
 | |
|       Preferences.get('enableWebGL').then(function resolved(value) {
 | |
|         PDFJS.disableWebGL = !value;
 | |
|       }),
 | |
|       Preferences.get('sidebarViewOnLoad').then(function resolved(value) {
 | |
|         self.preferenceSidebarViewOnLoad = value;
 | |
|       }),
 | |
|       Preferences.get('pdfBugEnabled').then(function resolved(value) {
 | |
|         self.preferencePdfBugEnabled = value;
 | |
|       }),
 | |
|       Preferences.get('showPreviousViewOnLoad').then(function resolved(value) {
 | |
|         self.preferenceShowPreviousViewOnLoad = value;
 | |
|       }),
 | |
|       Preferences.get('defaultZoomValue').then(function resolved(value) {
 | |
|         self.preferenceDefaultZoomValue = value;
 | |
|       }),
 | |
|       Preferences.get('disableTextLayer').then(function resolved(value) {
 | |
|         if (PDFJS.disableTextLayer === true) {
 | |
|           return;
 | |
|         }
 | |
|         PDFJS.disableTextLayer = value;
 | |
|       }),
 | |
|       Preferences.get('disableRange').then(function resolved(value) {
 | |
|         if (PDFJS.disableRange === true) {
 | |
|           return;
 | |
|         }
 | |
|         PDFJS.disableRange = value;
 | |
|       }),
 | |
|       Preferences.get('disableStream').then(function resolved(value) {
 | |
|         if (PDFJS.disableStream === true) {
 | |
|           return;
 | |
|         }
 | |
|         PDFJS.disableStream = value;
 | |
|       }),
 | |
|       Preferences.get('disableAutoFetch').then(function resolved(value) {
 | |
|         PDFJS.disableAutoFetch = value;
 | |
|       }),
 | |
|       Preferences.get('disableFontFace').then(function resolved(value) {
 | |
|         if (PDFJS.disableFontFace === true) {
 | |
|           return;
 | |
|         }
 | |
|         PDFJS.disableFontFace = value;
 | |
|       }),
 | |
|       Preferences.get('useOnlyCssZoom').then(function resolved(value) {
 | |
|         PDFJS.useOnlyCssZoom = value;
 | |
|       }),
 | |
|       Preferences.get('externalLinkTarget').then(function resolved(value) {
 | |
|         if (PDFJS.isExternalLinkTargetSet()) {
 | |
|           return;
 | |
|         }
 | |
|         PDFJS.externalLinkTarget = value;
 | |
|       }),
 | |
|       // TODO move more preferences and other async stuff here
 | |
|     ]).catch(function (reason) { });
 | |
| 
 | |
|     return initializedPromise.then(function () {
 | |
|       if (self.isViewerEmbedded && !PDFJS.isExternalLinkTargetSet()) {
 | |
|         // Prevent external links from "replacing" the viewer,
 | |
|         // when it's embedded in e.g. an iframe or an object.
 | |
|         PDFJS.externalLinkTarget = PDFJS.LinkTarget.TOP;
 | |
|       }
 | |
| 
 | |
|       self.initialized = true;
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   zoomIn: function pdfViewZoomIn(ticks) {
 | |
|     var newScale = this.pdfViewer.currentScale;
 | |
|     do {
 | |
|       newScale = (newScale * DEFAULT_SCALE_DELTA).toFixed(2);
 | |
|       newScale = Math.ceil(newScale * 10) / 10;
 | |
|       newScale = Math.min(MAX_SCALE, newScale);
 | |
|     } while (--ticks > 0 && newScale < MAX_SCALE);
 | |
|     this.pdfViewer.currentScaleValue = newScale;
 | |
|   },
 | |
| 
 | |
|   zoomOut: function pdfViewZoomOut(ticks) {
 | |
|     var newScale = this.pdfViewer.currentScale;
 | |
|     do {
 | |
|       newScale = (newScale / DEFAULT_SCALE_DELTA).toFixed(2);
 | |
|       newScale = Math.floor(newScale * 10) / 10;
 | |
|       newScale = Math.max(MIN_SCALE, newScale);
 | |
|     } while (--ticks > 0 && newScale > MIN_SCALE);
 | |
|     this.pdfViewer.currentScaleValue = newScale;
 | |
|   },
 | |
| 
 | |
|   get pagesCount() {
 | |
|     return this.pdfDocument.numPages;
 | |
|   },
 | |
| 
 | |
|   set page(val) {
 | |
|     this.pdfLinkService.page = val;
 | |
|   },
 | |
| 
 | |
|   get page() { // TODO remove
 | |
|     return this.pdfLinkService.page;
 | |
|   },
 | |
| 
 | |
|   get supportsPrinting() {
 | |
|     var canvas = document.createElement('canvas');
 | |
|     var value = 'mozPrintCallback' in canvas;
 | |
| 
 | |
|     return PDFJS.shadow(this, 'supportsPrinting', value);
 | |
|   },
 | |
| 
 | |
|   get supportsFullscreen() {
 | |
|     var doc = document.documentElement;
 | |
|     var support = !!(doc.requestFullscreen || doc.mozRequestFullScreen ||
 | |
|                      doc.webkitRequestFullScreen || doc.msRequestFullscreen);
 | |
| 
 | |
|     if (document.fullscreenEnabled === false ||
 | |
|         document.mozFullScreenEnabled === false ||
 | |
|         document.webkitFullscreenEnabled === false ||
 | |
|         document.msFullscreenEnabled === false) {
 | |
|       support = false;
 | |
|     }
 | |
|     if (support && PDFJS.disableFullscreen === true) {
 | |
|       support = false;
 | |
|     }
 | |
| 
 | |
|     return PDFJS.shadow(this, 'supportsFullscreen', support);
 | |
|   },
 | |
| 
 | |
|   get supportsIntegratedFind() {
 | |
|     var support = false;
 | |
| 
 | |
|     return PDFJS.shadow(this, 'supportsIntegratedFind', support);
 | |
|   },
 | |
| 
 | |
|   get supportsDocumentFonts() {
 | |
|     var support = true;
 | |
| 
 | |
|     return PDFJS.shadow(this, 'supportsDocumentFonts', support);
 | |
|   },
 | |
| 
 | |
|   get supportsDocumentColors() {
 | |
|     var support = true;
 | |
| 
 | |
|     return PDFJS.shadow(this, 'supportsDocumentColors', support);
 | |
|   },
 | |
| 
 | |
|   get loadingBar() {
 | |
|     var bar = new ProgressBar('#loadingBar', {});
 | |
| 
 | |
|     return PDFJS.shadow(this, 'loadingBar', bar);
 | |
|   },
 | |
| 
 | |
|   get supportedMouseWheelZoomModifierKeys() {
 | |
|     var support = {
 | |
|       ctrlKey: true,
 | |
|       metaKey: true,
 | |
|     };
 | |
| 
 | |
|     return PDFJS.shadow(this, 'supportedMouseWheelZoomModifierKeys', support);
 | |
|   },
 | |
| 
 | |
| 
 | |
|   setTitleUsingUrl: function pdfViewSetTitleUsingUrl(url) {
 | |
|     this.url = url;
 | |
|     try {
 | |
|       this.setTitle(decodeURIComponent(getFileName(url)) || url);
 | |
|     } catch (e) {
 | |
|       // decodeURIComponent may throw URIError,
 | |
|       // fall back to using the unprocessed url in that case
 | |
|       this.setTitle(url);
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   setTitle: function pdfViewSetTitle(title) {
 | |
|     if (this.isViewerEmbedded) {
 | |
|       // Embedded PDF viewers should not be changing their parent page's title.
 | |
|       return;
 | |
|     }
 | |
|     document.title = title;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Closes opened PDF document.
 | |
|    * @returns {Promise} - Returns the promise, which is resolved when all
 | |
|    *                      destruction is completed.
 | |
|    */
 | |
|   close: function pdfViewClose() {
 | |
|     var errorWrapper = document.getElementById('errorWrapper');
 | |
|     errorWrapper.setAttribute('hidden', 'true');
 | |
| 
 | |
|     if (!this.pdfLoadingTask) {
 | |
|       return Promise.resolve();
 | |
|     }
 | |
| 
 | |
|     var promise = this.pdfLoadingTask.destroy();
 | |
|     this.pdfLoadingTask = null;
 | |
| 
 | |
|     if (this.pdfDocument) {
 | |
|       this.pdfDocument = null;
 | |
| 
 | |
|       this.pdfThumbnailViewer.setDocument(null);
 | |
|       this.pdfViewer.setDocument(null);
 | |
|       this.pdfLinkService.setDocument(null, null);
 | |
|     }
 | |
| 
 | |
|     if (typeof PDFBug !== 'undefined') {
 | |
|       PDFBug.cleanup();
 | |
|     }
 | |
|     return promise;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Opens PDF document specified by URL or array with additional arguments.
 | |
|    * @param {string|TypedArray|ArrayBuffer} file - PDF location or binary data.
 | |
|    * @param {Object} args - (optional) Additional arguments for the getDocument
 | |
|    *                        call, e.g. HTTP headers ('httpHeaders') or
 | |
|    *                        alternative data transport ('range').
 | |
|    * @returns {Promise} - Returns the promise, which is resolved when document
 | |
|    *                      is opened.
 | |
|    */
 | |
|   open: function pdfViewOpen(file, args) {
 | |
|     var scale = 0;
 | |
|     if (arguments.length > 2 || typeof args === 'number') {
 | |
|       console.warn('Call of open() with obsolete signature.');
 | |
|       if (typeof args === 'number') {
 | |
|         scale = args; // scale argument was found
 | |
|       }
 | |
|       args = arguments[4] || null;
 | |
|       if (arguments[3] && typeof arguments[3] === 'object') {
 | |
|         // The pdfDataRangeTransport argument is present.
 | |
|         args = Object.create(args);
 | |
|         args.range = arguments[3];
 | |
|       }
 | |
|       if (typeof arguments[2] === 'string') {
 | |
|         // The password argument is present.
 | |
|         args = Object.create(args);
 | |
|         args.password = arguments[2];
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (this.pdfLoadingTask) {
 | |
|       // We need to destroy already opened document.
 | |
|       return this.close().then(function () {
 | |
|         // Reload the preferences if a document was previously opened.
 | |
|         Preferences.reload();
 | |
|         // ... and repeat the open() call.
 | |
|         return this.open(file, args);
 | |
|       }.bind(this));
 | |
|     }
 | |
| 
 | |
|     var parameters = Object.create(null);
 | |
|     if (typeof file === 'string') { // URL
 | |
|       this.setTitleUsingUrl(file);
 | |
|       parameters.url = file;
 | |
|     } else if (file && 'byteLength' in file) { // ArrayBuffer
 | |
|       parameters.data = file;
 | |
|     } else if (file.url && file.originalUrl) {
 | |
|       this.setTitleUsingUrl(file.originalUrl);
 | |
|       parameters.url = file.url;
 | |
|     }
 | |
|     if (args) {
 | |
|       for (var prop in args) {
 | |
|         parameters[prop] = args[prop];
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     var self = this;
 | |
|     self.downloadComplete = false;
 | |
| 
 | |
|     var loadingTask = PDFJS.getDocument(parameters);
 | |
|     this.pdfLoadingTask = loadingTask;
 | |
| 
 | |
|     loadingTask.onPassword = function passwordNeeded(updatePassword, reason) {
 | |
|       PasswordPrompt.updatePassword = updatePassword;
 | |
|       PasswordPrompt.reason = reason;
 | |
|       PasswordPrompt.open();
 | |
|     };
 | |
| 
 | |
|     loadingTask.onProgress = function getDocumentProgress(progressData) {
 | |
|       self.progress(progressData.loaded / progressData.total);
 | |
|     };
 | |
| 
 | |
|     // Listen for unsupported features to trigger the fallback UI.
 | |
|     loadingTask.onUnsupportedFeature = this.fallback.bind(this);
 | |
| 
 | |
|     var result = loadingTask.promise.then(
 | |
|       function getDocumentCallback(pdfDocument) {
 | |
|         self.load(pdfDocument, scale);
 | |
|       },
 | |
|       function getDocumentError(exception) {
 | |
|         var message = exception && exception.message;
 | |
|         var loadingErrorMessage = 'An error occurred while loading the PDF.';
 | |
| 
 | |
|         if (exception instanceof PDFJS.InvalidPDFException) {
 | |
|           // change error message also for other builds
 | |
|           loadingErrorMessage = 'Invalid or corrupted PDF file.';
 | |
|         } else if (exception instanceof PDFJS.MissingPDFException) {
 | |
|           // special message for missing PDF's
 | |
|           loadingErrorMessage = 'Missing PDF file.';
 | |
|         } else if (exception instanceof PDFJS.UnexpectedResponseException) {
 | |
|           loadingErrorMessage = 'Unexpected server response.';
 | |
|         }
 | |
| 
 | |
|         var moreInfo = {
 | |
|           message: message
 | |
|         };
 | |
|         self.error(loadingErrorMessage, moreInfo);
 | |
| 
 | |
|         throw new Error(loadingErrorMessage);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     if (args && args.length) {
 | |
|       PDFViewerApplication.pdfDocumentProperties.setFileSize(args.length);
 | |
|     }
 | |
|     return result;
 | |
|   },
 | |
| 
 | |
|   download: function pdfViewDownload() {
 | |
|     function downloadByUrl() {
 | |
|       downloadManager.downloadUrl(url, filename);
 | |
|     }
 | |
| 
 | |
|     var url = this.url.split('#')[0];
 | |
|     var filename = getPDFFileNameFromURL(url);
 | |
|     var downloadManager = new DownloadManager();
 | |
|     downloadManager.onerror = function (err) {
 | |
|       // This error won't really be helpful because it's likely the
 | |
|       // fallback won't work either (or is already open).
 | |
|       PDFViewerApplication.error('PDF failed to download.');
 | |
|     };
 | |
| 
 | |
|     if (!this.pdfDocument) { // the PDF is not ready yet
 | |
|       downloadByUrl();
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     if (!this.downloadComplete) { // the PDF is still downloading
 | |
|       downloadByUrl();
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     this.pdfDocument.getData().then(
 | |
|       function getDataSuccess(data) {
 | |
|         var blob = PDFJS.createBlob(data, 'application/pdf');
 | |
|         downloadManager.download(blob, url, filename);
 | |
|       },
 | |
|       downloadByUrl // Error occurred try downloading with just the url.
 | |
|     ).then(null, downloadByUrl);
 | |
|   },
 | |
| 
 | |
|   fallback: function pdfViewFallback(featureId) {
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Show the error box.
 | |
|    * @param {String} message A message that is human readable.
 | |
|    * @param {Object} moreInfo (optional) Further information about the error
 | |
|    *                            that is more technical.  Should have a 'message'
 | |
|    *                            and optionally a 'stack' property.
 | |
|    */
 | |
|   error: function pdfViewError(message, moreInfo) {
 | |
|     var moreInfoText = 'PDF.js v' + (PDFJS.version || '?') + '(build: ' + (PDFJS.build || '?') + ')\n';
 | |
|     if (moreInfo) {
 | |
|       moreInfoText += 'Message: ' + moreInfo.message
 | |
|       if (moreInfo.stack) {
 | |
|         moreInfoText += '\nStack: ' + moreInfo.stack;
 | |
|       } else {
 | |
|         if (moreInfo.filename) {
 | |
|           moreInfoText += '\nFile: ' + moreInfo.filename;
 | |
|         }
 | |
|         if (moreInfo.lineNumber) {
 | |
|           moreInfoText += '\nLine: ' + moreInfo.lineNumber;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     var errorWrapper = document.getElementById('errorWrapper');
 | |
|     errorWrapper.removeAttribute('hidden');
 | |
| 
 | |
|     var errorMessage = document.getElementById('errorMessage');
 | |
|     errorMessage.textContent = message;
 | |
| 
 | |
|     var closeButton = document.getElementById('errorClose');
 | |
|     closeButton.onclick = function() {
 | |
|       errorWrapper.setAttribute('hidden', 'true');
 | |
|     };
 | |
| 
 | |
|     var errorMoreInfo = document.getElementById('errorMoreInfo');
 | |
|     var moreInfoButton = document.getElementById('errorShowMore');
 | |
|     var lessInfoButton = document.getElementById('errorShowLess');
 | |
|     moreInfoButton.onclick = function() {
 | |
|       errorMoreInfo.removeAttribute('hidden');
 | |
|       moreInfoButton.setAttribute('hidden', 'true');
 | |
|       lessInfoButton.removeAttribute('hidden');
 | |
|       errorMoreInfo.style.height = errorMoreInfo.scrollHeight + 'px';
 | |
|     };
 | |
|     lessInfoButton.onclick = function() {
 | |
|       errorMoreInfo.setAttribute('hidden', 'true');
 | |
|       moreInfoButton.removeAttribute('hidden');
 | |
|       lessInfoButton.setAttribute('hidden', 'true');
 | |
|     };
 | |
|     moreInfoButton.oncontextmenu = noContextMenuHandler;
 | |
|     lessInfoButton.oncontextmenu = noContextMenuHandler;
 | |
|     closeButton.oncontextmenu = noContextMenuHandler;
 | |
|     moreInfoButton.removeAttribute('hidden');
 | |
|     lessInfoButton.setAttribute('hidden', 'true');
 | |
|     errorMoreInfo.value = moreInfoText;
 | |
|   },
 | |
| 
 | |
|   progress: function pdfViewProgress(level) {
 | |
|     var percent = Math.round(level * 100);
 | |
|     // When we transition from full request to range requests, it's possible
 | |
|     // that we discard some of the loaded data. This can cause the loading
 | |
|     // bar to move backwards. So prevent this by only updating the bar if it
 | |
|     // increases.
 | |
|     if (percent > this.loadingBar.percent || isNaN(percent)) {
 | |
|       this.loadingBar.percent = percent;
 | |
| 
 | |
|       // When disableAutoFetch is enabled, it's not uncommon for the entire file
 | |
|       // to never be fetched (depends on e.g. the file structure). In this case
 | |
|       // the loading bar will not be completely filled, nor will it be hidden.
 | |
|       // To prevent displaying a partially filled loading bar permanently, we
 | |
|       // hide it when no data has been loaded during a certain amount of time.
 | |
|       if (PDFJS.disableAutoFetch && percent) {
 | |
|         if (this.disableAutoFetchLoadingBarTimeout) {
 | |
|           clearTimeout(this.disableAutoFetchLoadingBarTimeout);
 | |
|           this.disableAutoFetchLoadingBarTimeout = null;
 | |
|         }
 | |
|         this.loadingBar.show();
 | |
| 
 | |
|         this.disableAutoFetchLoadingBarTimeout = setTimeout(function () {
 | |
|           this.loadingBar.hide();
 | |
|           this.disableAutoFetchLoadingBarTimeout = null;
 | |
|         }.bind(this), DISABLE_AUTO_FETCH_LOADING_BAR_TIMEOUT);
 | |
|       }
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   load: function pdfViewLoad(pdfDocument, scale) {
 | |
|     var self = this;
 | |
|     scale = scale || UNKNOWN_SCALE;
 | |
| 
 | |
|     this.findController.reset();
 | |
| 
 | |
|     this.pdfDocument = pdfDocument;
 | |
| 
 | |
|     this.pdfDocumentProperties.setDocumentAndUrl(pdfDocument, this.url);
 | |
| 
 | |
|     var downloadedPromise = pdfDocument.getDownloadInfo().then(function() {
 | |
|       self.downloadComplete = true;
 | |
|       self.loadingBar.hide();
 | |
|     });
 | |
| 
 | |
|     var pagesCount = pdfDocument.numPages;
 | |
|     document.getElementById('numPages').textContent = 'of ' + pagesCount;
 | |
|     document.getElementById('pageNumber').max = pagesCount;
 | |
| 
 | |
|     var id = this.documentFingerprint = pdfDocument.fingerprint;
 | |
|     var store = this.store = new ViewHistory(id);
 | |
| 
 | |
|     var baseDocumentUrl = null;
 | |
|     this.pdfLinkService.setDocument(pdfDocument, baseDocumentUrl);
 | |
| 
 | |
|     var pdfViewer = this.pdfViewer;
 | |
|     pdfViewer.currentScale = scale;
 | |
|     pdfViewer.setDocument(pdfDocument);
 | |
|     var firstPagePromise = pdfViewer.firstPagePromise;
 | |
|     var pagesPromise = pdfViewer.pagesPromise;
 | |
|     var onePageRendered = pdfViewer.onePageRendered;
 | |
| 
 | |
|     this.pageRotation = 0;
 | |
|     this.isInitialViewSet = false;
 | |
| 
 | |
|     this.pdfThumbnailViewer.setDocument(pdfDocument);
 | |
| 
 | |
|     firstPagePromise.then(function(pdfPage) {
 | |
|       downloadedPromise.then(function () {
 | |
|         var event = document.createEvent('CustomEvent');
 | |
|         event.initCustomEvent('documentload', true, true, {});
 | |
|         window.dispatchEvent(event);
 | |
|       });
 | |
| 
 | |
|       self.loadingBar.setWidth(document.getElementById('viewer'));
 | |
| 
 | |
|       if (!PDFJS.disableHistory && !self.isViewerEmbedded) {
 | |
|         // The browsing history is only enabled when the viewer is standalone,
 | |
|         // i.e. not when it is embedded in a web page.
 | |
|         if (!self.preferenceShowPreviousViewOnLoad) {
 | |
|           self.pdfHistory.clearHistoryState();
 | |
|         }
 | |
|         self.pdfHistory.initialize(self.documentFingerprint);
 | |
| 
 | |
|         if (self.pdfHistory.initialDestination) {
 | |
|           self.initialDestination = self.pdfHistory.initialDestination;
 | |
|         } else if (self.pdfHistory.initialBookmark) {
 | |
|           self.initialBookmark = self.pdfHistory.initialBookmark;
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       var initialParams = {
 | |
|         destination: self.initialDestination,
 | |
|         bookmark: self.initialBookmark,
 | |
|         hash: null,
 | |
|       };
 | |
| 
 | |
|       store.initializedPromise.then(function resolved() {
 | |
|         var storedHash = null;
 | |
|         if (self.preferenceShowPreviousViewOnLoad &&
 | |
|             store.get('exists', false)) {
 | |
|           var pageNum = store.get('page', '1');
 | |
|           var zoom = self.preferenceDefaultZoomValue ||
 | |
|                      store.get('zoom', DEFAULT_SCALE_VALUE);
 | |
|           var left = store.get('scrollLeft', '0');
 | |
|           var top = store.get('scrollTop', '0');
 | |
| 
 | |
|           storedHash = 'page=' + pageNum + '&zoom=' + zoom + ',' +
 | |
|                        left + ',' + top;
 | |
|         } else if (self.preferenceDefaultZoomValue) {
 | |
|           storedHash = 'page=1&zoom=' + self.preferenceDefaultZoomValue;
 | |
|         }
 | |
|         self.setInitialView(storedHash, scale);
 | |
| 
 | |
|         initialParams.hash = storedHash;
 | |
| 
 | |
|         // Make all navigation keys work on document load,
 | |
|         // unless the viewer is embedded in a web page.
 | |
|         if (!self.isViewerEmbedded) {
 | |
|           self.pdfViewer.focus();
 | |
|         }
 | |
|       }, function rejected(reason) {
 | |
|         console.error(reason);
 | |
|         self.setInitialView(null, scale);
 | |
|       });
 | |
| 
 | |
|       // For documents with different page sizes,
 | |
|       // ensure that the correct location becomes visible on load.
 | |
|       pagesPromise.then(function resolved() {
 | |
|         if (!initialParams.destination && !initialParams.bookmark &&
 | |
|             !initialParams.hash) {
 | |
|           return;
 | |
|         }
 | |
|         if (self.hasEqualPageSizes) {
 | |
|           return;
 | |
|         }
 | |
|         self.initialDestination = initialParams.destination;
 | |
|         self.initialBookmark = initialParams.bookmark;
 | |
| 
 | |
|         self.pdfViewer.currentScaleValue = self.pdfViewer.currentScaleValue;
 | |
|         self.setInitialView(initialParams.hash, scale);
 | |
|       });
 | |
|     });
 | |
| 
 | |
|     pagesPromise.then(function() {
 | |
|       if (self.supportsPrinting) {
 | |
|         pdfDocument.getJavaScript().then(function(javaScript) {
 | |
|           if (javaScript.length) {
 | |
|             console.warn('Warning: JavaScript is not supported');
 | |
|             self.fallback(PDFJS.UNSUPPORTED_FEATURES.javaScript);
 | |
|           }
 | |
|           // Hack to support auto printing.
 | |
|           var regex = /\bprint\s*\(/;
 | |
|           for (var i = 0, ii = javaScript.length; i < ii; i++) {
 | |
|             var js = javaScript[i];
 | |
|             if (js && regex.test(js)) {
 | |
|               setTimeout(function() {
 | |
|                 window.print();
 | |
|               });
 | |
|               return;
 | |
|             }
 | |
|           }
 | |
|         });
 | |
|       }
 | |
|     });
 | |
| 
 | |
|     // outline depends on pagesRefMap
 | |
|     var promises = [pagesPromise, this.animationStartedPromise];
 | |
|     Promise.all(promises).then(function() {
 | |
|       pdfDocument.getOutline().then(function(outline) {
 | |
|         var container = document.getElementById('outlineView');
 | |
|         self.outline = new PDFOutlineView({
 | |
|           container: container,
 | |
|           outline: outline,
 | |
|           linkService: self.pdfLinkService
 | |
|         });
 | |
|         self.outline.render();
 | |
|         document.getElementById('viewOutline').disabled = !outline;
 | |
| 
 | |
|         if (!outline && !container.classList.contains('hidden')) {
 | |
|           self.switchSidebarView('thumbs');
 | |
|         }
 | |
|         if (outline &&
 | |
|             self.preferenceSidebarViewOnLoad === SidebarView.OUTLINE) {
 | |
|           self.switchSidebarView('outline', true);
 | |
|         }
 | |
|       });
 | |
|       pdfDocument.getAttachments().then(function(attachments) {
 | |
|         var container = document.getElementById('attachmentsView');
 | |
|         self.attachments = new PDFAttachmentView({
 | |
|           container: container,
 | |
|           attachments: attachments,
 | |
|           downloadManager: new DownloadManager()
 | |
|         });
 | |
|         self.attachments.render();
 | |
|         document.getElementById('viewAttachments').disabled = !attachments;
 | |
| 
 | |
|         if (!attachments && !container.classList.contains('hidden')) {
 | |
|           self.switchSidebarView('thumbs');
 | |
|         }
 | |
|         if (attachments &&
 | |
|             self.preferenceSidebarViewOnLoad === SidebarView.ATTACHMENTS) {
 | |
|           self.switchSidebarView('attachments', true);
 | |
|         }
 | |
|       });
 | |
|     });
 | |
| 
 | |
|     if (self.preferenceSidebarViewOnLoad === SidebarView.THUMBS) {
 | |
|       Promise.all([firstPagePromise, onePageRendered]).then(function () {
 | |
|         self.switchSidebarView('thumbs', true);
 | |
|       });
 | |
|     }
 | |
| 
 | |
|     pdfDocument.getMetadata().then(function(data) {
 | |
|       var info = data.info, metadata = data.metadata;
 | |
|       self.documentInfo = info;
 | |
|       self.metadata = metadata;
 | |
| 
 | |
|       // Provides some basic debug information
 | |
|       console.log('PDF ' + pdfDocument.fingerprint + ' [' +
 | |
|                   info.PDFFormatVersion + ' ' + (info.Producer || '-').trim() +
 | |
|                   ' / ' + (info.Creator || '-').trim() + ']' +
 | |
|                   ' (PDF.js: ' + (PDFJS.version || '-') +
 | |
|                   (!PDFJS.disableWebGL ? ' [WebGL]' : '') + ')');
 | |
| 
 | |
|       var pdfTitle;
 | |
|       if (metadata && metadata.has('dc:title')) {
 | |
|         var title = metadata.get('dc:title');
 | |
|         // Ghostscript sometimes return 'Untitled', sets the title to 'Untitled'
 | |
|         if (title !== 'Untitled') {
 | |
|           pdfTitle = title;
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       if (!pdfTitle && info && info['Title']) {
 | |
|         pdfTitle = info['Title'];
 | |
|       }
 | |
| 
 | |
|       if (pdfTitle) {
 | |
|         self.setTitle(pdfTitle + ' - ' + document.title);
 | |
|       }
 | |
| 
 | |
|       if (info.IsAcroFormPresent) {
 | |
|         console.warn('Warning: AcroForm/XFA is not supported');
 | |
|         self.fallback(PDFJS.UNSUPPORTED_FEATURES.forms);
 | |
|       }
 | |
| 
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   setInitialView: function pdfViewSetInitialView(storedHash, scale) {
 | |
|     this.isInitialViewSet = true;
 | |
| 
 | |
|     // When opening a new file, when one is already loaded in the viewer,
 | |
|     // ensure that the 'pageNumber' element displays the correct value.
 | |
|     document.getElementById('pageNumber').value =
 | |
|       this.pdfViewer.currentPageNumber;
 | |
| 
 | |
|     if (this.initialDestination) {
 | |
|       this.pdfLinkService.navigateTo(this.initialDestination);
 | |
|       this.initialDestination = null;
 | |
|     } else if (this.initialBookmark) {
 | |
|       this.pdfLinkService.setHash(this.initialBookmark);
 | |
|       this.pdfHistory.push({ hash: this.initialBookmark }, true);
 | |
|       this.initialBookmark = null;
 | |
|     } else if (storedHash) {
 | |
|       this.pdfLinkService.setHash(storedHash);
 | |
|     } else if (scale) {
 | |
|       this.pdfViewer.currentScaleValue = scale;
 | |
|       this.page = 1;
 | |
|     }
 | |
| 
 | |
|     if (!this.pdfViewer.currentScaleValue) {
 | |
|       // Scale was not initialized: invalid bookmark or scale was not specified.
 | |
|       // Setting the default one.
 | |
|       this.pdfViewer.currentScaleValue = DEFAULT_SCALE_VALUE;
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   cleanup: function pdfViewCleanup() {
 | |
|     if (!this.pdfDocument) {
 | |
|       return; // run cleanup when document is loaded
 | |
|     }
 | |
|     this.pdfViewer.cleanup();
 | |
|     this.pdfThumbnailViewer.cleanup();
 | |
|     this.pdfDocument.cleanup();
 | |
|   },
 | |
| 
 | |
|   forceRendering: function pdfViewForceRendering() {
 | |
|     this.pdfRenderingQueue.printing = this.printing;
 | |
|     this.pdfRenderingQueue.isThumbnailViewEnabled = this.sidebarOpen;
 | |
|     this.pdfRenderingQueue.renderHighestPriority();
 | |
|   },
 | |
| 
 | |
|   refreshThumbnailViewer: function pdfViewRefreshThumbnailViewer() {
 | |
|     var pdfViewer = this.pdfViewer;
 | |
|     var thumbnailViewer = this.pdfThumbnailViewer;
 | |
| 
 | |
|     // set thumbnail images of rendered pages
 | |
|     var pagesCount = pdfViewer.pagesCount;
 | |
|     for (var pageIndex = 0; pageIndex < pagesCount; pageIndex++) {
 | |
|       var pageView = pdfViewer.getPageView(pageIndex);
 | |
|       if (pageView && pageView.renderingState === RenderingStates.FINISHED) {
 | |
|         var thumbnailView = thumbnailViewer.getThumbnail(pageIndex);
 | |
|         thumbnailView.setImage(pageView);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     thumbnailViewer.scrollThumbnailIntoView(this.page);
 | |
|   },
 | |
| 
 | |
|   switchSidebarView: function pdfViewSwitchSidebarView(view, openSidebar) {
 | |
|     if (openSidebar && !this.sidebarOpen) {
 | |
|       document.getElementById('sidebarToggle').click();
 | |
|     }
 | |
|     var thumbsView = document.getElementById('thumbnailView');
 | |
|     var outlineView = document.getElementById('outlineView');
 | |
|     var attachmentsView = document.getElementById('attachmentsView');
 | |
| 
 | |
|     var thumbsButton = document.getElementById('viewThumbnail');
 | |
|     var outlineButton = document.getElementById('viewOutline');
 | |
|     var attachmentsButton = document.getElementById('viewAttachments');
 | |
| 
 | |
|     switch (view) {
 | |
|       case 'thumbs':
 | |
|         var wasAnotherViewVisible = thumbsView.classList.contains('hidden');
 | |
| 
 | |
|         thumbsButton.classList.add('toggled');
 | |
|         outlineButton.classList.remove('toggled');
 | |
|         attachmentsButton.classList.remove('toggled');
 | |
|         thumbsView.classList.remove('hidden');
 | |
|         outlineView.classList.add('hidden');
 | |
|         attachmentsView.classList.add('hidden');
 | |
| 
 | |
|         this.forceRendering();
 | |
| 
 | |
|         if (wasAnotherViewVisible) {
 | |
|           this.pdfThumbnailViewer.ensureThumbnailVisible(this.page);
 | |
|         }
 | |
|         break;
 | |
| 
 | |
|       case 'outline':
 | |
|         if (outlineButton.disabled) {
 | |
|           return;
 | |
|         }
 | |
|         thumbsButton.classList.remove('toggled');
 | |
|         outlineButton.classList.add('toggled');
 | |
|         attachmentsButton.classList.remove('toggled');
 | |
|         thumbsView.classList.add('hidden');
 | |
|         outlineView.classList.remove('hidden');
 | |
|         attachmentsView.classList.add('hidden');
 | |
|         break;
 | |
| 
 | |
|       case 'attachments':
 | |
|         if (attachmentsButton.disabled) {
 | |
|           return;
 | |
|         }
 | |
|         thumbsButton.classList.remove('toggled');
 | |
|         outlineButton.classList.remove('toggled');
 | |
|         attachmentsButton.classList.add('toggled');
 | |
|         thumbsView.classList.add('hidden');
 | |
|         outlineView.classList.add('hidden');
 | |
|         attachmentsView.classList.remove('hidden');
 | |
|         break;
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   beforePrint: function pdfViewSetupBeforePrint() {
 | |
|     if (!this.supportsPrinting) {
 | |
|       var printMessage = 'Warning: Printing is not fully supported by this browser.';
 | |
|       this.error(printMessage);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     var alertNotReady = false;
 | |
|     var i, ii;
 | |
|     if (!this.pdfDocument || !this.pagesCount) {
 | |
|       alertNotReady = true;
 | |
|     } else {
 | |
|       for (i = 0, ii = this.pagesCount; i < ii; ++i) {
 | |
|         if (!this.pdfViewer.getPageView(i).pdfPage) {
 | |
|           alertNotReady = true;
 | |
|           break;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|     if (alertNotReady) {
 | |
|       var notReadyMessage = 'Warning: The PDF is not fully loaded for printing.';
 | |
|       window.alert(notReadyMessage);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     this.printing = true;
 | |
|     this.forceRendering();
 | |
| 
 | |
|     var body = document.querySelector('body');
 | |
|     body.setAttribute('data-mozPrintCallback', true);
 | |
| 
 | |
|     if (!this.hasEqualPageSizes) {
 | |
|       console.warn('Not all pages have the same size. The printed result ' +
 | |
|           'may be incorrect!');
 | |
|     }
 | |
| 
 | |
|     // Insert a @page + size rule to make sure that the page size is correctly
 | |
|     // set. Note that we assume that all pages have the same size, because
 | |
|     // variable-size pages are not supported yet (at least in Chrome & Firefox).
 | |
|     // TODO(robwu): Use named pages when size calculation bugs get resolved
 | |
|     // (e.g. https://crbug.com/355116) AND when support for named pages is
 | |
|     // added (http://www.w3.org/TR/css3-page/#using-named-pages).
 | |
|     // In browsers where @page + size is not supported (such as Firefox,
 | |
|     // https://bugzil.la/851441), the next stylesheet will be ignored and the
 | |
|     // user has to select the correct paper size in the UI if wanted.
 | |
|     this.pageStyleSheet = document.createElement('style');
 | |
|     var pageSize = this.pdfViewer.getPageView(0).pdfPage.getViewport(1);
 | |
|     this.pageStyleSheet.textContent =
 | |
|       // "size:<width> <height>" is what we need. But also add "A4" because
 | |
|       // Firefox incorrectly reports support for the other value.
 | |
|       '@supports ((size:A4) and (size:1pt 1pt)) {' +
 | |
|       '@page { size: ' + pageSize.width + 'pt ' + pageSize.height + 'pt;}' +
 | |
|       // The canvas and each ancestor node must have a height of 100% to make
 | |
|       // sure that each canvas is printed on exactly one page.
 | |
|       '#printContainer {height:100%}' +
 | |
|       '#printContainer > div {width:100% !important;height:100% !important;}' +
 | |
|       '}';
 | |
|     body.appendChild(this.pageStyleSheet);
 | |
| 
 | |
|     for (i = 0, ii = this.pagesCount; i < ii; ++i) {
 | |
|       this.pdfViewer.getPageView(i).beforePrint();
 | |
|     }
 | |
| 
 | |
|   },
 | |
| 
 | |
|   // Whether all pages of the PDF have the same width and height.
 | |
|   get hasEqualPageSizes() {
 | |
|     var firstPage = this.pdfViewer.getPageView(0);
 | |
|     for (var i = 1, ii = this.pagesCount; i < ii; ++i) {
 | |
|       var pageView = this.pdfViewer.getPageView(i);
 | |
|       if (pageView.width !== firstPage.width ||
 | |
|           pageView.height !== firstPage.height) {
 | |
|         return false;
 | |
|       }
 | |
|     }
 | |
|     return true;
 | |
|   },
 | |
| 
 | |
|   afterPrint: function pdfViewSetupAfterPrint() {
 | |
|     var div = document.getElementById('printContainer');
 | |
|     while (div.hasChildNodes()) {
 | |
|       div.removeChild(div.lastChild);
 | |
|     }
 | |
| 
 | |
|     if (this.pageStyleSheet && this.pageStyleSheet.parentNode) {
 | |
|       this.pageStyleSheet.parentNode.removeChild(this.pageStyleSheet);
 | |
|       this.pageStyleSheet = null;
 | |
|     }
 | |
| 
 | |
|     this.printing = false;
 | |
|     this.forceRendering();
 | |
|   },
 | |
| 
 | |
|   rotatePages: function pdfViewRotatePages(delta) {
 | |
|     var pageNumber = this.page;
 | |
|     this.pageRotation = (this.pageRotation + 360 + delta) % 360;
 | |
|     this.pdfViewer.pagesRotation = this.pageRotation;
 | |
|     this.pdfThumbnailViewer.pagesRotation = this.pageRotation;
 | |
| 
 | |
|     this.forceRendering();
 | |
| 
 | |
|     this.pdfViewer.scrollPageIntoView(pageNumber);
 | |
|   },
 | |
| 
 | |
|   requestPresentationMode: function pdfViewRequestPresentationMode() {
 | |
|     if (!this.pdfPresentationMode) {
 | |
|       return;
 | |
|     }
 | |
|     this.pdfPresentationMode.request();
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * @param {number} delta - The delta value from the mouse event.
 | |
|    */
 | |
|   scrollPresentationMode: function pdfViewScrollPresentationMode(delta) {
 | |
|     if (!this.pdfPresentationMode) {
 | |
|       return;
 | |
|     }
 | |
|     this.pdfPresentationMode.mouseScroll(delta);
 | |
|   }
 | |
| };
 | |
| window.PDFView = PDFViewerApplication; // obsolete name, using it as an alias
 | |
| 
 | |
| 
 | |
| var HOSTED_VIEWER_ORIGINS = ['null',
 | |
|   'http://mozilla.github.io', 'https://mozilla.github.io'];
 | |
| function validateFileURL(file) {
 | |
|   try {
 | |
|     var viewerOrigin = new URL(window.location.href).origin || 'null';
 | |
|     if (HOSTED_VIEWER_ORIGINS.indexOf(viewerOrigin) >= 0) {
 | |
|       // Hosted or local viewer, allow for any file locations
 | |
|       return;
 | |
|     }
 | |
|     var fileOrigin = new URL(file, window.location.href).origin;
 | |
|     // Removing of the following line will not guarantee that the viewer will
 | |
|     // start accepting URLs from foreign origin -- CORS headers on the remote
 | |
|     // server must be properly configured.
 | |
|     if (fileOrigin !== viewerOrigin) {
 | |
|       throw new Error('file origin does not match viewer\'s');
 | |
|     }
 | |
|   } catch (e) {
 | |
|     var message = e && e.message;
 | |
|     var loadingErrorMessage = 'An error occurred while loading the PDF.';
 | |
| 
 | |
|     var moreInfo = {
 | |
|       message: message
 | |
|     };
 | |
|     PDFViewerApplication.error(loadingErrorMessage, moreInfo);
 | |
|     throw e;
 | |
|   }
 | |
| }
 | |
| 
 | |
| function webViewerLoad(evt) {
 | |
|     configure(PDFJS);
 | |
|     PDFViewerApplication.initialize().then(webViewerInitialized);
 | |
| }
 | |
| 
 | |
| function webViewerInitialized() {
 | |
|   var queryString = document.location.search.substring(1);
 | |
|   var params = parseQueryString(queryString);
 | |
|   var file = 'file' in params ? params.file : DEFAULT_URL;
 | |
|   validateFileURL(file);
 | |
| 
 | |
|   var fileInput = document.createElement('input');
 | |
|   fileInput.id = 'fileInput';
 | |
|   fileInput.className = 'fileInput';
 | |
|   fileInput.setAttribute('type', 'file');
 | |
|   fileInput.oncontextmenu = noContextMenuHandler;
 | |
|   document.body.appendChild(fileInput);
 | |
| 
 | |
|   if (!window.File || !window.FileReader || !window.FileList || !window.Blob) {
 | |
|     document.getElementById('openFile').setAttribute('hidden', 'true');
 | |
|     document.getElementById('secondaryOpenFile').setAttribute('hidden', 'true');
 | |
|   } else {
 | |
|     document.getElementById('fileInput').value = null;
 | |
|   }
 | |
| 
 | |
|   var locale = PDFJS.locale || navigator.language;
 | |
| 
 | |
|   if (PDFViewerApplication.preferencePdfBugEnabled) {
 | |
|     // Special debugging flags in the hash section of the URL.
 | |
|     var hash = document.location.hash.substring(1);
 | |
|     var hashParams = parseQueryString(hash);
 | |
| 
 | |
|     if ('disableworker' in hashParams) {
 | |
|       PDFJS.disableWorker = (hashParams['disableworker'] === 'true');
 | |
|     }
 | |
|     if ('disablerange' in hashParams) {
 | |
|       PDFJS.disableRange = (hashParams['disablerange'] === 'true');
 | |
|     }
 | |
|     if ('disablestream' in hashParams) {
 | |
|       PDFJS.disableStream = (hashParams['disablestream'] === 'true');
 | |
|     }
 | |
|     if ('disableautofetch' in hashParams) {
 | |
|       PDFJS.disableAutoFetch = (hashParams['disableautofetch'] === 'true');
 | |
|     }
 | |
|     if ('disablefontface' in hashParams) {
 | |
|       PDFJS.disableFontFace = (hashParams['disablefontface'] === 'true');
 | |
|     }
 | |
|     if ('disablehistory' in hashParams) {
 | |
|       PDFJS.disableHistory = (hashParams['disablehistory'] === 'true');
 | |
|     }
 | |
|     if ('webgl' in hashParams) {
 | |
|       PDFJS.disableWebGL = (hashParams['webgl'] !== 'true');
 | |
|     }
 | |
|     if ('useonlycsszoom' in hashParams) {
 | |
|       PDFJS.useOnlyCssZoom = (hashParams['useonlycsszoom'] === 'true');
 | |
|     }
 | |
|     if ('verbosity' in hashParams) {
 | |
|       PDFJS.verbosity = hashParams['verbosity'] | 0;
 | |
|     }
 | |
|     if ('ignorecurrentpositiononzoom' in hashParams) {
 | |
|       IGNORE_CURRENT_POSITION_ON_ZOOM =
 | |
|         (hashParams['ignorecurrentpositiononzoom'] === 'true');
 | |
|     }
 | |
|     if ('locale' in hashParams) {
 | |
|       locale = hashParams['locale'];
 | |
|     }
 | |
|     if ('textlayer' in hashParams) {
 | |
|       switch (hashParams['textlayer']) {
 | |
|         case 'off':
 | |
|           PDFJS.disableTextLayer = true;
 | |
|           break;
 | |
|         case 'visible':
 | |
|         case 'shadow':
 | |
|         case 'hover':
 | |
|           var viewer = document.getElementById('viewer');
 | |
|           viewer.classList.add('textLayer-' + hashParams['textlayer']);
 | |
|           break;
 | |
|       }
 | |
|     }
 | |
|     if ('pdfbug' in hashParams) {
 | |
|       PDFJS.pdfBug = true;
 | |
|       var pdfBug = hashParams['pdfbug'];
 | |
|       var enabled = pdfBug.split(',');
 | |
|       PDFBug.enable(enabled);
 | |
|       PDFBug.init();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!PDFViewerApplication.supportsPrinting) {
 | |
|     document.getElementById('print').classList.add('hidden');
 | |
|     document.getElementById('secondaryPrint').classList.add('hidden');
 | |
|   }
 | |
| 
 | |
|   if (!PDFViewerApplication.supportsFullscreen) {
 | |
|     document.getElementById('presentationMode').classList.add('hidden');
 | |
|     document.getElementById('secondaryPresentationMode').
 | |
|       classList.add('hidden');
 | |
|   }
 | |
| 
 | |
|   if (PDFViewerApplication.supportsIntegratedFind) {
 | |
|     document.getElementById('viewFind').classList.add('hidden');
 | |
|   }
 | |
| 
 | |
|   // Suppress context menus for some controls
 | |
|   document.getElementById('scaleSelect').oncontextmenu = noContextMenuHandler;
 | |
| 
 | |
|   var mainContainer = document.getElementById('mainContainer');
 | |
|   var outerContainer = document.getElementById('outerContainer');
 | |
|   mainContainer.addEventListener('transitionend', function(e) {
 | |
|     if (e.target === mainContainer) {
 | |
|       var event = document.createEvent('UIEvents');
 | |
|       event.initUIEvent('resize', false, false, window, 0);
 | |
|       window.dispatchEvent(event);
 | |
|       outerContainer.classList.remove('sidebarMoving');
 | |
|     }
 | |
|   }, true);
 | |
| 
 | |
|   document.getElementById('sidebarToggle').addEventListener('click',
 | |
|     function() {
 | |
|       this.classList.toggle('toggled');
 | |
|       outerContainer.classList.add('sidebarMoving');
 | |
|       outerContainer.classList.toggle('sidebarOpen');
 | |
|       PDFViewerApplication.sidebarOpen =
 | |
|         outerContainer.classList.contains('sidebarOpen');
 | |
|       if (PDFViewerApplication.sidebarOpen) {
 | |
|         PDFViewerApplication.refreshThumbnailViewer();
 | |
|       }
 | |
|       PDFViewerApplication.forceRendering();
 | |
|     });
 | |
| 
 | |
|   document.getElementById('viewThumbnail').addEventListener('click',
 | |
|     function() {
 | |
|       PDFViewerApplication.switchSidebarView('thumbs');
 | |
|     });
 | |
| 
 | |
|   document.getElementById('viewOutline').addEventListener('click',
 | |
|     function() {
 | |
|       PDFViewerApplication.switchSidebarView('outline');
 | |
|     });
 | |
| 
 | |
|   document.getElementById('viewOutline').addEventListener('dblclick',
 | |
|     function() {
 | |
|       PDFViewerApplication.outline.toggleOutlineTree();
 | |
|     });
 | |
| 
 | |
|   document.getElementById('viewAttachments').addEventListener('click',
 | |
|     function() {
 | |
|       PDFViewerApplication.switchSidebarView('attachments');
 | |
|     });
 | |
| 
 | |
|   document.getElementById('previous').addEventListener('click',
 | |
|     function() {
 | |
|       PDFViewerApplication.page--;
 | |
|     });
 | |
| 
 | |
|   document.getElementById('next').addEventListener('click',
 | |
|     function() {
 | |
|       PDFViewerApplication.page++;
 | |
|     });
 | |
| 
 | |
|   document.getElementById('zoomIn').addEventListener('click',
 | |
|     function() {
 | |
|       PDFViewerApplication.zoomIn();
 | |
|     });
 | |
| 
 | |
|   document.getElementById('zoomOut').addEventListener('click',
 | |
|     function() {
 | |
|       PDFViewerApplication.zoomOut();
 | |
|     });
 | |
| 
 | |
|   document.getElementById('pageNumber').addEventListener('click', function() {
 | |
|     this.select();
 | |
|   });
 | |
| 
 | |
|   document.getElementById('pageNumber').addEventListener('change', function() {
 | |
|     // Handle the user inputting a floating point number.
 | |
|     PDFViewerApplication.page = (this.value | 0);
 | |
| 
 | |
|     if (this.value !== (this.value | 0).toString()) {
 | |
|       this.value = PDFViewerApplication.page;
 | |
|     }
 | |
|   });
 | |
| 
 | |
|   document.getElementById('scaleSelect').addEventListener('change', function() {
 | |
|     if (this.value === 'custom') {
 | |
|       return;
 | |
|     }
 | |
|     PDFViewerApplication.pdfViewer.currentScaleValue = this.value;
 | |
|   });
 | |
| 
 | |
|   document.getElementById('presentationMode').addEventListener('click',
 | |
|     SecondaryToolbar.presentationModeClick.bind(SecondaryToolbar));
 | |
| 
 | |
|   document.getElementById('openFile').addEventListener('click',
 | |
|     SecondaryToolbar.openFileClick.bind(SecondaryToolbar));
 | |
| 
 | |
|   document.getElementById('print').addEventListener('click',
 | |
|     SecondaryToolbar.printClick.bind(SecondaryToolbar));
 | |
| 
 | |
|   document.getElementById('download').addEventListener('click',
 | |
|     SecondaryToolbar.downloadClick.bind(SecondaryToolbar));
 | |
| 
 | |
| 
 | |
|   if (file && file.lastIndexOf('file:', 0) === 0) {
 | |
|     // file:-scheme. Load the contents in the main thread because QtWebKit
 | |
|     // cannot load file:-URLs in a Web Worker. file:-URLs are usually loaded
 | |
|     // very quickly, so there is no need to set up progress event listeners.
 | |
|     PDFViewerApplication.setTitleUsingUrl(file);
 | |
|     var xhr = new XMLHttpRequest();
 | |
|     xhr.onload = function() {
 | |
|       PDFViewerApplication.open(new Uint8Array(xhr.response));
 | |
|     };
 | |
|     try {
 | |
|       xhr.open('GET', file);
 | |
|       xhr.responseType = 'arraybuffer';
 | |
|       xhr.send();
 | |
|     } catch (e) {
 | |
|       PDFViewerApplication.error('An error occurred while loading the PDF.', e);
 | |
|     }
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (file) {
 | |
|     PDFViewerApplication.open(file);
 | |
|   }
 | |
| }
 | |
| 
 | |
| document.addEventListener('DOMContentLoaded', webViewerLoad, true);
 | |
| 
 | |
| document.addEventListener('pagerendered', function (e) {
 | |
|   var pageNumber = e.detail.pageNumber;
 | |
|   var pageIndex = pageNumber - 1;
 | |
|   var pageView = PDFViewerApplication.pdfViewer.getPageView(pageIndex);
 | |
| 
 | |
|   if (PDFViewerApplication.sidebarOpen) {
 | |
|     var thumbnailView = PDFViewerApplication.pdfThumbnailViewer.
 | |
|                         getThumbnail(pageIndex);
 | |
|     thumbnailView.setImage(pageView);
 | |
|   }
 | |
| 
 | |
|   if (PDFJS.pdfBug && Stats.enabled && pageView.stats) {
 | |
|     Stats.add(pageNumber, pageView.stats);
 | |
|   }
 | |
| 
 | |
|   if (pageView.error) {
 | |
|     PDFViewerApplication.error('An error occurred while rendering the page.', pageView.error);
 | |
|   }
 | |
| 
 | |
|   // If the page is still visible when it has finished rendering,
 | |
|   // ensure that the page number input loading indicator is hidden.
 | |
|   if (pageNumber === PDFViewerApplication.page) {
 | |
|     var pageNumberInput = document.getElementById('pageNumber');
 | |
|     pageNumberInput.classList.remove(PAGE_NUMBER_LOADING_INDICATOR);
 | |
|   }
 | |
| 
 | |
| }, true);
 | |
| 
 | |
| document.addEventListener('textlayerrendered', function (e) {
 | |
|   var pageIndex = e.detail.pageNumber - 1;
 | |
|   var pageView = PDFViewerApplication.pdfViewer.getPageView(pageIndex);
 | |
| 
 | |
| }, true);
 | |
| 
 | |
| document.addEventListener('pagemode', function (evt) {
 | |
|   if (!PDFViewerApplication.initialized) {
 | |
|     return;
 | |
|   }
 | |
|   // Handle the 'pagemode' hash parameter, see also `PDFLinkService_setHash`.
 | |
|   var mode = evt.detail.mode;
 | |
|   switch (mode) {
 | |
|     case 'bookmarks':
 | |
|       // Note: Our code calls this property 'outline', even though the
 | |
|       //       Open Parameter specification calls it 'bookmarks'.
 | |
|       mode = 'outline';
 | |
|       /* falls through */
 | |
|     case 'thumbs':
 | |
|     case 'attachments':
 | |
|       PDFViewerApplication.switchSidebarView(mode, true);
 | |
|       break;
 | |
|     case 'none':
 | |
|       if (PDFViewerApplication.sidebarOpen) {
 | |
|         document.getElementById('sidebarToggle').click();
 | |
|       }
 | |
|       break;
 | |
|   }
 | |
| }, true);
 | |
| 
 | |
| document.addEventListener('namedaction', function (e) {
 | |
|   if (!PDFViewerApplication.initialized) {
 | |
|     return;
 | |
|   }
 | |
|   // Processing couple of named actions that might be useful.
 | |
|   // See also PDFLinkService.executeNamedAction
 | |
|   var action = e.detail.action;
 | |
|   switch (action) {
 | |
|     case 'GoToPage':
 | |
|       document.getElementById('pageNumber').focus();
 | |
|       break;
 | |
| 
 | |
|     case 'Find':
 | |
|       if (!PDFViewerApplication.supportsIntegratedFind) {
 | |
|         PDFViewerApplication.findBar.toggle();
 | |
|       }
 | |
|       break;
 | |
|   }
 | |
| }, true);
 | |
| 
 | |
| window.addEventListener('presentationmodechanged', function (e) {
 | |
|   var active = e.detail.active;
 | |
|   var switchInProgress = e.detail.switchInProgress;
 | |
|   PDFViewerApplication.pdfViewer.presentationModeState =
 | |
|     switchInProgress ? PresentationModeState.CHANGING :
 | |
|     active ? PresentationModeState.FULLSCREEN : PresentationModeState.NORMAL;
 | |
| });
 | |
| 
 | |
| window.addEventListener('updateviewarea', function (evt) {
 | |
|   if (!PDFViewerApplication.initialized) {
 | |
|     return;
 | |
|   }
 | |
|   var location = evt.location;
 | |
| 
 | |
|   PDFViewerApplication.store.initializedPromise.then(function() {
 | |
|     PDFViewerApplication.store.setMultiple({
 | |
|       'exists': true,
 | |
|       'page': location.pageNumber,
 | |
|       'zoom': location.scale,
 | |
|       'scrollLeft': location.left,
 | |
|       'scrollTop': location.top
 | |
|     }).catch(function() {
 | |
|       // unable to write to storage
 | |
|     });
 | |
|   });
 | |
|   var href =
 | |
|     PDFViewerApplication.pdfLinkService.getAnchorUrl(location.pdfOpenParams);
 | |
|   document.getElementById('viewBookmark').href = href;
 | |
|   document.getElementById('secondaryViewBookmark').href = href;
 | |
| 
 | |
|   // Update the current bookmark in the browsing history.
 | |
|   PDFViewerApplication.pdfHistory.updateCurrentBookmark(location.pdfOpenParams,
 | |
|                                                         location.pageNumber);
 | |
| 
 | |
|   // Show/hide the loading indicator in the page number input element.
 | |
|   var pageNumberInput = document.getElementById('pageNumber');
 | |
|   var currentPage =
 | |
|     PDFViewerApplication.pdfViewer.getPageView(PDFViewerApplication.page - 1);
 | |
| 
 | |
|   if (currentPage.renderingState === RenderingStates.FINISHED) {
 | |
|     pageNumberInput.classList.remove(PAGE_NUMBER_LOADING_INDICATOR);
 | |
|   } else {
 | |
|     pageNumberInput.classList.add(PAGE_NUMBER_LOADING_INDICATOR);
 | |
|   }
 | |
| }, true);
 | |
| 
 | |
| window.addEventListener('resize', function webViewerResize(evt) {
 | |
|   if (PDFViewerApplication.initialized) {
 | |
|     var currentScaleValue = PDFViewerApplication.pdfViewer.currentScaleValue;
 | |
|     if (currentScaleValue === 'auto' ||
 | |
|         currentScaleValue === 'page-fit' ||
 | |
|         currentScaleValue === 'page-width') {
 | |
|       // Note: the scale is constant for 'page-actual'.
 | |
|       PDFViewerApplication.pdfViewer.currentScaleValue = currentScaleValue;
 | |
|     } else if (!currentScaleValue) {
 | |
|       // Normally this shouldn't happen, but if the scale wasn't initialized
 | |
|       // we set it to the default value in order to prevent any issues.
 | |
|       // (E.g. the document being rendered with the wrong scale on load.)
 | |
|       PDFViewerApplication.pdfViewer.currentScaleValue = DEFAULT_SCALE_VALUE;
 | |
|     }
 | |
|     PDFViewerApplication.pdfViewer.update();
 | |
|   }
 | |
| 
 | |
|   // Set the 'max-height' CSS property of the secondary toolbar.
 | |
|   SecondaryToolbar.setMaxHeight(document.getElementById('viewerContainer'));
 | |
| });
 | |
| 
 | |
| window.addEventListener('hashchange', function webViewerHashchange(evt) {
 | |
|   if (PDFViewerApplication.pdfHistory.isHashChangeUnlocked) {
 | |
|     var hash = document.location.hash.substring(1);
 | |
|     if (!hash) {
 | |
|       return;
 | |
|     }
 | |
|     if (!PDFViewerApplication.isInitialViewSet) {
 | |
|       PDFViewerApplication.initialBookmark = hash;
 | |
|     } else {
 | |
|       PDFViewerApplication.pdfLinkService.setHash(hash);
 | |
|     }
 | |
|   }
 | |
| });
 | |
| 
 | |
| window.addEventListener('change', function webViewerChange(evt) {
 | |
|   var files = evt.target.files;
 | |
|   if (!files || files.length === 0) {
 | |
|     return;
 | |
|   }
 | |
|   var file = files[0];
 | |
| 
 | |
|   if (!PDFJS.disableCreateObjectURL &&
 | |
|       typeof URL !== 'undefined' && URL.createObjectURL) {
 | |
|     PDFViewerApplication.open(URL.createObjectURL(file));
 | |
|   } else {
 | |
|     // Read the local file into a Uint8Array.
 | |
|     var fileReader = new FileReader();
 | |
|     fileReader.onload = function webViewerChangeFileReaderOnload(evt) {
 | |
|       var buffer = evt.target.result;
 | |
|       var uint8Array = new Uint8Array(buffer);
 | |
|       PDFViewerApplication.open(uint8Array);
 | |
|     };
 | |
|     fileReader.readAsArrayBuffer(file);
 | |
|   }
 | |
| 
 | |
|   PDFViewerApplication.setTitleUsingUrl(file.name);
 | |
| 
 | |
|   // URL does not reflect proper document location - hiding some icons.
 | |
|   document.getElementById('viewBookmark').setAttribute('hidden', 'true');
 | |
|   document.getElementById('secondaryViewBookmark').
 | |
|     setAttribute('hidden', 'true');
 | |
|   document.getElementById('download').setAttribute('hidden', 'true');
 | |
|   document.getElementById('secondaryDownload').setAttribute('hidden', 'true');
 | |
| }, true);
 | |
| 
 | |
| function selectScaleOption(value) {
 | |
|   var options = document.getElementById('scaleSelect').options;
 | |
|   var predefinedValueFound = false;
 | |
|   for (var i = 0, ii = options.length; i < ii; i++) {
 | |
|     var option = options[i];
 | |
|     if (option.value !== value) {
 | |
|       option.selected = false;
 | |
|       continue;
 | |
|     }
 | |
|     option.selected = true;
 | |
|     predefinedValueFound = true;
 | |
|   }
 | |
|   return predefinedValueFound;
 | |
| }
 | |
| 
 | |
| window.addEventListener('localized', function localized(evt) {
 | |
| //  document.getElementsByTagName('html')[0].dir = mozL10n.getDirection();
 | |
| 
 | |
|   PDFViewerApplication.animationStartedPromise.then(function() {
 | |
|     // Adjust the width of the zoom box to fit the content.
 | |
|     // Note: If the window is narrow enough that the zoom box is not visible,
 | |
|     //       we temporarily show it to be able to adjust its width.
 | |
|     var container = document.getElementById('scaleSelectContainer');
 | |
|     if (container.clientWidth === 0) {
 | |
|       container.setAttribute('style', 'display: inherit;');
 | |
|     }
 | |
|     if (container.clientWidth > 0) {
 | |
|       var select = document.getElementById('scaleSelect');
 | |
|       select.setAttribute('style', 'min-width: inherit;');
 | |
|       var width = select.clientWidth + SCALE_SELECT_CONTAINER_PADDING;
 | |
|       select.setAttribute('style', 'min-width: ' +
 | |
|                                    (width + SCALE_SELECT_PADDING) + 'px;');
 | |
|       container.setAttribute('style', 'min-width: ' + width + 'px; ' +
 | |
|                                       'max-width: ' + width + 'px;');
 | |
|     }
 | |
| 
 | |
|     // Set the 'max-height' CSS property of the secondary toolbar.
 | |
|     SecondaryToolbar.setMaxHeight(document.getElementById('viewerContainer'));
 | |
|   });
 | |
| }, true);
 | |
| 
 | |
| window.addEventListener('scalechange', function scalechange(evt) {
 | |
|   document.getElementById('zoomOut').disabled = (evt.scale === MIN_SCALE);
 | |
|   document.getElementById('zoomIn').disabled = (evt.scale === MAX_SCALE);
 | |
| 
 | |
|   // Update the 'scaleSelect' DOM element.
 | |
|   var predefinedValueFound = selectScaleOption(evt.presetValue ||
 | |
|                                                '' + evt.scale);
 | |
|   if (!predefinedValueFound) {
 | |
|     var customScaleOption = document.getElementById('customScaleOption');
 | |
|     var customScale = Math.round(evt.scale * 10000) / 100;
 | |
|     customScaleOption.textContent = customScale + '%';
 | |
|     customScaleOption.selected = true;
 | |
|   }
 | |
|   if (!PDFViewerApplication.initialized) {
 | |
|     return;
 | |
|   }
 | |
|   PDFViewerApplication.pdfViewer.update();
 | |
| }, true);
 | |
| 
 | |
| window.addEventListener('pagechange', function pagechange(evt) {
 | |
|   var page = evt.pageNumber;
 | |
|   if (evt.previousPageNumber !== page) {
 | |
|     document.getElementById('pageNumber').value = page;
 | |
|     if (PDFViewerApplication.sidebarOpen) {
 | |
|       PDFViewerApplication.pdfThumbnailViewer.scrollThumbnailIntoView(page);
 | |
|     }
 | |
|   }
 | |
|   var numPages = PDFViewerApplication.pagesCount;
 | |
| 
 | |
|   document.getElementById('previous').disabled = (page <= 1);
 | |
|   document.getElementById('next').disabled = (page >= numPages);
 | |
| 
 | |
|   document.getElementById('firstPage').disabled = (page <= 1);
 | |
|   document.getElementById('lastPage').disabled = (page >= numPages);
 | |
| 
 | |
|   // we need to update stats
 | |
|   if (PDFJS.pdfBug && Stats.enabled) {
 | |
|     var pageView = PDFViewerApplication.pdfViewer.getPageView(page - 1);
 | |
|     if (pageView.stats) {
 | |
|       Stats.add(page, pageView.stats);
 | |
|     }
 | |
|   }
 | |
| }, true);
 | |
| 
 | |
| function handleMouseWheel(evt) {
 | |
|   var MOUSE_WHEEL_DELTA_FACTOR = 40;
 | |
|   var ticks = (evt.type === 'DOMMouseScroll') ? -evt.detail :
 | |
|               evt.wheelDelta / MOUSE_WHEEL_DELTA_FACTOR;
 | |
|   var direction = (ticks < 0) ? 'zoomOut' : 'zoomIn';
 | |
| 
 | |
|   var pdfViewer = PDFViewerApplication.pdfViewer;
 | |
|   if (pdfViewer.isInPresentationMode) {
 | |
|     evt.preventDefault();
 | |
|     PDFViewerApplication.scrollPresentationMode(ticks *
 | |
|                                                 MOUSE_WHEEL_DELTA_FACTOR);
 | |
|   } else if (evt.ctrlKey || evt.metaKey) {
 | |
|     var support = PDFViewerApplication.supportedMouseWheelZoomModifierKeys;
 | |
|     if ((evt.ctrlKey && !support.ctrlKey) ||
 | |
|         (evt.metaKey && !support.metaKey)) {
 | |
|       return;
 | |
|     }
 | |
|     // Only zoom the pages, not the entire viewer.
 | |
|     evt.preventDefault();
 | |
| 
 | |
|     var previousScale = pdfViewer.currentScale;
 | |
| 
 | |
|     PDFViewerApplication[direction](Math.abs(ticks));
 | |
| 
 | |
|     var currentScale = pdfViewer.currentScale;
 | |
|     if (previousScale !== currentScale) {
 | |
|       // After scaling the page via zoomIn/zoomOut, the position of the upper-
 | |
|       // left corner is restored. When the mouse wheel is used, the position
 | |
|       // under the cursor should be restored instead.
 | |
|       var scaleCorrectionFactor = currentScale / previousScale - 1;
 | |
|       var rect = pdfViewer.container.getBoundingClientRect();
 | |
|       var dx = evt.clientX - rect.left;
 | |
|       var dy = evt.clientY - rect.top;
 | |
|       pdfViewer.container.scrollLeft += dx * scaleCorrectionFactor;
 | |
|       pdfViewer.container.scrollTop += dy * scaleCorrectionFactor;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| window.addEventListener('DOMMouseScroll', handleMouseWheel);
 | |
| window.addEventListener('mousewheel', handleMouseWheel);
 | |
| 
 | |
| window.addEventListener('click', function click(evt) {
 | |
|   if (SecondaryToolbar.opened &&
 | |
|       PDFViewerApplication.pdfViewer.containsElement(evt.target)) {
 | |
|     SecondaryToolbar.close();
 | |
|   }
 | |
| }, false);
 | |
| 
 | |
| window.addEventListener('keydown', function keydown(evt) {
 | |
|   if (OverlayManager.active) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   var handled = false;
 | |
|   var cmd = (evt.ctrlKey ? 1 : 0) |
 | |
|             (evt.altKey ? 2 : 0) |
 | |
|             (evt.shiftKey ? 4 : 0) |
 | |
|             (evt.metaKey ? 8 : 0);
 | |
| 
 | |
|   var pdfViewer = PDFViewerApplication.pdfViewer;
 | |
|   var isViewerInPresentationMode = pdfViewer && pdfViewer.isInPresentationMode;
 | |
| 
 | |
|   // First, handle the key bindings that are independent whether an input
 | |
|   // control is selected or not.
 | |
|   if (cmd === 1 || cmd === 8 || cmd === 5 || cmd === 12) {
 | |
|     // either CTRL or META key with optional SHIFT.
 | |
|     switch (evt.keyCode) {
 | |
|       case 70: // f
 | |
|         if (!PDFViewerApplication.supportsIntegratedFind) {
 | |
|           PDFViewerApplication.findBar.open();
 | |
|           handled = true;
 | |
|         }
 | |
|         break;
 | |
|       case 71: // g
 | |
|         if (!PDFViewerApplication.supportsIntegratedFind) {
 | |
|           PDFViewerApplication.findBar.dispatchEvent('again',
 | |
|                                                      cmd === 5 || cmd === 12);
 | |
|           handled = true;
 | |
|         }
 | |
|         break;
 | |
|       case 61: // FF/Mac '='
 | |
|       case 107: // FF '+' and '='
 | |
|       case 187: // Chrome '+'
 | |
|       case 171: // FF with German keyboard
 | |
|         if (!isViewerInPresentationMode) {
 | |
|           PDFViewerApplication.zoomIn();
 | |
|         }
 | |
|         handled = true;
 | |
|         break;
 | |
|       case 173: // FF/Mac '-'
 | |
|       case 109: // FF '-'
 | |
|       case 189: // Chrome '-'
 | |
|         if (!isViewerInPresentationMode) {
 | |
|           PDFViewerApplication.zoomOut();
 | |
|         }
 | |
|         handled = true;
 | |
|         break;
 | |
|       case 48: // '0'
 | |
|       case 96: // '0' on Numpad of Swedish keyboard
 | |
|         if (!isViewerInPresentationMode) {
 | |
|           // keeping it unhandled (to restore page zoom to 100%)
 | |
|           setTimeout(function () {
 | |
|             // ... and resetting the scale after browser adjusts its scale
 | |
|             pdfViewer.currentScaleValue = DEFAULT_SCALE_VALUE;
 | |
|           });
 | |
|           handled = false;
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // CTRL or META without shift
 | |
|   if (cmd === 1 || cmd === 8) {
 | |
|     switch (evt.keyCode) {
 | |
|       case 83: // s
 | |
|         PDFViewerApplication.download();
 | |
|         handled = true;
 | |
|         break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // CTRL+ALT or Option+Command
 | |
|   if (cmd === 3 || cmd === 10) {
 | |
|     switch (evt.keyCode) {
 | |
|       case 80: // p
 | |
|         PDFViewerApplication.requestPresentationMode();
 | |
|         handled = true;
 | |
|         break;
 | |
|       case 71: // g
 | |
|         // focuses input#pageNumber field
 | |
|         document.getElementById('pageNumber').select();
 | |
|         handled = true;
 | |
|         break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (handled) {
 | |
|     evt.preventDefault();
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Some shortcuts should not get handled if a control/input element
 | |
|   // is selected.
 | |
|   var curElement = document.activeElement || document.querySelector(':focus');
 | |
|   var curElementTagName = curElement && curElement.tagName.toUpperCase();
 | |
|   if (curElementTagName === 'INPUT' ||
 | |
|       curElementTagName === 'TEXTAREA' ||
 | |
|       curElementTagName === 'SELECT') {
 | |
|     // Make sure that the secondary toolbar is closed when Escape is pressed.
 | |
|     if (evt.keyCode !== 27) { // 'Esc'
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
|   var ensureViewerFocused = false;
 | |
| 
 | |
|   if (cmd === 0) { // no control key pressed at all.
 | |
|     switch (evt.keyCode) {
 | |
|       case 38: // up arrow
 | |
|       case 33: // pg up
 | |
|       case 8: // backspace
 | |
|         if (!isViewerInPresentationMode &&
 | |
|             pdfViewer.currentScaleValue !== 'page-fit') {
 | |
|           break;
 | |
|         }
 | |
|         /* in presentation mode */
 | |
|         /* falls through */
 | |
|       case 37: // left arrow
 | |
|         // horizontal scrolling using arrow keys
 | |
|         if (pdfViewer.isHorizontalScrollbarEnabled) {
 | |
|           break;
 | |
|         }
 | |
|         /* falls through */
 | |
|       case 75: // 'k'
 | |
|       case 80: // 'p'
 | |
|         PDFViewerApplication.page--;
 | |
|         handled = true;
 | |
|         break;
 | |
|       case 27: // esc key
 | |
|         if (SecondaryToolbar.opened) {
 | |
|           SecondaryToolbar.close();
 | |
|           handled = true;
 | |
|         }
 | |
|         if (!PDFViewerApplication.supportsIntegratedFind &&
 | |
|             PDFViewerApplication.findBar.opened) {
 | |
|           PDFViewerApplication.findBar.close();
 | |
|           handled = true;
 | |
|         }
 | |
|         break;
 | |
|       case 40: // down arrow
 | |
|       case 34: // pg down
 | |
|       case 32: // spacebar
 | |
|         if (!isViewerInPresentationMode &&
 | |
|             pdfViewer.currentScaleValue !== 'page-fit') {
 | |
|           break;
 | |
|         }
 | |
|         /* falls through */
 | |
|       case 39: // right arrow
 | |
|         // horizontal scrolling using arrow keys
 | |
|         if (pdfViewer.isHorizontalScrollbarEnabled) {
 | |
|           break;
 | |
|         }
 | |
|         /* falls through */
 | |
|       case 74: // 'j'
 | |
|       case 78: // 'n'
 | |
|         PDFViewerApplication.page++;
 | |
|         handled = true;
 | |
|         break;
 | |
| 
 | |
|       case 36: // home
 | |
|         if (isViewerInPresentationMode || PDFViewerApplication.page > 1) {
 | |
|           PDFViewerApplication.page = 1;
 | |
|           handled = true;
 | |
|           ensureViewerFocused = true;
 | |
|         }
 | |
|         break;
 | |
|       case 35: // end
 | |
|         if (isViewerInPresentationMode || (PDFViewerApplication.pdfDocument &&
 | |
|             PDFViewerApplication.page < PDFViewerApplication.pagesCount)) {
 | |
|           PDFViewerApplication.page = PDFViewerApplication.pagesCount;
 | |
|           handled = true;
 | |
|           ensureViewerFocused = true;
 | |
|         }
 | |
|         break;
 | |
| 
 | |
|       case 72: // 'h'
 | |
|         if (!isViewerInPresentationMode) {
 | |
|           HandTool.toggle();
 | |
|         }
 | |
|         break;
 | |
|       case 82: // 'r'
 | |
|         PDFViewerApplication.rotatePages(90);
 | |
|         break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (cmd === 4) { // shift-key
 | |
|     switch (evt.keyCode) {
 | |
|       case 32: // spacebar
 | |
|         if (!isViewerInPresentationMode &&
 | |
|             pdfViewer.currentScaleValue !== 'page-fit') {
 | |
|           break;
 | |
|         }
 | |
|         PDFViewerApplication.page--;
 | |
|         handled = true;
 | |
|         break;
 | |
| 
 | |
|       case 82: // 'r'
 | |
|         PDFViewerApplication.rotatePages(-90);
 | |
|         break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!handled && !isViewerInPresentationMode) {
 | |
|     // 33=Page Up  34=Page Down  35=End    36=Home
 | |
|     // 37=Left     38=Up         39=Right  40=Down
 | |
|     // 32=Spacebar
 | |
|     if ((evt.keyCode >= 33 && evt.keyCode <= 40) ||
 | |
|         (evt.keyCode === 32 && curElementTagName !== 'BUTTON')) {
 | |
|       ensureViewerFocused = true;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (cmd === 2) { // alt-key
 | |
|     switch (evt.keyCode) {
 | |
|       case 37: // left arrow
 | |
|         if (isViewerInPresentationMode) {
 | |
|           PDFViewerApplication.pdfHistory.back();
 | |
|           handled = true;
 | |
|         }
 | |
|         break;
 | |
|       case 39: // right arrow
 | |
|         if (isViewerInPresentationMode) {
 | |
|           PDFViewerApplication.pdfHistory.forward();
 | |
|           handled = true;
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (ensureViewerFocused && !pdfViewer.containsElement(curElement)) {
 | |
|     // The page container is not focused, but a page navigation key has been
 | |
|     // pressed. Change the focus to the viewer container to make sure that
 | |
|     // navigation by keyboard works as expected.
 | |
|     pdfViewer.focus();
 | |
|   }
 | |
| 
 | |
|   if (handled) {
 | |
|     evt.preventDefault();
 | |
|   }
 | |
| });
 | |
| 
 | |
| window.addEventListener('beforeprint', function beforePrint(evt) {
 | |
|   PDFViewerApplication.beforePrint();
 | |
| });
 | |
| 
 | |
| window.addEventListener('afterprint', function afterPrint(evt) {
 | |
|   PDFViewerApplication.afterPrint();
 | |
| });
 | |
| 
 | |
| (function animationStartedClosure() {
 | |
|   // The offsetParent is not set until the pdf.js iframe or object is visible.
 | |
|   // Waiting for first animation.
 | |
|   PDFViewerApplication.animationStartedPromise = new Promise(
 | |
|       function (resolve) {
 | |
|     window.requestAnimationFrame(resolve);
 | |
|   });
 | |
| })();
 | |
| 
 |