Merge pull request #1840 from yurydelendik/loadpdf-1
Moves loading of the binary PDF data to the worker
This commit is contained in:
commit
b3a603c199
74
src/api.js
74
src/api.js
@ -18,63 +18,26 @@
|
|||||||
* @return {Promise} A promise that is resolved with {PDFDocumentProxy} object.
|
* @return {Promise} A promise that is resolved with {PDFDocumentProxy} object.
|
||||||
*/
|
*/
|
||||||
PDFJS.getDocument = function getDocument(source) {
|
PDFJS.getDocument = function getDocument(source) {
|
||||||
var url, data, headers, password, parameters = {}, workerInitializedPromise,
|
var workerInitializedPromise, workerReadyPromise, transport;
|
||||||
workerReadyPromise, transport;
|
|
||||||
|
|
||||||
if (typeof source === 'string') {
|
if (typeof source === 'string') {
|
||||||
url = source;
|
source = { url: source };
|
||||||
} else if (isArrayBuffer(source)) {
|
} else if (isArrayBuffer(source)) {
|
||||||
data = source;
|
source = { data: source };
|
||||||
} else if (typeof source === 'object') {
|
} else if (typeof source !== 'object') {
|
||||||
url = source.url;
|
|
||||||
data = source.data;
|
|
||||||
headers = source.httpHeaders;
|
|
||||||
password = source.password;
|
|
||||||
parameters.password = password || null;
|
|
||||||
|
|
||||||
if (!url && !data)
|
|
||||||
error('Invalid parameter array, need either .data or .url');
|
|
||||||
} else {
|
|
||||||
error('Invalid parameter in getDocument, need either Uint8Array, ' +
|
error('Invalid parameter in getDocument, need either Uint8Array, ' +
|
||||||
'string or a parameter object');
|
'string or a parameter object');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!source.url && !source.data)
|
||||||
|
error('Invalid parameter array, need either .data or .url');
|
||||||
|
|
||||||
workerInitializedPromise = new PDFJS.Promise();
|
workerInitializedPromise = new PDFJS.Promise();
|
||||||
workerReadyPromise = new PDFJS.Promise();
|
workerReadyPromise = new PDFJS.Promise();
|
||||||
transport = new WorkerTransport(workerInitializedPromise, workerReadyPromise);
|
transport = new WorkerTransport(workerInitializedPromise, workerReadyPromise);
|
||||||
if (data) {
|
workerInitializedPromise.then(function transportInitialized() {
|
||||||
// assuming the data is array, instantiating directly from it
|
transport.fetchDocument(source);
|
||||||
transport.sendData(data, parameters);
|
});
|
||||||
} else if (url) {
|
|
||||||
// fetch url
|
|
||||||
PDFJS.getPdf(
|
|
||||||
{
|
|
||||||
url: url,
|
|
||||||
progress: function getPDFProgress(evt) {
|
|
||||||
if (evt.lengthComputable) {
|
|
||||||
workerReadyPromise.progress({
|
|
||||||
loaded: evt.loaded,
|
|
||||||
total: evt.total
|
|
||||||
});
|
|
||||||
}
|
|
||||||
},
|
|
||||||
error: function getPDFError(e) {
|
|
||||||
workerReadyPromise.reject('Unexpected server response of ' +
|
|
||||||
e.target.status + '.');
|
|
||||||
},
|
|
||||||
headers: headers
|
|
||||||
},
|
|
||||||
function getPDFLoad(data) {
|
|
||||||
// sometimes the pdf has finished downloading before the web worker-test
|
|
||||||
// has finished. In that case the rendering of the final pdf would cause
|
|
||||||
// errors. We have to wait for the WorkerTransport to finalize worker-
|
|
||||||
// support detection
|
|
||||||
workerInitializedPromise.then(function workerInitialized() {
|
|
||||||
transport.sendData(data, parameters);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return workerReadyPromise;
|
return workerReadyPromise;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -601,6 +564,17 @@ var WorkerTransport = (function WorkerTransportClosure() {
|
|||||||
}
|
}
|
||||||
}, this);
|
}, this);
|
||||||
|
|
||||||
|
messageHandler.on('DocProgress', function transportDocProgress(data) {
|
||||||
|
this.workerReadyPromise.progress({
|
||||||
|
loaded: data.loaded,
|
||||||
|
total: data.total
|
||||||
|
});
|
||||||
|
}, this);
|
||||||
|
|
||||||
|
messageHandler.on('DocError', function transportDocError(data) {
|
||||||
|
this.workerReadyPromise.reject(data);
|
||||||
|
}, this);
|
||||||
|
|
||||||
messageHandler.on('PageError', function transportError(data) {
|
messageHandler.on('PageError', function transportError(data) {
|
||||||
var page = this.pageCache[data.pageNum - 1];
|
var page = this.pageCache[data.pageNum - 1];
|
||||||
if (page.displayReadyPromise)
|
if (page.displayReadyPromise)
|
||||||
@ -645,11 +619,11 @@ var WorkerTransport = (function WorkerTransportClosure() {
|
|||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
sendData: function WorkerTransport_sendData(data, params) {
|
fetchDocument: function WorkerTransport_fetchDocument(source) {
|
||||||
this.messageHandler.send('GetDocRequest', {data: data, params: params});
|
this.messageHandler.send('GetDocRequest', {source: source});
|
||||||
},
|
},
|
||||||
|
|
||||||
getData: function WorkerTransport_sendData(promise) {
|
getData: function WorkerTransport_getData(promise) {
|
||||||
this.messageHandler.send('GetData', null, function(data) {
|
this.messageHandler.send('GetData', null, function(data) {
|
||||||
promise.resolve(data);
|
promise.resolve(data);
|
||||||
});
|
});
|
||||||
|
@ -45,8 +45,8 @@ function getPdf(arg, callback) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
xhr.mozResponseType = xhr.responseType = 'arraybuffer';
|
xhr.mozResponseType = xhr.responseType = 'arraybuffer';
|
||||||
var protocol = params.url.indexOf(':') < 0 ? window.location.protocol :
|
|
||||||
params.url.substring(0, params.url.indexOf(':') + 1);
|
var protocol = params.url.substring(0, params.url.indexOf(':') + 1);
|
||||||
xhr.expected = (protocol === 'http:' || protocol === 'https:') ? 200 : 0;
|
xhr.expected = (protocol === 'http:' || protocol === 'https:') ? 200 : 0;
|
||||||
|
|
||||||
if ('progress' in params)
|
if ('progress' in params)
|
||||||
|
23
src/util.js
23
src/util.js
@ -57,6 +57,29 @@ function assert(cond, msg) {
|
|||||||
error(msg);
|
error(msg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Combines two URLs. The baseUrl shall be absolute URL. If the url is an
|
||||||
|
// absolute URL, it will be returned as is.
|
||||||
|
function combineUrl(baseUrl, url) {
|
||||||
|
if (url.indexOf(':') >= 0)
|
||||||
|
return url;
|
||||||
|
if (url.charAt(0) == '/') {
|
||||||
|
// absolute path
|
||||||
|
var i = baseUrl.indexOf('://');
|
||||||
|
i = baseUrl.indexOf('/', i + 3);
|
||||||
|
return baseUrl.substring(0, i) + url;
|
||||||
|
} else {
|
||||||
|
// relative path
|
||||||
|
var pathLength = baseUrl.length, i;
|
||||||
|
i = baseUrl.lastIndexOf('#');
|
||||||
|
pathLength = i >= 0 ? i : pathLength;
|
||||||
|
i = baseUrl.lastIndexOf('?', pathLength);
|
||||||
|
pathLength = i >= 0 ? i : pathLength;
|
||||||
|
var prefixLength = baseUrl.lastIndexOf('/', pathLength);
|
||||||
|
return baseUrl.substring(0, prefixLength + 1) + url;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
PDFJS.combineUrl = combineUrl;
|
||||||
|
|
||||||
// In a well-formed PDF, |cond| holds. If it doesn't, subsequent
|
// In a well-formed PDF, |cond| holds. If it doesn't, subsequent
|
||||||
// behavior is undefined.
|
// behavior is undefined.
|
||||||
function assertWellFormed(cond, msg) {
|
function assertWellFormed(cond, msg) {
|
||||||
|
@ -84,15 +84,10 @@ var WorkerMessageHandler = {
|
|||||||
setup: function wphSetup(handler) {
|
setup: function wphSetup(handler) {
|
||||||
var pdfModel = null;
|
var pdfModel = null;
|
||||||
|
|
||||||
handler.on('test', function wphSetupTest(data) {
|
function loadDocument(pdfData, pdfModelSource) {
|
||||||
handler.send('test', data instanceof Uint8Array);
|
|
||||||
});
|
|
||||||
|
|
||||||
handler.on('GetDocRequest', function wphSetupDoc(data) {
|
|
||||||
// Create only the model of the PDFDoc, which is enough for
|
// Create only the model of the PDFDoc, which is enough for
|
||||||
// processing the content of the pdf.
|
// processing the content of the pdf.
|
||||||
var pdfData = data.data;
|
var pdfPassword = pdfModelSource.password;
|
||||||
var pdfPassword = data.params.password;
|
|
||||||
try {
|
try {
|
||||||
pdfModel = new PDFDocument(new Stream(pdfData), pdfPassword);
|
pdfModel = new PDFDocument(new Stream(pdfData), pdfPassword);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
@ -122,6 +117,40 @@ var WorkerMessageHandler = {
|
|||||||
encrypted: !!pdfModel.xref.encrypt
|
encrypted: !!pdfModel.xref.encrypt
|
||||||
};
|
};
|
||||||
handler.send('GetDoc', {pdfInfo: doc});
|
handler.send('GetDoc', {pdfInfo: doc});
|
||||||
|
}
|
||||||
|
|
||||||
|
handler.on('test', function wphSetupTest(data) {
|
||||||
|
handler.send('test', data instanceof Uint8Array);
|
||||||
|
});
|
||||||
|
|
||||||
|
handler.on('GetDocRequest', function wphSetupDoc(data) {
|
||||||
|
var source = data.source;
|
||||||
|
if (source.data) {
|
||||||
|
// the data is array, instantiating directly from it
|
||||||
|
loadDocument(source.data, source);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
PDFJS.getPdf(
|
||||||
|
{
|
||||||
|
url: source.url,
|
||||||
|
progress: function getPDFProgress(evt) {
|
||||||
|
if (evt.lengthComputable) {
|
||||||
|
handler.send('DocProgress', {
|
||||||
|
loaded: evt.loaded,
|
||||||
|
total: evt.total
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
error: function getPDFError(e) {
|
||||||
|
handler.send('DocError', 'Unexpected server response of ' +
|
||||||
|
e.target.status + '.');
|
||||||
|
},
|
||||||
|
headers: source.httpHeaders
|
||||||
|
},
|
||||||
|
function getPDFLoad(data) {
|
||||||
|
loadDocument(data, source);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
handler.on('GetPageRequest', function wphSetupGetPage(data) {
|
handler.on('GetPageRequest', function wphSetupGetPage(data) {
|
||||||
|
@ -98,7 +98,8 @@ function nextTask() {
|
|||||||
|
|
||||||
log('Loading file "' + task.file + '"\n');
|
log('Loading file "' + task.file + '"\n');
|
||||||
|
|
||||||
getPdf(task.file, function nextTaskGetPdf(data) {
|
var absoluteUrl = combineUrl(window.location.href, task.file);
|
||||||
|
getPdf(absoluteUrl, function nextTaskGetPdf(data) {
|
||||||
var failure;
|
var failure;
|
||||||
function continuation() {
|
function continuation() {
|
||||||
task.pageNum = task.firstPage || 1;
|
task.pageNum = task.firstPage || 1;
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
describe('api', function() {
|
describe('api', function() {
|
||||||
// TODO run with worker enabled
|
// TODO run with worker enabled
|
||||||
PDFJS.disableWorker = true;
|
PDFJS.disableWorker = true;
|
||||||
var basicApiUrl = '../pdfs/basicapi.pdf';
|
var basicApiUrl = combineUrl(window.location.href, '../pdfs/basicapi.pdf');
|
||||||
function waitsForPromise(promise) {
|
function waitsForPromise(promise) {
|
||||||
waitsFor(function() {
|
waitsFor(function() {
|
||||||
return promise.isResolved || promise.isRejected;
|
return promise.isResolved || promise.isRejected;
|
||||||
|
@ -43,6 +43,7 @@
|
|||||||
<script type="text/javascript" src="stream_spec.js"></script>
|
<script type="text/javascript" src="stream_spec.js"></script>
|
||||||
<script type="text/javascript" src="api_spec.js"></script>
|
<script type="text/javascript" src="api_spec.js"></script>
|
||||||
<script type="text/javascript" src="metadata_spec.js"></script>
|
<script type="text/javascript" src="metadata_spec.js"></script>
|
||||||
|
<script type="text/javascript" src="util_spec.js"></script>
|
||||||
<script type="text/javascript">
|
<script type="text/javascript">
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
|
51
test/unit/util_spec.js
Normal file
51
test/unit/util_spec.js
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
||||||
|
/* vim: set shiftwidth=2 tabstop=2 autoindent cindent expandtab: */
|
||||||
|
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
describe('util', function() {
|
||||||
|
|
||||||
|
describe('combineUrl', function() {
|
||||||
|
it('absolute url with protocol stays as is', function() {
|
||||||
|
var baseUrl = 'http://server/index.html';
|
||||||
|
var url = 'http://server2/test2.html';
|
||||||
|
var result = combineUrl(baseUrl, url);
|
||||||
|
var expected = 'http://server2/test2.html';
|
||||||
|
expect(result).toEqual(expected);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('absolute url without protocol uses prefix from base', function() {
|
||||||
|
var baseUrl = 'http://server/index.html';
|
||||||
|
var url = '/test2.html';
|
||||||
|
var result = combineUrl(baseUrl, url);
|
||||||
|
var expected = 'http://server/test2.html';
|
||||||
|
expect(result).toEqual(expected);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('combines relative url with base', function() {
|
||||||
|
var baseUrl = 'http://server/index.html';
|
||||||
|
var url = 'test2.html';
|
||||||
|
var result = combineUrl(baseUrl, url);
|
||||||
|
var expected = 'http://server/test2.html';
|
||||||
|
expect(result).toEqual(expected);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('combines relative url (w/hash) with base', function() {
|
||||||
|
var baseUrl = 'http://server/index.html#!/test';
|
||||||
|
var url = 'test2.html';
|
||||||
|
var result = combineUrl(baseUrl, url);
|
||||||
|
var expected = 'http://server/test2.html';
|
||||||
|
expect(result).toEqual(expected);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('combines relative url (w/query) with base', function() {
|
||||||
|
var baseUrl = 'http://server/index.html?search=/test';
|
||||||
|
var url = 'test2.html';
|
||||||
|
var result = combineUrl(baseUrl, url);
|
||||||
|
var expected = 'http://server/test2.html';
|
||||||
|
expect(result).toEqual(expected);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
|
|
@ -395,7 +395,7 @@ var PDFView = {
|
|||||||
if (typeof url === 'string') { // URL
|
if (typeof url === 'string') { // URL
|
||||||
this.url = url;
|
this.url = url;
|
||||||
document.title = decodeURIComponent(getFileName(url)) || url;
|
document.title = decodeURIComponent(getFileName(url)) || url;
|
||||||
parameters.url = url;
|
parameters.url = PDFJS.combineUrl(window.location.href, url);
|
||||||
} else if (url && 'byteLength' in url) { // ArrayBuffer
|
} else if (url && 'byteLength' in url) { // ArrayBuffer
|
||||||
parameters.data = url;
|
parameters.data = url;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user