web-dev-qa-db-ja.com

HTML5キャンバス画像のコントラスト

HTML5キャンバスピクセル処理を通じて効果を適用する画像処理プログラムを書いています。しきい値処理、Vintaging、およびColorGradientピクセル操作を達成しましたが、信じられないほど画像のコントラストを変更できません。私は複数の解決策を試しましたが、画像の明るさが多すぎてコントラスト効果が少なくなりますこれらの効果をネイティブに実現しようとしているため、Javascriptライブラリを使用する予定はありません。 ==

基本的なピクセル操作コード:

var data = imageData.data;
for (var i = 0; i < data.length; i += 4) {
 //Note: data[i], data[i+1], data[i+2] represent RGB respectively
data[i] = data[i];
data[i+1] = data[i+1];
data[i+2] = data[i+2];
}

ピクセル操作の例

値はRGBモードです。つまり、data [i]は赤色です。したがって、data [i] = data [i] * 2の場合;そのピクセルの赤チャンネルの明るさは2倍に増加します。例:

var data = imageData.data;
for (var i = 0; i < data.length; i += 4) {
 //Note: data[i], data[i+1], data[i+2] represent RGB respectively
 //Increases brightness of RGB channel by 2
data[i] = data[i]*2;
data[i+1] = data[i+1]*2;
data[i+2] = data[i+2]*2;
}

* 注:コードを完成させるようにお願いしているわけではありません!それはただの恩恵です!ピクセル操作のコントラストがどのように可能であるかを示すアルゴリズム(疑似コードでさえ)を求めています! =誰かがHTML5キャンバスの画像コントラストに優れたアルゴリズムを提供できれば幸いです。

より高速なオプション( エッシャーのアプローチ に基づく)は次のとおりです。

_function contrastImage(imgData, contrast){  //input range [-100..100]
    var d = imgData.data;
    contrast = (contrast/100) + 1;  //convert to decimal & shift range: [0..2]
    var intercept = 128 * (1 - contrast);
    for(var i=0;i<d.length;i+=4){   //r,g,b,a
        d[i] = d[i]*contrast + intercept;
        d[i+1] = d[i+1]*contrast + intercept;
        d[i+2] = d[i+2]*contrast + intercept;
    }
    return imgData;
}
_

以下のような派生。このバージョンは数学的に同じですが、はるかに高速に実行されます。


元の回答

これが簡略化されたバージョンです説明付きアプローチの すでに議論されています (これは この記事 に基づいていました):

_function contrastImage(imageData, contrast) {  // contrast as an integer percent  
    var data = imageData.data;  // original array modified, but canvas not updated
    contrast *= 2.55; // or *= 255 / 100; scale integer percent to full range
    var factor = (255 + contrast) / (255.01 - contrast);  //add .1 to avoid /0 error

    for(var i=0;i<data.length;i+=4)  //pixel values in 4-byte blocks (r,g,b,a)
    {
        data[i] = factor * (data[i] - 128) + 128;     //r value
        data[i+1] = factor * (data[i+1] - 128) + 128; //g value
        data[i+2] = factor * (data[i+2] - 128) + 128; //b value

    }
    return imageData;  //optional (e.g. for filter function chaining)
}
_

ノート

  1. 元の_+/- 100_の代わりに_+/- 255_のcontrast範囲を使用することを選択しました。パーセンテージ値は、基本的な概念を理解していないユーザーやプログラマーにとってはより直感的に思えます。また、私の使用法は常にUIコントロールに関連付けられています。 -100%から+ 100%の範囲では、コントロール値を調整または説明する代わりに、直接ラベルを付けてバインドできます。

  2. 計算値は許容範囲をはるかに超える可能性があります -これは、ImageDataオブジェクトの基になる配列が_Uint8ClampedArray_であるためですが、このアルゴリズムには範囲チェックは含まれていません。 MSDNで説明されているように 、_Uint8ClampedArray_を使用すると、範囲チェックが処理されます。

「[0,255]の範囲外の値を指定した場合は、代わりに0または255が設定されます。」

使用法

