Javascript圖像處理—亮度對(duì)比度應(yīng)用案例
前言
上一篇文章,我們講解了圖像處理中的卷積操作和平滑(也就是模糊)處理,這篇文章我們進(jìn)行亮度和對(duì)比度的變化。
其實(shí),亮度是啥玩意?
亮度就是比較亮眼咯……

實(shí)際上對(duì)于RGBA顏色空間,變亮其實(shí)就等于R、G、B三個(gè)通道同時(shí)加大,那么變暗就等于同時(shí)減小咯。
這比較好理解,因?yàn)樽畎档暮谏荝GB(0,0,0),而最亮的白色是RGB(255,255,255)。所以變亮應(yīng)該RGB各通道都要增大。
那么,對(duì)比度呢?
對(duì)比度,其實(shí)就是顏色差啦。
那么對(duì)于RGBA顏色空間,對(duì)比度變大其實(shí)就等于R、G、B三個(gè)通道同時(shí)乘以一個(gè)比例,因?yàn)檫@樣相近的顏色之間的差距就變大了,那么減小就是同時(shí)除以咯。
舉個(gè)例子,原來(lái)RGB(23,44,55)和RGB(33,44,55)相差只有10,但是一起乘以2以后,就變成了RGB(46,88,110)和RGB(66,88,110)
,相差變成了20了,也就是“顏色差”變大了。線性模型
newRGB = Contrast * RGB + Brightness
線性模型滿足上述公式,其中 Contrast表示對(duì)比度系數(shù),Brightness表示亮度系數(shù)。
線性模型實(shí)現(xiàn)比較簡(jiǎn)單,但是很容易就調(diào)出全白或者全黑的圖片,對(duì)于普通用戶來(lái)說(shuō)Contrast、Brightness選多少比較好也比較難確定。
所以,實(shí)際上在Photoshop里面使用的并不是線性模型,而是非線性模型。
非線性模型
非線性模型中對(duì)比度增大和閾值Threshold有關(guān):
當(dāng)Contrast >= 0時(shí):
newRGB = RGB + (RGB - Threshold) * (1 / (1 - Contrast / 255) - 1)
當(dāng)Contrast < 0時(shí):
newRGB = RGB + (RGB - Threshold) * Contrast / 255
那么當(dāng)對(duì)比度和亮度同時(shí)調(diào)整時(shí)候呢?
如果對(duì)比度大于0,先調(diào)整亮度,再調(diào)整對(duì)比度;當(dāng)對(duì)比度小于0時(shí),則相反,先調(diào)整對(duì)比度,再調(diào)整亮度。
最后一個(gè)問(wèn)題,閾值Threshold到底是什么,其實(shí)這個(gè)是圖片的灰度平均值。
實(shí)現(xiàn)代碼
var brightnessContrast = function(__src, __brightness, __contrast){
__src || error(arguments.callee, IS_UNDEFINED_OR_NULL/* {line} */);
if(__src.type === "CV_RGBA"){
var sData = __src.data,
width = __src.col,
height = __src.row,
dst = new Mat(height, width, CV_RGBA),
dData = dst.data,
brightness = Math.max(-255, Math.min(255, __brightness || 0)),
contrast = Math.max(-255, Math.min(255, __contrast || 0));
var gray = cvtColor(__src, CV_RGBA2GRAY),
allValue = 0,
gData = gray.data;
var y, x, c;
for(y = height; y--;){
for(x = width; x--;){
allValue += gData[y * width + x];
}
}
var r, g, b, offset, gAverage = (allValue / (height * width)) | 0;
for(y = height; y--;){
for(x = width; x--;){
offset = (y * width + x) * 4;
dData[offset] = sData[offset] + brightness;
dData[offset + 1] = sData[offset + 1] + brightness;
dData[offset + 2] = sData[offset + 2] + brightness;
if(contrast >= 0){
for(c = 3; c--;){
if(dData[offset + c] >= gAverage){
dData[offset + c] = dData[offset + c] + (255 - gAverage) * contrast / 255;
}else{
dData[offset + c] = dData[offset + c] - (gAverage * contrast / 255);
}
}
}else{
dData[offset] = dData[offset] + (dData[offset] - gAverage) * contrast / 255;
dData[offset + 1] = dData[offset + 1] + (dData[offset + 1] - gAverage) * contrast / 255;
dData[offset + 2] = dData[offset + 2] + (dData[offset + 2] - gAverage) * contrast / 255;
}
dData[offset + 3] = 255;
}
}
}else{
error(arguments.callee, UNSPPORT_DATA_TYPE/* {line} */);
}
return dst;
};
效果

- Javascript圖像處理思路及實(shí)現(xiàn)代碼
- Javascript圖像處理—為矩陣添加常用方法
- Javascript圖像處理—虛擬邊緣介紹及使用方法
- Javascript圖像處理—平滑處理實(shí)現(xiàn)原理
- Javascript圖像處理—閾值函數(shù)實(shí)例應(yīng)用
- Javascript圖像處理—圖像形態(tài)學(xué)(膨脹與腐蝕)
- javascript圖像處理—邊緣梯度計(jì)算函數(shù)
- javascript圖像處理—仿射變換深度理解
- java數(shù)字圖像處理基礎(chǔ)使用imageio寫(xiě)圖像文件示例
- Java圖像處理工具類
相關(guān)文章
2019年度web前端面試題總結(jié)(主要為Vue面試題)
轉(zhuǎn)眼2019又要過(guò)去了,作為一名前端碼農(nóng),又熬過(guò)一個(gè)沒(méi)日沒(méi)夜的年頭,頭發(fā)又少了不少,去年的學(xué)習(xí)計(jì)劃一半也沒(méi)完成,唉。。。 現(xiàn)在為大家總結(jié)一下這一年面試的幾家公司的關(guān)于前端面試題吧2020-01-01
基于js 字符串indexof與search方法的區(qū)別(詳解)
下面小編就為大家分享一篇基于js 字符串indexof與search方法的區(qū)別介紹,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2017-12-12
JavaScript CSS修改學(xué)習(xí)第五章 給“上傳”添加樣式
在所有的表單項(xiàng)里面,文件上傳部分是最難添加樣式的。IE支持一些(不是很多)樣式屬性,Mozilla很少,其他瀏覽器幾乎沒(méi)有?!盀g覽”按鈕在CSS操作里面也很難訪問(wèn)。2010-02-02
onkeyup,onkeydown和onkeypress的區(qū)別介紹
三者在事件的響應(yīng)上還有一點(diǎn)不同,就是onkeydown 、onkeypress事件響應(yīng)的時(shí)候輸入的字符并沒(méi)有被系統(tǒng)接受,而響應(yīng)onkeyup的時(shí)候,輸入流已經(jīng)被系統(tǒng)接受2013-10-10

