|
|
|
@ -1,4 +1,4 @@
@@ -1,4 +1,4 @@
|
|
|
|
|
/* Copyright 2012 Mozilla Foundation |
|
|
|
|
/* Copyright 2017 Mozilla Foundation |
|
|
|
|
* |
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License"); |
|
|
|
|
* you may not use this file except in compliance with the License. |
|
|
|
@ -12,415 +12,579 @@
@@ -12,415 +12,579 @@
|
|
|
|
|
* See the License for the specific language governing permissions and |
|
|
|
|
* limitations under the License. |
|
|
|
|
*/ |
|
|
|
|
/* globals chrome */ |
|
|
|
|
|
|
|
|
|
import { cloneObj, parseQueryString, waitOnEventOrTimeout } from './ui_utils'; |
|
|
|
|
import { getGlobalEventBus } from './dom_events'; |
|
|
|
|
|
|
|
|
|
function PDFHistory(options) { |
|
|
|
|
this.linkService = options.linkService; |
|
|
|
|
this.eventBus = options.eventBus || getGlobalEventBus(); |
|
|
|
|
// Heuristic value used when force-resetting `this._blockHashChange`.
|
|
|
|
|
const HASH_CHANGE_TIMEOUT = 1000; // milliseconds
|
|
|
|
|
// Heuristic value used when adding the current position to the browser history.
|
|
|
|
|
const POSITION_UPDATED_THRESHOLD = 50; |
|
|
|
|
// Heuristic value used when adding a temporary position to the browser history.
|
|
|
|
|
const UPDATE_VIEWAREA_TIMEOUT = 2000; // milliseconds
|
|
|
|
|
|
|
|
|
|
/** |
|
|
|
|
* @typedef {Object} PDFHistoryOptions |
|
|
|
|
* @property {IPDFLinkService} linkService - The navigation/linking service. |
|
|
|
|
* @property {EventBus} eventBus - The application event bus. |
|
|
|
|
*/ |
|
|
|
|
|
|
|
|
|
/** |
|
|
|
|
* @typedef {Object} PushParameters |
|
|
|
|
* @property {string} namedDest - (optional) The named destination. If absent, |
|
|
|
|
* a stringified version of `explicitDest` is used. |
|
|
|
|
* @property {Array} explicitDest - The explicit destination array. |
|
|
|
|
* @property {number} pageNumber - The page to which the destination points. |
|
|
|
|
*/ |
|
|
|
|
|
|
|
|
|
this.initialized = false; |
|
|
|
|
this.initialDestination = null; |
|
|
|
|
this.initialBookmark = null; |
|
|
|
|
function getCurrentHash() { |
|
|
|
|
return document.location.hash; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
PDFHistory.prototype = { |
|
|
|
|
function parseCurrentHash(linkService) { |
|
|
|
|
let hash = unescape(getCurrentHash()).substring(1); |
|
|
|
|
let params = parseQueryString(hash); |
|
|
|
|
|
|
|
|
|
let page = params.page | 0; |
|
|
|
|
if (!(Number.isInteger(page) && page > 0 && page <= linkService.pagesCount)) { |
|
|
|
|
page = null; |
|
|
|
|
} |
|
|
|
|
return { hash, page, }; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
class PDFHistory { |
|
|
|
|
/** |
|
|
|
|
* @param {string} fingerprint |
|
|
|
|
* @param {PDFHistoryOptions} options |
|
|
|
|
*/ |
|
|
|
|
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 = ''; |
|
|
|
|
constructor({ linkService, eventBus, }) { |
|
|
|
|
this.linkService = linkService; |
|
|
|
|
this.eventBus = eventBus || getGlobalEventBus(); |
|
|
|
|
|
|
|
|
|
this.initialized = false; |
|
|
|
|
this.initialBookmark = null; |
|
|
|
|
|
|
|
|
|
this._boundEvents = Object.create(null); |
|
|
|
|
this._isViewerInPresentationMode = false; |
|
|
|
|
this._isPagesLoaded = false; |
|
|
|
|
|
|
|
|
|
// Ensure that we don't miss either a 'presentationmodechanged' or a
|
|
|
|
|
// 'pagesloaded' event, by registering the listeners immediately.
|
|
|
|
|
this.eventBus.on('presentationmodechanged', (evt) => { |
|
|
|
|
this._isViewerInPresentationMode = evt.active || evt.switchInProgress; |
|
|
|
|
}); |
|
|
|
|
this.eventBus.on('pagesloaded', (evt) => { |
|
|
|
|
this._isPagesLoaded = !!evt.pagesCount; |
|
|
|
|
}); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/** |
|
|
|
|
* Initialize the history for the PDF document, using either the current |
|
|
|
|
* browser history entry or the document hash, whichever is present. |
|
|
|
|
* @param {string} fingerprint - The PDF document's unique fingerprint. |
|
|
|
|
* @param {boolean} resetHistory - (optional) Reset the browsing history. |
|
|
|
|
*/ |
|
|
|
|
initialize(fingerprint, resetHistory = false) { |
|
|
|
|
if (!fingerprint || typeof fingerprint !== 'string') { |
|
|
|
|
console.error( |
|
|
|
|
'PDFHistory.initialize: The "fingerprint" must be a non-empty string.'); |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
let reInitialized = this.initialized && this.fingerprint !== fingerprint; |
|
|
|
|
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); |
|
|
|
|
|
|
|
|
|
if (!this.initialized) { |
|
|
|
|
this._bindEvents(); |
|
|
|
|
} |
|
|
|
|
let state = window.history.state; |
|
|
|
|
|
|
|
|
|
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); |
|
|
|
|
this.initialized = true; |
|
|
|
|
this.initialBookmark = null; |
|
|
|
|
|
|
|
|
|
this._popStateInProgress = false; |
|
|
|
|
this._blockHashChange = 0; |
|
|
|
|
this._currentHash = getCurrentHash(); |
|
|
|
|
this._numPositionUpdates = 0; |
|
|
|
|
|
|
|
|
|
this._currentUid = this._uid = 0; |
|
|
|
|
this._destination = null; |
|
|
|
|
this._position = null; |
|
|
|
|
|
|
|
|
|
if (!this._isValidState(state) || resetHistory) { |
|
|
|
|
let { hash, page, } = parseCurrentHash(this.linkService); |
|
|
|
|
|
|
|
|
|
if (!hash || reInitialized || resetHistory) { |
|
|
|
|
// Ensure that the browser history is reset on PDF document load.
|
|
|
|
|
this._pushOrReplaceState(null, /* forceReplace = */ true); |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
// Ensure that the browser history is initialized correctly when
|
|
|
|
|
// the document hash is present on PDF document load.
|
|
|
|
|
this._pushOrReplaceState({ hash, page, }, /* forceReplace = */ true); |
|
|
|
|
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(); |
|
|
|
|
} |
|
|
|
|
}); |
|
|
|
|
// The browser history contains a valid entry, ensure that the history is
|
|
|
|
|
// initialized correctly on PDF document load.
|
|
|
|
|
let destination = state.destination; |
|
|
|
|
this._updateInternalState(destination, state.uid, |
|
|
|
|
/* removeTemporary = */ true); |
|
|
|
|
if (destination.dest) { |
|
|
|
|
this.initialBookmark = JSON.stringify(destination.dest); |
|
|
|
|
|
|
|
|
|
// If the history is updated, e.g. through the user changing the hash,
|
|
|
|
|
// before the initial destination has become visible, then we do *not*
|
|
|
|
|
// want to potentially add `this._position` to the browser history.
|
|
|
|
|
this._destination.page = null; |
|
|
|
|
} else if (destination.hash) { |
|
|
|
|
this.initialBookmark = destination.hash; |
|
|
|
|
} else if (destination.page) { |
|
|
|
|
// Fallback case; shouldn't be necessary, but better safe than sorry.
|
|
|
|
|
this.initialBookmark = `page=${destination.page}`; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/** |
|
|
|
|
* Push an internal destination to the browser history. |
|
|
|
|
* @param {PushParameters} |
|
|
|
|
*/ |
|
|
|
|
push({ namedDest, explicitDest, pageNumber, }) { |
|
|
|
|
if (!this.initialized) { |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
if ((namedDest && typeof namedDest !== 'string') || |
|
|
|
|
!(explicitDest instanceof Array) || |
|
|
|
|
!(Number.isInteger(pageNumber) && |
|
|
|
|
pageNumber > 0 && pageNumber <= this.linkService.pagesCount)) { |
|
|
|
|
console.error('PDFHistory.push: Invalid parameters.'); |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
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(); |
|
|
|
|
} |
|
|
|
|
let hash = namedDest || JSON.stringify(explicitDest); |
|
|
|
|
if (!hash) { |
|
|
|
|
// The hash *should* never be undefined, but if that were to occur,
|
|
|
|
|
// avoid any possible issues by not updating the browser history.
|
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
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(); |
|
|
|
|
let forceReplace = false; |
|
|
|
|
if (this._destination && |
|
|
|
|
(this._destination.hash === hash || |
|
|
|
|
isDestsEqual(this._destination.dest, explicitDest))) { |
|
|
|
|
// When the new destination is identical to `this._destination`, and
|
|
|
|
|
// its `page` is undefined, replace the current browser history entry.
|
|
|
|
|
// NOTE: This can only occur if `this._destination` was set either:
|
|
|
|
|
// - through the document hash being specified on load.
|
|
|
|
|
// - through the user changing the hash of the document.
|
|
|
|
|
if (this._destination.page) { |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
// Remove the event listener when navigating away from the document,
|
|
|
|
|
// since 'beforeunload' prevents Firefox from caching the document.
|
|
|
|
|
window.removeEventListener('beforeunload', pdfHistoryBeforeUnload); |
|
|
|
|
forceReplace = true; |
|
|
|
|
} |
|
|
|
|
if (this._popStateInProgress && !forceReplace) { |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
window.addEventListener('beforeunload', pdfHistoryBeforeUnload); |
|
|
|
|
this._pushOrReplaceState({ |
|
|
|
|
dest: explicitDest, |
|
|
|
|
hash, |
|
|
|
|
page: pageNumber, |
|
|
|
|
}, forceReplace); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
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); |
|
|
|
|
}); |
|
|
|
|
/** |
|
|
|
|
* Push the current position to the browser history. |
|
|
|
|
*/ |
|
|
|
|
pushCurrentPosition() { |
|
|
|
|
if (!this.initialized || this._popStateInProgress) { |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
this._tryPushCurrentPosition(); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
self.eventBus.on('presentationmodechanged', function(e) { |
|
|
|
|
self.isViewerInPresentationMode = e.active; |
|
|
|
|
}); |
|
|
|
|
}, |
|
|
|
|
/** |
|
|
|
|
* Go back one step in the browser history. |
|
|
|
|
* NOTE: Avoids navigating away from the document, useful for "named actions". |
|
|
|
|
*/ |
|
|
|
|
back() { |
|
|
|
|
if (!this.initialized || this._popStateInProgress) { |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
let state = window.history.state; |
|
|
|
|
if (this._isValidState(state) && state.uid > 0) { |
|
|
|
|
window.history.back(); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/** |
|
|
|
|
* Go forward one step in the browser history. |
|
|
|
|
* NOTE: Avoids navigating away from the document, useful for "named actions". |
|
|
|
|
*/ |
|
|
|
|
forward() { |
|
|
|
|
if (!this.initialized || this._popStateInProgress) { |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
let state = window.history.state; |
|
|
|
|
if (this._isValidState(state) && state.uid < (this._uid - 1)) { |
|
|
|
|
window.history.forward(); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
clearHistoryState: function pdfHistory_clearHistoryState() { |
|
|
|
|
this._pushOrReplaceState(null, true); |
|
|
|
|
}, |
|
|
|
|
/** |
|
|
|
|
* @returns {boolean} Indicating if the user is currently moving through the |
|
|
|
|
* browser history, useful e.g. for skipping the next 'hashchange' event. |
|
|
|
|
*/ |
|
|
|
|
get popStateInProgress() { |
|
|
|
|
return this.initialized && |
|
|
|
|
(this._popStateInProgress || this._blockHashChange > 0); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
_isStateObjectDefined: function pdfHistory_isStateObjectDefined(state) { |
|
|
|
|
return (state && state.uid >= 0 && |
|
|
|
|
state.fingerprint && this.fingerprint === state.fingerprint && |
|
|
|
|
state.target && state.target.hash) ? true : false; |
|
|
|
|
}, |
|
|
|
|
/** |
|
|
|
|
* @private |
|
|
|
|
*/ |
|
|
|
|
_pushOrReplaceState(destination, forceReplace = false) { |
|
|
|
|
let shouldReplace = forceReplace || !this._destination; |
|
|
|
|
let newState = { |
|
|
|
|
fingerprint: this.fingerprint, |
|
|
|
|
uid: shouldReplace ? this._currentUid : this._uid, |
|
|
|
|
destination, |
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
_pushOrReplaceState: function pdfHistory_pushOrReplaceState(stateObj, |
|
|
|
|
replace) { |
|
|
|
|
// history.state.chromecomState is managed by chromecom.js.
|
|
|
|
|
if (typeof PDFJSDev !== 'undefined' && PDFJSDev.test('CHROME') && |
|
|
|
|
window.history.state && 'chromecomState' in window.history.state) { |
|
|
|
|
stateObj = stateObj || {}; |
|
|
|
|
stateObj.chromecomState = window.history.state.chromecomState; |
|
|
|
|
} |
|
|
|
|
if (replace) { |
|
|
|
|
if (typeof PDFJSDev === 'undefined' || |
|
|
|
|
PDFJSDev.test('GENERIC || CHROME')) { |
|
|
|
|
window.history.replaceState(stateObj, '', document.URL); |
|
|
|
|
window.history.state && window.history.state.chromecomState) { |
|
|
|
|
// history.state.chromecomState is managed by chromecom.js.
|
|
|
|
|
newState.chromecomState = window.history.state.chromecomState; |
|
|
|
|
} |
|
|
|
|
this._updateInternalState(destination, newState.uid); |
|
|
|
|
|
|
|
|
|
if (shouldReplace) { |
|
|
|
|
if (typeof PDFJSDev !== 'undefined' && |
|
|
|
|
PDFJSDev.test('FIREFOX || MOZCENTRAL')) { |
|
|
|
|
// Providing the third argument causes a SecurityError for file:// URLs.
|
|
|
|
|
window.history.replaceState(newState, ''); |
|
|
|
|
} else { |
|
|
|
|
window.history.replaceState(stateObj, ''); |
|
|
|
|
window.history.replaceState(newState, '', document.URL); |
|
|
|
|
} |
|
|
|
|
} else { |
|
|
|
|
if (typeof PDFJSDev === 'undefined' || |
|
|
|
|
PDFJSDev.test('GENERIC || CHROME')) { |
|
|
|
|
window.history.pushState(stateObj, '', document.URL); |
|
|
|
|
if (typeof PDFJSDev !== 'undefined' && |
|
|
|
|
PDFJSDev.test('FIREFOX || MOZCENTRAL')) { |
|
|
|
|
// Providing the third argument causes a SecurityError for file:// URLs.
|
|
|
|
|
window.history.pushState(newState, ''); |
|
|
|
|
} else { |
|
|
|
|
window.history.pushState(stateObj, ''); |
|
|
|
|
} |
|
|
|
|
if (typeof PDFJSDev !== 'undefined' && PDFJSDev.test('CHROME') && |
|
|
|
|
top === window) { |
|
|
|
|
chrome.runtime.sendMessage('showPageAction'); |
|
|
|
|
window.history.pushState(newState, '', document.URL); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
get isHashChangeUnlocked() { |
|
|
|
|
if (!this.initialized) { |
|
|
|
|
return true; |
|
|
|
|
if (typeof PDFJSDev !== 'undefined' && PDFJSDev.test('CHROME') && |
|
|
|
|
top === window) { |
|
|
|
|
// eslint-disable-next-line no-undef
|
|
|
|
|
chrome.runtime.sendMessage('showPageAction'); |
|
|
|
|
} |
|
|
|
|
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; |
|
|
|
|
/** |
|
|
|
|
* @private |
|
|
|
|
*/ |
|
|
|
|
_tryPushCurrentPosition(temporary = false) { |
|
|
|
|
if (!this._position) { |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
updateCurrentBookmark: function pdfHistoryUpdateCurrentBookmark(bookmark, |
|
|
|
|
pageNum) { |
|
|
|
|
if (this.initialized) { |
|
|
|
|
this.currentBookmark = bookmark.substring(1); |
|
|
|
|
this.currentPage = pageNum | 0; |
|
|
|
|
this._updatePreviousBookmark(); |
|
|
|
|
let position = this._position; |
|
|
|
|
if (temporary) { |
|
|
|
|
position = cloneObj(this._position); |
|
|
|
|
position.temporary = true; |
|
|
|
|
} |
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
updateNextHashParam: function pdfHistoryUpdateNextHashParam(param) { |
|
|
|
|
if (this.initialized) { |
|
|
|
|
this.nextHashParam = param; |
|
|
|
|
if (!this._destination) { |
|
|
|
|
this._pushOrReplaceState(position); |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
push: function pdfHistoryPush(params, isInitialBookmark) { |
|
|
|
|
if (!(this.initialized && this.historyUnlocked)) { |
|
|
|
|
if (this._destination.temporary) { |
|
|
|
|
// Always replace a previous *temporary* position.
|
|
|
|
|
this._pushOrReplaceState(position, /* forceReplace = */ true); |
|
|
|
|
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(); |
|
|
|
|
} |
|
|
|
|
if (this._destination.hash === position.hash) { |
|
|
|
|
return; // The current document position has not changed.
|
|
|
|
|
} |
|
|
|
|
if (!this._destination.page && |
|
|
|
|
(POSITION_UPDATED_THRESHOLD <= 0 || |
|
|
|
|
this._numPositionUpdates <= POSITION_UPDATED_THRESHOLD)) { |
|
|
|
|
// `this._destination` was set through the user changing the hash of
|
|
|
|
|
// the document. Do not add `this._position` to the browser history,
|
|
|
|
|
// to avoid "flooding" it with lots of (nearly) identical entries,
|
|
|
|
|
// since we cannot ensure that the document position has changed.
|
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
if (this.nextHashParam) { |
|
|
|
|
if (this.nextHashParam === params.hash) { |
|
|
|
|
this.nextHashParam = null; |
|
|
|
|
this.updatePreviousBookmark = true; |
|
|
|
|
|
|
|
|
|
let forceReplace = false; |
|
|
|
|
if (this._destination.page === position.first || |
|
|
|
|
this._destination.page === position.page) { |
|
|
|
|
// When the `page` of `this._destination` is still visible, do not
|
|
|
|
|
// update the browsing history when `this._destination` either:
|
|
|
|
|
// - contains an internal destination, since in this case we
|
|
|
|
|
// cannot ensure that the document position has actually changed.
|
|
|
|
|
// - was set through the user changing the hash of the document.
|
|
|
|
|
if (this._destination.dest || !this._destination.first) { |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
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; |
|
|
|
|
// To avoid "flooding" the browser history, replace the current entry.
|
|
|
|
|
forceReplace = true; |
|
|
|
|
} |
|
|
|
|
this._pushOrReplaceState(position, forceReplace); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/** |
|
|
|
|
* @private |
|
|
|
|
*/ |
|
|
|
|
_isValidState(state) { |
|
|
|
|
if (!state) { |
|
|
|
|
return false; |
|
|
|
|
} |
|
|
|
|
if (state.fingerprint !== this.fingerprint) { |
|
|
|
|
// This should only occur in viewers with support for opening more than
|
|
|
|
|
// one PDF document, e.g. the GENERIC viewer.
|
|
|
|
|
return false; |
|
|
|
|
} |
|
|
|
|
if (!Number.isInteger(state.uid) || state.uid < 0) { |
|
|
|
|
return false; |
|
|
|
|
} |
|
|
|
|
if (state.destination === null || typeof state.destination !== 'object') { |
|
|
|
|
return false; |
|
|
|
|
} |
|
|
|
|
var params = { hash: this.currentBookmark, page: this.currentPage, }; |
|
|
|
|
if (this.isViewerInPresentationMode) { |
|
|
|
|
params.hash = null; |
|
|
|
|
return true; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/** |
|
|
|
|
* @private |
|
|
|
|
*/ |
|
|
|
|
_updateInternalState(destination, uid, removeTemporary = false) { |
|
|
|
|
if (removeTemporary && destination && destination.temporary) { |
|
|
|
|
// When the `destination` comes from the browser history,
|
|
|
|
|
// we no longer treat it as a *temporary* position.
|
|
|
|
|
delete destination.temporary; |
|
|
|
|
} |
|
|
|
|
return params; |
|
|
|
|
}, |
|
|
|
|
this._destination = destination; |
|
|
|
|
this._currentUid = uid; |
|
|
|
|
this._uid = this._currentUid + 1; |
|
|
|
|
// This should always be reset when `this._destination` is updated.
|
|
|
|
|
this._numPositionUpdates = 0; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
_stateObj: function pdfHistory_stateObj(params) { |
|
|
|
|
return { fingerprint: this.fingerprint, uid: this.uid, target: params, }; |
|
|
|
|
}, |
|
|
|
|
/** |
|
|
|
|
* @private |
|
|
|
|
*/ |
|
|
|
|
_updateViewarea({ location, }) { |
|
|
|
|
if (this._updateViewareaTimeout) { |
|
|
|
|
clearTimeout(this._updateViewareaTimeout); |
|
|
|
|
this._updateViewareaTimeout = null; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
_pushToHistory: function pdfHistory_pushToHistory(params, |
|
|
|
|
addPrevious, overwrite) { |
|
|
|
|
if (!this.initialized) { |
|
|
|
|
this._position = { |
|
|
|
|
hash: this._isViewerInPresentationMode ? |
|
|
|
|
`page=${location.pageNumber}` : location.pdfOpenParams.substring(1), |
|
|
|
|
page: this.linkService.page, |
|
|
|
|
first: location.pageNumber, |
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
if (this._popStateInProgress) { |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
if (!params.hash && params.page) { |
|
|
|
|
params.hash = ('page=' + params.page); |
|
|
|
|
|
|
|
|
|
if (POSITION_UPDATED_THRESHOLD > 0 && this._isPagesLoaded && |
|
|
|
|
this._destination && !this._destination.page) { |
|
|
|
|
// If the current destination was set through the user changing the hash
|
|
|
|
|
// of the document, we will usually not try to push the current position
|
|
|
|
|
// to the browser history; see `this._tryPushCurrentPosition()`.
|
|
|
|
|
//
|
|
|
|
|
// To prevent `this._tryPushCurrentPosition()` from effectively being
|
|
|
|
|
// reduced to a no-op in this case, we will assume that the position
|
|
|
|
|
// *did* in fact change if the 'updateviewarea' event was dispatched
|
|
|
|
|
// more than `POSITION_UPDATED_THRESHOLD` times.
|
|
|
|
|
this._numPositionUpdates++; |
|
|
|
|
} |
|
|
|
|
if (addPrevious && !overwrite) { |
|
|
|
|
var previousParams = this._getPreviousParams(); |
|
|
|
|
if (previousParams) { |
|
|
|
|
var replacePrevious = (!this.current.dest && |
|
|
|
|
this.current.hash !== this.previousHash); |
|
|
|
|
this._pushToHistory(previousParams, false, replacePrevious); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if (UPDATE_VIEWAREA_TIMEOUT > 0) { |
|
|
|
|
// When closing the browser, a 'pagehide' event will be dispatched which
|
|
|
|
|
// *should* allow us to push the current position to the browser history.
|
|
|
|
|
// In practice, it seems that the event is arriving too late in order for
|
|
|
|
|
// the session history to be successfully updated.
|
|
|
|
|
// (For additional details, please refer to the discussion in
|
|
|
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=1153393.)
|
|
|
|
|
//
|
|
|
|
|
// To workaround this we attempt to *temporarily* add the current position
|
|
|
|
|
// to the browser history only when the viewer is *idle*,
|
|
|
|
|
// i.e. when scrolling and/or zooming does not occur.
|
|
|
|
|
//
|
|
|
|
|
// PLEASE NOTE: It's absolutely imperative that the browser history is
|
|
|
|
|
// *not* updated too often, since that would render the viewer more or
|
|
|
|
|
// less unusable. Hence the use of a timeout to delay the update until
|
|
|
|
|
// the viewer has been idle for `UPDATE_VIEWAREA_TIMEOUT` milliseconds.
|
|
|
|
|
this._updateViewareaTimeout = setTimeout(() => { |
|
|
|
|
if (!this._popStateInProgress) { |
|
|
|
|
this._tryPushCurrentPosition(/* temporary = */ true); |
|
|
|
|
} |
|
|
|
|
this._updateViewareaTimeout = null; |
|
|
|
|
}, UPDATE_VIEWAREA_TIMEOUT); |
|
|
|
|
} |
|
|
|
|
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))) { |
|
|
|
|
/** |
|
|
|
|
* @private |
|
|
|
|
*/ |
|
|
|
|
_popState({ state, }) { |
|
|
|
|
let newHash = getCurrentHash(), hashChanged = this._currentHash !== newHash; |
|
|
|
|
this._currentHash = newHash; |
|
|
|
|
|
|
|
|
|
if (!state || |
|
|
|
|
(typeof PDFJSDev !== 'undefined' && PDFJSDev.test('CHROME') && |
|
|
|
|
state.chromecomState && !this._isValidState(state))) { |
|
|
|
|
// This case corresponds to the user changing the hash of the document.
|
|
|
|
|
this._currentUid = this._uid; |
|
|
|
|
|
|
|
|
|
let { hash, page, } = parseCurrentHash(this.linkService); |
|
|
|
|
this._pushOrReplaceState({ hash, page, }, /* forceReplace */ true); |
|
|
|
|
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; |
|
|
|
|
} |
|
|
|
|
if (!this._isValidState(state)) { |
|
|
|
|
// This should only occur in viewers with support for opening more than
|
|
|
|
|
// one PDF document, e.g. the GENERIC viewer.
|
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
this.historyUnlocked = false; |
|
|
|
|
|
|
|
|
|
if (state.target.dest) { |
|
|
|
|
this.linkService.navigateTo(state.target.dest); |
|
|
|
|
} else { |
|
|
|
|
this.linkService.setHash(state.target.hash); |
|
|
|
|
// Prevent the browser history from updating until the new destination,
|
|
|
|
|
// as stored in the browser history, has been scrolled into view.
|
|
|
|
|
this._popStateInProgress = true; |
|
|
|
|
|
|
|
|
|
if (hashChanged) { |
|
|
|
|
// When the hash changed, implying that the 'popstate' event will be
|
|
|
|
|
// followed by a 'hashchange' event, then we do *not* want to update the
|
|
|
|
|
// browser history when handling the 'hashchange' event (in web/app.js)
|
|
|
|
|
// since that would *overwrite* the new destination navigated to below.
|
|
|
|
|
//
|
|
|
|
|
// To avoid accidentally disabling all future user-initiated hash changes,
|
|
|
|
|
// if there's e.g. another 'hashchange' listener that stops the event
|
|
|
|
|
// propagation, we make sure to always force-reset `this._blockHashChange`
|
|
|
|
|
// after `HASH_CHANGE_TIMEOUT` milliseconds have passed.
|
|
|
|
|
this._blockHashChange++; |
|
|
|
|
waitOnEventOrTimeout({ |
|
|
|
|
target: window, |
|
|
|
|
name: 'hashchange', |
|
|
|
|
delay: HASH_CHANGE_TIMEOUT, |
|
|
|
|
}).then(() => { |
|
|
|
|
this._blockHashChange--; |
|
|
|
|
}); |
|
|
|
|
} |
|
|
|
|
this.currentUid = state.uid; |
|
|
|
|
if (state.uid > this.uid) { |
|
|
|
|
this.uid = state.uid; |
|
|
|
|
|
|
|
|
|
// This case corresponds to navigation backwards in the browser history.
|
|
|
|
|
if (state.uid < this._currentUid && this._position && this._destination) { |
|
|
|
|
let shouldGoBack = false; |
|
|
|
|
|
|
|
|
|
if (this._destination.temporary) { |
|
|
|
|
// If the `this._destination` contains a *temporary* position, always
|
|
|
|
|
// push the `this._position` to the browser history before moving back.
|
|
|
|
|
this._pushOrReplaceState(this._position); |
|
|
|
|
shouldGoBack = true; |
|
|
|
|
} else if (this._destination.page && |
|
|
|
|
this._destination.page !== this._position.first && |
|
|
|
|
this._destination.page !== this._position.page) { |
|
|
|
|
// If the `page` of the `this._destination` is no longer visible,
|
|
|
|
|
// push the `this._position` to the browser history before moving back.
|
|
|
|
|
this._pushOrReplaceState(this._destination); |
|
|
|
|
this._pushOrReplaceState(this._position); |
|
|
|
|
shouldGoBack = true; |
|
|
|
|
} |
|
|
|
|
if (shouldGoBack) { |
|
|
|
|
// After `window.history.back()`, we must not enter this block on the
|
|
|
|
|
// resulting 'popstate' event, since that may cause an infinite loop.
|
|
|
|
|
this._currentUid = state.uid; |
|
|
|
|
|
|
|
|
|
window.history.back(); |
|
|
|
|
return; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
this.current = state.target; |
|
|
|
|
this.updatePreviousBookmark = true; |
|
|
|
|
|
|
|
|
|
var currentHash = window.location.hash.substring(1); |
|
|
|
|
if (this.previousHash !== currentHash) { |
|
|
|
|
this.allowHashChange = false; |
|
|
|
|
// Navigate to the new destination.
|
|
|
|
|
let destination = state.destination; |
|
|
|
|
this._updateInternalState(destination, state.uid, |
|
|
|
|
/* removeTemporary = */ true); |
|
|
|
|
if (destination.dest) { |
|
|
|
|
this.linkService.navigateTo(destination.dest); |
|
|
|
|
} else if (destination.hash) { |
|
|
|
|
this.linkService.setHash(destination.hash); |
|
|
|
|
} else if (destination.page) { |
|
|
|
|
// Fallback case; shouldn't be necessary, but better safe than sorry.
|
|
|
|
|
this.linkService.page = destination.page; |
|
|
|
|
} |
|
|
|
|
this.previousHash = currentHash; |
|
|
|
|
|
|
|
|
|
this.historyUnlocked = true; |
|
|
|
|
}, |
|
|
|
|
// Since `PDFLinkService.navigateTo` is asynchronous, we thus defer the
|
|
|
|
|
// resetting of `this._popStateInProgress` slightly.
|
|
|
|
|
Promise.resolve().then(() => { |
|
|
|
|
this._popStateInProgress = false; |
|
|
|
|
}); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
back: function pdfHistoryBack() { |
|
|
|
|
this.go(-1); |
|
|
|
|
}, |
|
|
|
|
/** |
|
|
|
|
* @private |
|
|
|
|
*/ |
|
|
|
|
_bindEvents() { |
|
|
|
|
let { _boundEvents, eventBus, } = this; |
|
|
|
|
|
|
|
|
|
_boundEvents.updateViewarea = this._updateViewarea.bind(this); |
|
|
|
|
_boundEvents.popState = this._popState.bind(this); |
|
|
|
|
_boundEvents.pageHide = (evt) => { |
|
|
|
|
// Attempt to push the `this._position` into the browser history when
|
|
|
|
|
// navigating away from the document. This is *only* done if the history
|
|
|
|
|
// is currently empty, since otherwise an existing browser history entry
|
|
|
|
|
// will end up being overwritten (given that new entries cannot be pushed
|
|
|
|
|
// into the browser history when the 'unload' event has already fired).
|
|
|
|
|
if (!this._destination) { |
|
|
|
|
this._tryPushCurrentPosition(); |
|
|
|
|
} |
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
forward: function pdfHistoryForward() { |
|
|
|
|
this.go(1); |
|
|
|
|
}, |
|
|
|
|
eventBus.on('updateviewarea', _boundEvents.updateViewarea); |
|
|
|
|
window.addEventListener('popstate', _boundEvents.popState); |
|
|
|
|
window.addEventListener('pagehide', _boundEvents.pageHide); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
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(); |
|
|
|
|
function isDestsEqual(firstDest, secondDest) { |
|
|
|
|
function isEntryEqual(first, second) { |
|
|
|
|
if (typeof first !== typeof second) { |
|
|
|
|
return false; |
|
|
|
|
} |
|
|
|
|
if (first instanceof Array || second instanceof Array) { |
|
|
|
|
return false; |
|
|
|
|
} |
|
|
|
|
if (first !== null && typeof first === 'object' && second !== null) { |
|
|
|
|
if (Object.keys(first).length !== Object.keys(second).length) { |
|
|
|
|
return false; |
|
|
|
|
} |
|
|
|
|
for (var key in first) { |
|
|
|
|
if (!isEntryEqual(first[key], second[key])) { |
|
|
|
|
return false; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
return true; |
|
|
|
|
} |
|
|
|
|
}, |
|
|
|
|
}; |
|
|
|
|
return first === second || (Number.isNaN(first) && Number.isNaN(second)); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if (!(firstDest instanceof Array && secondDest instanceof Array)) { |
|
|
|
|
return false; |
|
|
|
|
} |
|
|
|
|
if (firstDest.length !== secondDest.length) { |
|
|
|
|
return false; |
|
|
|
|
} |
|
|
|
|
for (let i = 0, ii = firstDest.length; i < ii; i++) { |
|
|
|
|
if (!isEntryEqual(firstDest[i], secondDest[i])) { |
|
|
|
|
return false; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
return true; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
export { |
|
|
|
|
PDFHistory, |
|
|
|
|
isDestsEqual, |
|
|
|
|
}; |
|
|
|
|