更快的选择(基于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)
}
笔记
我选择使用contrast
范围+/- 100
代替原来的+/- 255
. 对于不了解基本概念的用户或程序员来说,百分比值似乎更直观。此外,我的使用总是与 UI 控件相关联;从 -100% 到 +100% 的范围允许我直接标记和绑定控制值,而不是调整或解释它。
该算法不包括范围检查,即使计算的值可能远远超出允许的范围- 这是因为 ImageData 对象下的数组是Uint8ClampedArray
. 正如 MSDN 解释的那样,Uint8ClampedArray
为您处理范围检查:
“如果您指定的值超出 [0,255] 的范围,将改为设置 0 或 255。”
用法
请注意,虽然基础公式是相当对称的(允许往返),但在高级过滤时数据会丢失,因为像素只允许整数值。例如,当您将图像去饱和到极端水平(> 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 规范中所暗示的,只需选择直线的斜率和截距:
<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 偏移量是水平移动直线。
出于我们的目的,该图表示输入值(x 轴)和结果(y 轴)。我们已经知道b
,y 截距(对于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。
所以现在y=m(x-a)+b
,我们什么都知道,除了m
。回想一下几何课中的两个更重要的点:
- 即使位置发生变化,线也具有相同的斜率;也就是说,无论和
m
的值如何,都保持不变。a
b
- 可以使用线上的任意 2 个点找到线的斜率
为了简化斜率讨论,让我们将坐标原点移动到 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
)。我们将其代入斜率方程,然后将其代入全线方程,使用之前的所有部分:
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
量子点
有数学头脑的人会注意到结果m
orfactor
是一个标量(无单位)值;您可以使用任何您想要的范围contrast
,只要它与计算中的常数 ( 255
)匹配即可factor
。例如,and 的范围contrast
,这是我真正用来消除缩放到 255 的步骤;我只是留在顶部的代码中以简化解释。+/-100
factor = (100 + contrast)/(100.01 - contrast)
255
关于“魔法”的注意事项 259
源文章使用了“魔法”259,尽管作者承认他不记得为什么:
“我不记得是我自己计算的,还是在书本或网上读到的。”。
259 实际上应该是 255 或者可能是 256 - 每个像素的每个通道的可能非零值的数量。请注意,在原始factor
计算中,259/255 抵消了 - 技术上是 1.01,但最终值是整数,因此 1 用于所有实际目的。所以这个外项可以被丢弃。但是,实际上使用 255 作为分母中的常数会在公式中引入除以零错误的可能性;调整到稍大的值(例如,259)可以避免这个问题,而不会给结果带来明显的错误。我选择使用 255.01 代替,因为错误较低,并且(希望)对新手来说似乎不那么“神奇”。
但据我所知,您使用的并没有太大区别- 除了在具有低正对比度增加的低对比度值的窄带中存在微小的对称差异外,您会得到相同的值。我很想反复往返两个版本并与原始数据进行比较,但是这个答案已经花费了太长时间。:)