pdf.js.mirror/src/shared/image_utils.js
Jonas Jenwald 55c8516944 Use the convertRGBToRGBA helper with RGB images in putBinaryImageData
This removes a little bit of code duplication, which only exist since the `src/display/canvas.js` code pre-dates the helper function by many years.

Note: Given that `OffscreenCanvas` is enabled by default there's currently not a lot of test coverage for this code-path, hence the added browser-test.
2026-06-13 13:14:50 +02:00

148 lines
4.4 KiB
JavaScript

/* Copyright 2022 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.
*/
import { FeatureTest, ImageKind } from "./util.js";
function convertToRGBA(params) {
switch (params.kind) {
case ImageKind.GRAYSCALE_1BPP:
return convertBlackAndWhiteToRGBA(params);
case ImageKind.RGB_24BPP:
return convertRGBToRGBA(params);
}
return null;
}
function convertBlackAndWhiteToRGBA({
src,
srcPos = 0,
dest,
width,
height,
nonBlackColor = 0xffffffff,
inverseDecode = false,
}) {
const black = FeatureTest.isLittleEndian ? 0xff000000 : 0x000000ff;
const [zeroMapping, oneMapping] = inverseDecode
? [nonBlackColor, black]
: [black, nonBlackColor];
const widthInSource = width >> 3;
const widthRemainder = width & 7;
const xorMask = zeroMapping ^ oneMapping;
const srcLength = src.length;
dest = new Uint32Array(dest.buffer);
let destPos = 0;
for (let i = 0; i < height; ++i) {
for (
const max = srcPos + widthInSource;
srcPos < max;
++srcPos, destPos += 8
) {
const elem = src[srcPos];
dest[destPos] = zeroMapping ^ (-((elem >> 7) & 1) & xorMask);
dest[destPos + 1] = zeroMapping ^ (-((elem >> 6) & 1) & xorMask);
dest[destPos + 2] = zeroMapping ^ (-((elem >> 5) & 1) & xorMask);
dest[destPos + 3] = zeroMapping ^ (-((elem >> 4) & 1) & xorMask);
dest[destPos + 4] = zeroMapping ^ (-((elem >> 3) & 1) & xorMask);
dest[destPos + 5] = zeroMapping ^ (-((elem >> 2) & 1) & xorMask);
dest[destPos + 6] = zeroMapping ^ (-((elem >> 1) & 1) & xorMask);
dest[destPos + 7] = zeroMapping ^ (-(elem & 1) & xorMask);
}
if (widthRemainder === 0) {
continue;
}
const elem = srcPos < srcLength ? src[srcPos++] : 255;
for (let j = 0; j < widthRemainder; ++j, ++destPos) {
dest[destPos] = zeroMapping ^ (-((elem >> (7 - j)) & 1) & xorMask);
}
}
return { srcPos, destPos };
}
function convertRGBToRGBA({
src,
srcPos = 0,
dest,
destPos = 0,
width,
height,
}) {
let i = 0;
const len = width * height * 3;
const len32 = len >> 2;
const src32 = new Uint32Array(src.buffer, srcPos, len32);
const alphaMask = FeatureTest.isLittleEndian ? 0xff000000 : 0xff;
if (FeatureTest.isLittleEndian) {
// It's a way faster to do the shuffle manually instead of working
// component by component with some Uint8 arrays.
for (; i < len32 - 2; i += 3, destPos += 4) {
const s1 = src32[i], // R2B1G1R1
s2 = src32[i + 1], // G3R3B2G2
s3 = src32[i + 2]; // B4G4R4B3
dest[destPos] = s1 | alphaMask;
dest[destPos + 1] = (s1 >>> 24) | (s2 << 8) | alphaMask;
dest[destPos + 2] = (s2 >>> 16) | (s3 << 16) | alphaMask;
dest[destPos + 3] = (s3 >>> 8) | alphaMask;
}
for (let j = i * 4, jj = srcPos + len; j < jj; j += 3) {
dest[destPos++] =
src[j] | (src[j + 1] << 8) | (src[j + 2] << 16) | alphaMask;
}
} else {
for (; i < len32 - 2; i += 3, destPos += 4) {
const s1 = src32[i], // R1G1B1R2
s2 = src32[i + 1], // G2B2R3G3
s3 = src32[i + 2]; // B3R4G4B4
dest[destPos] = s1 | alphaMask;
dest[destPos + 1] = (s1 << 24) | (s2 >>> 8) | alphaMask;
dest[destPos + 2] = (s2 << 16) | (s3 >>> 16) | alphaMask;
dest[destPos + 3] = (s3 << 8) | alphaMask;
}
for (let j = i * 4, jj = srcPos + len; j < jj; j += 3) {
dest[destPos++] =
(src[j] << 24) | (src[j + 1] << 16) | (src[j + 2] << 8) | alphaMask;
}
}
return { srcPos: srcPos + len, destPos };
}
function grayToRGBA(src, dest) {
if (FeatureTest.isLittleEndian) {
for (let i = 0, ii = src.length; i < ii; i++) {
dest[i] = (src[i] * 0x10101) | 0xff000000;
}
} else {
for (let i = 0, ii = src.length; i < ii; i++) {
dest[i] = (src[i] * 0x1010100) | 0x000000ff;
}
}
}
export {
convertBlackAndWhiteToRGBA,
convertRGBToRGBA,
convertToRGBA,
grayToRGBA,
};