基になる式はかなり対称的ですが(ラウンドトリップが可能)、ピクセルは整数値しか許可しないため、高レベルのフィルタリングではデータが失われることに注意してください。たとえば、画像の彩度を極端なレベル(> 95%程度)に下げるまでに、すべてのピクセルは基本的に均一なミディアムグレーになります(平均値128の数桁以内)。コントラストを再び上げると、色の範囲がフラットになります。

また、複数のコントラスト調整を適用する場合は、操作の順序が重要です。飽和値はすぐに「吹き飛ばされ」(クランプされた最大値の255を超える)、つまり、高度に飽和してから非飽和にすると、画像全体が暗くなります。ただし、ハイライト値とシャドウ値がクリップされるのではなくミュートされるため、彩度を下げてから彩度を上げることによるデータ損失はそれほど多くありません(以下の説明を参照)。

一般的に、複数のフィルターを適用する場合は、少なくとも画質に関しては、前の変更を元に戻そうとするのではなく、元のデータで各操作を開始し、各調整を順番に再適用することをお勧めします。パフォーマンスの速度やその他の要求は、状況ごとに異なる場合があります。

Mandrill contrast examples

コード例:

_function contrastImage(imageData, contrast) {  // contrast input as percent; range [-1..1]
    var data = imageData.data;  // Note: original dataset modified directly!
    contrast *= 255;
    var factor = (contrast + 255) / (255.01 - contrast);  //add .1 to avoid /0 error.

    for(var i=0;i<data.length;i+=4)
    {
        data[i] = factor * (data[i] - 128) + 128;
        data[i+1] = factor * (data[i+1] - 128) + 128;
        data[i+2] = factor * (data[i+2] - 128) + 128;
    }
    return imageData;  //optional (e.g. for filter function chaining)
}

$(document).ready(function(){
  var ctxOrigMinus100 = document.getElementById('canvOrigMinus100').getContext("2d");
  var ctxOrigMinus50 = document.getElementById('canvOrigMinus50').getContext("2d");
  var ctxOrig = document.getElementById('canvOrig').getContext("2d");
  var ctxOrigPlus50 = document.getElementById('canvOrigPlus50').getContext("2d");
  var ctxOrigPlus100 = document.getElementById('canvOrigPlus100').getContext("2d");
  
  var ctxRoundMinus90 = document.getElementById('canvRoundMinus90').getContext("2d");
  var ctxRoundMinus50 = document.getElementById('canvRoundMinus50').getContext("2d");
  var ctxRound0 = document.getElementById('canvRound0').getContext("2d");
  var ctxRoundPlus50 = document.getElementById('canvRoundPlus50').getContext("2d");
  var ctxRoundPlus90 = document.getElementById('canvRoundPlus90').getContext("2d");
  
  
  var img = new Image();
  img.onload = function() {
    //draw orig
    ctxOrig.drawImage(img, 0, 0, img.width, img.height, 0, 0, 100, 100); //100 = canvas width, height
    
    //reduce contrast
    var origBits = ctxOrig.getImageData(0, 0, 100, 100);
    contrastImage(origBits, -.98);
    ctxOrigMinus100.putImageData(origBits, 0, 0);
    
    var origBits = ctxOrig.getImageData(0, 0, 100, 100);
    contrastImage(origBits, -.5);
    ctxOrigMinus50.putImageData(origBits, 0, 0);
    
    // add contrast
    var origBits = ctxOrig.getImageData(0, 0, 100, 100);
    contrastImage(origBits, .5);
    ctxOrigPlus50.putImageData(origBits, 0, 0);
    
    var origBits = ctxOrig.getImageData(0, 0, 100, 100);
    contrastImage(origBits, .98);
    ctxOrigPlus100.putImageData(origBits, 0, 0);
    
    
    //round-trip, de-saturate first
    origBits = ctxOrig.getImageData(0, 0, 100, 100);
    contrastImage(origBits, -.98);
    contrastImage(origBits, .98);
    ctxRoundMinus90.putImageData(origBits, 0, 0);
    
    origBits = ctxOrig.getImageData(0, 0, 100, 100);
    contrastImage(origBits, -.5);
    contrastImage(origBits, .5);
    ctxRoundMinus50.putImageData(origBits, 0, 0);
    
    //do nothing 100 times
    origBits = ctxOrig.getImageData(0, 0, 100, 100);
    for(i=0;i<100;i++){
      contrastImage(origBits, 0);
    }
    ctxRound0.putImageData(origBits, 0, 0);
    
    //round-trip, saturate first
    origBits = ctxOrig.getImageData(0, 0, 100, 100);
    contrastImage(origBits, .5);
    contrastImage(origBits, -.5);
    ctxRoundPlus50.putImageData(origBits, 0, 0);
    
    origBits = ctxOrig.getImageData(0, 0, 100, 100);
    contrastImage(origBits, .98);
    contrastImage(origBits, -.98);
    ctxRoundPlus90.putImageData(origBits, 0, 0);
  };
  
  img.src = "";
  
});_
_canvas {width: 100px; height: 100px}
div {text-align:center; width:120px; float:left}_
_<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>

