我一直在编写一个图像处理程序,该程序通过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)控制值,而不用进行调整或解释。 Uint8ClampedArray
。 As MSDN explains,使用Uint8ClampedArray
为您处理范围检查:用法
请注意,尽管基本公式是相当对称的(允许往返),但由于像素仅允许整数值,因此在高级别过滤时会丢失数据。例如,当您将图像去饱和度达到极高的级别(> 95%左右)时,所有像素基本上都是均匀的中等灰度(在128个可能的平均值的几位之内)。再次将对比度调高会导致色彩范围变平。
同样,在应用多个对比度调整时,操作顺序也很重要-饱和值会迅速“爆裂”(超出最大值255),这意味着高度饱和然后降低饱和度会导致整体图像变暗。然而,去饱和然后再饱和不会造成太多数据丢失,因为高光和阴影值被静音,而不是被裁剪(请参阅下面的说明)。
一般来说,当应用多个滤镜时,最好从原始数据开始每个操作,然后依次重新应用每个调整,而不是尝试撤消以前的更改-至少对于图像质量而言。对于每种情况,性能速度或其他要求可能会有所不同。
代码示例:
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+b
。 y
是我们要得到的最终值,坡度m
是对比度(或factor
),x
是初始像素值,b
是y轴的截距(x = 0),垂直移动该线。还请记住,由于y截距不在原点(0,0)处,所以该公式也可以表示为 y=m(x-a)+b
,其中a
是水平偏移线的x偏移量。就我们的目的而言,该图表示输入值(x轴)和结果(y轴)。我们已经知道,y截距
b
(对于m=0
,没有对比度)必须为128(我们可以根据规格检查0.5-0.5 * 256的整个范围= 128)。 x
是我们的原始值,因此我们需要找出斜率m
和x-offset a
。首先,斜率
m
是“上升越过”或(y2-y1)/(x2-x1)
-因此我们需要2个点已知在所需的线上。找到这些要点需要结合以下几点:b = 128
-不管斜率(对比度)如何。 0
的调整应导致输入和输出之间没有变化;即1:1的斜率。 综合考虑所有这些因素,我们可以得出结论,无论施加了什么对比度(斜率),我们得到的线都将以
128,128
为中心(并绕其旋转)。由于我们的y截距非零,因此x截距也非零;我们知道x范围为256宽,并且居中居中,因此它必须偏移可能范围的一半:256/2 = 128。因此,现在对于
y=m(x-a)+b
,我们知道m
以外的所有信息。回想一下几何类中的两个要点:m
和a
的值如何,b
都保持不变。 为了简化斜率讨论,让我们将坐标原点移动到x轴截距(-128),暂时忽略
a
和b
。现在,我们的原始行将绕过(0,0),我们知道该行的第二点位于(255,255)处的x
(输入)和y
(输出)的整个范围内。我们将使新线在(0,0)处旋转,因此可以将其用作新线上将遵循最终对比度斜率
m
的点之一。第二点可以通过将(255,255)处的当前端移动一定量来确定;由于我们仅限于单个输入(contrast
)并使用线性函数,因此第二个点将在图形上的x
和y
方向上平均移动。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)的初始点可以转换,从而产生一条垂直线(全对比度,全黑或全白)或一条水平线(无对比度,全灰色)。所以现在我们在新行上有了两个点的坐标-(0,0)和(
255 - contrast
,255 + contrast
)。我们将其插入斜率方程式,然后使用之前的所有部分将其插入全线方程式:数学上会注意到结果
m
或factor
是标量(无单位)值;您可以为contrast
使用任何想要的范围,只要它与255
计算中的常数(factor
)相匹配即可。例如,contrast
范围为+/-100
和factor = (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/