206 lines
5.5 KiB
JavaScript
206 lines
5.5 KiB
JavaScript
/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set shiftwidth=2 tabstop=2 autoindent cindent expandtab: */
|
|
|
|
'use strict';
|
|
|
|
function MessageHandler(name, comObj) {
|
|
this.name = name;
|
|
this.comObj = comObj;
|
|
var ah = this.actionHandler = {};
|
|
|
|
ah['console_log'] = [function ahConsoleLog(data) {
|
|
console.log.apply(console, data);
|
|
}];
|
|
ah['console_error'] = [function ahConsoleError(data) {
|
|
console.error.apply(console, data);
|
|
}];
|
|
comObj.onmessage = function messageHandlerComObjOnMessage(event) {
|
|
var data = event.data;
|
|
if (data.action in ah) {
|
|
var action = ah[data.action];
|
|
action[0].call(action[1], data.data);
|
|
} else {
|
|
throw 'Unkown action from worker: ' + data.action;
|
|
}
|
|
};
|
|
}
|
|
|
|
MessageHandler.prototype = {
|
|
on: function messageHandlerOn(actionName, handler, scope) {
|
|
var ah = this.actionHandler;
|
|
if (ah[actionName]) {
|
|
throw 'There is already an actionName called "' + actionName + '"';
|
|
}
|
|
ah[actionName] = [handler, scope];
|
|
},
|
|
|
|
send: function messageHandlerSend(actionName, data) {
|
|
this.comObj.postMessage({
|
|
action: actionName,
|
|
data: data
|
|
});
|
|
}
|
|
};
|
|
|
|
var WorkerMessageHandler = {
|
|
setup: function wphSetup(handler) {
|
|
var pdfDoc = null;
|
|
|
|
handler.on('test', function wphSetupTest(data) {
|
|
handler.send('test', data instanceof Uint8Array);
|
|
});
|
|
|
|
handler.on('workerSrc', function wphSetupWorkerSrc(data) {
|
|
// In development, the `workerSrc` message is handled in the
|
|
// `worker_loader.js` file. In production the workerProcessHandler is
|
|
// called for this. This servers as a dummy to prevent calling an
|
|
// undefined action `workerSrc`.
|
|
});
|
|
|
|
handler.on('doc', function wphSetupDoc(data) {
|
|
// Create only the model of the PDFDoc, which is enough for
|
|
// processing the content of the pdf.
|
|
pdfDoc = new PDFDocModel(new Stream(data));
|
|
});
|
|
|
|
handler.on('page_request', function wphSetupPageRequest(pageNum) {
|
|
pageNum = parseInt(pageNum);
|
|
|
|
|
|
// 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.
|
|
var gfx = new CanvasGraphics(null);
|
|
|
|
var start = Date.now();
|
|
|
|
var dependency = [];
|
|
var IRQueue = null;
|
|
try {
|
|
var page = pdfDoc.getPage(pageNum);
|
|
// Pre compile the pdf page and fetch the fonts/images.
|
|
IRQueue = page.getIRQueue(handler, dependency);
|
|
} catch (e) {
|
|
// Turn the error into an obj that can be serialized
|
|
e = {
|
|
message: typeof e === 'object' ? e.message : e,
|
|
stack: typeof e === 'object' ? e.stack : null
|
|
};
|
|
handler.send('page_error', {
|
|
pageNum: pageNum,
|
|
error: e
|
|
});
|
|
return;
|
|
}
|
|
|
|
console.log('page=%d - getIRQueue: time=%dms, len=%d', pageNum,
|
|
Date.now() - start, IRQueue.fnArray.length);
|
|
|
|
// Filter the dependecies for fonts.
|
|
var fonts = {};
|
|
for (var i = 0, ii = dependency.length; i < ii; i++) {
|
|
var dep = dependency[i];
|
|
if (dep.indexOf('font_') == 0) {
|
|
fonts[dep] = true;
|
|
}
|
|
}
|
|
|
|
handler.send('page', {
|
|
pageNum: pageNum,
|
|
IRQueue: IRQueue,
|
|
depFonts: Object.keys(fonts)
|
|
});
|
|
}, this);
|
|
|
|
handler.on('font', function wphSetupFont(data) {
|
|
var objId = data[0];
|
|
var name = data[1];
|
|
var file = data[2];
|
|
var properties = data[3];
|
|
|
|
var font = {
|
|
name: name,
|
|
file: file,
|
|
properties: properties
|
|
};
|
|
|
|
// Some fonts don't have a file, e.g. the build in ones like Arial.
|
|
if (file) {
|
|
var fontFileDict = new Dict();
|
|
fontFileDict.map = file.dict.map;
|
|
|
|
var fontFile = new Stream(file.bytes, file.start,
|
|
file.end - file.start, fontFileDict);
|
|
|
|
// Check if this is a FlateStream. Otherwise just use the created
|
|
// Stream one. This makes complex_ttf_font.pdf work.
|
|
var cmf = file.bytes[0];
|
|
if ((cmf & 0x0f) == 0x08) {
|
|
font.file = new FlateStream(fontFile);
|
|
} else {
|
|
font.file = fontFile;
|
|
}
|
|
}
|
|
|
|
var obj = new Font(font.name, font.file, font.properties);
|
|
|
|
var str = '';
|
|
var objData = obj.data;
|
|
if (objData) {
|
|
var length = objData.length;
|
|
for (var j = 0; j < length; ++j)
|
|
str += String.fromCharCode(objData[j]);
|
|
}
|
|
|
|
obj.str = str;
|
|
|
|
// Remove the data array form the font object, as it's not needed
|
|
// anymore as we sent over the ready str.
|
|
delete obj.data;
|
|
|
|
handler.send('font_ready', [objId, obj]);
|
|
});
|
|
}
|
|
};
|
|
|
|
var consoleTimer = {};
|
|
|
|
var workerConsole = {
|
|
log: function log() {
|
|
var args = Array.prototype.slice.call(arguments);
|
|
postMessage({
|
|
action: 'console_log',
|
|
data: args
|
|
});
|
|
},
|
|
|
|
error: function error() {
|
|
var args = Array.prototype.slice.call(arguments);
|
|
postMessage({
|
|
action: 'console_error',
|
|
data: args
|
|
});
|
|
},
|
|
|
|
time: function time(name) {
|
|
consoleTimer[name] = Date.now();
|
|
},
|
|
|
|
timeEnd: function timeEnd(name) {
|
|
var time = consoleTimer[name];
|
|
if (time == null) {
|
|
throw 'Unkown timer name ' + name;
|
|
}
|
|
this.log('Timer:', name, Date.now() - time);
|
|
}
|
|
};
|
|
|
|
// Worker thread?
|
|
if (typeof window === 'undefined') {
|
|
globalScope.console = workerConsole;
|
|
|
|
var handler = new MessageHandler('worker_processor', this);
|
|
WorkerMessageHandler.setup(handler);
|
|
}
|
|
|