Контраст изображения холста HTML5
Я пишу программу обработки изображений, которая применяет эффекты через обработку пиксельных холстов HTML5. Я добился манипуляций с пикселами Thresholding, 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, что означает, что данные [i] - это красный цвет. Поэтому, если данные [i] = данные [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.
Ответы
Ответ 1
После того, как Шариар Саффар Шарг попытался ответить, он не вел себя так, как должен вести себя контраст. Я, наконец, натолкнулся на этот алгоритм, и он работает как шарм!
За дополнительной информацией об алгоритме читайте эту статью и раздел комментариев.
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);
Надеюсь, это будет время для кого-то. Ура!
Ответ 2
Более быстрый вариант (основанный на подход 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)
}
Примечания
-
Я выбрал вместо +/- 255
диапазон contrast
+/- 100
. Процентное значение кажется более интуитивным для пользователей или программистов, которые не понимают базовые концепции. Кроме того, мое использование всегда связано с элементами управления пользовательского интерфейса; диапазон от -100% до + 100% позволяет мне наклеивать и связывать контрольное значение напрямую, а не корректировать или объяснять его.
-
Этот алгоритм не включает проверку диапазона, хотя расчетные значения могут значительно превышать допустимый диапазон - это потому, что массив, лежащий в основе объекта ImageData, является Uint8ClampedArray
. Как объясняется MSDN, при Uint8ClampedArray
проверка диапазона обрабатывается для вас:
", если вы указали значение, которое вне диапазона [0,255], вместо него будут установлены 0 или 255."
Использование
Обратите внимание, что хотя базовая формула является довольно симметричной (допускает круговое отключение), данные теряются при высоких уровнях фильтрации, поскольку пиксели допускают только целочисленные значения. Например, к тому времени, когда вы удалите изображение на экстремальные уровни ( > 95% или около того), все пиксели в основном представляют собой однородный серый цвет (в пределах нескольких цифр от среднего возможного значения 128). Поворот контрастности снова приводит к уплощенному цветовому диапазону.
Кроме того, порядок операций имеет важное значение при применении нескольких корректировок контрастности - насыщенные значения "выдувают" (превышают максимальное сжатие 255), что означает насыщение, а затем удаление насыщения приведет к получению более темного изображения. Однако при насыщении и затем насыщении не так много потерь данных, потому что значения подсветки и тени приглушаются, а не обрезаны (см. Пояснение ниже).
Вообще говоря, при применении нескольких фильтров лучше начинать каждую операцию с исходных данных и повторно применять каждую настройку по очереди, вместо того, чтобы пытаться изменить предыдущее изменение - по крайней мере, на качество изображения. Производительность или другие требования могут различаться по-разному для каждой ситуации.
![Контрастные примеры Mandrill]()
Пример кода:
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>
Ответ 3
Я выяснил, что вы должны использовать эффект, разделяя темные и огни или технически все, что меньше 127 (среднее значение R + G + B/3) в шкале rgb - черное, и более 127 - это белый, поэтому по вашему уровню контрастности вы минус значение, указывающее 10 контрастов с черными, и добавляете то же значение к белым!
Вот пример:
У меня два пикселя с цветами RGB, [105,40,200] | [255200150]
Поэтому я знаю, что для моего первого пикселя 105 + 40 + 200 = 345, 345/3 = 115
и 115 меньше, чем моя половина 255, что составляет 127, поэтому я считаю, что пиксель ближе к [0,0,0], поэтому, если я хочу минус 10 контрастов, тогда я убираю 10 из каждого цвета на нем в среднем
Таким образом, я должен разделить каждое значение цвета на общее среднее значение, равное 115 для этого случая, и умножить его на мой контраст и минус окончательное значение от этого конкретного цвета:
Например, я возьму 105 (красный) из моего пикселя, поэтому я разделил его на общий RGB avg. что составляет 115 и умножает на это значение моего контраста 10, (105/115) * 10, которое дает вам что-то около 9 (вы должны округлить его!), а затем отнести это 9 от 105, чтобы цвет стал 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;
}
}
Ответ 4
Вы можете взглянуть на документы OpenCV, чтобы узнать, как вы могли бы это сделать: Настройки яркости и контрастности.
Затем появится демо-код:
double alpha; // Simple contrast control: value [1.0-3.0]
int beta; // Simple brightness control: value [0-100]
for( int y = 0; y < image.rows; y++ )
{
for( int x = 0; x < image.cols; x++ )
{
for( int c = 0; c < 3; c++ )
{
new_image.at<Vec3b>(y,x)[c] = saturate_cast<uchar>( alpha*( image.at<Vec3b>(y,x)[c] ) + beta );
}
}
}
который, я думаю, вы можете перевести на javascript.
Ответ 5
Посредством vintaging я предполагаю, что вы пытаетесь применить LUTS. В последнее время я пытаюсь добавить обработку цвета в окна холста. Если вы хотите применить "LUTS" к окну canvas, я считаю, что вам нужно на самом деле сопоставить массив, который возвращает imageData в массив RGB LUT.
(Из иллюзии Света)
В качестве примера начало 1D LUT может выглядеть примерно так:
Примечание: строго говоря, это 3x 1D LUT, так как каждый цвет (R, G, B) является 1D LUT
R, G, B
3, 0, 0
5, 2, 1
7, 5, 3
9, 9, 9
Это означает, что:
For an input value of 0 for R, G, and B, the output is R=3, G=0, B=0
For an input value of 1 for R, G, and B, the output is R=5, G=2, B=1
For an input value of 2 for R, G, and B, the output is R=7, G=5, B=3
For an input value of 3 for R, G, and B, the output is R=9, G=9, B=9
Что является странным LUT, но вы видите, что для заданного значения R, G или B входных данных есть заданное значение выходных данных R, G и B.
Итак, если пиксель имел входное значение 3, 1, 0 для RGB, выходной пиксель был бы равен 9, 2, 0.
Во время этого я также понял после игры с imageData, что он возвращает Uint8Array и что значения в этом массиве являются десятичными. Большинство 3D-сгустков являются шестнадцатеричными. Поэтому сначала вам нужно сделать какой-либо тип преобразования hex-dec в весь массив до всего этого сопоставления.
Ответ 6
Эта реализация 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;
}
Ответ 7
Это формула, которую вы ищете...
var data = imageData.data;
if (contrast > 0) {
for(var i = 0; i < data.length; i += 4) {
data[i] += (255 - data[i]) * contrast / 255; // red
data[i + 1] += (255 - data[i + 1]) * contrast / 255; // green
data[i + 2] += (255 - data[i + 2]) * contrast / 255; // blue
}
} else if (contrast < 0) {
for (var i = 0; i < data.length; i += 4) {
data[i] += data[i] * (contrast) / 255; // red
data[i + 1] += data[i + 1] * (contrast) / 255; // green
data[i + 2] += data[i + 2] * (contrast) / 255; // blue
}
}
Надеюсь, что это поможет!