<div>
  <canvas id="canvOrigMinus100" width="100" height="100"></canvas>
  -98%
</div>

<div>
  <canvas id="canvOrigMinus50" width="100" height="100"></canvas>
  -50%
</div>

<div>
  <canvas id="canvOrig" width="100" height="100"></canvas>
  Original
</div>

<div>
  <canvas id="canvOrigPlus50" width="100" height="100"></canvas>
  +50%
</div>

<div>
  <canvas id="canvOrigPlus100" width="100" height="100"></canvas>
  +98%
</div>

  <hr/>

<div style="clear:left">
  <canvas id="canvRoundMinus90" width="100" height="100"></canvas>
  Round-trip <br/> (-98%, +98%)
</div>

<div>
  <canvas id="canvRoundMinus50" width="100" height="100"></canvas>
  Round-trip <br/> (-50%, +50%)
</div>

<div>
  <canvas id="canvRound0" width="100" height="100"></canvas>
  Round-trip <br/> (0% 100x)
</div>

<div>
  <canvas id="canvRoundPlus50" width="100" height="100"></canvas>
  Round-trip <br/> (+50%, -50%)
</div>

<div>
  <canvas id="canvRoundPlus90" width="100" height="100"></canvas>
  Round-trip <br/> (+98%, -98%)
</div>_

説明

免責事項-私は画像の専門家でも数学者でもありません。最小限の技術的な詳細で常識的な説明を提供しようとしています。インデックスの問題を回避するために、255 = 256など、以下のように手を振っています。数値を単純化するための127.5 = 128。

与えられたピクセルについて、カラーチャネルのゼロ以外の値の可能な数は255であるため、「コントラストなし」、ピクセルの平均値は128 (または127、または議論したい場合は127.5ですが、違いはごくわずかです)。この説明のために、"コントラスト"の量は、現在の値から平均値(128)までの距離です。コントラストを調整するということは、現在の値と平均値の差を増減することを意味します。

アルゴリズムが解決する問題は次のとおりです。

  1. 一定の係数を選択して、コントラストを調整します
  2. 各ピクセルの各カラーチャネルについて、「コントラスト」(平均からの距離)をその一定の係数でスケーリングします

または、 CSS仕様 で示唆されているように、線の傾きと切片を選択するだけです。

<feFuncR type="linear" slope="[amount]" intercept="-(0.5 * [amount]) + 0.5"/>

_type='linear'_という用語に注意してください。二次スケーリング関数、 輝度ベース 調整、または ヒストグラムマッチング とは対照的に、 RGB色空間 で線形コントラスト調整を行っています。

ジオメトリクラスから思い出すと、線の式は_y=mx+b_です。 yは後の最終値、勾配mはコントラスト(またはfactor)、xは初期ピクセル値、bはy軸の切片(x = 0)で、線を垂直方向にシフトします。 y切片が原点(0,0)にないため、式はy=m(x-a)+bとして表すこともできます。ここで、aはxオフセットをシフトします。水平に並べます。

Formula for slope of a line

