2011-10-26 10:18:22 +09:00
|
|
|
/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* 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-10-26 10:18:22 +09:00
|
|
|
|
|
|
|
'use strict';
|
|
|
|
|
2012-05-15 09:19:09 +09:00
|
|
|
// Use only for debugging purposes. This should not be used in any code that is
|
|
|
|
// in mozilla master.
|
2012-08-31 20:40:37 +09:00
|
|
|
var log = (function() {
|
|
|
|
if ('console' in globalScope && 'log' in globalScope['console']) {
|
|
|
|
return globalScope['console']['log'].bind(globalScope['console']);
|
|
|
|
} else {
|
|
|
|
return function nop() {
|
|
|
|
};
|
|
|
|
}
|
|
|
|
})();
|
2011-10-25 08:55:23 +09:00
|
|
|
|
2012-05-15 09:19:09 +09:00
|
|
|
// A notice for devs that will not trigger the fallback UI. These are good
|
|
|
|
// for things that are helpful to devs, such as warning that Workers were
|
|
|
|
// disabled, which is important to devs but not end users.
|
|
|
|
function info(msg) {
|
|
|
|
if (verbosity >= INFOS) {
|
|
|
|
log('Info: ' + msg);
|
|
|
|
PDFJS.LogManager.notify('info', msg);
|
|
|
|
}
|
2011-10-25 08:55:23 +09:00
|
|
|
}
|
|
|
|
|
2012-05-15 09:19:09 +09:00
|
|
|
// Non-fatal warnings that should trigger the fallback UI.
|
|
|
|
function warn(msg) {
|
|
|
|
if (verbosity >= WARNINGS) {
|
|
|
|
log('Warning: ' + msg);
|
|
|
|
PDFJS.LogManager.notify('warn', msg);
|
2011-10-25 08:55:23 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-15 09:19:09 +09:00
|
|
|
// Fatal errors that should trigger the fallback UI and halt execution by
|
|
|
|
// throwing an exception.
|
2011-10-25 08:55:23 +09:00
|
|
|
function error(msg) {
|
2012-08-31 20:40:37 +09:00
|
|
|
// If multiple arguments were passed, pass them all to the log function.
|
|
|
|
if (arguments.length > 1) {
|
|
|
|
var logArguments = ['Error:'];
|
|
|
|
logArguments.push.apply(logArguments, arguments);
|
|
|
|
log.apply(null, logArguments);
|
|
|
|
// Join the arguments into a single string for the lines below.
|
|
|
|
msg = [].join.call(arguments, ' ');
|
|
|
|
} else {
|
|
|
|
log('Error: ' + msg);
|
|
|
|
}
|
2011-10-25 08:55:23 +09:00
|
|
|
log(backtrace());
|
2012-05-15 09:19:09 +09:00
|
|
|
PDFJS.LogManager.notify('error', msg);
|
2011-10-25 08:55:23 +09:00
|
|
|
throw new Error(msg);
|
|
|
|
}
|
|
|
|
|
2012-05-15 09:19:09 +09:00
|
|
|
// Missing features that should trigger the fallback UI.
|
2011-10-25 08:55:23 +09:00
|
|
|
function TODO(what) {
|
2012-05-15 09:19:09 +09:00
|
|
|
warn('TODO: ' + what);
|
2011-10-25 08:55:23 +09:00
|
|
|
}
|
|
|
|
|
2012-05-15 09:19:09 +09:00
|
|
|
function backtrace() {
|
|
|
|
try {
|
|
|
|
throw new Error();
|
|
|
|
} catch (e) {
|
|
|
|
return e.stack ? e.stack.split('\n').slice(2).join('\n') : '';
|
|
|
|
}
|
2011-10-25 08:55:23 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
function assert(cond, msg) {
|
|
|
|
if (!cond)
|
|
|
|
error(msg);
|
|
|
|
}
|
|
|
|
|
2012-06-24 05:35:59 +09:00
|
|
|
// Combines two URLs. The baseUrl shall be absolute URL. If the url is an
|
|
|
|
// absolute URL, it will be returned as is.
|
2012-06-24 04:48:33 +09:00
|
|
|
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
|
2012-06-24 05:35:59 +09:00
|
|
|
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;
|
2012-06-24 04:48:33 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-25 08:55:23 +09:00
|
|
|
// In a well-formed PDF, |cond| holds. If it doesn't, subsequent
|
|
|
|
// behavior is undefined.
|
|
|
|
function assertWellFormed(cond, msg) {
|
|
|
|
if (!cond)
|
2012-05-15 09:19:09 +09:00
|
|
|
error(msg);
|
2011-10-25 08:55:23 +09:00
|
|
|
}
|
|
|
|
|
2012-05-15 09:19:09 +09:00
|
|
|
var LogManager = PDFJS.LogManager = (function LogManagerClosure() {
|
|
|
|
var loggers = [];
|
|
|
|
return {
|
|
|
|
addLogger: function logManager_addLogger(logger) {
|
|
|
|
loggers.push(logger);
|
|
|
|
},
|
|
|
|
notify: function(type, message) {
|
|
|
|
for (var i = 0, ii = loggers.length; i < ii; i++) {
|
|
|
|
var logger = loggers[i];
|
|
|
|
if (logger[type])
|
|
|
|
logger[type](message);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
})();
|
|
|
|
|
2011-10-25 08:55:23 +09:00
|
|
|
function shadow(obj, prop, value) {
|
|
|
|
Object.defineProperty(obj, prop, { value: value,
|
|
|
|
enumerable: true,
|
|
|
|
configurable: true,
|
|
|
|
writable: false });
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2012-05-18 04:34:39 +09:00
|
|
|
var PasswordException = (function PasswordExceptionClosure() {
|
|
|
|
function PasswordException(msg, code) {
|
|
|
|
this.name = 'PasswordException';
|
|
|
|
this.message = msg;
|
|
|
|
this.code = code;
|
|
|
|
}
|
|
|
|
|
|
|
|
PasswordException.prototype = new Error();
|
|
|
|
PasswordException.constructor = PasswordException;
|
|
|
|
|
|
|
|
return PasswordException;
|
|
|
|
})();
|
2012-05-15 03:45:07 +09:00
|
|
|
|
2011-10-25 08:55:23 +09:00
|
|
|
function bytesToString(bytes) {
|
|
|
|
var str = '';
|
|
|
|
var length = bytes.length;
|
|
|
|
for (var n = 0; n < length; ++n)
|
|
|
|
str += String.fromCharCode(bytes[n]);
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
function stringToBytes(str) {
|
|
|
|
var length = str.length;
|
|
|
|
var bytes = new Uint8Array(length);
|
|
|
|
for (var n = 0; n < length; ++n)
|
|
|
|
bytes[n] = str.charCodeAt(n) & 0xFF;
|
|
|
|
return bytes;
|
|
|
|
}
|
|
|
|
|
|
|
|
var IDENTITY_MATRIX = [1, 0, 0, 1, 0, 0];
|
|
|
|
|
2012-04-17 01:45:49 +09:00
|
|
|
var Util = PDFJS.Util = (function UtilClosure() {
|
2011-12-09 07:18:43 +09:00
|
|
|
function Util() {}
|
2012-02-02 07:48:44 +09:00
|
|
|
|
2012-04-05 05:43:26 +09:00
|
|
|
Util.makeCssRgb = function Util_makeCssRgb(r, g, b) {
|
2011-10-25 08:55:23 +09:00
|
|
|
var ri = (255 * r) | 0, gi = (255 * g) | 0, bi = (255 * b) | 0;
|
|
|
|
return 'rgb(' + ri + ',' + gi + ',' + bi + ')';
|
|
|
|
};
|
2012-02-02 07:48:44 +09:00
|
|
|
|
2012-04-05 05:43:26 +09:00
|
|
|
Util.makeCssCmyk = function Util_makeCssCmyk(c, m, y, k) {
|
2011-10-25 08:55:23 +09:00
|
|
|
c = (new DeviceCmykCS()).getRgb([c, m, y, k]);
|
|
|
|
var ri = (255 * c[0]) | 0, gi = (255 * c[1]) | 0, bi = (255 * c[2]) | 0;
|
|
|
|
return 'rgb(' + ri + ',' + gi + ',' + bi + ')';
|
|
|
|
};
|
2012-02-02 07:48:44 +09:00
|
|
|
|
|
|
|
// For 2d affine transforms
|
2012-04-05 05:43:26 +09:00
|
|
|
Util.applyTransform = function Util_applyTransform(p, m) {
|
2011-10-25 08:55:23 +09:00
|
|
|
var xt = p[0] * m[0] + p[1] * m[2] + m[4];
|
|
|
|
var yt = p[0] * m[1] + p[1] * m[3] + m[5];
|
|
|
|
return [xt, yt];
|
|
|
|
};
|
|
|
|
|
2012-05-02 02:48:07 +09:00
|
|
|
Util.applyInverseTransform = function Util_applyInverseTransform(p, m) {
|
2012-04-12 00:29:44 +09:00
|
|
|
var d = m[0] * m[3] - m[1] * m[2];
|
|
|
|
var xt = (p[0] * m[3] - p[1] * m[2] + m[2] * m[5] - m[4] * m[3]) / d;
|
|
|
|
var yt = (-p[0] * m[1] + p[1] * m[0] + m[4] * m[1] - m[5] * m[0]) / d;
|
|
|
|
return [xt, yt];
|
|
|
|
};
|
|
|
|
|
|
|
|
Util.inverseTransform = function Util_inverseTransform(m) {
|
|
|
|
var d = m[0] * m[3] - m[1] * m[2];
|
|
|
|
return [m[3] / d, -m[1] / d, -m[2] / d, m[0] / d,
|
|
|
|
(m[2] * m[5] - m[4] * m[3]) / d, (m[4] * m[1] - m[5] * m[0]) / d];
|
|
|
|
};
|
|
|
|
|
2012-02-02 07:48:44 +09:00
|
|
|
// Apply a generic 3d matrix M on a 3-vector v:
|
|
|
|
// | a b c | | X |
|
|
|
|
// | d e f | x | Y |
|
|
|
|
// | g h i | | Z |
|
|
|
|
// M is assumed to be serialized as [a,b,c,d,e,f,g,h,i],
|
|
|
|
// with v as [X,Y,Z]
|
2012-04-05 05:43:26 +09:00
|
|
|
Util.apply3dTransform = function Util_apply3dTransform(m, v) {
|
2012-02-02 07:48:44 +09:00
|
|
|
return [
|
|
|
|
m[0] * v[0] + m[1] * v[1] + m[2] * v[2],
|
|
|
|
m[3] * v[0] + m[4] * v[1] + m[5] * v[2],
|
|
|
|
m[6] * v[0] + m[7] * v[1] + m[8] * v[2]
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2012-02-15 04:48:58 +09:00
|
|
|
// Normalize rectangle rect=[x1, y1, x2, y2] so that (x1,y1) < (x2,y2)
|
|
|
|
// For coordinate systems whose origin lies in the bottom-left, this
|
|
|
|
// means normalization to (BL,TR) ordering. For systems with origin in the
|
|
|
|
// top-left, this means (TL,BR) ordering.
|
2012-04-05 05:43:26 +09:00
|
|
|
Util.normalizeRect = function Util_normalizeRect(rect) {
|
2012-02-15 04:48:58 +09:00
|
|
|
var r = rect.slice(0); // clone rect
|
|
|
|
if (rect[0] > rect[2]) {
|
|
|
|
r[0] = rect[2];
|
|
|
|
r[2] = rect[0];
|
|
|
|
}
|
|
|
|
if (rect[1] > rect[3]) {
|
|
|
|
r[1] = rect[3];
|
|
|
|
r[3] = rect[1];
|
|
|
|
}
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns a rectangle [x1, y1, x2, y2] corresponding to the
|
|
|
|
// intersection of rect1 and rect2. If no intersection, returns 'false'
|
|
|
|
// The rectangle coordinates of rect1, rect2 should be [x1, y1, x2, y2]
|
2012-04-05 05:43:26 +09:00
|
|
|
Util.intersect = function Util_intersect(rect1, rect2) {
|
2012-02-15 04:48:58 +09:00
|
|
|
function compare(a, b) {
|
|
|
|
return a - b;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Order points along the axes
|
|
|
|
var orderedX = [rect1[0], rect1[2], rect2[0], rect2[2]].sort(compare),
|
|
|
|
orderedY = [rect1[1], rect1[3], rect2[1], rect2[3]].sort(compare),
|
|
|
|
result = [];
|
|
|
|
|
|
|
|
rect1 = Util.normalizeRect(rect1);
|
|
|
|
rect2 = Util.normalizeRect(rect2);
|
|
|
|
|
|
|
|
// X: first and second points belong to different rectangles?
|
|
|
|
if ((orderedX[0] === rect1[0] && orderedX[1] === rect2[0]) ||
|
|
|
|
(orderedX[0] === rect2[0] && orderedX[1] === rect1[0])) {
|
|
|
|
// Intersection must be between second and third points
|
|
|
|
result[0] = orderedX[1];
|
|
|
|
result[2] = orderedX[2];
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Y: first and second points belong to different rectangles?
|
|
|
|
if ((orderedY[0] === rect1[1] && orderedY[1] === rect2[1]) ||
|
|
|
|
(orderedY[0] === rect2[1] && orderedY[1] === rect1[1])) {
|
|
|
|
// Intersection must be between second and third points
|
|
|
|
result[1] = orderedY[1];
|
|
|
|
result[3] = orderedY[2];
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
2012-04-12 00:29:44 +09:00
|
|
|
};
|
2012-02-15 04:48:58 +09:00
|
|
|
|
2012-04-05 05:43:26 +09:00
|
|
|
Util.sign = function Util_sign(num) {
|
2012-01-24 05:23:09 +09:00
|
|
|
return num < 0 ? -1 : 1;
|
|
|
|
};
|
|
|
|
|
2011-12-09 07:18:43 +09:00
|
|
|
return Util;
|
2011-10-25 08:55:23 +09:00
|
|
|
})();
|
|
|
|
|
2012-04-12 00:29:44 +09:00
|
|
|
var PageViewport = PDFJS.PageViewport = (function PageViewportClosure() {
|
|
|
|
function PageViewport(viewBox, scale, rotate, offsetX, offsetY) {
|
|
|
|
// creating transform to convert pdf coordinate system to the normal
|
|
|
|
// canvas like coordinates taking in account scale and rotation
|
|
|
|
var centerX = (viewBox[2] + viewBox[0]) / 2;
|
|
|
|
var centerY = (viewBox[3] + viewBox[1]) / 2;
|
|
|
|
var rotateA, rotateB, rotateC, rotateD;
|
2012-10-11 00:27:35 +09:00
|
|
|
switch (rotate % 360) {
|
2012-04-12 00:29:44 +09:00
|
|
|
case -180:
|
|
|
|
case 180:
|
|
|
|
rotateA = -1; rotateB = 0; rotateC = 0; rotateD = 1;
|
|
|
|
break;
|
|
|
|
case -270:
|
|
|
|
case 90:
|
|
|
|
rotateA = 0; rotateB = 1; rotateC = 1; rotateD = 0;
|
|
|
|
break;
|
|
|
|
case -90:
|
|
|
|
case 270:
|
|
|
|
rotateA = 0; rotateB = -1; rotateC = -1; rotateD = 0;
|
|
|
|
break;
|
|
|
|
case 360:
|
|
|
|
case 0:
|
|
|
|
default:
|
|
|
|
rotateA = 1; rotateB = 0; rotateC = 0; rotateD = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
var offsetCanvasX, offsetCanvasY;
|
|
|
|
var width, height;
|
|
|
|
if (rotateA == 0) {
|
|
|
|
offsetCanvasX = Math.abs(centerY - viewBox[1]) * scale + offsetX;
|
|
|
|
offsetCanvasY = Math.abs(centerX - viewBox[0]) * scale + offsetY;
|
|
|
|
width = Math.abs(viewBox[3] - viewBox[1]) * scale;
|
|
|
|
height = Math.abs(viewBox[2] - viewBox[0]) * scale;
|
|
|
|
} else {
|
|
|
|
offsetCanvasX = Math.abs(centerX - viewBox[0]) * scale + offsetX;
|
|
|
|
offsetCanvasY = Math.abs(centerY - viewBox[1]) * scale + offsetY;
|
|
|
|
width = Math.abs(viewBox[2] - viewBox[0]) * scale;
|
|
|
|
height = Math.abs(viewBox[3] - viewBox[1]) * scale;
|
|
|
|
}
|
|
|
|
// creating transform for the following operations:
|
|
|
|
// translate(-centerX, -centerY), rotate and flip vertically,
|
|
|
|
// scale, and translate(offsetCanvasX, offsetCanvasY)
|
|
|
|
this.transform = [
|
|
|
|
rotateA * scale,
|
|
|
|
rotateB * scale,
|
|
|
|
rotateC * scale,
|
|
|
|
rotateD * scale,
|
|
|
|
offsetCanvasX - rotateA * scale * centerX - rotateC * scale * centerY,
|
|
|
|
offsetCanvasY - rotateB * scale * centerX - rotateD * scale * centerY
|
|
|
|
];
|
|
|
|
|
|
|
|
this.offsetX = offsetX;
|
|
|
|
this.offsetY = offsetY;
|
|
|
|
this.width = width;
|
|
|
|
this.height = height;
|
2012-04-13 00:23:38 +09:00
|
|
|
this.fontScale = scale;
|
2012-04-12 00:29:44 +09:00
|
|
|
}
|
|
|
|
PageViewport.prototype = {
|
2012-04-12 02:18:29 +09:00
|
|
|
convertToViewportPoint: function PageViewport_convertToViewportPoint(x, y) {
|
2012-04-12 00:29:44 +09:00
|
|
|
return Util.applyTransform([x, y], this.transform);
|
|
|
|
},
|
2012-04-12 02:18:29 +09:00
|
|
|
convertToViewportRectangle:
|
|
|
|
function PageViewport_convertToViewportRectangle(rect) {
|
2012-04-12 00:29:44 +09:00
|
|
|
var tl = Util.applyTransform([rect[0], rect[1]], this.transform);
|
|
|
|
var br = Util.applyTransform([rect[2], rect[3]], this.transform);
|
|
|
|
return [tl[0], tl[1], br[0], br[1]];
|
|
|
|
},
|
2012-04-12 02:18:29 +09:00
|
|
|
convertToPdfPoint: function PageViewport_convertToPdfPoint(x, y) {
|
2012-04-12 00:29:44 +09:00
|
|
|
return Util.applyInverseTransform([x, y], this.transform);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
return PageViewport;
|
|
|
|
})();
|
|
|
|
|
2011-10-25 08:55:23 +09:00
|
|
|
var PDFStringTranslateTable = [
|
|
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
|
|
0x2D8, 0x2C7, 0x2C6, 0x2D9, 0x2DD, 0x2DB, 0x2DA, 0x2DC, 0, 0, 0, 0, 0, 0, 0,
|
|
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0x2022, 0x2020, 0x2021, 0x2026, 0x2014,
|
|
|
|
0x2013, 0x192, 0x2044, 0x2039, 0x203A, 0x2212, 0x2030, 0x201E, 0x201C,
|
|
|
|
0x201D, 0x2018, 0x2019, 0x201A, 0x2122, 0xFB01, 0xFB02, 0x141, 0x152, 0x160,
|
|
|
|
0x178, 0x17D, 0x131, 0x142, 0x153, 0x161, 0x17E, 0, 0x20AC
|
|
|
|
];
|
|
|
|
|
|
|
|
function stringToPDFString(str) {
|
|
|
|
var i, n = str.length, str2 = '';
|
|
|
|
if (str[0] === '\xFE' && str[1] === '\xFF') {
|
|
|
|
// UTF16BE BOM
|
|
|
|
for (i = 2; i < n; i += 2)
|
|
|
|
str2 += String.fromCharCode(
|
|
|
|
(str.charCodeAt(i) << 8) | str.charCodeAt(i + 1));
|
|
|
|
} else {
|
|
|
|
for (i = 0; i < n; ++i) {
|
|
|
|
var code = PDFStringTranslateTable[str.charCodeAt(i)];
|
|
|
|
str2 += code ? String.fromCharCode(code) : str.charAt(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return str2;
|
|
|
|
}
|
|
|
|
|
2012-05-28 05:49:28 +09:00
|
|
|
function stringToUTF8String(str) {
|
|
|
|
return decodeURIComponent(escape(str));
|
|
|
|
}
|
|
|
|
|
2011-10-25 08:55:23 +09:00
|
|
|
function isBool(v) {
|
|
|
|
return typeof v == 'boolean';
|
|
|
|
}
|
|
|
|
|
|
|
|
function isInt(v) {
|
|
|
|
return typeof v == 'number' && ((v | 0) == v);
|
|
|
|
}
|
|
|
|
|
|
|
|
function isNum(v) {
|
|
|
|
return typeof v == 'number';
|
|
|
|
}
|
|
|
|
|
|
|
|
function isString(v) {
|
|
|
|
return typeof v == 'string';
|
|
|
|
}
|
|
|
|
|
|
|
|
function isNull(v) {
|
|
|
|
return v === null;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isName(v) {
|
|
|
|
return v instanceof Name;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isCmd(v, cmd) {
|
|
|
|
return v instanceof Cmd && (!cmd || v.cmd == cmd);
|
|
|
|
}
|
|
|
|
|
|
|
|
function isDict(v, type) {
|
|
|
|
return v instanceof Dict && (!type || v.get('Type').name == type);
|
|
|
|
}
|
|
|
|
|
|
|
|
function isArray(v) {
|
|
|
|
return v instanceof Array;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isStream(v) {
|
|
|
|
return typeof v == 'object' && v != null && ('getChar' in v);
|
|
|
|
}
|
|
|
|
|
|
|
|
function isArrayBuffer(v) {
|
|
|
|
return typeof v == 'object' && v != null && ('byteLength' in v);
|
|
|
|
}
|
|
|
|
|
|
|
|
function isRef(v) {
|
|
|
|
return v instanceof Ref;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isPDFFunction(v) {
|
|
|
|
var fnDict;
|
|
|
|
if (typeof v != 'object')
|
|
|
|
return false;
|
|
|
|
else if (isDict(v))
|
|
|
|
fnDict = v;
|
|
|
|
else if (isStream(v))
|
|
|
|
fnDict = v.dict;
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
return fnDict.has('FunctionType');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 'Promise' object.
|
|
|
|
* 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.
|
|
|
|
*/
|
2012-04-10 14:20:57 +09:00
|
|
|
var Promise = PDFJS.Promise = (function PromiseClosure() {
|
2011-10-25 08:55:23 +09:00
|
|
|
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;
|
2012-01-05 09:13:53 +09:00
|
|
|
this.isRejected = false;
|
|
|
|
this.error = null;
|
2011-10-25 08:55:23 +09:00
|
|
|
// If you build a promise and pass in some data it's already resolved.
|
|
|
|
if (data != null) {
|
|
|
|
this.isResolved = true;
|
|
|
|
this._data = data;
|
|
|
|
this.hasData = true;
|
|
|
|
} else {
|
|
|
|
this.isResolved = false;
|
|
|
|
this._data = EMPTY_PROMISE;
|
|
|
|
}
|
|
|
|
this.callbacks = [];
|
2012-01-05 09:13:53 +09:00
|
|
|
this.errbacks = [];
|
2012-04-10 14:20:57 +09:00
|
|
|
this.progressbacks = [];
|
2011-10-25 08:55:23 +09:00
|
|
|
};
|
2011-12-12 09:56:45 +09:00
|
|
|
/**
|
|
|
|
* Builds a promise that is resolved when all the passed in promises are
|
|
|
|
* resolved.
|
2011-12-13 02:26:24 +09:00
|
|
|
* @param {Promise[]} promises Array of promises to wait for.
|
|
|
|
* @return {Promise} New dependant promise.
|
2011-12-12 09:56:45 +09:00
|
|
|
*/
|
2012-04-05 05:43:26 +09:00
|
|
|
Promise.all = function Promise_all(promises) {
|
2011-12-12 09:56:45 +09:00
|
|
|
var deferred = new Promise();
|
|
|
|
var unresolved = promises.length;
|
|
|
|
var results = [];
|
|
|
|
if (unresolved === 0) {
|
|
|
|
deferred.resolve(results);
|
|
|
|
return deferred;
|
|
|
|
}
|
2012-04-10 14:20:57 +09:00
|
|
|
for (var i = 0, ii = promises.length; i < ii; ++i) {
|
2011-12-12 09:56:45 +09:00
|
|
|
var promise = promises[i];
|
|
|
|
promise.then((function(i) {
|
|
|
|
return function(value) {
|
|
|
|
results[i] = value;
|
|
|
|
unresolved--;
|
|
|
|
if (unresolved === 0)
|
|
|
|
deferred.resolve(results);
|
|
|
|
};
|
|
|
|
})(i));
|
|
|
|
}
|
|
|
|
return deferred;
|
|
|
|
};
|
2011-10-25 08:55:23 +09:00
|
|
|
Promise.prototype = {
|
|
|
|
hasData: false,
|
|
|
|
|
2011-10-28 20:34:56 +09:00
|
|
|
set data(value) {
|
|
|
|
if (value === undefined) {
|
2011-10-25 08:55:23 +09:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (this._data !== EMPTY_PROMISE) {
|
2012-01-25 05:04:59 +09:00
|
|
|
error('Promise ' + this.name +
|
|
|
|
': Cannot set the data of a promise twice');
|
2011-10-25 08:55:23 +09:00
|
|
|
}
|
2011-10-28 20:34:56 +09:00
|
|
|
this._data = value;
|
2011-10-25 08:55:23 +09:00
|
|
|
this.hasData = true;
|
|
|
|
|
|
|
|
if (this.onDataCallback) {
|
2011-10-28 20:34:56 +09:00
|
|
|
this.onDataCallback(value);
|
2011-10-25 08:55:23 +09:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
get data() {
|
|
|
|
if (this._data === EMPTY_PROMISE) {
|
2012-01-25 05:04:59 +09:00
|
|
|
error('Promise ' + this.name + ': Cannot get data that isn\'t set');
|
2011-10-25 08:55:23 +09:00
|
|
|
}
|
|
|
|
return this._data;
|
|
|
|
},
|
|
|
|
|
2012-04-05 05:43:26 +09:00
|
|
|
onData: function Promise_onData(callback) {
|
2011-10-25 08:55:23 +09:00
|
|
|
if (this._data !== EMPTY_PROMISE) {
|
|
|
|
callback(this._data);
|
|
|
|
} else {
|
|
|
|
this.onDataCallback = callback;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2012-04-05 05:43:26 +09:00
|
|
|
resolve: function Promise_resolve(data) {
|
2011-10-25 08:55:23 +09:00
|
|
|
if (this.isResolved) {
|
2012-01-25 05:04:59 +09:00
|
|
|
error('A Promise can be resolved only once ' + this.name);
|
2011-10-25 08:55:23 +09:00
|
|
|
}
|
2012-01-05 09:13:53 +09:00
|
|
|
if (this.isRejected) {
|
2012-01-25 05:04:59 +09:00
|
|
|
error('The Promise was already rejected ' + this.name);
|
2012-01-05 09:13:53 +09:00
|
|
|
}
|
2011-10-25 08:55:23 +09:00
|
|
|
|
|
|
|
this.isResolved = true;
|
2012-05-15 03:45:07 +09:00
|
|
|
this.data = (typeof data !== 'undefined') ? data : null;
|
2011-10-25 08:55:23 +09:00
|
|
|
var callbacks = this.callbacks;
|
|
|
|
|
2011-11-03 04:11:33 +09:00
|
|
|
for (var i = 0, ii = callbacks.length; i < ii; i++) {
|
2011-10-25 08:55:23 +09:00
|
|
|
callbacks[i].call(null, data);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2012-04-10 14:20:57 +09:00
|
|
|
progress: function Promise_progress(data) {
|
|
|
|
var callbacks = this.progressbacks;
|
|
|
|
for (var i = 0, ii = callbacks.length; i < ii; i++) {
|
|
|
|
callbacks[i].call(null, data);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2012-05-15 03:45:07 +09:00
|
|
|
reject: function Promise_reject(reason, exception) {
|
2012-01-05 09:13:53 +09:00
|
|
|
if (this.isRejected) {
|
2012-01-25 05:04:59 +09:00
|
|
|
error('A Promise can be rejected only once ' + this.name);
|
2012-01-05 09:13:53 +09:00
|
|
|
}
|
|
|
|
if (this.isResolved) {
|
2012-01-25 05:04:59 +09:00
|
|
|
error('The Promise was already resolved ' + this.name);
|
2012-01-05 09:13:53 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
this.isRejected = true;
|
|
|
|
this.error = reason || null;
|
|
|
|
var errbacks = this.errbacks;
|
|
|
|
|
|
|
|
for (var i = 0, ii = errbacks.length; i < ii; i++) {
|
2012-05-15 03:45:07 +09:00
|
|
|
errbacks[i].call(null, reason, exception);
|
2012-01-05 09:13:53 +09:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2012-04-10 14:20:57 +09:00
|
|
|
then: function Promise_then(callback, errback, progressback) {
|
2011-10-25 08:55:23 +09:00
|
|
|
if (!callback) {
|
2012-01-25 05:04:59 +09:00
|
|
|
error('Requiring callback' + this.name);
|
2011-10-25 08:55:23 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
// If the promise is already resolved, call the callback directly.
|
|
|
|
if (this.isResolved) {
|
|
|
|
var data = this.data;
|
|
|
|
callback.call(null, data);
|
2012-03-16 03:59:32 +09:00
|
|
|
} else if (this.isRejected && errback) {
|
2012-01-05 09:13:53 +09:00
|
|
|
var error = this.error;
|
|
|
|
errback.call(null, error);
|
2011-10-25 08:55:23 +09:00
|
|
|
} else {
|
|
|
|
this.callbacks.push(callback);
|
2012-01-05 09:13:53 +09:00
|
|
|
if (errback)
|
|
|
|
this.errbacks.push(errback);
|
2011-10-25 08:55:23 +09:00
|
|
|
}
|
2012-04-10 14:20:57 +09:00
|
|
|
|
|
|
|
if (progressback)
|
|
|
|
this.progressbacks.push(progressback);
|
2011-10-25 08:55:23 +09:00
|
|
|
}
|
|
|
|
};
|
2011-10-28 03:51:10 +09:00
|
|
|
|
2011-10-25 08:55:23 +09:00
|
|
|
return Promise;
|
|
|
|
})();
|
2011-10-28 03:51:10 +09:00
|
|
|
|
2012-02-22 02:52:09 +09:00
|
|
|
var StatTimer = (function StatTimerClosure() {
|
2012-01-12 09:48:51 +09:00
|
|
|
function rpad(str, pad, length) {
|
|
|
|
while (str.length < length)
|
|
|
|
str += pad;
|
|
|
|
return str;
|
|
|
|
}
|
2012-02-22 02:52:09 +09:00
|
|
|
function StatTimer() {
|
2012-01-12 09:48:51 +09:00
|
|
|
this.started = {};
|
|
|
|
this.times = [];
|
|
|
|
this.enabled = true;
|
|
|
|
}
|
2012-02-22 02:52:09 +09:00
|
|
|
StatTimer.prototype = {
|
2012-04-05 05:43:26 +09:00
|
|
|
time: function StatTimer_time(name) {
|
2012-01-12 09:48:51 +09:00
|
|
|
if (!this.enabled)
|
|
|
|
return;
|
|
|
|
if (name in this.started)
|
|
|
|
throw 'Timer is already running for ' + name;
|
|
|
|
this.started[name] = Date.now();
|
|
|
|
},
|
2012-04-05 05:43:26 +09:00
|
|
|
timeEnd: function StatTimer_timeEnd(name) {
|
2012-01-12 09:48:51 +09:00
|
|
|
if (!this.enabled)
|
|
|
|
return;
|
|
|
|
if (!(name in this.started))
|
|
|
|
throw 'Timer has not been started for ' + name;
|
|
|
|
this.times.push({
|
|
|
|
'name': name,
|
|
|
|
'start': this.started[name],
|
|
|
|
'end': Date.now()
|
|
|
|
});
|
|
|
|
// Remove timer from started so it can be called again.
|
|
|
|
delete this.started[name];
|
|
|
|
},
|
2012-04-05 05:43:26 +09:00
|
|
|
toString: function StatTimer_toString() {
|
2012-01-12 09:48:51 +09:00
|
|
|
var times = this.times;
|
|
|
|
var out = '';
|
|
|
|
// Find the longest name for padding purposes.
|
|
|
|
var longest = 0;
|
|
|
|
for (var i = 0, ii = times.length; i < ii; ++i) {
|
|
|
|
var name = times[i]['name'];
|
|
|
|
if (name.length > longest)
|
|
|
|
longest = name.length;
|
|
|
|
}
|
|
|
|
for (var i = 0, ii = times.length; i < ii; ++i) {
|
|
|
|
var span = times[i];
|
|
|
|
var duration = span.end - span.start;
|
|
|
|
out += rpad(span['name'], ' ', longest) + ' ' + duration + 'ms\n';
|
|
|
|
}
|
|
|
|
return out;
|
|
|
|
}
|
2012-02-09 09:38:43 +09:00
|
|
|
};
|
2012-02-22 02:52:09 +09:00
|
|
|
return StatTimer;
|
2012-01-12 09:48:51 +09:00
|
|
|
})();
|
2012-09-23 01:44:49 +09:00
|
|
|
|
|
|
|
PDFJS.createBlob = function createBlob(data, contentType) {
|
|
|
|
if (typeof Blob === 'function')
|
|
|
|
return new Blob([data], { type: contentType });
|
|
|
|
// Blob builder is deprecated in FF14 and removed in FF18.
|
|
|
|
var bb = new MozBlobBuilder();
|
|
|
|
bb.append(data);
|
|
|
|
return bb.getBlob(contentType);
|
|
|
|
};
|