Merge remote-tracking branch 'upstream/master'
This commit is contained in:
commit
aea65f8d7d
31
external/jpgjs/LICENSE
vendored
31
external/jpgjs/LICENSE
vendored
@ -1,24 +1,17 @@
|
|||||||
|
|
||||||
Copyright (C) 2011 by notmasteryet
|
Copyright (C) 2011 notmasteryet
|
||||||
|
|
||||||
Contributors: Yury Delendik
|
Contributors: Yury Delendik <ydelendik@mozilla.com>
|
||||||
Brendan Dahl <bdahl@mozilla.com>
|
Brendan Dahl <bdahl@mozilla.com>
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
a copy of this software and associated documentation files (the
|
you may not use this file except in compliance with the License.
|
||||||
"Software"), to deal in the Software without restriction, including
|
You may obtain a copy of the License at
|
||||||
without limitation the rights to use, copy, modify, merge, publish,
|
|
||||||
distribute, sublicense, and/or sell copies of the Software, and to
|
|
||||||
permit persons to whom the Software is furnished to do so, subject to
|
|
||||||
the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
included in all copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
Unless required by applicable law or agreed to in writing, software
|
||||||
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
See the License for the specific language governing permissions and
|
||||||
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
limitations under the License.
|
||||||
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
||||||
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
||||||
|
27
external/jpgjs/jpg.js
vendored
27
external/jpgjs/jpg.js
vendored
@ -1,5 +1,20 @@
|
|||||||
/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- /
|
/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- /
|
||||||
/* vim: set shiftwidth=2 tabstop=2 autoindent cindent expandtab: */
|
/* vim: set shiftwidth=2 tabstop=2 autoindent cindent expandtab: */
|
||||||
|
/*
|
||||||
|
Copyright 2011 notmasteryet
|
||||||
|
|
||||||
|
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.
|
||||||
|
*/
|
||||||
|
|
||||||
// - The JPEG specification can be found in the ITU CCITT Recommendation T.81
|
// - The JPEG specification can be found in the ITU CCITT Recommendation T.81
|
||||||
// (www.w3.org/Graphics/JPEG/itu-t81.pdf)
|
// (www.w3.org/Graphics/JPEG/itu-t81.pdf)
|
||||||
@ -627,8 +642,9 @@ var JpegImage = (function jpegImage() {
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case 0xFFDB: // DQT (Define Quantization Tables)
|
case 0xFFDB: // DQT (Define Quantization Tables)
|
||||||
var quantizationTableCount = Math.floor((readUint16() - 2) / 65);
|
var quantizationTablesLength = readUint16();
|
||||||
for (i = 0; i < quantizationTableCount; i++) {
|
var quantizationTablesEnd = quantizationTablesLength + offset - 2;
|
||||||
|
while (offset < quantizationTablesEnd) {
|
||||||
var quantizationTableSpec = data[offset++];
|
var quantizationTableSpec = data[offset++];
|
||||||
var tableData = new Int32Array(64);
|
var tableData = new Int32Array(64);
|
||||||
if ((quantizationTableSpec >> 4) === 0) { // 8 bit values
|
if ((quantizationTableSpec >> 4) === 0) { // 8 bit values
|
||||||
@ -721,6 +737,13 @@ var JpegImage = (function jpegImage() {
|
|||||||
offset += processed;
|
offset += processed;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
if (data[offset - 3] == 0xFF &&
|
||||||
|
data[offset - 2] >= 0xC0 && data[offset - 2] <= 0xFE) {
|
||||||
|
// could be incorrect encoding -- last 0xFF byte of the previous
|
||||||
|
// block was eaten by the encoder
|
||||||
|
offset -= 3;
|
||||||
|
break;
|
||||||
|
}
|
||||||
throw "unknown JPEG marker " + fileMarker.toString(16);
|
throw "unknown JPEG marker " + fileMarker.toString(16);
|
||||||
}
|
}
|
||||||
fileMarker = readUint16();
|
fileMarker = readUint16();
|
||||||
|
68
make.js
68
make.js
@ -6,6 +6,7 @@ var ROOT_DIR = __dirname + '/', // absolute path to project's root
|
|||||||
BUILD_DIR = 'build/',
|
BUILD_DIR = 'build/',
|
||||||
BUILD_TARGET = BUILD_DIR + 'pdf.js',
|
BUILD_TARGET = BUILD_DIR + 'pdf.js',
|
||||||
FIREFOX_BUILD_DIR = BUILD_DIR + '/firefox/',
|
FIREFOX_BUILD_DIR = BUILD_DIR + '/firefox/',
|
||||||
|
CHROME_BUILD_DIR = BUILD_DIR + '/chrome/',
|
||||||
EXTENSION_SRC_DIR = 'extensions/',
|
EXTENSION_SRC_DIR = 'extensions/',
|
||||||
LOCALE_SRC_DIR = 'l10n/',
|
LOCALE_SRC_DIR = 'l10n/',
|
||||||
GH_PAGES_DIR = BUILD_DIR + 'gh-pages/',
|
GH_PAGES_DIR = BUILD_DIR + 'gh-pages/',
|
||||||
@ -107,6 +108,8 @@ target.web = function() {
|
|||||||
cp('-R', GENERIC_DIR + '/*', GH_PAGES_DIR);
|
cp('-R', GENERIC_DIR + '/*', GH_PAGES_DIR);
|
||||||
cp(FIREFOX_BUILD_DIR + '/*.xpi', FIREFOX_BUILD_DIR + '/*.rdf',
|
cp(FIREFOX_BUILD_DIR + '/*.xpi', FIREFOX_BUILD_DIR + '/*.rdf',
|
||||||
GH_PAGES_DIR + EXTENSION_SRC_DIR + 'firefox/');
|
GH_PAGES_DIR + EXTENSION_SRC_DIR + 'firefox/');
|
||||||
|
cp(CHROME_BUILD_DIR + '/*.crx', FIREFOX_BUILD_DIR + '/*.rdf',
|
||||||
|
GH_PAGES_DIR + EXTENSION_SRC_DIR + 'chrome/');
|
||||||
cp('web/index.html.template', GH_PAGES_DIR + '/index.html');
|
cp('web/index.html.template', GH_PAGES_DIR + '/index.html');
|
||||||
|
|
||||||
cd(GH_PAGES_DIR);
|
cd(GH_PAGES_DIR);
|
||||||
@ -258,6 +261,7 @@ target.pagesrepo = function() {
|
|||||||
mkdir('-p', GH_PAGES_DIR + '/web/images');
|
mkdir('-p', GH_PAGES_DIR + '/web/images');
|
||||||
mkdir('-p', GH_PAGES_DIR + BUILD_DIR);
|
mkdir('-p', GH_PAGES_DIR + BUILD_DIR);
|
||||||
mkdir('-p', GH_PAGES_DIR + EXTENSION_SRC_DIR + '/firefox');
|
mkdir('-p', GH_PAGES_DIR + EXTENSION_SRC_DIR + '/firefox');
|
||||||
|
mkdir('-p', GH_PAGES_DIR + EXTENSION_SRC_DIR + '/chrome');
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -574,6 +578,70 @@ target.chrome = function() {
|
|||||||
]
|
]
|
||||||
};
|
};
|
||||||
builder.build(setup);
|
builder.build(setup);
|
||||||
|
|
||||||
|
// Bundle the files to a Chrome extension file .crx if path to key is set
|
||||||
|
var pem = env['PDFJS_CHROME_KEY'];
|
||||||
|
if (!pem) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
echo();
|
||||||
|
echo('### Bundling .crx extension into ' + CHROME_BUILD_DIR);
|
||||||
|
|
||||||
|
if (!test('-f', pem)) {
|
||||||
|
echo('Incorrect PDFJS_CHROME_KEY path');
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
var browserManifest = env['PDF_BROWSERS'] ||
|
||||||
|
'test/resources/browser_manifests/browser_manifest.json';
|
||||||
|
|
||||||
|
if (!test('-f', browserManifest)) {
|
||||||
|
echo('Browser manifest file ' + browserManifest + ' does not exist.');
|
||||||
|
echo('Try copying one of the examples in test/resources/browser_manifests');
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
var manifest = JSON.parse(cat(browserManifest));
|
||||||
|
} catch (e) {
|
||||||
|
echo('Malformed browser manifest file');
|
||||||
|
echo(e.message);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
var executable;
|
||||||
|
manifest.forEach(function(browser) {
|
||||||
|
if (browser.name === 'chrome') {
|
||||||
|
executable = browser.path;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// If there was no chrome entry in the browser manifest, exit
|
||||||
|
if (!executable) {
|
||||||
|
echo('There was no \'chrome\' entry in the browser manifest');
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we're on a Darwin (Mac) OS, then let's check for an .app path
|
||||||
|
if (process.platform === 'darwin' && executable.indexOf('.app') !== -1) {
|
||||||
|
executable = executable + '/Contents/MacOS/Google Chrome';
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the chrome executable doesn't exist
|
||||||
|
if (!test('-f', executable)) {
|
||||||
|
echo('Incorrect executable path to chrome');
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Let chrome pack the extension for us
|
||||||
|
exec('"' + executable + '"' +
|
||||||
|
' --no-message-box' +
|
||||||
|
' "--pack-extension=' + ROOT_DIR + CHROME_BUILD_DIR + '"' +
|
||||||
|
' "--pack-extension-key=' + pem + '"');
|
||||||
|
|
||||||
|
// Rename to pdf.js.crx
|
||||||
|
mv(BUILD_DIR + 'chrome.crx', CHROME_BUILD_DIR + 'pdf.js.crx');
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
17
src/api.js
17
src/api.js
@ -311,13 +311,19 @@ var PDFPageProxy = (function PDFPageProxyClosure() {
|
|||||||
ensureFonts: function PDFPageProxy_ensureFonts(fonts, callback) {
|
ensureFonts: function PDFPageProxy_ensureFonts(fonts, callback) {
|
||||||
this.stats.time('Font Loading');
|
this.stats.time('Font Loading');
|
||||||
// Convert the font names to the corresponding font obj.
|
// Convert the font names to the corresponding font obj.
|
||||||
|
var fontObjs = [];
|
||||||
for (var i = 0, ii = fonts.length; i < ii; i++) {
|
for (var i = 0, ii = fonts.length; i < ii; i++) {
|
||||||
fonts[i] = this.objs.objs[fonts[i]].data;
|
var obj = this.objs.objs[fonts[i]].data;
|
||||||
|
if (obj.error) {
|
||||||
|
warn('Error during font loading: ' + obj.error);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
fontObjs.push(obj);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Load all the fonts
|
// Load all the fonts
|
||||||
FontLoader.bind(
|
FontLoader.bind(
|
||||||
fonts,
|
fontObjs,
|
||||||
function pageEnsureFontsFontObjs(fontObjs) {
|
function pageEnsureFontsFontObjs(fontObjs) {
|
||||||
this.stats.timeEnd('Font Loading');
|
this.stats.timeEnd('Font Loading');
|
||||||
|
|
||||||
@ -565,7 +571,12 @@ var WorkerTransport = (function WorkerTransportClosure() {
|
|||||||
|
|
||||||
// At this point, only the font object is created but the font is
|
// At this point, only the font object is created but the font is
|
||||||
// not yet attached to the DOM. This is done in `FontLoader.bind`.
|
// not yet attached to the DOM. This is done in `FontLoader.bind`.
|
||||||
var font = new Font(name, file, properties);
|
var font;
|
||||||
|
try {
|
||||||
|
font = new Font(name, file, properties);
|
||||||
|
} catch (e) {
|
||||||
|
font = new ErrorFont(e);
|
||||||
|
}
|
||||||
this.objs.resolve(id, font);
|
this.objs.resolve(id, font);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
166
src/fonts.js
166
src/fonts.js
@ -404,28 +404,15 @@ function mapPrivateUseChars(code) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var FontLoader = {
|
var FontLoader = {
|
||||||
listeningForFontLoad: false,
|
loadingContext: {
|
||||||
|
requests: [],
|
||||||
|
nextRequestId: 0
|
||||||
|
},
|
||||||
|
|
||||||
bind: function fontLoaderBind(fonts, callback) {
|
bind: function fontLoaderBind(fonts, callback) {
|
||||||
function checkFontsLoaded() {
|
assert(!isWorker, 'bind() shall be called from main thread');
|
||||||
for (var i = 0, ii = fonts.length; i < ii; i++) {
|
|
||||||
var fontObj = fonts[i];
|
|
||||||
if (fontObj.loading) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
document.documentElement.removeEventListener(
|
|
||||||
'pdfjsFontLoad', checkFontsLoaded, false);
|
|
||||||
|
|
||||||
// Use timeout to fix chrome intermittent failures on font loading.
|
|
||||||
setTimeout(callback, 0);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
var rules = [], names = [], fontsToLoad = [];
|
|
||||||
var fontCreateTimer = 0;
|
|
||||||
|
|
||||||
|
var rules = [], fontsToLoad = [];
|
||||||
for (var i = 0, ii = fonts.length; i < ii; i++) {
|
for (var i = 0, ii = fonts.length; i < ii; i++) {
|
||||||
var font = fonts[i];
|
var font = fonts[i];
|
||||||
|
|
||||||
@ -436,8 +423,6 @@ var FontLoader = {
|
|||||||
}
|
}
|
||||||
font.attached = true;
|
font.attached = true;
|
||||||
|
|
||||||
fontsToLoad.push(font);
|
|
||||||
|
|
||||||
var str = '';
|
var str = '';
|
||||||
var data = font.data;
|
var data = font.data;
|
||||||
if (data) {
|
if (data) {
|
||||||
@ -448,28 +433,51 @@ var FontLoader = {
|
|||||||
var rule = font.bindDOM(str);
|
var rule = font.bindDOM(str);
|
||||||
if (rule) {
|
if (rule) {
|
||||||
rules.push(rule);
|
rules.push(rule);
|
||||||
names.push(font.loadedName);
|
fontsToLoad.push(font);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this.listeningForFontLoad = false;
|
var request = FontLoader.queueLoadingCallback(callback);
|
||||||
if (!isWorker && rules.length) {
|
if (rules.length > 0) {
|
||||||
FontLoader.prepareFontLoadEvent(rules, names, fontsToLoad);
|
FontLoader.prepareFontLoadEvent(rules, fontsToLoad, request);
|
||||||
}
|
} else {
|
||||||
|
request.complete();
|
||||||
if (!checkFontsLoaded()) {
|
|
||||||
document.documentElement.addEventListener(
|
|
||||||
'pdfjsFontLoad', checkFontsLoaded, false);
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
queueLoadingCallback: function FontLoader_queueLoadingCallback(callback) {
|
||||||
|
function LoadLoader_completeRequest() {
|
||||||
|
assert(!request.end, 'completeRequest() cannot be called twice');
|
||||||
|
request.end = Date.now();
|
||||||
|
|
||||||
|
// sending all completed requests in order how they were queued
|
||||||
|
while (context.requests.length > 0 && context.requests[0].end) {
|
||||||
|
var otherRequest = context.requests.shift();
|
||||||
|
setTimeout(otherRequest.callback, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var context = FontLoader.loadingContext;
|
||||||
|
var requestId = 'pdfjs-font-loading-' + (context.nextRequestId++);
|
||||||
|
var request = {
|
||||||
|
id: requestId,
|
||||||
|
complete: LoadLoader_completeRequest,
|
||||||
|
callback: callback,
|
||||||
|
started: Date.now()
|
||||||
|
};
|
||||||
|
context.requests.push(request);
|
||||||
|
return request;
|
||||||
|
},
|
||||||
|
|
||||||
// Set things up so that at least one pdfjsFontLoad event is
|
// Set things up so that at least one pdfjsFontLoad event is
|
||||||
// dispatched when all the @font-face |rules| for |names| have been
|
// dispatched when all the @font-face |rules| for |fonts| have been
|
||||||
// loaded in a subdocument. It's expected that the load of |rules|
|
// loaded in a subdocument. It's expected that the load of |rules|
|
||||||
// has already started in this (outer) document, so that they should
|
// has already started in this (outer) document, so that they should
|
||||||
// be ordered before the load in the subdocument.
|
// be ordered before the load in the subdocument.
|
||||||
prepareFontLoadEvent: function fontLoaderPrepareFontLoadEvent(rules, names,
|
prepareFontLoadEvent: function fontLoaderPrepareFontLoadEvent(rules,
|
||||||
fonts) {
|
fonts,
|
||||||
|
request) {
|
||||||
/** Hack begin */
|
/** Hack begin */
|
||||||
// There's no event when a font has finished downloading so the
|
// There's no event when a font has finished downloading so the
|
||||||
// following code is a dirty hack to 'guess' when a font is
|
// following code is a dirty hack to 'guess' when a font is
|
||||||
@ -493,6 +501,20 @@ var FontLoader = {
|
|||||||
// The postMessage() hackery was added to work around chrome bug
|
// The postMessage() hackery was added to work around chrome bug
|
||||||
// 82402.
|
// 82402.
|
||||||
|
|
||||||
|
var requestId = request.id;
|
||||||
|
// Validate the requestId parameter -- the value used to construct HTML.
|
||||||
|
if (!/^[\w\-]+$/.test(requestId)) {
|
||||||
|
error('Invalid request id: ' + requestId);
|
||||||
|
|
||||||
|
// Normally the error-function throws. But if a malicious code
|
||||||
|
// intercepts the function call then the return is needed.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var names = [];
|
||||||
|
for (var i = 0, ii = fonts.length; i < ii; i++)
|
||||||
|
names.push(fonts[i].loadedName);
|
||||||
|
|
||||||
// Validate the names parameter -- the values can used to construct HTML.
|
// Validate the names parameter -- the values can used to construct HTML.
|
||||||
if (!/^\w+$/.test(names.join(''))) {
|
if (!/^\w+$/.test(names.join(''))) {
|
||||||
error('Invalid font name(s): ' + names.join());
|
error('Invalid font name(s): ' + names.join());
|
||||||
@ -514,22 +536,21 @@ var FontLoader = {
|
|||||||
div.innerHTML = html;
|
div.innerHTML = html;
|
||||||
document.body.appendChild(div);
|
document.body.appendChild(div);
|
||||||
|
|
||||||
if (!this.listeningForFontLoad) {
|
window.addEventListener(
|
||||||
window.addEventListener(
|
'message',
|
||||||
'message',
|
function fontLoaderMessage(e) {
|
||||||
function fontLoaderMessage(e) {
|
if (e.data !== requestId)
|
||||||
var fontNames = JSON.parse(e.data);
|
return;
|
||||||
for (var i = 0, ii = fonts.length; i < ii; ++i) {
|
for (var i = 0, ii = fonts.length; i < ii; ++i) {
|
||||||
var font = fonts[i];
|
var font = fonts[i];
|
||||||
font.loading = false;
|
font.loading = false;
|
||||||
}
|
}
|
||||||
var evt = document.createEvent('Events');
|
request.complete();
|
||||||
evt.initEvent('pdfjsFontLoad', true, false);
|
// cleanup
|
||||||
document.documentElement.dispatchEvent(evt);
|
document.body.removeChild(frame);
|
||||||
},
|
window.removeEventListener('message', fontLoaderMessage, false);
|
||||||
false);
|
},
|
||||||
this.listeningForFontLoad = true;
|
false);
|
||||||
}
|
|
||||||
|
|
||||||
// XXX we should have a time-out here too, and maybe fire
|
// XXX we should have a time-out here too, and maybe fire
|
||||||
// pdfjsFontLoadFailed?
|
// pdfjsFontLoadFailed?
|
||||||
@ -540,13 +561,8 @@ var FontLoader = {
|
|||||||
}
|
}
|
||||||
src += '</style>';
|
src += '</style>';
|
||||||
src += '<script type="application/javascript">';
|
src += '<script type="application/javascript">';
|
||||||
var fontNamesArray = '';
|
|
||||||
for (var i = 0, ii = names.length; i < ii; ++i) {
|
|
||||||
fontNamesArray += '"' + names[i] + '", ';
|
|
||||||
}
|
|
||||||
src += ' var fontNames=[' + fontNamesArray + '];\n';
|
|
||||||
src += ' window.onload = function fontLoaderOnload() {\n';
|
src += ' window.onload = function fontLoaderOnload() {\n';
|
||||||
src += ' parent.postMessage(JSON.stringify(fontNames), "*");\n';
|
src += ' parent.postMessage("' + requestId + '", "*");\n';
|
||||||
src += ' }';
|
src += ' }';
|
||||||
// Hack so the end script tag isn't counted if this is inline JS.
|
// Hack so the end script tag isn't counted if this is inline JS.
|
||||||
src += '</scr' + 'ipt></head><body>';
|
src += '</scr' + 'ipt></head><body>';
|
||||||
@ -1561,13 +1577,19 @@ var Font = (function FontClosure() {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Some fonts might use wrong font types for Type1C or CIDFontType0C
|
||||||
|
var subtype = properties.subtype;
|
||||||
|
if (subtype == 'Type1C' && (type != 'Type1' && type != 'MMType1'))
|
||||||
|
type = 'Type1';
|
||||||
|
if (subtype == 'CIDFontType0C' && type != 'CIDFontType0')
|
||||||
|
type = 'CIDFontType0';
|
||||||
|
|
||||||
var data;
|
var data;
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case 'Type1':
|
case 'Type1':
|
||||||
case 'CIDFontType0':
|
case 'CIDFontType0':
|
||||||
this.mimetype = 'font/opentype';
|
this.mimetype = 'font/opentype';
|
||||||
|
|
||||||
var subtype = properties.subtype;
|
|
||||||
var cff = (subtype == 'Type1C' || subtype == 'CIDFontType0C') ?
|
var cff = (subtype == 'Type1C' || subtype == 'CIDFontType0C') ?
|
||||||
new CFFFont(file, properties) : new Type1Font(name, file, properties);
|
new CFFFont(file, properties) : new Type1Font(name, file, properties);
|
||||||
|
|
||||||
@ -2922,6 +2944,7 @@ var Font = (function FontClosure() {
|
|||||||
}
|
}
|
||||||
this.toFontChar = toFontChar;
|
this.toFontChar = toFontChar;
|
||||||
}
|
}
|
||||||
|
var unitsPerEm = properties.unitsPerEm || 1000; // defaulting to 1000
|
||||||
|
|
||||||
var fields = {
|
var fields = {
|
||||||
// PostScript Font Program
|
// PostScript Font Program
|
||||||
@ -2942,7 +2965,7 @@ var Font = (function FontClosure() {
|
|||||||
'\x00\x00\x00\x00' + // checksumAdjustement
|
'\x00\x00\x00\x00' + // checksumAdjustement
|
||||||
'\x5F\x0F\x3C\xF5' + // magicNumber
|
'\x5F\x0F\x3C\xF5' + // magicNumber
|
||||||
'\x00\x00' + // Flags
|
'\x00\x00' + // Flags
|
||||||
'\x03\xE8' + // unitsPerEM (defaulting to 1000)
|
safeString16(unitsPerEm) + // unitsPerEM
|
||||||
'\x00\x00\x00\x00\x9e\x0b\x7e\x27' + // creation date
|
'\x00\x00\x00\x00\x9e\x0b\x7e\x27' + // creation date
|
||||||
'\x00\x00\x00\x00\x9e\x0b\x7e\x27' + // modifification date
|
'\x00\x00\x00\x00\x9e\x0b\x7e\x27' + // modifification date
|
||||||
'\x00\x00' + // xMin
|
'\x00\x00' + // xMin
|
||||||
@ -3293,6 +3316,20 @@ var Font = (function FontClosure() {
|
|||||||
return Font;
|
return Font;
|
||||||
})();
|
})();
|
||||||
|
|
||||||
|
var ErrorFont = (function ErrorFontClosure() {
|
||||||
|
function ErrorFont(error) {
|
||||||
|
this.error = error;
|
||||||
|
}
|
||||||
|
|
||||||
|
ErrorFont.prototype = {
|
||||||
|
charsToGlyphs: function ErrorFont_charsToGlyphs() {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return ErrorFont;
|
||||||
|
})();
|
||||||
|
|
||||||
var CallothersubrCmd = (function CallothersubrCmdClosure() {
|
var CallothersubrCmd = (function CallothersubrCmdClosure() {
|
||||||
function CallothersubrCmd(index) {
|
function CallothersubrCmd(index) {
|
||||||
this.index = index;
|
this.index = index;
|
||||||
@ -4397,6 +4434,19 @@ var CFFParser = (function CFFParserClosure() {
|
|||||||
var charStringOffset = topDict.getByName('CharStrings');
|
var charStringOffset = topDict.getByName('CharStrings');
|
||||||
cff.charStrings = this.parseCharStrings(charStringOffset);
|
cff.charStrings = this.parseCharStrings(charStringOffset);
|
||||||
|
|
||||||
|
var fontMatrix = topDict.getByName('FontMatrix');
|
||||||
|
if (fontMatrix) {
|
||||||
|
// estimating unitsPerEM for the font
|
||||||
|
properties.unitsPerEm = 1 / fontMatrix[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
var fontBBox = topDict.getByName('FontBBox');
|
||||||
|
if (fontBBox) {
|
||||||
|
// adjusting ascent/descent
|
||||||
|
properties.ascent = fontBBox[3];
|
||||||
|
properties.descent = fontBBox[1];
|
||||||
|
}
|
||||||
|
|
||||||
var charset, encoding;
|
var charset, encoding;
|
||||||
if (cff.isCIDFont) {
|
if (cff.isCIDFont) {
|
||||||
var fdArrayIndex = this.parseIndex(topDict.getByName('FDArray')).obj;
|
var fdArrayIndex = this.parseIndex(topDict.getByName('FDArray')).obj;
|
||||||
@ -4470,7 +4520,7 @@ var CFFParser = (function CFFParserClosure() {
|
|||||||
return parseFloatOperand(pos);
|
return parseFloatOperand(pos);
|
||||||
} else if (value === 28) {
|
} else if (value === 28) {
|
||||||
value = dict[pos++];
|
value = dict[pos++];
|
||||||
value = (value << 8) | dict[pos++];
|
value = ((value << 24) | (dict[pos++] << 16)) >> 16;
|
||||||
return value;
|
return value;
|
||||||
} else if (value === 29) {
|
} else if (value === 29) {
|
||||||
value = dict[pos++];
|
value = dict[pos++];
|
||||||
|
67
src/image.js
67
src/image.js
@ -97,7 +97,12 @@ var PDFImage = (function PDFImageClosure() {
|
|||||||
if (smask) {
|
if (smask) {
|
||||||
this.smask = new PDFImage(xref, res, smask, false);
|
this.smask = new PDFImage(xref, res, smask, false);
|
||||||
} else if (mask) {
|
} else if (mask) {
|
||||||
this.mask = new PDFImage(xref, res, mask, false);
|
if (isStream(mask)) {
|
||||||
|
this.mask = new PDFImage(xref, res, mask, false);
|
||||||
|
} else {
|
||||||
|
// Color key mask (just an array).
|
||||||
|
this.mask = mask;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/**
|
/**
|
||||||
@ -129,11 +134,15 @@ var PDFImage = (function PDFImageClosure() {
|
|||||||
maskPromise.resolve(null);
|
maskPromise.resolve(null);
|
||||||
} else {
|
} else {
|
||||||
smaskPromise.resolve(null);
|
smaskPromise.resolve(null);
|
||||||
if (mask && isStream(mask)) {
|
if (mask) {
|
||||||
handleImageData(handler, xref, res, mask, maskPromise);
|
if (isStream(mask)) {
|
||||||
} else if (mask) {
|
handleImageData(handler, xref, res, mask, maskPromise);
|
||||||
TODO('handle color key masking');
|
} else if (isArray(mask)) {
|
||||||
maskPromise.resolve(null);
|
maskPromise.resolve(mask);
|
||||||
|
} else {
|
||||||
|
warn('Unsupported mask format.');
|
||||||
|
maskPromise.resolve(null);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
maskPromise.resolve(null);
|
maskPromise.resolve(null);
|
||||||
}
|
}
|
||||||
@ -279,7 +288,7 @@ var PDFImage = (function PDFImageClosure() {
|
|||||||
}
|
}
|
||||||
return output;
|
return output;
|
||||||
},
|
},
|
||||||
getOpacity: function PDFImage_getOpacity(width, height) {
|
getOpacity: function PDFImage_getOpacity(width, height, image) {
|
||||||
var smask = this.smask;
|
var smask = this.smask;
|
||||||
var mask = this.mask;
|
var mask = this.mask;
|
||||||
var originalWidth = this.width;
|
var originalWidth = this.width;
|
||||||
@ -294,18 +303,40 @@ var PDFImage = (function PDFImageClosure() {
|
|||||||
if (sw != width || sh != height)
|
if (sw != width || sh != height)
|
||||||
buf = PDFImage.resize(buf, smask.bpc, 1, sw, sh, width, height);
|
buf = PDFImage.resize(buf, smask.bpc, 1, sw, sh, width, height);
|
||||||
} else if (mask) {
|
} else if (mask) {
|
||||||
var sw = mask.width;
|
if (mask instanceof PDFImage) {
|
||||||
var sh = mask.height;
|
var sw = mask.width;
|
||||||
buf = new Uint8Array(sw * sh);
|
var sh = mask.height;
|
||||||
mask.numComps = 1;
|
buf = new Uint8Array(sw * sh);
|
||||||
mask.fillGrayBuffer(buf);
|
mask.numComps = 1;
|
||||||
|
mask.fillGrayBuffer(buf);
|
||||||
|
|
||||||
// Need to invert values in buffer
|
// Need to invert values in buffer
|
||||||
for (var i = 0, ii = sw * sh; i < ii; ++i)
|
for (var i = 0, ii = sw * sh; i < ii; ++i)
|
||||||
buf[i] = 255 - buf[i];
|
buf[i] = 255 - buf[i];
|
||||||
|
|
||||||
if (sw != width || sh != height)
|
if (sw != width || sh != height)
|
||||||
buf = PDFImage.resize(buf, mask.bpc, 1, sw, sh, width, height);
|
buf = PDFImage.resize(buf, mask.bpc, 1, sw, sh, width, height);
|
||||||
|
} else if (isArray(mask)) {
|
||||||
|
// Color key mask: if any of the compontents are outside the range
|
||||||
|
// then they should be painted.
|
||||||
|
buf = new Uint8Array(width * height);
|
||||||
|
var numComps = this.numComps;
|
||||||
|
for (var i = 0, ii = width * height; i < ii; ++i) {
|
||||||
|
var opacity = 0;
|
||||||
|
var imageOffset = i * numComps;
|
||||||
|
for (var j = 0; j < numComps; ++j) {
|
||||||
|
var color = image[imageOffset + j];
|
||||||
|
var maskOffset = j * 2;
|
||||||
|
if (color < mask[maskOffset] || color > mask[maskOffset + 1]) {
|
||||||
|
opacity = 255;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
buf[i] = opacity;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
error('Unknown mask format.');
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
buf = new Uint8Array(width * height);
|
buf = new Uint8Array(width * height);
|
||||||
for (var i = 0, ii = width * height; i < ii; ++i)
|
for (var i = 0, ii = width * height; i < ii; ++i)
|
||||||
@ -357,7 +388,7 @@ var PDFImage = (function PDFImageClosure() {
|
|||||||
comps = PDFImage.resize(comps, this.bpc, 3, originalWidth,
|
comps = PDFImage.resize(comps, this.bpc, 3, originalWidth,
|
||||||
originalHeight, width, height);
|
originalHeight, width, height);
|
||||||
var compsPos = 0;
|
var compsPos = 0;
|
||||||
var opacity = this.getOpacity(width, height);
|
var opacity = this.getOpacity(width, height, imgArray);
|
||||||
var opacityPos = 0;
|
var opacityPos = 0;
|
||||||
var length = width * actualHeight * 4;
|
var length = width * actualHeight * 4;
|
||||||
|
|
||||||
|
BIN
test/pdfs/colorkeymask.pdf
Normal file
BIN
test/pdfs/colorkeymask.pdf
Normal file
Binary file not shown.
@ -1 +1 @@
|
|||||||
http://www.freepatentsonline.com/pdf/documents/uspt/D661/USD661296/USD661296S1.pdf
|
http://assets.sbnation.com/assets/1167519/USD661296S1.pdf
|
||||||
|
@ -673,5 +673,11 @@
|
|||||||
"md5": "2ac7c68e26a8ef797aead15e4875cc6d",
|
"md5": "2ac7c68e26a8ef797aead15e4875cc6d",
|
||||||
"rounds": 1,
|
"rounds": 1,
|
||||||
"type": "load"
|
"type": "load"
|
||||||
|
},
|
||||||
|
{ "id": "colorkeymask",
|
||||||
|
"file": "pdfs/colorkeymask.pdf",
|
||||||
|
"md5": "9f11e815b485f7f0e1fa5c116c636cf9",
|
||||||
|
"rounds": 1,
|
||||||
|
"type": "eq"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
@ -42,7 +42,7 @@ describe('font', function() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
describe('CFFParser', function() {
|
describe('CFFParser', function() {
|
||||||
var parser = new CFFParser(fontData);
|
var parser = new CFFParser(fontData, {});
|
||||||
var cff = parser.parse();
|
var cff = parser.parse();
|
||||||
|
|
||||||
it('parses header', function() {
|
it('parses header', function() {
|
||||||
|
@ -29,29 +29,17 @@ select {
|
|||||||
|
|
||||||
#viewerContainer:-webkit-full-screen {
|
#viewerContainer:-webkit-full-screen {
|
||||||
top: 0px;
|
top: 0px;
|
||||||
padding-top: 6px;
|
border-top: 5px solid transparent;
|
||||||
padding-bottom: 24px;
|
|
||||||
background-color: #404040;
|
background-color: #404040;
|
||||||
background-image: url(images/texture.png);
|
background-image: url(images/texture.png);
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
overflow: auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
:-webkit-full-screen #viewer {
|
|
||||||
margin: 0pt;
|
|
||||||
padding: 0pt;
|
|
||||||
height: 100%;
|
|
||||||
width: 100%;
|
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
:-webkit-full-screen .page {
|
|
||||||
margin: 0px auto;
|
|
||||||
margin-bottom: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
#viewerContainer:-moz-full-screen {
|
#viewerContainer:-moz-full-screen {
|
||||||
|
top: 0px;
|
||||||
|
border-top: 5px solid transparent;
|
||||||
background-color: #404040;
|
background-color: #404040;
|
||||||
background-image: url(images/texture.png);
|
background-image: url(images/texture.png);
|
||||||
width: 100%;
|
width: 100%;
|
||||||
@ -59,6 +47,10 @@ select {
|
|||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
:-webkit-full-screen .page:last-child {
|
||||||
|
margin-bottom: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
:-moz-full-screen .page:last-child {
|
:-moz-full-screen .page:last-child {
|
||||||
margin-bottom: 40px;
|
margin-bottom: 40px;
|
||||||
}
|
}
|
||||||
|
@ -33,6 +33,19 @@ function getFileName(url) {
|
|||||||
return url.substring(url.lastIndexOf('/', end) + 1, end);
|
return url.substring(url.lastIndexOf('/', end) + 1, end);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function scrollIntoView(element, spot) {
|
||||||
|
var parent = element.offsetParent, offsetY = element.offsetTop;
|
||||||
|
while (parent.clientHeight == parent.scrollHeight) {
|
||||||
|
offsetY += parent.offsetTop;
|
||||||
|
parent = parent.offsetParent;
|
||||||
|
if (!parent)
|
||||||
|
return; // no need to scroll
|
||||||
|
}
|
||||||
|
if (spot)
|
||||||
|
offsetY += spot.top;
|
||||||
|
parent.scrollTop = offsetY;
|
||||||
|
}
|
||||||
|
|
||||||
var Cache = function cacheCache(size) {
|
var Cache = function cacheCache(size) {
|
||||||
var data = [];
|
var data = [];
|
||||||
this.push = function cachePush(view) {
|
this.push = function cachePush(view) {
|
||||||
@ -385,7 +398,13 @@ var PDFView = {
|
|||||||
|
|
||||||
setTitleUsingUrl: function pdfViewSetTitleUsingUrl(url) {
|
setTitleUsingUrl: function pdfViewSetTitleUsingUrl(url) {
|
||||||
this.url = url;
|
this.url = url;
|
||||||
document.title = decodeURIComponent(getFileName(url)) || url;
|
try {
|
||||||
|
document.title = decodeURIComponent(getFileName(url)) || url;
|
||||||
|
} catch (e) {
|
||||||
|
// decodeURIComponent may throw URIError,
|
||||||
|
// fall back to using the unprocessed url in that case
|
||||||
|
document.title = url;
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
open: function pdfViewOpen(url, scale, password) {
|
open: function pdfViewOpen(url, scale, password) {
|
||||||
@ -1309,7 +1328,7 @@ var PageView = function pageView(container, pdfPage, id, scale,
|
|||||||
|
|
||||||
this.scrollIntoView = function pageViewScrollIntoView(dest) {
|
this.scrollIntoView = function pageViewScrollIntoView(dest) {
|
||||||
if (!dest) {
|
if (!dest) {
|
||||||
div.scrollIntoView(true);
|
scrollIntoView(div);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1368,16 +1387,7 @@ var PageView = function pageView(container, pdfPage, id, scale,
|
|||||||
var width = Math.abs(boundingRect[0][0] - boundingRect[1][0]);
|
var width = Math.abs(boundingRect[0][0] - boundingRect[1][0]);
|
||||||
var height = Math.abs(boundingRect[0][1] - boundingRect[1][1]);
|
var height = Math.abs(boundingRect[0][1] - boundingRect[1][1]);
|
||||||
|
|
||||||
// using temporary div to scroll it into view
|
scrollIntoView(div, {left: x, top: y, width: width, height: height});
|
||||||
var tempDiv = document.createElement('div');
|
|
||||||
tempDiv.style.position = 'absolute';
|
|
||||||
tempDiv.style.left = Math.floor(x) + 'px';
|
|
||||||
tempDiv.style.top = Math.floor(y) + 'px';
|
|
||||||
tempDiv.style.width = Math.ceil(width) + 'px';
|
|
||||||
tempDiv.style.height = Math.ceil(height) + 'px';
|
|
||||||
div.appendChild(tempDiv);
|
|
||||||
tempDiv.scrollIntoView(true);
|
|
||||||
div.removeChild(tempDiv);
|
|
||||||
}, 0);
|
}, 0);
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -2078,7 +2088,7 @@ window.addEventListener('pagechange', function pagechange(evt) {
|
|||||||
var last = numVisibleThumbs > 1 ?
|
var last = numVisibleThumbs > 1 ?
|
||||||
visibleThumbs.last.id : first;
|
visibleThumbs.last.id : first;
|
||||||
if (page <= first || page >= last)
|
if (page <= first || page >= last)
|
||||||
thumbnail.scrollIntoView();
|
scrollIntoView(thumbnail);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user