2013-08-14 01:28:13 +09:00
|
|
|
/* Copyright 2013 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.
|
|
|
|
*/
|
2017-02-10 18:42:26 +09:00
|
|
|
/* globals chrome */
|
Switch to using ESLint, instead of JSHint, for linting
*Please note that most of the necessary code adjustments were made in PR 7890.*
ESLint has a number of advantageous properties, compared to JSHint. Among those are:
- The ability to find subtle bugs, thanks to more rules (e.g. PR 7881).
- Much more customizable in general, and many rules allow fine-tuned behaviour rather than the just the on/off rules in JSHint.
- Many more rules that can help developers avoid bugs, and a lot of rules that can be used to enforce a consistent coding style. The latter should be particularily useful for new contributors (and reduce the amount of stylistic review comments necessary).
- The ability to easily specify exactly what rules to use/not to use, as opposed to JSHint which has a default set. *Note:* in future JSHint version some of the rules we depend on will be removed, according to warnings in http://jshint.com/docs/options/, so we wouldn't be able to update without losing lint coverage.
- More easily disable one, or more, rules temporarily. In JSHint this requires using a numeric code, which isn't very user friendly, whereas in ESLint the rule name is simply used instead.
By default there's no rules enabled in ESLint, but there are some default rule sets available. However, to prevent linting failures if we update ESLint in the future, it seemed easier to just explicitly specify what rules we want.
Obviously this makes the ESLint config file somewhat bigger than the old JSHint config file, but given how rarely that one has been updated over the years I don't think that matters too much.
I've tried, to the best of my ability, to ensure that we enable the same rules for ESLint that we had for JSHint. Furthermore, I've also enabled a number of rules that seemed to make sense, both to catch possible errors *and* various style guide violations.
Despite the ESLint README claiming that it's slower that JSHint, https://github.com/eslint/eslint#how-does-eslint-performance-compare-to-jshint, locally this patch actually reduces the runtime for `gulp` lint (by approximately 20-25%).
A couple of stylistic rules that would have been nice to enable, but where our code currently differs to much to make it feasible:
- `comma-dangle`, controls trailing commas in Objects and Arrays (among others).
- `object-curly-spacing`, controls spacing inside of Objects.
- `spaced-comment`, used to enforce spaces after `//` and `/*. (This is made difficult by the fact that there's still some usage of the old preprocessor left.)
Rules that I indend to look into possibly enabling in follow-ups, if it seems to make sense: `no-else-return`, `no-lonely-if`, `brace-style` with the `allowSingleLine` parameter removed.
Useful links:
- http://eslint.org/docs/user-guide/configuring
- http://eslint.org/docs/rules/
2016-12-15 23:52:29 +09:00
|
|
|
|
2017-04-15 00:32:36 +09:00
|
|
|
import { DefaultExternalServices, PDFViewerApplication } from './app';
|
2018-03-09 02:44:12 +09:00
|
|
|
import { AppOptions } from './app_options';
|
2017-04-17 20:21:11 +09:00
|
|
|
import { BasePreferences } from './preferences';
|
2017-04-17 08:55:45 +09:00
|
|
|
import { DownloadManager } from './download_manager';
|
2017-05-04 10:05:53 +09:00
|
|
|
import { GenericL10n } from './genericl10n';
|
2013-08-14 01:28:13 +09:00
|
|
|
|
2017-03-28 08:07:27 +09:00
|
|
|
if (typeof PDFJSDev === 'undefined' || !PDFJSDev.test('CHROME')) {
|
|
|
|
throw new Error('Module "pdfjs-web/chromecom" shall not be used outside ' +
|
|
|
|
'CHROME build.');
|
|
|
|
}
|
2016-10-15 00:57:53 +09:00
|
|
|
|
2017-11-05 21:26:06 +09:00
|
|
|
let ChromeCom = {
|
|
|
|
/**
|
|
|
|
* Creates an event that the extension is listening for and will
|
|
|
|
* asynchronously respond by calling the callback.
|
|
|
|
*
|
|
|
|
* @param {String} action The action to trigger.
|
|
|
|
* @param {String} data Optional data to send.
|
|
|
|
* @param {Function} callback Optional response callback that will be called
|
|
|
|
* with one data argument. When the request cannot be handled, the callback
|
|
|
|
* is immediately invoked with no arguments.
|
|
|
|
*/
|
|
|
|
request(action, data, callback) {
|
|
|
|
let message = {
|
|
|
|
action,
|
|
|
|
data,
|
|
|
|
};
|
|
|
|
if (!chrome.runtime) {
|
|
|
|
console.error('chrome.runtime is undefined.');
|
|
|
|
if (callback) {
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
} else if (callback) {
|
|
|
|
chrome.runtime.sendMessage(message, callback);
|
|
|
|
} else {
|
|
|
|
chrome.runtime.sendMessage(message);
|
2017-03-28 08:07:27 +09:00
|
|
|
}
|
2017-11-05 21:26:06 +09:00
|
|
|
},
|
2014-04-12 00:28:35 +09:00
|
|
|
|
2017-11-05 21:26:06 +09:00
|
|
|
/**
|
|
|
|
* Resolves a PDF file path and attempts to detects length.
|
|
|
|
*
|
|
|
|
* @param {String} file - Absolute URL of PDF file.
|
|
|
|
* @param {OverlayManager} overlayManager - Manager for the viewer overlays.
|
|
|
|
* @param {Function} callback - A callback with resolved URL and file length.
|
|
|
|
*/
|
|
|
|
resolvePDFFile(file, overlayManager, callback) {
|
|
|
|
// Expand drive:-URLs to filesystem URLs (Chrome OS)
|
|
|
|
file = file.replace(/^drive:/i,
|
|
|
|
'filesystem:' + location.origin + '/external/');
|
2014-04-12 00:28:35 +09:00
|
|
|
|
2017-11-05 21:26:06 +09:00
|
|
|
if (/^https?:/.test(file)) {
|
|
|
|
// Assumption: The file being opened is the file that was requested.
|
|
|
|
// There is no UI to input a different URL, so this assumption will hold
|
|
|
|
// for now.
|
|
|
|
setReferer(file, function() {
|
|
|
|
callback(file);
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (/^file?:/.test(file)) {
|
|
|
|
getEmbedderOrigin(function(origin) {
|
|
|
|
// If the origin cannot be determined, let Chrome decide whether to
|
|
|
|
// allow embedding files. Otherwise, only allow local files to be
|
|
|
|
// embedded from local files or Chrome extensions.
|
|
|
|
// Even without this check, the file load in frames is still blocked,
|
|
|
|
// but this may change in the future (https://crbug.com/550151).
|
|
|
|
if (origin && !/^file:|^chrome-extension:/.test(origin)) {
|
|
|
|
PDFViewerApplication.error('Blocked ' + origin + ' from loading ' +
|
|
|
|
file + '. Refused to load a local file in a non-local page ' +
|
|
|
|
'for security reasons.');
|
|
|
|
return;
|
2014-04-12 00:28:35 +09:00
|
|
|
}
|
2017-11-05 21:26:06 +09:00
|
|
|
isAllowedFileSchemeAccess(function(isAllowedAccess) {
|
|
|
|
if (isAllowedAccess) {
|
|
|
|
callback(file);
|
|
|
|
} else {
|
|
|
|
requestAccessToLocalFile(file, overlayManager);
|
|
|
|
}
|
|
|
|
});
|
2016-05-10 05:49:47 +09:00
|
|
|
});
|
2017-11-05 21:26:06 +09:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
callback(file);
|
|
|
|
},
|
2017-03-28 08:07:27 +09:00
|
|
|
};
|
2016-01-30 22:21:54 +09:00
|
|
|
|
2017-03-28 08:07:27 +09:00
|
|
|
function getEmbedderOrigin(callback) {
|
2017-11-05 21:26:06 +09:00
|
|
|
let origin = window === top ? location.origin : location.ancestorOrigins[0];
|
2017-03-28 08:07:27 +09:00
|
|
|
if (origin === 'null') {
|
|
|
|
// file:-URLs, data-URLs, sandboxed frames, etc.
|
|
|
|
getParentOrigin(callback);
|
|
|
|
} else {
|
|
|
|
callback(origin);
|
2016-01-30 22:21:54 +09:00
|
|
|
}
|
2017-03-28 08:07:27 +09:00
|
|
|
}
|
2016-01-30 22:21:54 +09:00
|
|
|
|
2017-03-28 08:07:27 +09:00
|
|
|
function getParentOrigin(callback) {
|
|
|
|
ChromeCom.request('getParentOrigin', null, callback);
|
|
|
|
}
|
2015-07-19 00:11:33 +09:00
|
|
|
|
2017-03-28 08:07:27 +09:00
|
|
|
function isAllowedFileSchemeAccess(callback) {
|
|
|
|
ChromeCom.request('isAllowedFileSchemeAccess', null, callback);
|
|
|
|
}
|
2015-07-19 00:11:33 +09:00
|
|
|
|
2017-03-28 08:07:27 +09:00
|
|
|
function isRuntimeAvailable() {
|
|
|
|
try {
|
|
|
|
// When the extension is reloaded, the extension runtime is destroyed and
|
|
|
|
// the extension APIs become unavailable.
|
|
|
|
if (chrome.runtime && chrome.runtime.getManifest()) {
|
|
|
|
return true;
|
2015-07-19 00:11:33 +09:00
|
|
|
}
|
2017-03-28 08:07:27 +09:00
|
|
|
} catch (e) {}
|
|
|
|
return false;
|
|
|
|
}
|
2015-07-19 00:11:33 +09:00
|
|
|
|
2017-03-28 08:07:27 +09:00
|
|
|
function reloadIfRuntimeIsUnavailable() {
|
|
|
|
if (!isRuntimeAvailable()) {
|
|
|
|
location.reload();
|
|
|
|
}
|
|
|
|
}
|
2015-07-19 00:11:33 +09:00
|
|
|
|
2017-11-05 21:26:06 +09:00
|
|
|
let chromeFileAccessOverlayPromise;
|
2017-05-26 21:52:23 +09:00
|
|
|
function requestAccessToLocalFile(fileUrl, overlayManager) {
|
2017-11-05 21:26:06 +09:00
|
|
|
let onCloseOverlay = null;
|
2017-03-28 08:07:27 +09:00
|
|
|
if (top !== window) {
|
|
|
|
// When the extension reloads after receiving new permissions, the pages
|
|
|
|
// have to be reloaded to restore the extension runtime. Auto-reload
|
|
|
|
// frames, because users should not have to reload the whole page just to
|
|
|
|
// update the viewer.
|
|
|
|
// Top-level frames are closed by Chrome upon reload, so there is no need
|
|
|
|
// for detecting unload of the top-level frame. Should this ever change
|
|
|
|
// (crbug.com/511670), then the user can just reload the tab.
|
|
|
|
window.addEventListener('focus', reloadIfRuntimeIsUnavailable);
|
|
|
|
onCloseOverlay = function() {
|
|
|
|
window.removeEventListener('focus', reloadIfRuntimeIsUnavailable);
|
|
|
|
reloadIfRuntimeIsUnavailable();
|
2017-05-26 21:52:23 +09:00
|
|
|
overlayManager.close('chromeFileAccessOverlay');
|
2017-03-28 08:07:27 +09:00
|
|
|
};
|
|
|
|
}
|
|
|
|
if (!chromeFileAccessOverlayPromise) {
|
2017-05-26 21:52:23 +09:00
|
|
|
chromeFileAccessOverlayPromise = overlayManager.register(
|
2017-03-28 08:07:27 +09:00
|
|
|
'chromeFileAccessOverlay',
|
|
|
|
document.getElementById('chromeFileAccessOverlay'),
|
|
|
|
onCloseOverlay, true);
|
|
|
|
}
|
|
|
|
chromeFileAccessOverlayPromise.then(function() {
|
2017-11-05 21:26:06 +09:00
|
|
|
let iconPath = chrome.runtime.getManifest().icons[48];
|
2017-03-28 08:07:27 +09:00
|
|
|
document.getElementById('chrome-pdfjs-logo-bg').style.backgroundImage =
|
|
|
|
'url(' + chrome.runtime.getURL(iconPath) + ')';
|
2015-07-19 01:26:04 +09:00
|
|
|
|
2017-03-28 08:07:27 +09:00
|
|
|
// Use Chrome's definition of UI language instead of PDF.js's #lang=...,
|
|
|
|
// because the shown string should match the UI at chrome://extensions.
|
|
|
|
// These strings are from chrome/app/resources/generated_resources_*.xtb.
|
|
|
|
/* eslint-disable no-unexpected-multiline */
|
2017-11-05 21:26:06 +09:00
|
|
|
let i18nFileAccessLabel =
|
2017-03-28 08:07:27 +09:00
|
|
|
PDFJSDev.json('$ROOT/web/chrome-i18n-allow-access-to-file-urls.json')
|
|
|
|
[chrome.i18n.getUILanguage && chrome.i18n.getUILanguage()];
|
|
|
|
/* eslint-enable no-unexpected-multiline */
|
2015-07-19 01:26:04 +09:00
|
|
|
|
2017-03-28 08:07:27 +09:00
|
|
|
if (i18nFileAccessLabel) {
|
|
|
|
document.getElementById('chrome-file-access-label').textContent =
|
|
|
|
i18nFileAccessLabel;
|
|
|
|
}
|
2015-07-19 00:11:33 +09:00
|
|
|
|
2017-11-05 21:26:06 +09:00
|
|
|
let link = document.getElementById('chrome-link-to-extensions-page');
|
2017-03-28 08:07:27 +09:00
|
|
|
link.href = 'chrome://extensions/?id=' + chrome.runtime.id;
|
|
|
|
link.onclick = function(e) {
|
|
|
|
// Direct navigation to chrome:// URLs is blocked by Chrome, so we
|
|
|
|
// have to ask the background page to open chrome://extensions/?id=...
|
|
|
|
e.preventDefault();
|
|
|
|
// Open in the current tab by default, because toggling the file access
|
|
|
|
// checkbox causes the extension to reload, and Chrome will close all
|
|
|
|
// tabs upon reload.
|
|
|
|
ChromeCom.request('openExtensionsPageForFileAccess', {
|
Fix inconsistent spacing and trailing commas in objects in `web/` files, so we can enable the `comma-dangle` and `object-curly-spacing` ESLint rules later on
http://eslint.org/docs/rules/comma-dangle
http://eslint.org/docs/rules/object-curly-spacing
Given that we currently have quite inconsistent object formatting, fixing this in in *one* big patch probably wouldn't be feasible (since I cannot imagine anyone wanting to review that); hence I've opted to try and do this piecewise instead.
*Please note:* This patch was created automatically, using the ESLint `--fix` command line option. In a couple of places this caused lines to become too long, and I've fixed those manually; please refer to the interdiff below for the only hand-edits in this patch.
```diff
diff --git a/web/pdf_thumbnail_view.js b/web/pdf_thumbnail_view.js
index 002dbf29..1de4e530 100644
--- a/web/pdf_thumbnail_view.js
+++ b/web/pdf_thumbnail_view.js
@@ -420,8 +420,8 @@ var PDFThumbnailView = (function PDFThumbnailViewClosure() {
setPageLabel: function PDFThumbnailView_setPageLabel(label) {
this.pageLabel = (typeof label === 'string' ? label : null);
- this.l10n.get('thumb_page_title', { page: this.pageId, }, 'Page {{page}}').
- then((msg) => {
+ this.l10n.get('thumb_page_title', { page: this.pageId, },
+ 'Page {{page}}').then((msg) => {
this.anchor.title = msg;
});
diff --git a/web/secondary_toolbar.js b/web/secondary_toolbar.js
index 160e0410..6495fc5e 100644
--- a/web/secondary_toolbar.js
+++ b/web/secondary_toolbar.js
@@ -65,7 +65,8 @@ class SecondaryToolbar {
{ element: options.printButton, eventName: 'print', close: true, },
{ element: options.downloadButton, eventName: 'download', close: true, },
{ element: options.viewBookmarkButton, eventName: null, close: true, },
- { element: options.firstPageButton, eventName: 'firstpage', close: true, },
+ { element: options.firstPageButton, eventName: 'firstpage',
+ close: true, },
{ element: options.lastPageButton, eventName: 'lastpage', close: true, },
{ element: options.pageRotateCwButton, eventName: 'rotatecw',
close: false, },
@@ -76,7 +77,7 @@ class SecondaryToolbar {
{ element: options.cursorHandToolButton, eventName: 'switchcursortool',
eventDetails: { tool: CursorTool.HAND, }, close: true, },
{ element: options.documentPropertiesButton,
- eventName: 'documentproperties', close: true, }
+ eventName: 'documentproperties', close: true, },
];
this.items = {
firstPage: options.firstPageButton,
```
2017-06-01 19:46:12 +09:00
|
|
|
newTab: e.ctrlKey || e.metaKey || e.button === 1 || window !== top,
|
2017-03-28 08:07:27 +09:00
|
|
|
});
|
|
|
|
};
|
2015-07-19 00:11:33 +09:00
|
|
|
|
2017-03-28 08:07:27 +09:00
|
|
|
// Show which file is being opened to help the user with understanding
|
|
|
|
// why this permission request is shown.
|
|
|
|
document.getElementById('chrome-url-of-local-file').textContent = fileUrl;
|
2015-07-19 00:11:33 +09:00
|
|
|
|
2017-05-26 21:52:23 +09:00
|
|
|
overlayManager.open('chromeFileAccessOverlay');
|
2017-03-28 08:07:27 +09:00
|
|
|
});
|
|
|
|
}
|
2015-07-20 08:29:37 +09:00
|
|
|
|
2017-03-28 08:07:27 +09:00
|
|
|
if (window === top) {
|
|
|
|
// Chrome closes all extension tabs (crbug.com/511670) when the extension
|
|
|
|
// reloads. To counter this, the tab URL and history state is saved to
|
|
|
|
// localStorage and restored by extension-router.js.
|
|
|
|
// Unfortunately, the window and tab index are not restored. And if it was
|
|
|
|
// the only tab in an incognito window, then the tab is not restored either.
|
|
|
|
addEventListener('unload', function() {
|
|
|
|
// If the runtime is still available, the unload is most likely a normal
|
|
|
|
// tab closure. Otherwise it is most likely an extension reload.
|
|
|
|
if (!isRuntimeAvailable()) {
|
|
|
|
localStorage.setItem(
|
|
|
|
'unload-' + Date.now() + '-' + document.hidden + '-' + location.href,
|
|
|
|
JSON.stringify(history.state));
|
2015-03-10 23:22:32 +09:00
|
|
|
}
|
2017-03-28 08:07:27 +09:00
|
|
|
});
|
|
|
|
}
|
2015-03-10 23:22:32 +09:00
|
|
|
|
2017-03-28 08:07:27 +09:00
|
|
|
// This port is used for several purposes:
|
|
|
|
// 1. When disconnected, the background page knows that the frame has unload.
|
|
|
|
// 2. When the referrer was saved in history.state.chromecomState, it is sent
|
|
|
|
// to the background page.
|
|
|
|
// 3. When the background page knows the referrer of the page, the referrer is
|
|
|
|
// saved in history.state.chromecomState.
|
2017-11-05 21:26:06 +09:00
|
|
|
let port;
|
2017-03-28 08:07:27 +09:00
|
|
|
// Set the referer for the given URL.
|
|
|
|
// 0. Background: If loaded via a http(s) URL: Save referer.
|
|
|
|
// 1. Page -> background: send URL and referer from history.state
|
|
|
|
// 2. Background: Bind referer to URL (via webRequest).
|
|
|
|
// 3. Background -> page: Send latest referer and save to history.
|
|
|
|
// 4. Page: Invoke callback.
|
|
|
|
function setReferer(url, callback) {
|
|
|
|
if (!port) {
|
|
|
|
// The background page will accept the port, and keep adding the Referer
|
|
|
|
// request header to requests to |url| until the port is disconnected.
|
Fix inconsistent spacing and trailing commas in objects in `web/` files, so we can enable the `comma-dangle` and `object-curly-spacing` ESLint rules later on
http://eslint.org/docs/rules/comma-dangle
http://eslint.org/docs/rules/object-curly-spacing
Given that we currently have quite inconsistent object formatting, fixing this in in *one* big patch probably wouldn't be feasible (since I cannot imagine anyone wanting to review that); hence I've opted to try and do this piecewise instead.
*Please note:* This patch was created automatically, using the ESLint `--fix` command line option. In a couple of places this caused lines to become too long, and I've fixed those manually; please refer to the interdiff below for the only hand-edits in this patch.
```diff
diff --git a/web/pdf_thumbnail_view.js b/web/pdf_thumbnail_view.js
index 002dbf29..1de4e530 100644
--- a/web/pdf_thumbnail_view.js
+++ b/web/pdf_thumbnail_view.js
@@ -420,8 +420,8 @@ var PDFThumbnailView = (function PDFThumbnailViewClosure() {
setPageLabel: function PDFThumbnailView_setPageLabel(label) {
this.pageLabel = (typeof label === 'string' ? label : null);
- this.l10n.get('thumb_page_title', { page: this.pageId, }, 'Page {{page}}').
- then((msg) => {
+ this.l10n.get('thumb_page_title', { page: this.pageId, },
+ 'Page {{page}}').then((msg) => {
this.anchor.title = msg;
});
diff --git a/web/secondary_toolbar.js b/web/secondary_toolbar.js
index 160e0410..6495fc5e 100644
--- a/web/secondary_toolbar.js
+++ b/web/secondary_toolbar.js
@@ -65,7 +65,8 @@ class SecondaryToolbar {
{ element: options.printButton, eventName: 'print', close: true, },
{ element: options.downloadButton, eventName: 'download', close: true, },
{ element: options.viewBookmarkButton, eventName: null, close: true, },
- { element: options.firstPageButton, eventName: 'firstpage', close: true, },
+ { element: options.firstPageButton, eventName: 'firstpage',
+ close: true, },
{ element: options.lastPageButton, eventName: 'lastpage', close: true, },
{ element: options.pageRotateCwButton, eventName: 'rotatecw',
close: false, },
@@ -76,7 +77,7 @@ class SecondaryToolbar {
{ element: options.cursorHandToolButton, eventName: 'switchcursortool',
eventDetails: { tool: CursorTool.HAND, }, close: true, },
{ element: options.documentPropertiesButton,
- eventName: 'documentproperties', close: true, }
+ eventName: 'documentproperties', close: true, },
];
this.items = {
firstPage: options.firstPageButton,
```
2017-06-01 19:46:12 +09:00
|
|
|
port = chrome.runtime.connect({ name: 'chromecom-referrer', });
|
2017-03-28 08:07:27 +09:00
|
|
|
}
|
|
|
|
port.onDisconnect.addListener(onDisconnect);
|
|
|
|
port.onMessage.addListener(onMessage);
|
|
|
|
// Initiate the information exchange.
|
|
|
|
port.postMessage({
|
|
|
|
referer: window.history.state && window.history.state.chromecomState,
|
Fix inconsistent spacing and trailing commas in objects in `web/` files, so we can enable the `comma-dangle` and `object-curly-spacing` ESLint rules later on
http://eslint.org/docs/rules/comma-dangle
http://eslint.org/docs/rules/object-curly-spacing
Given that we currently have quite inconsistent object formatting, fixing this in in *one* big patch probably wouldn't be feasible (since I cannot imagine anyone wanting to review that); hence I've opted to try and do this piecewise instead.
*Please note:* This patch was created automatically, using the ESLint `--fix` command line option. In a couple of places this caused lines to become too long, and I've fixed those manually; please refer to the interdiff below for the only hand-edits in this patch.
```diff
diff --git a/web/pdf_thumbnail_view.js b/web/pdf_thumbnail_view.js
index 002dbf29..1de4e530 100644
--- a/web/pdf_thumbnail_view.js
+++ b/web/pdf_thumbnail_view.js
@@ -420,8 +420,8 @@ var PDFThumbnailView = (function PDFThumbnailViewClosure() {
setPageLabel: function PDFThumbnailView_setPageLabel(label) {
this.pageLabel = (typeof label === 'string' ? label : null);
- this.l10n.get('thumb_page_title', { page: this.pageId, }, 'Page {{page}}').
- then((msg) => {
+ this.l10n.get('thumb_page_title', { page: this.pageId, },
+ 'Page {{page}}').then((msg) => {
this.anchor.title = msg;
});
diff --git a/web/secondary_toolbar.js b/web/secondary_toolbar.js
index 160e0410..6495fc5e 100644
--- a/web/secondary_toolbar.js
+++ b/web/secondary_toolbar.js
@@ -65,7 +65,8 @@ class SecondaryToolbar {
{ element: options.printButton, eventName: 'print', close: true, },
{ element: options.downloadButton, eventName: 'download', close: true, },
{ element: options.viewBookmarkButton, eventName: null, close: true, },
- { element: options.firstPageButton, eventName: 'firstpage', close: true, },
+ { element: options.firstPageButton, eventName: 'firstpage',
+ close: true, },
{ element: options.lastPageButton, eventName: 'lastpage', close: true, },
{ element: options.pageRotateCwButton, eventName: 'rotatecw',
close: false, },
@@ -76,7 +77,7 @@ class SecondaryToolbar {
{ element: options.cursorHandToolButton, eventName: 'switchcursortool',
eventDetails: { tool: CursorTool.HAND, }, close: true, },
{ element: options.documentPropertiesButton,
- eventName: 'documentproperties', close: true, }
+ eventName: 'documentproperties', close: true, },
];
this.items = {
firstPage: options.firstPageButton,
```
2017-06-01 19:46:12 +09:00
|
|
|
requestUrl: url,
|
2017-03-28 08:07:27 +09:00
|
|
|
});
|
|
|
|
|
|
|
|
function onMessage(referer) {
|
|
|
|
if (referer) {
|
|
|
|
// The background extracts the Referer from the initial HTTP request for
|
|
|
|
// the PDF file. When the viewer is reloaded or when the user navigates
|
|
|
|
// back and forward, the background page will not observe a HTTP request
|
|
|
|
// with Referer. To make sure that the Referer is preserved, store it in
|
|
|
|
// history.state, which is preserved across reloads/navigations.
|
2017-11-05 21:26:06 +09:00
|
|
|
let state = window.history.state || {};
|
2017-03-28 08:07:27 +09:00
|
|
|
state.chromecomState = referer;
|
|
|
|
window.history.replaceState(state, '');
|
2015-03-10 23:22:32 +09:00
|
|
|
}
|
2017-03-28 08:07:27 +09:00
|
|
|
onCompleted();
|
2015-03-10 23:22:32 +09:00
|
|
|
}
|
2017-03-28 08:07:27 +09:00
|
|
|
function onDisconnect() {
|
|
|
|
// When the connection fails, ignore the error and call the callback.
|
|
|
|
port = null;
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
function onCompleted() {
|
|
|
|
port.onDisconnect.removeListener(onDisconnect);
|
|
|
|
port.onMessage.removeListener(onMessage);
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
}
|
2015-03-10 23:22:32 +09:00
|
|
|
|
2017-03-28 08:07:27 +09:00
|
|
|
// chrome.storage.sync is not supported in every Chromium-derivate.
|
|
|
|
// Note: The background page takes care of migrating values from
|
|
|
|
// chrome.storage.local to chrome.storage.sync when needed.
|
2017-11-05 21:26:06 +09:00
|
|
|
let storageArea = chrome.storage.sync || chrome.storage.local;
|
2016-05-23 22:50:07 +09:00
|
|
|
|
2017-04-17 20:21:11 +09:00
|
|
|
class ChromePreferences extends BasePreferences {
|
|
|
|
_writeToStorage(prefObj) {
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
if (prefObj === this.defaults) {
|
2017-11-05 21:26:06 +09:00
|
|
|
let keysToRemove = Object.keys(this.defaults);
|
2017-04-17 20:21:11 +09:00
|
|
|
// If the storage is reset, remove the keys so that the values from
|
|
|
|
// managed storage are applied again.
|
|
|
|
storageArea.remove(keysToRemove, function() {
|
|
|
|
resolve();
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
storageArea.set(prefObj, function() {
|
|
|
|
resolve();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2016-04-16 00:02:14 +09:00
|
|
|
|
2017-04-17 20:21:11 +09:00
|
|
|
_readFromStorage(prefObj) {
|
|
|
|
return new Promise((resolve) => {
|
2017-11-05 21:26:06 +09:00
|
|
|
let getPreferences = (defaultPrefs) => {
|
2017-04-17 20:21:11 +09:00
|
|
|
if (chrome.runtime.lastError) {
|
|
|
|
// Managed storage not supported, e.g. in Opera.
|
|
|
|
defaultPrefs = this.defaults;
|
|
|
|
}
|
|
|
|
storageArea.get(defaultPrefs, function(readPrefs) {
|
|
|
|
resolve(readPrefs);
|
|
|
|
});
|
|
|
|
};
|
2016-04-16 00:02:14 +09:00
|
|
|
|
2017-04-17 20:21:11 +09:00
|
|
|
if (chrome.storage.managed) {
|
|
|
|
// Get preferences as set by the system administrator.
|
|
|
|
// See extensions/chromium/preferences_schema.json for more information.
|
|
|
|
// These preferences can be overridden by the user.
|
2018-02-17 01:34:34 +09:00
|
|
|
|
|
|
|
// Deprecated preferences are removed from web/default_preferences.json,
|
|
|
|
// but kept in extensions/chromium/preferences_schema.json for backwards
|
|
|
|
// compatibility with managed preferences.
|
|
|
|
let defaultManagedPrefs = Object.assign({
|
|
|
|
enableHandToolOnLoad: false,
|
[CRX] Make textLayerMode pref visible and add migration logic
In a1cfa5f4d7c8fcf55e9f3b51a23885dca8782915, the textLayerMode
preference was introduced, to replace the disableTextLayer and
enhanceTextSelection preferences.
As a result, the text selection preference was no longer visible
in Chrome (because preferences are only rendered by default for
boolean preferences, not for enumerations).
This commit adds the necessary bits to
extensions/chromium/options/options.{html,js}
so that the textLayerMode preference can be changed again.
Also, migration logic has been added to move over preferences
from the old to the new names:
- In web/chromecom.js, the logic is added to translate
preferences that were set by an administrator (it is read-only,
so this layer is unavoidable).
- In extensions/chromium/options/migration.js, similar logic is
added, except in this case the preference storage is writable,
so this migration logic happens only once.
The "enhanced text selection" mode is still experimental, so it
has been marked as experimental to signal that there may be bugs.
The list of tasks that block promotion to stable is at #7584.
2018-02-17 02:02:05 +09:00
|
|
|
disableTextLayer: false,
|
|
|
|
enhanceTextSelection: false,
|
2018-02-17 01:34:34 +09:00
|
|
|
}, this.defaults);
|
|
|
|
|
|
|
|
chrome.storage.managed.get(defaultManagedPrefs, function(items) {
|
|
|
|
items = items || defaultManagedPrefs;
|
[CRX] Make textLayerMode pref visible and add migration logic
In a1cfa5f4d7c8fcf55e9f3b51a23885dca8782915, the textLayerMode
preference was introduced, to replace the disableTextLayer and
enhanceTextSelection preferences.
As a result, the text selection preference was no longer visible
in Chrome (because preferences are only rendered by default for
boolean preferences, not for enumerations).
This commit adds the necessary bits to
extensions/chromium/options/options.{html,js}
so that the textLayerMode preference can be changed again.
Also, migration logic has been added to move over preferences
from the old to the new names:
- In web/chromecom.js, the logic is added to translate
preferences that were set by an administrator (it is read-only,
so this layer is unavoidable).
- In extensions/chromium/options/migration.js, similar logic is
added, except in this case the preference storage is writable,
so this migration logic happens only once.
The "enhanced text selection" mode is still experimental, so it
has been marked as experimental to signal that there may be bugs.
The list of tasks that block promotion to stable is at #7584.
2018-02-17 02:02:05 +09:00
|
|
|
// Migration logic for deprecated preferences: If the new preference
|
|
|
|
// is not defined by an administrator (i.e. the value is the same as
|
|
|
|
// the default value), and a deprecated preference is set with a
|
|
|
|
// non-default value, migrate the deprecated preference value to the
|
|
|
|
// new preference value.
|
2017-07-15 08:50:15 +09:00
|
|
|
// Never remove this, because we have no means of modifying managed
|
|
|
|
// preferences.
|
[CRX] Make textLayerMode pref visible and add migration logic
In a1cfa5f4d7c8fcf55e9f3b51a23885dca8782915, the textLayerMode
preference was introduced, to replace the disableTextLayer and
enhanceTextSelection preferences.
As a result, the text selection preference was no longer visible
in Chrome (because preferences are only rendered by default for
boolean preferences, not for enumerations).
This commit adds the necessary bits to
extensions/chromium/options/options.{html,js}
so that the textLayerMode preference can be changed again.
Also, migration logic has been added to move over preferences
from the old to the new names:
- In web/chromecom.js, the logic is added to translate
preferences that were set by an administrator (it is read-only,
so this layer is unavoidable).
- In extensions/chromium/options/migration.js, similar logic is
added, except in this case the preference storage is writable,
so this migration logic happens only once.
The "enhanced text selection" mode is still experimental, so it
has been marked as experimental to signal that there may be bugs.
The list of tasks that block promotion to stable is at #7584.
2018-02-17 02:02:05 +09:00
|
|
|
|
|
|
|
// Migration code for https://github.com/mozilla/pdf.js/pull/7635.
|
2018-02-17 01:34:34 +09:00
|
|
|
if (items.enableHandToolOnLoad && !items.cursorToolOnLoad) {
|
2017-07-15 08:50:15 +09:00
|
|
|
items.cursorToolOnLoad = 1;
|
|
|
|
}
|
2018-02-17 01:34:34 +09:00
|
|
|
delete items.enableHandToolOnLoad;
|
[CRX] Make textLayerMode pref visible and add migration logic
In a1cfa5f4d7c8fcf55e9f3b51a23885dca8782915, the textLayerMode
preference was introduced, to replace the disableTextLayer and
enhanceTextSelection preferences.
As a result, the text selection preference was no longer visible
in Chrome (because preferences are only rendered by default for
boolean preferences, not for enumerations).
This commit adds the necessary bits to
extensions/chromium/options/options.{html,js}
so that the textLayerMode preference can be changed again.
Also, migration logic has been added to move over preferences
from the old to the new names:
- In web/chromecom.js, the logic is added to translate
preferences that were set by an administrator (it is read-only,
so this layer is unavoidable).
- In extensions/chromium/options/migration.js, similar logic is
added, except in this case the preference storage is writable,
so this migration logic happens only once.
The "enhanced text selection" mode is still experimental, so it
has been marked as experimental to signal that there may be bugs.
The list of tasks that block promotion to stable is at #7584.
2018-02-17 02:02:05 +09:00
|
|
|
|
|
|
|
// Migration code for https://github.com/mozilla/pdf.js/pull/9479.
|
|
|
|
if (items.textLayerMode !== 1) {
|
|
|
|
if (items.disableTextLayer) {
|
|
|
|
items.textLayerMode = 0;
|
|
|
|
} else if (items.enhanceTextSelection) {
|
|
|
|
items.textLayerMode = 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
delete items.disableTextLayer;
|
|
|
|
delete items.enhanceTextSelection;
|
|
|
|
|
2017-07-15 08:50:15 +09:00
|
|
|
getPreferences(items);
|
|
|
|
});
|
2017-04-17 20:21:11 +09:00
|
|
|
} else {
|
|
|
|
// Managed storage not supported, e.g. in old Chromium versions.
|
|
|
|
getPreferences(this.defaults);
|
2016-04-16 00:02:14 +09:00
|
|
|
}
|
2017-04-17 20:21:11 +09:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2016-04-16 00:02:14 +09:00
|
|
|
|
2017-11-05 21:26:06 +09:00
|
|
|
let ChromeExternalServices = Object.create(DefaultExternalServices);
|
|
|
|
ChromeExternalServices.initPassiveLoading = function(callbacks) {
|
2018-03-09 02:44:12 +09:00
|
|
|
let { overlayManager, } = PDFViewerApplication;
|
|
|
|
// defaultUrl is set in viewer.js
|
|
|
|
ChromeCom.resolvePDFFile(AppOptions.get('defaultUrl'), overlayManager,
|
2017-11-05 21:26:06 +09:00
|
|
|
function(url, length, originalURL) {
|
2017-03-28 08:07:27 +09:00
|
|
|
callbacks.onOpenWithURL(url, length, originalURL);
|
|
|
|
});
|
|
|
|
};
|
2018-02-18 06:51:03 +09:00
|
|
|
ChromeExternalServices.createDownloadManager = function(options) {
|
|
|
|
return new DownloadManager(options);
|
2017-04-17 08:55:45 +09:00
|
|
|
};
|
2017-04-17 20:21:11 +09:00
|
|
|
ChromeExternalServices.createPreferences = function() {
|
|
|
|
return new ChromePreferences();
|
|
|
|
};
|
2018-02-13 22:22:14 +09:00
|
|
|
ChromeExternalServices.createL10n = function(options) {
|
2017-05-04 10:05:53 +09:00
|
|
|
return new GenericL10n(navigator.language);
|
|
|
|
};
|
2017-03-28 08:07:27 +09:00
|
|
|
PDFViewerApplication.externalServices = ChromeExternalServices;
|
2016-04-16 00:02:14 +09:00
|
|
|
|
2017-03-28 08:07:27 +09:00
|
|
|
export {
|
|
|
|
ChromeCom,
|
|
|
|
};
|