我一直在编写一个图像处理程序,该程序通过HTML5 Canvas 像素处理来应用效果。我已经实现了阈值处理,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中的图像对比度提供良好的算法,我将感到非常高兴。

最佳答案

更快的选项(基于Escher's approach)是:

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;
}

推导类似于以下内容;这个版本在数学上是相同的,但是运行速度要快得多。

原始答案

这是简化版,其中包含方法already discussed(基于this article)的解释:
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)
}

笔记
  • 我选择使用contrast+/- 100范围代替原始的+/- 255。对于不了解基本概念的用户或程序员而言,百分比值似乎更直观。另外,我的用法始终与UI控件相关;从-100%到+ 100%的范围允许我直接标记和绑定(bind)控制值,而不用进行调整或解释。
  • 即使calculated values can far exceed the allowable range,此算法也不包括范围检查-这是因为ImageData对象基础的数组是Uint8ClampedArrayAs MSDN explains,使用Uint8ClampedArray为您处理范围检查:



  • 用法

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

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

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

    html - HTML5 Canvas图像对比度-LMLPHP

    代码示例:

    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)的距离。调整对比度意味着增大或减小当前值和平均值之间的差异。

    该算法然后解决的问题是:
  • 选择一个常数因子以通过
  • 调整对比度
  • 对于每个像素的每个颜色通道,按该常数因子
  • 缩放“对比度”(与平均值的距离)

    或者,如CSS spec所示,只需选择直线的斜率和截距即可:



    注意术语type='linear';我们正在RGB color space中进行线性对比度调整,这与二次缩放函数,luminence-based调整或histogram matching相反。

    如果从几何类中回想起,则直线的公式为y=mx+by是我们要得到的最终值,坡度m是对比度(或factor),x是初始像素值,b是y轴的截距(x = 0),垂直移动该线。还请记住,由于y截距不在原点(0,0)处,所以该公式也可以表示为 y=m(x-a)+b ,其中a是水平偏移线的x偏移量。

    html - HTML5 Canvas图像对比度-LMLPHP

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

    html - HTML5 Canvas图像对比度-LMLPHP

    因此,现在对于y=m(x-a)+b,我们知道m以外的所有信息。回想一下几何类中的两个要点:
  • 线即使位置发生变化也具有相同的斜率;也就是说,不管ma的值如何,b都保持不变。
  • 可以使用
  • 线上的任意2个点找到直线的斜率

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

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

    html - HTML5 Canvas图像对比度-LMLPHP

    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)。我们将其插入斜率方程式,然后使用之前的所有部分将其插入全线方程式:



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

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

    source article使用“魔术” 259,尽管作者承认他不记得为什么:



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

    据我所知,it doesn't make much difference which you use-您获得相同的值,除了在低对比度值的窄带中具有较小的对称差异,而正对比度的增加较低。我很好奇要反复往返两个版本并与原始数据进行比较,但是这个答案已经花了太长时间。 :)

    关于html - HTML5 Canvas图像对比度,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/10521978/

    10-11 00:51