2011-10-26 10:18:22 +09:00
|
|
|
/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
2011-09-08 02:16:02 +09:00
|
|
|
/* vim: set shiftwidth=2 tabstop=2 autoindent cindent expandtab: */
|
2012-09-01 07:48:21 +09:00
|
|
|
/* 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.
|
|
|
|
*/
|
2011-09-08 02:16:02 +09:00
|
|
|
|
2011-10-26 10:18:22 +09:00
|
|
|
'use strict';
|
|
|
|
|
2011-10-26 02:16:20 +09:00
|
|
|
function MessageHandler(name, comObj) {
|
|
|
|
this.name = name;
|
|
|
|
this.comObj = comObj;
|
2011-12-08 08:36:27 +09:00
|
|
|
this.callbackIndex = 1;
|
2011-12-09 07:51:26 +09:00
|
|
|
var callbacks = this.callbacks = {};
|
2011-10-26 02:16:20 +09:00
|
|
|
var ah = this.actionHandler = {};
|
|
|
|
|
2011-10-29 03:23:30 +09:00
|
|
|
ah['console_log'] = [function ahConsoleLog(data) {
|
2012-08-31 20:40:37 +09:00
|
|
|
log.apply(null, data);
|
2012-05-15 09:19:09 +09:00
|
|
|
}];
|
2012-08-31 22:51:31 +09:00
|
|
|
// If there's no console available, console_error in the
|
|
|
|
// action handler will do nothing.
|
2012-08-31 20:40:37 +09:00
|
|
|
if ('console' in globalScope) {
|
|
|
|
ah['console_error'] = [function ahConsoleError(data) {
|
|
|
|
globalScope['console'].error.apply(null, data);
|
|
|
|
}];
|
|
|
|
} else {
|
|
|
|
ah['console_error'] = [function ahConsoleError(data) {
|
2012-10-05 00:01:53 +09:00
|
|
|
log.apply(null, data);
|
2012-08-31 20:40:37 +09:00
|
|
|
}];
|
|
|
|
}
|
2012-05-15 09:19:09 +09:00
|
|
|
ah['_warn'] = [function ah_Warn(data) {
|
|
|
|
warn(data);
|
2011-10-26 02:16:20 +09:00
|
|
|
}];
|
2011-12-09 07:51:26 +09:00
|
|
|
|
|
|
|
comObj.onmessage = function messageHandlerComObjOnMessage(event) {
|
|
|
|
var data = event.data;
|
|
|
|
if (data.isReply) {
|
2011-12-08 08:36:27 +09:00
|
|
|
var callbackId = data.callbackId;
|
2011-12-09 07:51:26 +09:00
|
|
|
if (data.callbackId in callbacks) {
|
|
|
|
var callback = callbacks[callbackId];
|
|
|
|
delete callbacks[callbackId];
|
2011-12-08 08:36:27 +09:00
|
|
|
callback(data.data);
|
|
|
|
} else {
|
2012-01-25 05:04:59 +09:00
|
|
|
error('Cannot resolve callback ' + callbackId);
|
2011-12-08 08:36:27 +09:00
|
|
|
}
|
2011-12-09 07:51:26 +09:00
|
|
|
} else if (data.action in ah) {
|
2011-10-26 02:16:20 +09:00
|
|
|
var action = ah[data.action];
|
2011-12-15 07:02:00 +09:00
|
|
|
if (data.callbackId) {
|
|
|
|
var promise = new Promise();
|
|
|
|
promise.then(function(resolvedData) {
|
|
|
|
comObj.postMessage({
|
|
|
|
isReply: true,
|
|
|
|
callbackId: data.callbackId,
|
|
|
|
data: resolvedData
|
|
|
|
});
|
|
|
|
});
|
|
|
|
action[0].call(action[1], data.data, promise);
|
|
|
|
} else {
|
|
|
|
action[0].call(action[1], data.data);
|
|
|
|
}
|
2011-10-26 02:16:20 +09:00
|
|
|
} else {
|
2012-01-25 05:04:59 +09:00
|
|
|
error('Unkown action from worker: ' + data.action);
|
2011-10-26 02:16:20 +09:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
MessageHandler.prototype = {
|
2011-10-29 03:23:30 +09:00
|
|
|
on: function messageHandlerOn(actionName, handler, scope) {
|
2011-10-26 02:16:20 +09:00
|
|
|
var ah = this.actionHandler;
|
|
|
|
if (ah[actionName]) {
|
2012-01-25 05:04:59 +09:00
|
|
|
error('There is already an actionName called "' + actionName + '"');
|
2011-10-26 02:16:20 +09:00
|
|
|
}
|
|
|
|
ah[actionName] = [handler, scope];
|
|
|
|
},
|
2011-12-13 04:38:16 +09:00
|
|
|
/**
|
|
|
|
* Sends a message to the comObj to invoke the action with the supplied data.
|
|
|
|
* @param {String} actionName Action to call.
|
|
|
|
* @param {JSON} data JSON data to send.
|
|
|
|
* @param {function} [callback] Optional callback that will handle a reply.
|
|
|
|
*/
|
2011-12-08 08:36:27 +09:00
|
|
|
send: function messageHandlerSend(actionName, data, callback) {
|
|
|
|
var message = {
|
2011-10-26 02:16:20 +09:00
|
|
|
action: actionName,
|
|
|
|
data: data
|
2011-12-08 08:36:27 +09:00
|
|
|
};
|
|
|
|
if (callback) {
|
|
|
|
var callbackId = this.callbackIndex++;
|
|
|
|
this.callbacks[callbackId] = callback;
|
|
|
|
message.callbackId = callbackId;
|
|
|
|
}
|
|
|
|
this.comObj.postMessage(message);
|
2011-10-26 02:16:20 +09:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2011-11-03 23:30:53 +09:00
|
|
|
var WorkerMessageHandler = {
|
2011-10-29 03:23:30 +09:00
|
|
|
setup: function wphSetup(handler) {
|
2012-02-21 22:10:44 +09:00
|
|
|
var pdfModel = null;
|
2011-10-09 17:37:53 +09:00
|
|
|
|
2012-06-24 05:49:17 +09:00
|
|
|
function loadDocument(pdfData, pdfModelSource) {
|
2011-10-08 22:19:48 +09:00
|
|
|
// Create only the model of the PDFDoc, which is enough for
|
|
|
|
// processing the content of the pdf.
|
2012-06-24 04:48:33 +09:00
|
|
|
var pdfPassword = pdfModelSource.password;
|
2012-05-15 03:45:07 +09:00
|
|
|
try {
|
|
|
|
pdfModel = new PDFDocument(new Stream(pdfData), pdfPassword);
|
|
|
|
} catch (e) {
|
|
|
|
if (e instanceof PasswordException) {
|
|
|
|
if (e.code === 'needpassword') {
|
|
|
|
handler.send('NeedPassword', {
|
|
|
|
exception: e
|
|
|
|
});
|
|
|
|
} else if (e.code === 'incorrectpassword') {
|
|
|
|
handler.send('IncorrectPassword', {
|
|
|
|
exception: e
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2012-10-16 19:10:37 +09:00
|
|
|
return;
|
|
|
|
} else if (e instanceof InvalidPDFException) {
|
|
|
|
handler.send('InvalidPDF', {
|
|
|
|
exception: e
|
|
|
|
});
|
|
|
|
|
2013-01-30 03:13:28 +09:00
|
|
|
return;
|
|
|
|
} else if (e instanceof MissingPDFException) {
|
|
|
|
handler.send('MissingPDF', {
|
|
|
|
exception: e
|
|
|
|
});
|
|
|
|
|
2012-05-15 03:45:07 +09:00
|
|
|
return;
|
|
|
|
} else {
|
2012-10-16 19:10:37 +09:00
|
|
|
handler.send('UnknownError', {
|
|
|
|
exception: new UnknownErrorException(e.message, e.toString())
|
|
|
|
});
|
|
|
|
|
|
|
|
return;
|
2012-05-15 03:45:07 +09:00
|
|
|
}
|
|
|
|
}
|
2012-04-12 07:52:15 +09:00
|
|
|
var doc = {
|
|
|
|
numPages: pdfModel.numPages,
|
2012-04-13 07:14:18 +09:00
|
|
|
fingerprint: pdfModel.getFingerprint(),
|
2012-04-12 07:52:15 +09:00
|
|
|
destinations: pdfModel.catalog.destinations,
|
|
|
|
outline: pdfModel.catalog.documentOutline,
|
2012-04-13 07:56:17 +09:00
|
|
|
info: pdfModel.getDocumentInfo(),
|
2012-05-15 03:45:07 +09:00
|
|
|
metadata: pdfModel.catalog.metadata,
|
|
|
|
encrypted: !!pdfModel.xref.encrypt
|
2012-04-12 07:52:15 +09:00
|
|
|
};
|
2012-04-13 04:11:22 +09:00
|
|
|
handler.send('GetDoc', {pdfInfo: doc});
|
2012-06-24 04:48:33 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
handler.on('test', function wphSetupTest(data) {
|
2012-12-12 02:07:58 +09:00
|
|
|
// check if Uint8Array can be sent to worker
|
|
|
|
if (!(data instanceof Uint8Array)) {
|
|
|
|
handler.send('test', false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// check if the response property is supported by xhr
|
|
|
|
var xhr = new XMLHttpRequest();
|
|
|
|
if (!('response' in xhr || 'mozResponse' in xhr ||
|
|
|
|
'responseArrayBuffer' in xhr || 'mozResponseArrayBuffer' in xhr)) {
|
|
|
|
handler.send('test', false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
handler.send('test', true);
|
2012-06-24 04:48:33 +09:00
|
|
|
});
|
|
|
|
|
|
|
|
handler.on('GetDocRequest', function wphSetupDoc(data) {
|
|
|
|
var source = data.source;
|
|
|
|
if (source.data) {
|
|
|
|
// the data is array, instantiating directly from it
|
2012-06-24 05:49:17 +09:00
|
|
|
loadDocument(source.data, source);
|
2012-06-24 04:48:33 +09:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
PDFJS.getPdf(
|
|
|
|
{
|
|
|
|
url: source.url,
|
|
|
|
progress: function getPDFProgress(evt) {
|
2012-08-23 00:48:56 +09:00
|
|
|
handler.send('DocProgress', {
|
|
|
|
loaded: evt.loaded,
|
|
|
|
total: evt.lengthComputable ? evt.total : void(0)
|
|
|
|
});
|
2012-06-24 04:48:33 +09:00
|
|
|
},
|
|
|
|
error: function getPDFError(e) {
|
2013-01-30 03:13:28 +09:00
|
|
|
if (e.target.status == 404) {
|
|
|
|
handler.send('MissingPDF', {
|
|
|
|
exception: new MissingPDFException(
|
|
|
|
'Missing PDF \"' + source.url + '\".')});
|
|
|
|
} else {
|
|
|
|
handler.send('DocError', 'Unexpected server response (' +
|
|
|
|
e.target.status + ') while retrieving PDF \"' +
|
|
|
|
source.url + '\".');
|
|
|
|
}
|
2012-06-24 04:48:33 +09:00
|
|
|
},
|
|
|
|
headers: source.httpHeaders
|
|
|
|
},
|
|
|
|
function getPDFLoad(data) {
|
2012-06-24 05:49:17 +09:00
|
|
|
loadDocument(data, source);
|
2012-06-24 04:48:33 +09:00
|
|
|
});
|
2012-04-12 07:52:15 +09:00
|
|
|
});
|
|
|
|
|
2012-04-15 05:54:31 +09:00
|
|
|
handler.on('GetPageRequest', function wphSetupGetPage(data) {
|
2012-04-13 02:01:07 +09:00
|
|
|
var pageNumber = data.pageIndex + 1;
|
|
|
|
var pdfPage = pdfModel.getPage(pageNumber);
|
2012-04-12 07:52:15 +09:00
|
|
|
var page = {
|
2012-04-13 02:01:07 +09:00
|
|
|
pageIndex: data.pageIndex,
|
2012-04-12 07:52:15 +09:00
|
|
|
rotate: pdfPage.rotate,
|
|
|
|
ref: pdfPage.ref,
|
2012-04-15 05:54:31 +09:00
|
|
|
view: pdfPage.view
|
2012-04-12 07:52:15 +09:00
|
|
|
};
|
2012-04-13 04:11:22 +09:00
|
|
|
handler.send('GetPage', {pageInfo: page});
|
2011-09-08 02:16:02 +09:00
|
|
|
});
|
2011-10-09 17:37:53 +09:00
|
|
|
|
2012-06-02 06:17:09 +09:00
|
|
|
handler.on('GetData', function wphSetupGetData(data, promise) {
|
|
|
|
promise.resolve(pdfModel.stream.bytes);
|
|
|
|
});
|
|
|
|
|
2012-04-15 05:54:31 +09:00
|
|
|
handler.on('GetAnnotationsRequest', function wphSetupGetAnnotations(data) {
|
|
|
|
var pdfPage = pdfModel.getPage(data.pageIndex + 1);
|
|
|
|
handler.send('GetAnnotations', {
|
|
|
|
pageIndex: data.pageIndex,
|
|
|
|
annotations: pdfPage.getAnnotations()
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
handler.on('RenderPageRequest', function wphSetupRenderPage(data) {
|
2012-04-13 02:01:07 +09:00
|
|
|
var pageNum = data.pageIndex + 1;
|
2011-09-08 02:16:02 +09:00
|
|
|
|
2011-10-09 17:37:53 +09:00
|
|
|
// The following code does quite the same as
|
|
|
|
// Page.prototype.startRendering, but stops at one point and sends the
|
|
|
|
// result back to the main thread.
|
2011-09-08 02:16:02 +09:00
|
|
|
var gfx = new CanvasGraphics(null);
|
|
|
|
|
2011-09-10 03:20:55 +09:00
|
|
|
var start = Date.now();
|
2011-09-30 22:38:54 +09:00
|
|
|
|
|
|
|
var dependency = [];
|
2012-03-13 02:41:40 +09:00
|
|
|
var operatorList = null;
|
2011-11-29 09:55:09 +09:00
|
|
|
try {
|
2012-02-21 22:10:44 +09:00
|
|
|
var page = pdfModel.getPage(pageNum);
|
2011-11-29 09:55:09 +09:00
|
|
|
// Pre compile the pdf page and fetch the fonts/images.
|
2012-03-13 02:41:40 +09:00
|
|
|
operatorList = page.getOperatorList(handler, dependency);
|
2011-11-30 04:32:01 +09:00
|
|
|
} catch (e) {
|
2012-01-20 04:19:19 +09:00
|
|
|
var minimumStackMessage =
|
2012-03-13 02:41:40 +09:00
|
|
|
'worker.js: while trying to getPage() and getOperatorList()';
|
2012-01-20 04:19:19 +09:00
|
|
|
|
2011-11-29 09:55:09 +09:00
|
|
|
// Turn the error into an obj that can be serialized
|
2012-01-20 04:19:19 +09:00
|
|
|
if (typeof e === 'string') {
|
|
|
|
e = {
|
|
|
|
message: e,
|
|
|
|
stack: minimumStackMessage
|
|
|
|
};
|
|
|
|
} else if (typeof e === 'object') {
|
|
|
|
e = {
|
|
|
|
message: e.message || e.toString(),
|
|
|
|
stack: e.stack || minimumStackMessage
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
e = {
|
|
|
|
message: 'Unknown exception type: ' + (typeof e),
|
|
|
|
stack: minimumStackMessage
|
2012-01-21 08:44:51 +09:00
|
|
|
};
|
2012-01-20 04:19:19 +09:00
|
|
|
}
|
|
|
|
|
2012-04-13 04:11:22 +09:00
|
|
|
handler.send('PageError', {
|
2011-11-29 09:55:09 +09:00
|
|
|
pageNum: pageNum,
|
|
|
|
error: e
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
2011-09-08 02:16:02 +09:00
|
|
|
|
2012-08-31 20:37:44 +09:00
|
|
|
log('page=%d - getOperatorList: time=%dms, len=%d', pageNum,
|
2012-03-13 02:41:40 +09:00
|
|
|
Date.now() - start, operatorList.fnArray.length);
|
2011-09-08 02:16:02 +09:00
|
|
|
|
2011-09-30 22:38:54 +09:00
|
|
|
// Filter the dependecies for fonts.
|
2011-10-09 05:22:30 +09:00
|
|
|
var fonts = {};
|
2011-11-03 04:11:33 +09:00
|
|
|
for (var i = 0, ii = dependency.length; i < ii; i++) {
|
2011-09-30 22:38:54 +09:00
|
|
|
var dep = dependency[i];
|
2012-10-29 05:10:34 +09:00
|
|
|
if (dep.indexOf('g_font_') == 0) {
|
2011-10-09 05:22:30 +09:00
|
|
|
fonts[dep] = true;
|
2011-09-30 22:38:54 +09:00
|
|
|
}
|
|
|
|
}
|
2012-04-13 04:11:22 +09:00
|
|
|
handler.send('RenderPage', {
|
2012-04-13 02:01:07 +09:00
|
|
|
pageIndex: data.pageIndex,
|
2012-03-13 02:41:40 +09:00
|
|
|
operatorList: operatorList,
|
2011-10-12 02:51:35 +09:00
|
|
|
depFonts: Object.keys(fonts)
|
2011-09-08 02:16:02 +09:00
|
|
|
});
|
|
|
|
}, this);
|
2011-10-09 17:37:53 +09:00
|
|
|
|
2012-04-21 08:49:08 +09:00
|
|
|
handler.on('GetTextContent', function wphExtractText(data, promise) {
|
|
|
|
var pageNum = data.pageIndex + 1;
|
2011-12-12 09:14:52 +09:00
|
|
|
var start = Date.now();
|
|
|
|
|
2012-04-09 08:18:43 +09:00
|
|
|
var textContent = '';
|
|
|
|
try {
|
|
|
|
var page = pdfModel.getPage(pageNum);
|
|
|
|
textContent = page.extractTextContent();
|
2012-04-21 08:49:08 +09:00
|
|
|
promise.resolve(textContent);
|
2012-04-09 08:18:43 +09:00
|
|
|
} catch (e) {
|
|
|
|
// Skip errored pages
|
2012-04-21 08:49:08 +09:00
|
|
|
promise.reject(e);
|
2011-12-12 09:14:52 +09:00
|
|
|
}
|
2011-12-11 08:24:54 +09:00
|
|
|
|
2012-08-31 20:37:44 +09:00
|
|
|
log('text indexing: page=%d - time=%dms',
|
2012-04-09 08:18:43 +09:00
|
|
|
pageNum, Date.now() - start);
|
2011-12-11 08:24:54 +09:00
|
|
|
});
|
2011-09-08 02:16:02 +09:00
|
|
|
}
|
2011-10-09 17:37:53 +09:00
|
|
|
};
|
2011-10-26 02:16:20 +09:00
|
|
|
|
2011-10-26 07:43:41 +09:00
|
|
|
var consoleTimer = {};
|
|
|
|
|
2011-10-26 02:16:20 +09:00
|
|
|
var workerConsole = {
|
|
|
|
log: function log() {
|
|
|
|
var args = Array.prototype.slice.call(arguments);
|
2012-08-31 20:40:37 +09:00
|
|
|
globalScope.postMessage({
|
2011-10-26 02:16:20 +09:00
|
|
|
action: 'console_log',
|
|
|
|
data: args
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
error: function error() {
|
|
|
|
var args = Array.prototype.slice.call(arguments);
|
2012-08-31 20:40:37 +09:00
|
|
|
globalScope.postMessage({
|
2011-10-26 02:16:20 +09:00
|
|
|
action: 'console_error',
|
|
|
|
data: args
|
|
|
|
});
|
2012-01-30 05:25:06 +09:00
|
|
|
throw 'pdf.js execution error';
|
2011-10-26 02:16:20 +09:00
|
|
|
},
|
|
|
|
|
2011-10-29 03:23:30 +09:00
|
|
|
time: function time(name) {
|
2011-10-26 02:16:20 +09:00
|
|
|
consoleTimer[name] = Date.now();
|
|
|
|
},
|
|
|
|
|
2011-10-29 03:23:30 +09:00
|
|
|
timeEnd: function timeEnd(name) {
|
2011-10-26 02:16:20 +09:00
|
|
|
var time = consoleTimer[name];
|
|
|
|
if (time == null) {
|
2012-01-25 05:04:59 +09:00
|
|
|
error('Unkown timer name ' + name);
|
2011-10-26 02:16:20 +09:00
|
|
|
}
|
|
|
|
this.log('Timer:', name, Date.now() - time);
|
|
|
|
}
|
2011-10-26 02:43:28 +09:00
|
|
|
};
|
2011-10-26 07:43:41 +09:00
|
|
|
|
|
|
|
// Worker thread?
|
|
|
|
if (typeof window === 'undefined') {
|
2011-11-02 03:32:20 +09:00
|
|
|
globalScope.console = workerConsole;
|
2011-10-26 07:43:41 +09:00
|
|
|
|
2012-05-15 09:19:09 +09:00
|
|
|
// Add a logger so we can pass warnings on to the main thread, errors will
|
|
|
|
// throw an exception which will be forwarded on automatically.
|
|
|
|
PDFJS.LogManager.addLogger({
|
|
|
|
warn: function(msg) {
|
2012-08-31 20:40:37 +09:00
|
|
|
globalScope.postMessage({
|
2012-05-15 09:19:09 +09:00
|
|
|
action: '_warn',
|
|
|
|
data: msg
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2011-10-28 21:32:36 +09:00
|
|
|
var handler = new MessageHandler('worker_processor', this);
|
2011-11-03 23:30:53 +09:00
|
|
|
WorkerMessageHandler.setup(handler);
|
2011-10-26 07:43:41 +09:00
|
|
|
}
|