pdf.js/worker.js

276 lines
7.2 KiB
JavaScript
Raw Normal View History

2011-09-06 08:07:03 +09:00
/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* vim: set shiftwidth=2 tabstop=2 autoindent cindent expandtab: */
'use strict';
// Set this to true if you want to use workers.
2011-10-09 05:35:35 +09:00
var useWorker = false;
2011-09-06 08:07:03 +09:00
var WorkerPage = (function() {
2011-10-03 05:17:32 +09:00
function constructor(workerPDF, page, objs) {
2011-09-06 08:07:03 +09:00
this.workerPDF = workerPDF;
this.page = page;
2011-10-03 05:17:32 +09:00
this.objs = objs;
2011-10-09 17:37:53 +09:00
2011-09-06 08:07:03 +09:00
this.ref = page.ref;
}
2011-10-09 17:37:53 +09:00
2011-09-06 08:07:03 +09:00
constructor.prototype = {
get width() {
return this.page.width;
},
2011-10-09 17:37:53 +09:00
2011-09-06 08:07:03 +09:00
get height() {
return this.page.height;
},
2011-10-09 17:37:53 +09:00
2011-09-06 08:07:03 +09:00
get stats() {
return this.page.stats;
},
2011-10-09 17:37:53 +09:00
2011-10-08 21:18:23 +09:00
get view() {
return this.page.view;
},
2011-09-06 08:07:03 +09:00
startRendering: function(ctx, callback, errback) {
2011-09-06 09:42:58 +09:00
this.ctx = ctx;
this.callback = callback;
2011-09-06 08:07:03 +09:00
// TODO: Place the worker magic HERE.
2011-09-06 09:42:58 +09:00
// this.page.startRendering(ctx, callback, errback);
2011-10-09 17:37:53 +09:00
this.startRenderingTime = Date.now();
2011-10-09 17:37:53 +09:00
this.workerPDF.startRendering(this);
2011-09-06 09:42:58 +09:00
},
2011-10-09 17:37:53 +09:00
startRenderingFromIRQueue: function(IRQueue, fonts) {
2011-10-03 05:17:32 +09:00
var gfx = new CanvasGraphics(this.ctx, this.objs);
2011-10-09 17:37:53 +09:00
var startTime = Date.now();
var callback = function(err) {
var pageNum = this.page.pageNumber + 1;
2011-10-09 17:37:53 +09:00
console.log('page=%d - rendering time: time=%dms',
pageNum, Date.now() - startTime);
2011-10-09 17:37:53 +09:00
console.log('page=%d - total time: time=%dms',
pageNum, Date.now() - this.startRenderingTime);
this.callback(err);
}.bind(this);
this.page.startRenderingFromIRQueue(gfx, IRQueue, fonts, callback);
2011-09-06 08:07:03 +09:00
},
2011-10-09 17:37:53 +09:00
2011-09-06 08:07:03 +09:00
getLinks: function() {
return this.page.getLinks();
}
};
2011-10-09 17:37:53 +09:00
2011-09-06 08:07:03 +09:00
return constructor;
})();
2011-10-09 00:08:17 +09:00
/**
* A PDF document and page is build up of many objects. E.g. there are objects
* for fonts, images, rendering code and such. These objects might get processed
2011-10-09 17:37:53 +09:00
* inside of a worker. The `PDFObjects` implements some basic functions to
* manage these objects.
2011-10-09 00:08:17 +09:00
*/
var PDFObjects = (function() {
function PDFObjects() {
this.objs = {};
}
PDFObjects.prototype = {
objs: null,
/**
2011-10-09 00:08:17 +09:00
* Internal function.
* Ensures there is an object defined for `objId`. Stores `data` on the
* object *if* it is created.
*/
ensureObj: function(objId, data) {
if (!this.objs[objId]) {
return this.objs[objId] = new Promise(objId, data);
} else {
return this.objs[objId];
}
},
/**
* If called *without* callback, this returns the data of `objId` but the
* object needs to be resolved. If it isn't, this function throws.
*
* If called *with* a callback, the callback is called with the data of the
2011-10-09 17:37:53 +09:00
* object once the object is resolved. That means, if you call this
* function and the object is already resolved, the callback gets called
* right away.
*/
get: function(objId, callback) {
2011-10-09 17:37:53 +09:00
// If there is a callback, then the get can be async and the object is
// not required to be resolved right now
if (callback) {
this.ensureObj(objId).then(callback);
2011-10-09 17:37:53 +09:00
}
// If there isn't a callback, the user expects to get the resolved data
// directly.
else {
var obj = this.objs[objId];
// If there isn't an object yet or the object isn't resolved, then the
// data isn't ready yet!
if (!obj || !obj.isResolved) {
2011-10-09 17:37:53 +09:00
throw 'Requesting object that isn\'t resolved yet ' + objId;
}
// Direct access.
else {
return obj.data;
}
}
},
/**
* Resolves the object `objId` with optional `data`.
*/
resolve: function(objId, data) {
var objs = this.objs;
2011-10-09 17:37:53 +09:00
// In case there is a promise already on this object, just resolve it.
if (objs[objId]) {
objs[objId].resolve(data);
} else {
this.ensureObj(objId, data);
}
},
onData: function(objId, callback) {
this.ensureObj(objId).onData(callback);
},
isResolved: function(objId) {
var objs = this.objs;
if (!objs[objId]) {
return false;
} else {
return objs[objId].isResolved;
}
},
hasData: function(objId) {
var objs = this.objs;
if (!objs[objId]) {
return false;
} else {
return objs[objId].hasData;
}
},
/**
* Sets the data of an object but *doesn't* resolve it.
*/
setData: function(objId, data) {
// Watchout! If you call `this.ensureObj(objId, data)` you'll gone create
// a *resolved* promise which shouldn't be the case!
this.ensureObj(objId).data = data;
}
2011-10-09 17:37:53 +09:00
};
return PDFObjects;
})();
/**
2011-10-09 17:37:53 +09:00
* 'Promise' object.
2011-10-09 00:08:17 +09:00
* Each object that is stored in PDFObjects is based on a Promise object that
* contains the status of the object and the data. There migth be situations,
* where a function want to use the value of an object, but it isn't ready at
* that time. To get a notification, once the object is ready to be used, s.o.
* can add a callback using the `then` method on the promise that then calls
* the callback once the object gets resolved.
* A promise can get resolved only once and only once the data of the promise
* can be set. If any of these happens twice or the data is required before
* it was set, an exception is throw.
*/
2011-09-08 09:15:35 +09:00
var Promise = (function() {
var EMPTY_PROMISE = {};
/**
* If `data` is passed in this constructor, the promise is created resolved.
* If there isn't data, it isn't resolved at the beginning.
*/
function Promise(name, data) {
this.name = name;
// If you build a promise and pass in some data it's already resolved.
if (data != null) {
this.isResolved = true;
2011-10-02 00:16:11 +09:00
this._data = data;
2011-10-01 18:44:01 +09:00
this.hasData = true;
} else {
2011-10-09 17:37:53 +09:00
this.isResolved = false;
2011-10-02 00:16:11 +09:00
this._data = EMPTY_PROMISE;
}
this.callbacks = [];
2011-09-08 09:15:35 +09:00
};
2011-10-09 17:37:53 +09:00
2011-09-08 09:15:35 +09:00
Promise.prototype = {
hasData: false,
set data(data) {
if (data === undefined) {
return;
}
2011-10-02 00:16:11 +09:00
if (this._data !== EMPTY_PROMISE) {
2011-10-09 17:37:53 +09:00
throw 'Promise ' + this.name +
': Cannot set the data of a promise twice';
}
2011-10-02 00:16:11 +09:00
this._data = data;
this.hasData = true;
2011-10-02 00:16:11 +09:00
if (this.onDataCallback) {
this.onDataCallback(data);
}
},
2011-10-09 17:37:53 +09:00
get data() {
2011-10-02 00:16:11 +09:00
if (this._data === EMPTY_PROMISE) {
2011-10-09 17:37:53 +09:00
throw 'Promise ' + this.name + ': Cannot get data that isn\'t set';
}
2011-10-02 00:16:11 +09:00
return this._data;
},
onData: function(callback) {
2011-10-02 00:16:11 +09:00
if (this._data !== EMPTY_PROMISE) {
callback(this._data);
} else {
2011-10-02 00:16:11 +09:00
this.onDataCallback = callback;
}
},
2011-10-09 17:37:53 +09:00
2011-09-08 09:15:35 +09:00
resolve: function(data) {
if (this.isResolved) {
2011-10-09 17:37:53 +09:00
throw 'A Promise can be resolved only once ' + this.name;
2011-09-08 09:15:35 +09:00
}
2011-09-08 09:15:35 +09:00
this.isResolved = true;
this.data = data;
var callbacks = this.callbacks;
2011-10-09 17:37:53 +09:00
2011-09-08 09:15:35 +09:00
for (var i = 0; i < callbacks.length; i++) {
callbacks[i].call(null, data);
}
},
2011-10-09 17:37:53 +09:00
2011-09-08 09:15:35 +09:00
then: function(callback) {
if (!callback) {
2011-10-09 17:37:53 +09:00
throw 'Requiring callback' + this.name;
}
2011-10-09 17:37:53 +09:00
2011-09-08 09:15:35 +09:00
// If the promise is already resolved, call the callback directly.
if (this.isResolved) {
var data = this.data;
callback.call(null, data);
2011-09-08 09:15:35 +09:00
} else {
2011-10-09 17:37:53 +09:00
this.callbacks.push(callback);
2011-09-08 09:15:35 +09:00
}
}
2011-10-09 17:37:53 +09:00
};
return Promise;
2011-09-08 09:15:35 +09:00
})();
2011-09-06 08:07:03 +09:00