ここでは、このグラフは入力値(x軸)と結果(y軸)を表しています。 b、y切片(_m=0_の場合、コントラストなし)は128でなければならないことはすでにわかっています(これは、仕様から0.5に対してチェックできます-0.5 * 256 = 128の全範囲)。 xは元の値であるため、必要なのは勾配mとxオフセットaを把握することだけです。

まず、勾配mは「ライズオーバーラン」または_(y2-y1)/(x2-x1)_であるため、目的の線上にあることがわかっている2つのポイントが必要です。これらのポイントを見つけるには、いくつかのものをまとめる必要があります。

  • 私たちの関数は線切片グラフの形をしています
  • Y切片は_b = 128_にあります-傾き(コントラスト)に関係なく。
  • 予想される最大の「y」値は255で、最小値は0です。
  • 可能な「x」値の範囲は256です
  • ニュートラル値は常にニュートラルのままである必要があります:勾配に関係なく128 => 128
  • _0_のコントラスト調整では、入力と出力の間に変化はありません。つまり、1:1の勾配です。

これらすべてを総合すると、適用されるコントラスト(勾配)に関係なく、結果の線は_128,128_を中心に(そして中心に回転して)いると推測できます。 y切片がゼロ以外であるため、x切片もゼロではありません。 x範囲の幅は256で、中央が中央にあることがわかっているため、可能な範囲の半分(256/2 = 128)だけオフセットする必要があります。

Contrast function slopes

したがって、y=m(x-a)+bについては、m以外のすべてがわかっています。ジオメトリクラスからさらに2つの重要なポイントを思い出してください。

  • 線の位置が変わっても、線の傾きは同じです。つまり、mは、abの値に関係なく、同じままです。
  • 線の傾きは、線上の任意の2点を使用して見つけることができます

傾きの説明を簡単にするために、座標の原点をx切片(-128)に移動し、abをしばらく無視してみましょう。元の線は(0,0)を通過し、線の2番目の点が(255,255)のx(入力)とy(出力)の両方の全範囲から離れていることがわかります。

新しい線を(0,0)でピボットさせるので、それを、最終的なコントラスト勾配mに従う新しい線上の点の1つとして使用できます。 2番目のポイントは、現在の端を(255,255)である程度移動することによって決定できます。単一の入力(contrast)に制限されており、線形関数を使用しているため、この2番目のポイントはグラフのx方向とy方向に等しく移動します。

Adjusting the contrast slope

4つの可能な新しいポイントの(x、y)座標は_255 +/- contrast_になります。 xとyの両方を増減すると、元の1:1の行が維持されるため、図のように_+x, -y_と_-x, +y_を見てみましょう。

急な線(-x、+ y)は、正のcontrast調整に関連付けられています。 (x、y)座標は(_255 - contrast_、_255 + contrast_)です。浅い線(負のcontrast)の座標も同じ方法で見つかります。 contrastの最大の意味のある値は255になります-垂直線(完全なコントラスト、すべて黒または白)または水平線になる前に(255,255)の初期点を変換できる最大値に注意してください線(コントラストなし、すべて灰色)。

これで、新しい線上の2つの点の座標((0,0)と(_255 - contrast_、_255 + contrast_))が得られました。これを勾配方程式に接続し、次に、前のすべての部分を使用して、それを実線方程式に接続します。

y = m(x-a) + b

m = _(y2-y1)/(x2-x1)_ =>
_((255 + contrast) - 0)/((255 - contrast) - 0)_ =>
_(255 + contrast)/(255 - contrast)_

_a = 128_
_b = 128_

