HTML5 Canvas图像对比度
image-processing
4
0

我一直在编写一个图像处理程序,该程序通过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;该像素的红色通道的亮度将增加到两倍。例:

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 canvas中的图像对比度提供良好的算法,我将感到非常高兴。

参考资料:
Stack Overflow
收藏
评论
共 4 个回答
高赞 时间 活跃

在尝试了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);

希望这将为某人节省时间。干杯!

收藏
评论

更快的选择(基于Escher的方法 )是:

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. 我选择使用+/- 100contrast范围,而不是原来的+/- 255 。对于不了解基本概念的用户或程序员来说,百分比值似乎更直观。另外,我的用法始终与UI控件相关;从-100%到+ 100%的范围允许我直接标记和绑定控制值,而不用进行调整或解释。

  2. 该算法不包括范围检查,即使计算出的值可能会远远超出允许的范围 ,这是因为ImageData对象下面的数组是Uint8ClampedArray正如MSDN解释的那样 ,使用Uint8ClampedArray可以为您处理范围检查:

“如果您指定的值超出[0,255]的范围,则将设置为0或255。”

用法

请注意,虽然基本公式是相当对称的(允许往返),但由于像素仅允许整数值,因此在高级别过滤时会丢失数据。例如,当您将图像去饱和度达到极高的级别(> 95%左右)时,所有像素基本上都是均匀的中等灰度(在128的平均可能值的几位之内)。再次将对比度调高会导致色彩范围变平。

同样,在应用多个对比度调整时,操作顺序也很重要-饱和值会迅速“吹灭”(超出最大值255),这意味着高度饱和然后再去饱和将导致整体图像变暗。然而,去饱和然后再饱和不会造成太多数据丢失,因为高光和阴影值被静音,而不是被裁剪(请参阅下面的说明)。

一般来说,当应用多个滤镜时,最好从原始数据开始每个操作,然后依次重新应用每个调整,而不是尝试撤消以前的更改-至少对于图像质量而言。对于每种情况,性能速度或其他要求可能会有所不同。

山d对比实例

代码示例:

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+by是我们要获得的最终值,斜率m是对比度(或factor ), x是初始像素值, b是y轴(x = 0)的截距,该轴垂直移动。还请记住,由于y截距不在原点(0,0)处,所以公式也可以表示为y=m(xa)+b ,其中a是水平偏移线的x偏移量。

直线斜率的公式

就我们的目的而言,该图表示输入值(x轴)和结果(y轴)。我们已经知道b ,y轴截距(对于m=0 ,没有对比度)必须为128(我们可以根据规格检查0.5的值-0.5 * 256的整个范围= 128)。 x是我们的原始值,因此我们需要找出斜率m和x偏移a

首先,斜率m为“ (y2-y1)/(x2-x1)程上升”,即(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。

对比度函数斜率

所以现在对于y=m(xa)+b ,我们知道除m之外的所有东西。回想一下几何类中的两个要点:

  • 直线即使位置发生变化也具有相同的斜率;也就是说,无论ab的值如何, m相同。
  • 可以使用直线上的任意两个点来找到直线的斜率

为了简化斜率讨论,让我们将坐标原点移动到x轴截距(-128),暂时忽略ab 。现在,我们的原始线将绕过(0,0),并且我们知道该线上的第二点位于(255,255)处的x (输入)和y (输出)的整个范围内。

我们将使新线在(0,0)处枢轴旋转,因此可以将其用作新线上将遵循最终对比度斜率m 。第二点可以通过将(255,255)处的当前端移动一定量来确定;由于我们仅限于单个输入( contrast )并使用线性函数,因此第二个点将在图形上的xy方向上均等移动。

调整对比度斜率

4个可能的新点的(x,y)坐标将为255 +/- contrast 。由于增加和减少x和y会使我们停留在原始的1:1线上,因此让我们仅看一下+x, -y-x, +y ,如图所示。

较陡的线(-x,+ y)与正的contrast调整相关;它的(x,y)坐标是( 255 - contrast255 + contrast )。较浅的线(负contrast )的坐标以相同的方式找到。请注意, contrast的最大有意义值为255- (255,255)的初始点在产生垂直线(全对比度,全黑或白色)或水平线(无对比度,全灰色)之前可以转换的最大值)。

所以现在我们在新行上有了两个点的坐标-(0,0)和( 255 - contrast255 + 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

数学上会注意到,所得的mfactor是标量(无单位)值;您可以使用任何想要的contrast范围,只要它与factor计算中的常数( 255 )相匹配即可。例如, contrast范围为+/-100并且factor = (100 + contrast)/(100.01 - contrast) ,这是我真正用来消除缩放到255的步骤;为了简化说明,我只在顶部的代码中保留了255


关于“魔术”的注意事项259

原始文章使用的是“魔术” 259,尽管作者承认他不记得为什么:

“我不记得自己是自己计算过的,还是在书中或在线阅读过的。”

259实际上应该是255或256-每个像素的每个通道可能的非零值的数量。请注意,在原始factor计算中,259/255抵消了-从技术上讲是1.01,但是最终值是整数,因此对于所有实际目的而言都是1。因此,该外部术语可以被丢弃。但是,实际上使用255作为分母中的常数会导致公式中被零除的错误;调整为稍大的值(例如259)可以避免此问题,而不会给结果带来明显的误差。我选择使用255.01来代替,因为错误降低了(希望)对于新手来说似乎不太“神奇”。

据我所知, 使用哪种方法并没有多大区别 -您可以得到相同的值,只是在低对比度值的窄带中具有较小的对称差异,而正对比度的增加较低。我很好奇要反复往返两个版本并与原始数据进行比较,但是这个答案已经花了太长时间。 :)

收藏
评论

此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;
}
收藏
评论

我发现您必须使用这种效果,即将黑暗和明亮分开,或者从技术上讲,rgb比例中小于127(R + G + B / 3的平均值)的是黑色,大于127是白色,因此根据您的对比度水平,您减去一个值,例如与黑人的对比度为10,并向白人添加相同的值!

这是一个示例:我有两个具有RGB颜色的像素,[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(105/115)* 10,它会为您提供9左右的值(您必须将其四舍五入!),然后从105取9以便颜色变为96,所以我在暗像素上具有10对比度后呈红色。

因此,如果继续进行下去,像素的值将变为[96,37,183]! (注意:对比度的大小取决于您!但是最后我应该将其转换为1到255之类的比例)

对于较亮的像素,我也执行相同的操作,只是不减去对比度值,而是添加对比度值!如果达到255或0的限制,则停止对该特定颜色进行加减运算!因此,我的第二个像素(较亮的像素)变为[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;
  }
}
收藏
评论
新手导航
  • 社区规范
  • 提出问题
  • 进行投票
  • 个人资料
  • 优化问题
  • 回答问题

关于我们

常见问题

内容许可

联系我们

@2020 AskGo
京ICP备20001863号