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: */
|
|
|
|
|
|
|
|
'use strict';
|
|
|
|
|
2011-12-09 07:18:43 +09:00
|
|
|
var PDFImage = (function PDFImageClosure() {
|
2011-12-12 09:56:45 +09:00
|
|
|
/**
|
|
|
|
* Decode the image in the main thread if it supported. Resovles the promise
|
|
|
|
* when the image data is ready.
|
|
|
|
*/
|
|
|
|
function handleImageData(handler, xref, res, image, promise) {
|
2012-01-04 07:26:19 +09:00
|
|
|
if (image instanceof JpegStream && image.isNativelyDecodable(xref, res)) {
|
2011-12-12 09:56:45 +09:00
|
|
|
// For natively supported jpegs send them to the main thread for decoding.
|
|
|
|
var dict = image.dict;
|
|
|
|
var colorSpace = dict.get('ColorSpace', 'CS');
|
2011-12-13 02:26:24 +09:00
|
|
|
colorSpace = ColorSpace.parse(colorSpace, xref, res);
|
2011-12-12 09:56:45 +09:00
|
|
|
var numComps = colorSpace.numComps;
|
|
|
|
handler.send('jpeg_decode', [image.getIR(), numComps], function(message) {
|
|
|
|
var data = message.data;
|
|
|
|
var stream = new Stream(data, 0, data.length, image.dict);
|
|
|
|
promise.resolve(stream);
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
promise.resolve(image);
|
|
|
|
}
|
|
|
|
}
|
2011-12-14 06:53:22 +09:00
|
|
|
/**
|
2011-12-14 07:35:46 +09:00
|
|
|
* Decode and clamp a value. The formula is different from the spec because we
|
|
|
|
* don't decode to float range [0,1], we decode it in the [0,max] range.
|
2011-12-14 06:53:22 +09:00
|
|
|
*/
|
2011-12-14 07:35:46 +09:00
|
|
|
function decodeAndClamp(value, addend, coefficient, max) {
|
2011-12-14 06:53:22 +09:00
|
|
|
value = addend + value * coefficient;
|
|
|
|
// Clamp the value to the range
|
|
|
|
return value < 0 ? 0 : value > max ? max : value;
|
|
|
|
}
|
2011-12-12 14:13:53 +09:00
|
|
|
function PDFImage(xref, res, image, inline, smask) {
|
2011-10-25 08:55:23 +09:00
|
|
|
this.image = image;
|
|
|
|
if (image.getParams) {
|
|
|
|
// JPX/JPEG2000 streams directly contain bits per component
|
|
|
|
// and color space mode information.
|
|
|
|
TODO('get params from actual stream');
|
|
|
|
// var bits = ...
|
|
|
|
// var colorspace = ...
|
|
|
|
}
|
|
|
|
// TODO cache rendered images?
|
|
|
|
|
|
|
|
var dict = image.dict;
|
|
|
|
this.width = dict.get('Width', 'W');
|
|
|
|
this.height = dict.get('Height', 'H');
|
|
|
|
|
|
|
|
if (this.width < 1 || this.height < 1)
|
|
|
|
error('Invalid image width: ' + this.width + ' or height: ' +
|
|
|
|
this.height);
|
|
|
|
|
|
|
|
this.interpolate = dict.get('Interpolate', 'I') || false;
|
|
|
|
this.imageMask = dict.get('ImageMask', 'IM') || false;
|
|
|
|
|
|
|
|
var bitsPerComponent = image.bitsPerComponent;
|
|
|
|
if (!bitsPerComponent) {
|
|
|
|
bitsPerComponent = dict.get('BitsPerComponent', 'BPC');
|
|
|
|
if (!bitsPerComponent) {
|
|
|
|
if (this.imageMask)
|
|
|
|
bitsPerComponent = 1;
|
|
|
|
else
|
|
|
|
error('Bits per component missing in image: ' + this.imageMask);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.bpc = bitsPerComponent;
|
|
|
|
|
|
|
|
if (!this.imageMask) {
|
|
|
|
var colorSpace = dict.get('ColorSpace', 'CS');
|
|
|
|
if (!colorSpace) {
|
|
|
|
TODO('JPX images (which don"t require color spaces');
|
|
|
|
colorSpace = new Name('DeviceRGB');
|
|
|
|
}
|
|
|
|
this.colorSpace = ColorSpace.parse(colorSpace, xref, res);
|
|
|
|
this.numComps = this.colorSpace.numComps;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.decode = dict.get('Decode', 'D');
|
2011-12-14 07:35:46 +09:00
|
|
|
this.needsDecode = false;
|
|
|
|
if (this.decode && this.colorSpace &&
|
|
|
|
!this.colorSpace.isDefaultDecode(this.decode)) {
|
|
|
|
this.needsDecode = true;
|
2011-12-14 06:53:22 +09:00
|
|
|
// Do some preprocessing to avoid more math.
|
|
|
|
var max = (1 << bitsPerComponent) - 1;
|
|
|
|
this.decodeCoefficients = [];
|
|
|
|
this.decodeAddends = [];
|
|
|
|
for (var i = 0, j = 0; i < this.decode.length; i += 2, ++j) {
|
|
|
|
var dmin = this.decode[i];
|
|
|
|
var dmax = this.decode[i + 1];
|
|
|
|
this.decodeCoefficients[j] = dmax - dmin;
|
|
|
|
this.decodeAddends[j] = max * dmin;
|
|
|
|
}
|
|
|
|
}
|
2011-10-25 08:55:23 +09:00
|
|
|
|
|
|
|
var mask = xref.fetchIfRef(dict.get('Mask'));
|
|
|
|
|
|
|
|
if (mask) {
|
|
|
|
TODO('masked images');
|
|
|
|
} else if (smask) {
|
2011-12-12 09:56:45 +09:00
|
|
|
this.smask = new PDFImage(xref, res, smask, false);
|
2011-10-25 08:55:23 +09:00
|
|
|
}
|
|
|
|
}
|
2011-12-12 09:56:45 +09:00
|
|
|
/**
|
|
|
|
* Handles processing of image data and calls the callback with an argument
|
|
|
|
* of a PDFImage when the image is ready to be used.
|
|
|
|
*/
|
2011-12-12 14:13:53 +09:00
|
|
|
PDFImage.buildImage = function buildImage(callback, handler, xref, res,
|
2011-12-12 09:56:45 +09:00
|
|
|
image, inline) {
|
2011-12-13 08:09:05 +09:00
|
|
|
var imageDataPromise = new Promise();
|
2011-12-12 09:56:45 +09:00
|
|
|
var smaskPromise = new Promise();
|
|
|
|
// The image data and smask data may not be ready yet, wait till both are
|
|
|
|
// resolved.
|
2011-12-13 08:09:05 +09:00
|
|
|
Promise.all([imageDataPromise, smaskPromise]).then(function(results) {
|
|
|
|
var imageData = results[0], smaskData = results[1];
|
|
|
|
var image = new PDFImage(xref, res, imageData, inline, smaskData);
|
2011-12-12 09:56:45 +09:00
|
|
|
callback(image);
|
|
|
|
});
|
|
|
|
|
2011-12-13 08:09:05 +09:00
|
|
|
handleImageData(handler, xref, res, image, imageDataPromise);
|
2011-12-12 09:56:45 +09:00
|
|
|
|
|
|
|
var smask = xref.fetchIfRef(image.dict.get('SMask'));
|
|
|
|
if (smask)
|
|
|
|
handleImageData(handler, xref, res, smask, smaskPromise);
|
|
|
|
else
|
|
|
|
smaskPromise.resolve(null);
|
|
|
|
};
|
2011-10-25 08:55:23 +09:00
|
|
|
|
2011-12-15 05:27:56 +09:00
|
|
|
/**
|
|
|
|
* Resize an image using the nearest neighbor algorithm. Currently only
|
2011-12-20 07:58:01 +09:00
|
|
|
* supports one and three component images.
|
2011-12-15 05:27:56 +09:00
|
|
|
* @param {TypedArray} pixels The original image with one component.
|
2011-12-19 10:18:36 +09:00
|
|
|
* @param {Number} bpc Number of bits per component.
|
|
|
|
* @param {Number} components Number of color components, 1 or 3 is supported.
|
2011-12-15 05:27:56 +09:00
|
|
|
* @param {Number} w1 Original width.
|
|
|
|
* @param {Number} h1 Original height.
|
|
|
|
* @param {Number} w2 New width.
|
|
|
|
* @param {Number} h2 New height.
|
|
|
|
* @return {TypedArray} Resized image data.
|
|
|
|
*/
|
2011-12-19 10:18:36 +09:00
|
|
|
PDFImage.resize = function resize(pixels, bpc, components, w1, h1, w2, h2) {
|
|
|
|
var length = w2 * h2 * components;
|
|
|
|
var temp = bpc <= 8 ? new Uint8Array(length) :
|
|
|
|
bpc <= 16 ? new Uint16Array(length) : new Uint32Array(length);
|
2011-12-15 05:27:56 +09:00
|
|
|
var xRatio = w1 / w2;
|
|
|
|
var yRatio = h1 / h2;
|
2011-12-19 10:18:36 +09:00
|
|
|
var px, py, newIndex, oldIndex;
|
2011-12-15 05:27:56 +09:00
|
|
|
for (var i = 0; i < h2; i++) {
|
|
|
|
for (var j = 0; j < w2; j++) {
|
|
|
|
px = Math.floor(j * xRatio);
|
|
|
|
py = Math.floor(i * yRatio);
|
2011-12-19 10:18:36 +09:00
|
|
|
newIndex = (i * w2) + j;
|
|
|
|
oldIndex = ((py * w1) + px);
|
|
|
|
if (components === 1) {
|
|
|
|
temp[newIndex] = pixels[oldIndex];
|
2011-12-19 10:29:08 +09:00
|
|
|
} else if (components === 3) {
|
2011-12-19 10:18:36 +09:00
|
|
|
newIndex *= 3;
|
|
|
|
oldIndex *= 3;
|
|
|
|
temp[newIndex] = pixels[oldIndex];
|
|
|
|
temp[newIndex + 1] = pixels[oldIndex + 1];
|
|
|
|
temp[newIndex + 2] = pixels[oldIndex + 2];
|
|
|
|
}
|
2011-12-15 05:27:56 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return temp;
|
|
|
|
};
|
|
|
|
|
2011-12-09 07:18:43 +09:00
|
|
|
PDFImage.prototype = {
|
2011-12-20 09:31:47 +09:00
|
|
|
get drawWidth() {
|
2011-12-19 10:18:36 +09:00
|
|
|
if (!this.smask)
|
|
|
|
return this.width;
|
|
|
|
return Math.max(this.width, this.smask.width);
|
|
|
|
},
|
2011-12-20 09:31:47 +09:00
|
|
|
get drawHeight() {
|
2011-12-19 10:18:36 +09:00
|
|
|
if (!this.smask)
|
|
|
|
return this.height;
|
|
|
|
return Math.max(this.height, this.smask.height);
|
|
|
|
},
|
2011-12-14 01:48:27 +09:00
|
|
|
getComponents: function getComponents(buffer) {
|
2011-10-25 08:55:23 +09:00
|
|
|
var bpc = this.bpc;
|
2011-12-14 07:35:46 +09:00
|
|
|
var needsDecode = this.needsDecode;
|
2011-12-14 01:48:27 +09:00
|
|
|
var decodeMap = this.decode;
|
2011-12-14 07:35:46 +09:00
|
|
|
|
|
|
|
// This image doesn't require any extra work.
|
|
|
|
if (bpc == 8 && !needsDecode)
|
2011-12-14 06:53:22 +09:00
|
|
|
return buffer;
|
2011-12-14 07:35:46 +09:00
|
|
|
|
2011-12-14 06:53:22 +09:00
|
|
|
var bufferLength = buffer.length;
|
2011-10-25 08:55:23 +09:00
|
|
|
var width = this.width;
|
|
|
|
var height = this.height;
|
|
|
|
var numComps = this.numComps;
|
|
|
|
|
2011-12-14 06:53:22 +09:00
|
|
|
var length = width * height * numComps;
|
2011-10-25 08:55:23 +09:00
|
|
|
var bufferPos = 0;
|
|
|
|
var output = bpc <= 8 ? new Uint8Array(length) :
|
|
|
|
bpc <= 16 ? new Uint16Array(length) : new Uint32Array(length);
|
|
|
|
var rowComps = width * numComps;
|
2011-12-14 06:53:22 +09:00
|
|
|
var decodeAddends, decodeCoefficients;
|
2011-12-14 07:35:46 +09:00
|
|
|
if (needsDecode) {
|
2011-12-14 06:53:22 +09:00
|
|
|
decodeAddends = this.decodeAddends;
|
|
|
|
decodeCoefficients = this.decodeCoefficients;
|
|
|
|
}
|
|
|
|
var max = (1 << bpc) - 1;
|
2011-10-25 08:55:23 +09:00
|
|
|
|
2011-12-14 06:53:22 +09:00
|
|
|
if (bpc == 8) {
|
|
|
|
// Optimization for reading 8 bpc images that have a decode.
|
|
|
|
for (var i = 0, ii = length; i < ii; ++i) {
|
|
|
|
var compIndex = i % numComps;
|
|
|
|
var value = buffer[i];
|
2011-12-14 07:35:46 +09:00
|
|
|
value = decodeAndClamp(value, decodeAddends[compIndex],
|
2011-12-14 06:53:22 +09:00
|
|
|
decodeCoefficients[compIndex], max);
|
|
|
|
output[i] = value;
|
|
|
|
}
|
|
|
|
} else if (bpc == 1) {
|
|
|
|
// Optimization for reading 1 bpc images.
|
2011-10-25 08:55:23 +09:00
|
|
|
var valueZero = 0, valueOne = 1;
|
|
|
|
if (decodeMap) {
|
|
|
|
valueZero = decodeMap[0] ? 1 : 0;
|
|
|
|
valueOne = decodeMap[1] ? 1 : 0;
|
|
|
|
}
|
|
|
|
var mask = 0;
|
|
|
|
var buf = 0;
|
|
|
|
|
|
|
|
for (var i = 0, ii = length; i < ii; ++i) {
|
|
|
|
if (i % rowComps == 0) {
|
|
|
|
mask = 0;
|
|
|
|
buf = 0;
|
|
|
|
} else {
|
|
|
|
mask >>= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mask <= 0) {
|
|
|
|
buf = buffer[bufferPos++];
|
|
|
|
mask = 128;
|
|
|
|
}
|
|
|
|
|
|
|
|
output[i] = !(buf & mask) ? valueZero : valueOne;
|
|
|
|
}
|
|
|
|
} else {
|
2011-12-14 06:53:22 +09:00
|
|
|
// The general case that handles all other bpc values.
|
2011-10-25 08:55:23 +09:00
|
|
|
var bits = 0, buf = 0;
|
|
|
|
for (var i = 0, ii = length; i < ii; ++i) {
|
|
|
|
if (i % rowComps == 0) {
|
|
|
|
buf = 0;
|
|
|
|
bits = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (bits < bpc) {
|
|
|
|
buf = (buf << 8) | buffer[bufferPos++];
|
|
|
|
bits += 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
var remainingBits = bits - bpc;
|
2011-12-14 06:53:22 +09:00
|
|
|
var value = buf >> remainingBits;
|
2011-12-14 07:35:46 +09:00
|
|
|
if (needsDecode) {
|
2011-12-14 06:53:22 +09:00
|
|
|
var compIndex = i % numComps;
|
2011-12-14 07:35:46 +09:00
|
|
|
value = decodeAndClamp(value, decodeAddends[compIndex],
|
2011-12-14 06:53:22 +09:00
|
|
|
decodeCoefficients[compIndex], max);
|
2011-12-14 01:48:27 +09:00
|
|
|
}
|
2011-12-14 06:53:22 +09:00
|
|
|
output[i] = value;
|
2011-10-25 08:55:23 +09:00
|
|
|
buf = buf & ((1 << remainingBits) - 1);
|
|
|
|
bits = remainingBits;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return output;
|
|
|
|
},
|
2011-12-19 10:18:36 +09:00
|
|
|
getOpacity: function getOpacity(width, height) {
|
2011-10-25 08:55:23 +09:00
|
|
|
var smask = this.smask;
|
2011-12-19 10:18:36 +09:00
|
|
|
var originalWidth = this.width;
|
|
|
|
var originalHeight = this.height;
|
2011-12-15 05:27:56 +09:00
|
|
|
var buf;
|
2011-10-25 08:55:23 +09:00
|
|
|
|
|
|
|
if (smask) {
|
|
|
|
var sw = smask.width;
|
|
|
|
var sh = smask.height;
|
2011-12-16 09:30:51 +09:00
|
|
|
buf = new Uint8Array(sw * sh);
|
2011-10-25 08:55:23 +09:00
|
|
|
smask.fillGrayBuffer(buf);
|
2011-12-19 10:18:36 +09:00
|
|
|
if (sw != width || sh != height)
|
|
|
|
buf = PDFImage.resize(buf, smask.bps, 1, sw, sh, width, height);
|
2011-10-25 08:55:23 +09:00
|
|
|
} else {
|
2011-12-16 09:30:51 +09:00
|
|
|
buf = new Uint8Array(width * height);
|
2011-10-25 08:55:23 +09:00
|
|
|
for (var i = 0, ii = width * height; i < ii; ++i)
|
|
|
|
buf[i] = 255;
|
|
|
|
}
|
|
|
|
return buf;
|
|
|
|
},
|
|
|
|
applyStencilMask: function applyStencilMask(buffer, inverseDecode) {
|
|
|
|
var width = this.width, height = this.height;
|
|
|
|
var bitStrideLength = (width + 7) >> 3;
|
2011-12-09 14:18:04 +09:00
|
|
|
var imgArray = this.getImageBytes(bitStrideLength * height);
|
2011-10-25 08:55:23 +09:00
|
|
|
var imgArrayPos = 0;
|
|
|
|
var i, j, mask, buf;
|
|
|
|
// removing making non-masked pixels transparent
|
|
|
|
var bufferPos = 3; // alpha component offset
|
|
|
|
for (i = 0; i < height; i++) {
|
|
|
|
mask = 0;
|
|
|
|
for (j = 0; j < width; j++) {
|
|
|
|
if (!mask) {
|
|
|
|
buf = imgArray[imgArrayPos++];
|
|
|
|
mask = 128;
|
|
|
|
}
|
|
|
|
if (!(buf & mask) == inverseDecode) {
|
|
|
|
buffer[bufferPos] = 0;
|
|
|
|
}
|
|
|
|
bufferPos += 4;
|
|
|
|
mask >>= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
2011-12-19 10:18:36 +09:00
|
|
|
fillRgbaBuffer: function fillRgbaBuffer(buffer, width, height) {
|
2011-10-25 08:55:23 +09:00
|
|
|
var numComps = this.numComps;
|
2011-12-19 10:18:36 +09:00
|
|
|
var originalWidth = this.width;
|
|
|
|
var originalHeight = this.height;
|
2011-10-25 08:55:23 +09:00
|
|
|
var bpc = this.bpc;
|
|
|
|
|
|
|
|
// rows start at byte boundary;
|
2011-12-19 10:18:36 +09:00
|
|
|
var rowBytes = (originalWidth * numComps * bpc + 7) >> 3;
|
|
|
|
var imgArray = this.getImageBytes(originalHeight * rowBytes);
|
2011-10-25 08:55:23 +09:00
|
|
|
|
|
|
|
var comps = this.colorSpace.getRgbBuffer(
|
2011-12-14 01:48:27 +09:00
|
|
|
this.getComponents(imgArray), bpc);
|
2011-12-19 10:18:36 +09:00
|
|
|
if (originalWidth != width || originalHeight != height)
|
2011-12-19 10:29:08 +09:00
|
|
|
comps = PDFImage.resize(comps, this.bpc, 3, originalWidth,
|
|
|
|
originalHeight, width, height);
|
2011-10-25 08:55:23 +09:00
|
|
|
var compsPos = 0;
|
2011-12-19 10:18:36 +09:00
|
|
|
var opacity = this.getOpacity(width, height);
|
2011-10-25 08:55:23 +09:00
|
|
|
var opacityPos = 0;
|
|
|
|
var length = width * height * 4;
|
|
|
|
|
|
|
|
for (var i = 0; i < length; i += 4) {
|
|
|
|
buffer[i] = comps[compsPos++];
|
|
|
|
buffer[i + 1] = comps[compsPos++];
|
|
|
|
buffer[i + 2] = comps[compsPos++];
|
|
|
|
buffer[i + 3] = opacity[opacityPos++];
|
|
|
|
}
|
|
|
|
},
|
|
|
|
fillGrayBuffer: function fillGrayBuffer(buffer) {
|
|
|
|
var numComps = this.numComps;
|
|
|
|
if (numComps != 1)
|
|
|
|
error('Reading gray scale from a color image: ' + numComps);
|
|
|
|
|
|
|
|
var width = this.width;
|
|
|
|
var height = this.height;
|
|
|
|
var bpc = this.bpc;
|
|
|
|
|
|
|
|
// rows start at byte boundary;
|
|
|
|
var rowBytes = (width * numComps * bpc + 7) >> 3;
|
2011-12-09 14:18:04 +09:00
|
|
|
var imgArray = this.getImageBytes(height * rowBytes);
|
2011-10-25 08:55:23 +09:00
|
|
|
|
|
|
|
var comps = this.getComponents(imgArray);
|
|
|
|
var length = width * height;
|
2011-12-19 10:18:36 +09:00
|
|
|
// we aren't using a colorspace so we need to scale the value
|
|
|
|
var scale = 255 / ((1 << bpc) - 1);
|
2011-10-25 08:55:23 +09:00
|
|
|
for (var i = 0; i < length; ++i)
|
2011-12-19 10:18:36 +09:00
|
|
|
buffer[i] = (scale * comps[i]) | 0;
|
2011-12-09 05:50:34 +09:00
|
|
|
},
|
2011-12-09 14:18:04 +09:00
|
|
|
getImageBytes: function getImageBytes(length) {
|
|
|
|
this.image.reset();
|
|
|
|
return this.image.getBytes(length);
|
2011-10-25 08:55:23 +09:00
|
|
|
}
|
|
|
|
};
|
2011-12-09 07:18:43 +09:00
|
|
|
return PDFImage;
|
2011-10-25 08:55:23 +09:00
|
|
|
})();
|
|
|
|
|
2011-12-09 14:18:04 +09:00
|
|
|
function loadJpegStream(id, imageData, objs) {
|
|
|
|
var img = new Image();
|
|
|
|
img.onload = (function jpegImageLoaderOnload() {
|
|
|
|
objs.resolve(id, img);
|
|
|
|
});
|
|
|
|
img.src = 'data:image/jpeg;base64,' + window.btoa(imageData);
|
|
|
|
}
|
2011-10-28 03:51:10 +09:00
|
|
|
|