2022-04-06 22:34:08 +09:00
|
|
|
/* 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.
|
|
|
|
*/
|
|
|
|
|
2023-02-16 01:14:04 +09:00
|
|
|
import { FeatureTest, ImageKind } from "./util.js";
|
2022-04-10 03:18:29 +09:00
|
|
|
|
2023-02-16 01:14:04 +09:00
|
|
|
function convertToRGBA(params) {
|
|
|
|
switch (params.kind) {
|
|
|
|
case ImageKind.GRAYSCALE_1BPP:
|
|
|
|
return convertBlackAndWhiteToRGBA(params);
|
|
|
|
case ImageKind.RGB_24BPP:
|
|
|
|
return convertRGBToRGBA(params);
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
function convertBlackAndWhiteToRGBA({
|
2022-04-06 22:34:08 +09:00
|
|
|
src,
|
|
|
|
srcPos = 0,
|
|
|
|
dest,
|
|
|
|
width,
|
|
|
|
height,
|
2023-02-16 01:14:04 +09:00
|
|
|
nonBlackColor = 0xffffffff,
|
2022-04-06 22:34:08 +09:00
|
|
|
inverseDecode = false,
|
|
|
|
}) {
|
2023-02-16 01:14:04 +09:00
|
|
|
const black = FeatureTest.isLittleEndian ? 0xff000000 : 0x000000ff;
|
|
|
|
const [zeroMapping, oneMapping] = inverseDecode
|
|
|
|
? [nonBlackColor, black]
|
|
|
|
: [black, nonBlackColor];
|
2022-04-10 03:18:29 +09:00
|
|
|
const widthInSource = width >> 3;
|
|
|
|
const widthRemainder = width & 7;
|
|
|
|
const srcLength = src.length;
|
|
|
|
dest = new Uint32Array(dest.buffer);
|
2023-02-16 01:14:04 +09:00
|
|
|
let destPos = 0;
|
2022-04-06 22:34:08 +09:00
|
|
|
|
2022-04-10 03:18:29 +09:00
|
|
|
for (let i = 0; i < height; i++) {
|
|
|
|
for (const max = srcPos + widthInSource; srcPos < max; srcPos++) {
|
|
|
|
const elem = srcPos < srcLength ? src[srcPos] : 255;
|
|
|
|
dest[destPos++] = elem & 0b10000000 ? oneMapping : zeroMapping;
|
|
|
|
dest[destPos++] = elem & 0b1000000 ? oneMapping : zeroMapping;
|
|
|
|
dest[destPos++] = elem & 0b100000 ? oneMapping : zeroMapping;
|
|
|
|
dest[destPos++] = elem & 0b10000 ? oneMapping : zeroMapping;
|
|
|
|
dest[destPos++] = elem & 0b1000 ? oneMapping : zeroMapping;
|
|
|
|
dest[destPos++] = elem & 0b100 ? oneMapping : zeroMapping;
|
|
|
|
dest[destPos++] = elem & 0b10 ? oneMapping : zeroMapping;
|
|
|
|
dest[destPos++] = elem & 0b1 ? oneMapping : zeroMapping;
|
|
|
|
}
|
|
|
|
if (widthRemainder === 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const elem = srcPos < srcLength ? src[srcPos++] : 255;
|
|
|
|
for (let j = 0; j < widthRemainder; j++) {
|
|
|
|
dest[destPos++] = elem & (1 << (7 - j)) ? oneMapping : zeroMapping;
|
2022-04-06 22:34:08 +09:00
|
|
|
}
|
|
|
|
}
|
2023-02-16 01:14:04 +09:00
|
|
|
return { srcPos, destPos };
|
|
|
|
}
|
|
|
|
|
|
|
|
function convertRGBToRGBA({
|
|
|
|
src,
|
|
|
|
srcPos = 0,
|
|
|
|
dest,
|
|
|
|
destPos = 0,
|
|
|
|
width,
|
|
|
|
height,
|
|
|
|
}) {
|
|
|
|
let i = 0;
|
|
|
|
const len32 = src.length >> 2;
|
|
|
|
const src32 = new Uint32Array(src.buffer, srcPos, len32);
|
|
|
|
|
|
|
|
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
|
|
|
|
const s2 = src32[i + 1]; // G3R3B2G2
|
|
|
|
const s3 = src32[i + 2]; // B4G4R4B3
|
|
|
|
|
|
|
|
dest[destPos] = s1 | 0xff000000;
|
|
|
|
dest[destPos + 1] = (s1 >>> 24) | (s2 << 8) | 0xff000000;
|
|
|
|
dest[destPos + 2] = (s2 >>> 16) | (s3 << 16) | 0xff000000;
|
|
|
|
dest[destPos + 3] = (s3 >>> 8) | 0xff000000;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let j = i * 4, jj = src.length; j < jj; j += 3) {
|
|
|
|
dest[destPos++] =
|
|
|
|
src[j] | (src[j + 1] << 8) | (src[j + 2] << 16) | 0xff000000;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (; i < len32 - 2; i += 3, destPos += 4) {
|
|
|
|
const s1 = src32[i]; // R1G1B1R2
|
|
|
|
const s2 = src32[i + 1]; // G2B2R3G3
|
|
|
|
const s3 = src32[i + 2]; // B3R4G4B4
|
|
|
|
|
|
|
|
dest[destPos] = s1 | 0xff;
|
|
|
|
dest[destPos + 1] = (s1 << 24) | (s2 >>> 8) | 0xff;
|
|
|
|
dest[destPos + 2] = (s2 << 16) | (s3 >>> 16) | 0xff;
|
|
|
|
dest[destPos + 3] = (s3 << 8) | 0xff;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let j = i * 4, jj = src.length; j < jj; j += 3) {
|
|
|
|
dest[destPos++] =
|
|
|
|
(src[j] << 24) | (src[j + 1] << 16) | (src[j + 2] << 8) | 0xff;
|
|
|
|
}
|
|
|
|
}
|
2022-04-06 22:34:08 +09:00
|
|
|
|
|
|
|
return { srcPos, destPos };
|
|
|
|
}
|
|
|
|
|
2023-02-16 01:14:04 +09:00
|
|
|
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, convertToRGBA, grayToRGBA };
|