y = (255 + contrast)/(255 - contrast) * (x - 128) + 128[〜#〜] qed [〜#〜]

数学に関心のある人は、結果のmまたはfactorがスカラー(単位のない)値であることに気付くでしょう。 contrast計算の定数(_255_)と一致する限り、factorに任意の範囲を使用できます。たとえば、_+/-100_とfactor = (100 + contrast)/(100.01 - contrast)contrast範囲は、255にスケーリングするステップを排除するために実際に使用しました。説明を簡単にするために、上部のコードに_255_を残しました。


「魔法」についての注意259

ソース記事 は「魔法」259を使用していますが、著者は理由を覚えていないことを認めています。

「これを自分で計算したのか、本やオンラインで読んだのか思い出せません。」.

259は実際には255またはおそらく256である必要があります-各ピクセルの各チャネルで可能なゼロ以外の値の数。元のfactor計算では、259/255はキャンセルされます-技術的には1.01ですが、最終値は整数であるため、すべての実用的な目的で1になります。したがって、この外側の項は破棄できます。ただし、実際には分母の定数に255を使用すると、数式にゼロ除算エラーが発生する可能性があります。わずかに大きい値(たとえば、259)に調整すると、結果に重大なエラーを発生させることなく、この問題を回避できます。エラーが少なく、(うまくいけば)初心者には「魔法」が少ないように見えるので、代わりに255.01を使用することにしました。

私が知る限り、 使用するものに大きな違いはありません -正のコントラストの増加が少ない低コントラスト値の狭帯域でのわずかな対称差を除いて、同じ値が得られます。両方のバージョンを繰り返しラウンドトリップして元のデータと比較したいのですが、この回答にはすでに時間がかかりすぎていました。 :)

21
brichins

Schahriar SaffarSharghによる回答を試した後、コントラストが動作するように動作していませんでした。私はついにこのアルゴリズムに出くわしました、そしてそれは魅力のように機能します!

アルゴリズムの詳細については、 この記事 とそのコメントセクションをお読みください。

function contrastImage(imageData, contrast) {

    var data = imageData.data;
    var factor = (259 * (contrast + 255)) / (255 * (259 - contrast));

    for(var i=0;i<data.length;i+=4)
    {
        data[i] = factor * (data[i] - 128) + 128;
        data[i+1] = factor * (data[i+1] - 128) + 128;
        data[i+2] = factor * (data[i+2] - 128) + 128;
    }
    return imageData;
}

使用法:

var newImageData = contrastImage(imageData, 30);

うまくいけば、これは誰かにとって時間の節約になるでしょう。乾杯!

26
Brian

暗闇と明かりを分離するか、技術的にはrgbスケールで127(R + G + B/3の平均)未満のものは黒で、127を超えるものは白であるため、この効果を使用する必要があることがわかりました。コントラストのレベルによって、値を引いた値を黒から10コントラストと言い、同じ値を白に追加します。

次に例を示します。RGBカラーの2つのピクセルがあります[105,40,200] | [255,200,150]つまり、最初のピクセルの場合、105 + 40 + 200 = 345、345/3 = 115、115は255の半分である127よりも小さいので、[0,0,0]に近いピクセルと見なします。したがって、10のコントラストをマイナスしたい場合は、平均で各色から10を取ります。したがって、各色の値を、この場合は115であった合計の平均で割り、コントラストで乗算し、から最終値を差し引く必要があります。その特定の色:

たとえば、ピクセルから105(赤)を取得するので、RGBの平均の合計で割ります。これは115で、コントラスト値10の10倍(105/115)* 10で、約9になります(切り上げる必要があります)。次に、その9を105から離して、色が96になるようにします。暗いピクセルに10のコントラストを付けた後の赤。

したがって、ピクセルの値を続行すると、[96,37,183]になります。 (注:コントラストのスケールはあなた次第です!しかし、私の最終的には、1から255のようなスケールに変換する必要があります)

明るいピクセルについても、コントラスト値を減算する代わりに加算することを除いて、同じことを行います。そして、255または0の制限に達すると、その特定の色の加算と減算を停止します。したがって、より明るいピクセルである私の2番目のピクセルは[255,210,157]になります。

コントラストを追加すると、明るい色が明るくなり、暗い色が暗くなるため、画像にコントラストが追加されます!

これがサンプルのJavascriptコードです(私はまだ試していません):

var data = imageData.data;
for (var i = 0; i < data.length; i += 4) {
 var contrast = 10;
 var average = Math.round( ( data[i] + data[i+1] + data[i+2] ) / 3 );
  if (average > 127){
    data[i] += ( data[i]/average ) * contrast;
    data[i+1] += ( data[i+1]/average ) * contrast;
    data[i+2] += ( data[i+2]/average ) * contrast;
  }else{
    data[i] -= ( data[i]/average ) * contrast;
    data[i+1] -= ( data[i+1]/average ) * contrast;
    data[i+2] -= ( data[i+2]/average ) * contrast;
  }
}

