ここからのコードに基づいて、画像を(HTMLキャンバスを使用して)RGBからYCoCg色空間に変換し、キャンバスからの出力をロスレスで変換できるPNGとして保存しようとしています:
https://stackoverflow.com/a/12146329/2399347
このプロセスは元に戻すことができますが、私の試行は現在、中間形式が使用されていない場合にのみ機能します。これがないと、コードの実際の適用が制限されます。
「リフティング」プロセスで 8 ビットの符号付き整数 (疑似コードでは -128 から 127、モジュロ 0x100 であると考えられます) を使用し、キャンバスに表示される値 (0 から 255) を変換する必要があります。保存できます。成功した場合、変換された画像がキャンバスに配置されたときに、値を再変換して逆リフトすると、元の画像が生成されます。現在生成されている値は、キャンバスに正しく表示できません (-255 から 255 まで)。
上記の問題は私の現在の知識を超えており、オンライン検索や Javascript のドキュメントを調べても、これを行う方法が見つかりませんでした。知っている(または興味がある)人は、コードを見て、これを行う実用的な方法、またはその効果に対する回避策を提案または提供できますか?
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="content-type" content="charset=UTF-8" />
<style>
img, canvas {
display: block;
padding-bottom: 1em;
}
</style>
<script>
function go() {
// Extract PNG image to Canvas
var canvas = document.getElementById('ycocg');
var ctx = canvas.getContext('2d');
var source = document.querySelector("#original");
canvas.width = source.width;
canvas.height = source.height;
ctx.drawImage(source, 0, 0);
var canvasData = ctx.getImageData(0, 0, canvas.width, canvas.height);
// Canvas to convert image to YCoCg and back in one step
var canvasBack = document.getElementById('back2rgb');
var ctxBack = canvasBack.getContext('2d');
canvasBack.width = source.width;
canvasBack.height = source.height;
ctxBack.drawImage(canvas, 0, 0);
var canvasBackData = ctxBack.getImageData(0, 0, canvas.width, canvas.height);
// Colour space functions
function forward_lift( x1, y1 ) {
var diff = ( ( y1 - x1 ) ) % 256;
var average = ( x1 + ( diff >> 1 ) ) % 256;
return { c1: average, c2: diff };
}
function reverse_lift( average, diff ) {
var x2 = ( average - ( diff >> 1 ) ) % 256;
var y2 = ( ( x2 + diff ) ) % 256;
return { c1: x2, c2: y2 };
}
function RGB_to_YCoCg24( red, green, blue ) {
// Conversion and back
var lift1 = forward_lift( red, blue );
var lift2 = forward_lift( green, lift1.c1 );
var rlift1 = reverse_lift( lift2.c1, lift2.c2 );
var rlift2 = reverse_lift( rlift1.c2, lift1.c2 );
canvasBackData.data[idx + 0] = rlift2.c1 % 256; // R
canvasBackData.data[idx + 1] = rlift1.c1 % 256; // G
canvasBackData.data[idx + 2] = rlift2.c2 % 256; // B
// Attempt at conversion only and store
canvasData.data[idx + 0] = lift2.c1 % 256; // Y
canvasData.data[idx + 1] = lift2.c2 % 256; // Cg
canvasData.data[idx + 2] = lift1.c2 % 256; // Co
}
function YCoCg24_to_RGB( Yimg, Cg, Co ) {
// Attempt at reversing stored image
var rlift1 = reverse_lift( Yimg, Cg );
var rlift2 = reverse_lift( rlift1.c2, Co );
canvasStepData.data[idx + 0] = rlift2.c1 % 256; // R
canvasStepData.data[idx + 1] = rlift1.c1 % 256; // G
canvasStepData.data[idx + 2] = rlift2.c2 % 256; // B
}
// Conversion of RGB image to YCoCg
for (var x = 0; x < canvasData.width; x++) {
for (var y = 0; y < canvasData.height; y++) {
// Index of the pixel in the array
var idx = (x + y * canvas.width) * 4;
RGB_to_YCoCg24( canvasData.data[idx + 0], canvasData.data[idx + 1], canvasData.data[idx + 2] );
}
}
ctxBack.putImageData(canvasBackData, 0, 0);
ctx.putImageData(canvasData, 0, 0);
// Convert from YCoCg to RGB, in two steps
var canvasStep = document.getElementById('twostep');
var ctxStep = canvasStep.getContext('2d');
canvasStep.width = source.width;
canvasStep.height = source.height;
ctxStep.drawImage(canvas, 0, 0);
var canvasStepData = ctxStep.getImageData(0, 0, canvas.width, canvas.height);
for (var x = 0; x < canvasData.width; x++) {
for (var y = 0; y < canvasData.height; y++) {
// Index of the pixel in the array
var idx = (x + y * canvas.width) * 4;
YCoCg24_to_RGB( canvasStepData.data[idx + 0], canvasStepData.data[idx + 1], canvasStepData.data[idx + 2] );
}
}
ctxStep.putImageData(canvasStepData, 0, 0);
}
</script>
</head>
<body onload="go()">
<div id="container">
Original:
<img id="original" src="" />
Converted and back in one step:
<canvas id="back2rgb"></canvas>
Converted and stored (right click, save image as):
<canvas id="ycocg"></canvas>
Restored previous image:
<canvas id="twostep"></canvas>
</div>
</body>
</html>