You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
173 lines
6.8 KiB
173 lines
6.8 KiB
/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ |
|
/* 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. |
|
*/ |
|
|
|
/* globals chrome, PDFJS, PDFViewerApplication */ |
|
'use strict'; |
|
|
|
var ChromeCom = (function ChromeComClosure() { |
|
var 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. |
|
*/ |
|
ChromeCom.request = function ChromeCom_request(action, data, callback) { |
|
var message = { |
|
action: action, |
|
data: 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); |
|
} |
|
}; |
|
|
|
/** |
|
* Opens a PDF file with the PDF viewer. |
|
* |
|
* @param {String} file Absolute URL of PDF file. |
|
*/ |
|
ChromeCom.openPDFFile = function ChromeCom_openPDFFile(file) { |
|
// Expand drive:-URLs to filesystem URLs (Chrome OS) |
|
file = file.replace(/^drive:/i, |
|
'filesystem:' + location.origin + '/external/'); |
|
|
|
ChromeCom.request('getPDFStream', file, function(response) { |
|
if (response) { |
|
// We will only get a response when the streamsPrivate API is available. |
|
|
|
var isFTPFile = /^ftp:/i.test(file); |
|
var streamUrl = response.streamUrl; |
|
if (streamUrl) { |
|
console.log('Found data stream for ' + file); |
|
PDFViewerApplication.open(streamUrl, 0, undefined, undefined, { |
|
length: response.contentLength |
|
}); |
|
PDFViewerApplication.setTitleUsingUrl(file); |
|
return; |
|
} |
|
if (isFTPFile && !response.extensionSupportsFTP) { |
|
// Stream not found, and it's loaded from FTP. |
|
// When the browser does not support loading ftp resources over |
|
// XMLHttpRequest, just reload the page. |
|
// NOTE: This will not lead to an infinite redirect loop, because |
|
// if the file exists, then the streamsPrivate API will capture the |
|
// stream and send back the response. If the stream does not exist, |
|
// a "Webpage not available" error will be shown (not the PDF Viewer). |
|
location.replace(file); |
|
return; |
|
} |
|
} |
|
if (/^filesystem:/.test(file) && !PDFJS.disableWorker) { |
|
// The security origin of filesystem:-URLs are not preserved when the |
|
// URL is passed to a Web worker, (http://crbug.com/362061), so we have |
|
// to create an intermediate blob:-URL as a work-around. |
|
var resolveLocalFileSystemURL = window.resolveLocalFileSystemURL || |
|
window.webkitResolveLocalFileSystemURL; |
|
resolveLocalFileSystemURL(file, function onResolvedFSURL(fileEntry) { |
|
fileEntry.file(function(fileObject) { |
|
var blobUrl = URL.createObjectURL(fileObject); |
|
PDFViewerApplication.open(blobUrl, 0, undefined, undefined, { |
|
length: fileObject.size |
|
}); |
|
}); |
|
}, function onFileSystemError(error) { |
|
// This should not happen. When it happens, just fall back to the |
|
// usual way of getting the File's data (via the Web worker). |
|
console.warn('Cannot resolve file ' + file + ', ' + error.name + ' ' + |
|
error.message); |
|
PDFViewerApplication.open(file, 0); |
|
}); |
|
return; |
|
} |
|
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() { |
|
PDFViewerApplication.open(file, 0); |
|
}); |
|
return; |
|
} |
|
PDFViewerApplication.open(file, 0); |
|
}); |
|
}; |
|
|
|
// 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. |
|
var port; |
|
// 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. |
|
port = chrome.runtime.connect({name: 'chromecom-referrer'}); |
|
} |
|
port.onDisconnect.addListener(onDisconnect); |
|
port.onMessage.addListener(onMessage); |
|
// Initiate the information exchange. |
|
port.postMessage({ |
|
referer: window.history.state && window.history.state.chromecomState, |
|
requestUrl: url |
|
}); |
|
|
|
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 accross reloads/navigations. |
|
var state = window.history.state || {}; |
|
state.chromecomState = referer; |
|
window.history.replaceState(state, ''); |
|
} |
|
onCompleted(); |
|
} |
|
function onDisconnect() { |
|
// When the connection fails, ignore the error and call the callback. |
|
port = null; |
|
onCompleted(); |
|
} |
|
function onCompleted() { |
|
port.onDisconnect.removeListener(onDisconnect); |
|
port.onMessage.removeListener(onMessage); |
|
callback(); |
|
} |
|
} |
|
|
|
return ChromeCom; |
|
})();
|
|
|