このjavascriptの実装は、「コントラスト」のSVG/CSS3定義に準拠しています(次のコードは、キャンバスイメージを同じようにレンダリングします)。

/*contrast filter function*/
//See definition at https://drafts.fxtf.org/filters/#contrastEquivalent
//pixels come from your getImageData() function call on your canvas image
contrast = function(pixels, value){
    var d = pixels.data;
    var intercept = 255*(-value/2 + 0.5);
    for(var i=0;i<d.length;i+=4){
        d[i] = d[i]*value + intercept;
        d[i+1] = d[i+1]*value + intercept;
        d[i+2] = d[i+2]*value + intercept;
        //implement clamping in a separate function if using in production
        if(d[i] > 255) d[i] = 255;
        if(d[i+1] > 255) d[i+1] = 255;
        if(d[i+2] > 255) d[i+2] = 255;
        if(d[i] < 0) d[i] = 0;
        if(d[i+1] < 0) d[i+1] = 0;
        if(d[i+2] < 0) d[i+2] = 0;
    }
    return pixels;
}
2
Escher

ヴィンティングによって、あなたがLUTSを適用しようとしていると思います。最近、私はキャンバスウィンドウにカラートリートメントを追加しようとしています。キャンバスウィンドウに実際に「LUTS」を適用したい場合は、imageDataが返す配列をLUTのRGB配列に実際にマップする必要があると思います。

(光の錯覚から)例として、1D LUTの開始は次のようになります。注:厳密に言えば、これは3x 1D LUTであり、各色(R、G、B)は1DLUTです。

R, G, B 
3, 0, 0 
5, 2, 1 
7, 5, 3 
9, 9, 9

つまり、次のことを意味します。

For an input value of 0 for R, G, and B, the output is R=3, G=0, B=0 
For an input value of 1 for R, G, and B, the output is R=5, G=2, B=1 
For an input value of 2 for R, G, and B, the output is R=7, G=5, B=3 
For an input value of 3 for R, G, and B, the output is R=9, G=9, B=9

これは奇妙なLUTですが、R、G、またはB入力の特定の値に対して、R、G、およびB出力の特定の値があることがわかります。

したがって、ピクセルのRGBの入力値が3、1、0の場合、出力ピクセルは9、2、0になります。

この間、imageDataを操作した後、Uint8Arrayが返され、その配列の値が10進数であることに気付きました。ほとんどの3DLUTSは16進数です。したがって、このすべてのマッピングの前に、最初に配列全体で何らかのタイプの16進数から12進数への変換を行う必要があります。

1
Jay

OpenCVのドキュメントを見て、これをどのように達成できるかを確認できます: 明るさとコントラストの調整

次に、デモコードがあります。

 double alpha; // Simple contrast control: value [1.0-3.0]
 int beta;     // Simple brightness control: value [0-100]

 for( int y = 0; y < image.rows; y++ )
 { 
      for( int x = 0; x < image.cols; x++ )
      { 
          for( int c = 0; c < 3; c++ )
          {
              new_image.at<Vec3b>(y,x)[c] = saturate_cast<uchar>( alpha*( image.at<Vec3b>(y,x)[c] ) + beta );
          }
      }
 }

あなたはjavascriptに翻訳できると思います。

1
karlphillip

これはあなたが探している式です...

var data = imageData.data;
if (contrast > 0) {

    for(var i = 0; i < data.length; i += 4) {
        data[i] += (255 - data[i]) * contrast / 255;            // red
        data[i + 1] += (255 - data[i + 1]) * contrast / 255;    // green
        data[i + 2] += (255 - data[i + 2]) * contrast / 255;    // blue
    }

} else if (contrast < 0) {
    for (var i = 0; i < data.length; i += 4) {
        data[i] += data[i] * (contrast) / 255;                  // red
        data[i + 1] += data[i + 1] * (contrast) / 255;          // green
        data[i + 2] += data[i + 2] * (contrast) / 255;          // blue
    }
}

それが役に立てば幸い!

0
fforgoso