HTML5 холст контраст изображения
Я пишу программу обработки изображений, которая применяет эффекты через обработку пикселей холста HTML5. Я достиг порогового значения, винтажа и цветовых манипуляций с пикселями, но невероятно, что я не могу изменить контраст изображения! Я пробовал несколько решений, но я всегда получаю слишком много яркости в изображении и меньше контрастного эффекта , и я не планирую использовать какие-либо библиотеки 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] имеют красный цвет. Таким образом, если 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.
7 ответов:
Более быстрый вариант (основанный на подходе Эшера ):
Вывод аналогичен нижеприведенному; эта версия математически та же, но работает гораздо быстрее.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
. Процентное значение кажется более интуитивным для пользователей или программистов, которые не понимают основных понятий. Кроме того, мое использование всегда привязано к элементам управления пользовательского интерфейса; диапазон от -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 spec, просто выбирая наклон и перехват линии:
<feFuncR type="linear" slope="[amount]" intercept="-(0.5 * [amount]) + 0.5"/>
Обратите внимание на термин
type='linear'
; мы выполняем линейную настройку контраста в цветовом пространстве RGB , в отличие от квадратичной функции масштабирования, регулировка на основе яркости илисогласование гистограммы .Если вы помните из класса геометрии, формула для линии
Для наших целей этот график представляет входное значение (ось x) и результат (ось y). Мы уже знаем, чтоy=mx+b
.y
- это конечное значение, которое мы ищем, наклонm
- это контраст (илиfactor
),x
является начальным значением пикселя, аb
- перехватом оси y (x=0), которая сдвигает линию по вертикали. Напомним также, что поскольку y-перехват не находится в начале координат (0,0), то формула также может быть представлена в видеy=m(x-a)+b
, где именноa
- смещение x, смещающее линию по горизонтали.b
, y-Перехват (дляm=0
, без контраста) должен быть 128 (что мы можем проверить против 0.5 из спецификации - 0.5 * полный диапазон 256 = 128).x
-это наше исходное значение, поэтому все, что нам нужно, это вычислить наклонm
и смещение xa
.Во-первых, склон
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), и мы знаем, что вторая точка на линии лежит далеко от полного диапазона какx
(вход), так иy
(выход) в (255,255).Мы позволим новой линии развернуться на (0,0), поэтому мы можем использовать ее как одну из точек на новой линии, которая будет следовать за нашим конечным контрастным наклоном
m
. Вторая точка может быть определена путем перемещения текущего конца в (255,255) на некоторую величину; так как мы ограниченная одним входом (contrast
) и использующая линейную функцию, эта вторая точка будет одинаково перемещаться в направленияхx
иy
на нашем графике.Координаты (x,y) 4 возможных новых точек будут
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
QEDМатематик заметит, что результирующее
m
илиfactor
является скалярным (безразмерным) значением; вы можете использовать любой диапазон, который вы хотите дляcontrast
, если он соответствует константе (255
) вfactor
расчет. Например, диапазонcontrast
+/-100
иfactor = (100 + contrast)/(100.01 - contrast)
, который я действительно использую, чтобы исключить шаг масштабирования до 255; я просто оставил255
в коде вверху, чтобы упростить объяснение.
Примечание о "магии" 259
Исходная статья использует "магию" 259, хотя автор признает, что не помнит, почему:
259 действительно должно быть 255 или, возможно, 256-число возможных ненулевых значений для каждого канала каждого пикселя. Обратите внимание, что в исходном вычислении" я не могу вспомнить, рассчитал ли я это сам, или я прочитал это в книге или онлайн.".
factor
259/255 отменяет - технически 1.01, но конечные значения являются целыми числами, поэтому 1 для всех практических целей. Так что этот внешний термин можно отбросить. На самом деле использование 255 для константы в знаменателе, однако, вводит возможность деления на нулевую ошибку в Формуле; корректировка немного большее значение (скажем, 259) позволяет избежать этой проблемы, не внося существенных ошибок в результаты. Я решил использовать 255.01 вместо этого, так как ошибка ниже, и это (надеюсь) кажется менее "волшебным" новичку.Насколько я могу судить, не имеет большого значения, какие вы используете - вы получаете одинаковые значения, за исключением незначительных симметричных различий в узкой полосе низких значений контраста с низким положительным увеличением контраста. Мне было бы любопытно обойти обе версии неоднократно и сравнить с исходными данными, но этот ответ уже занял слишком много времени. :)
После попытки ответа Шахрияра Саффаршарга, он вел себя не так, как должен был бы вести себя контраст. Я наконец-то наткнулся на этот алгоритм, и он работает как заклинание!
Для получения дополнительной информации об алгоритме прочитайтеЭту статью и раздел комментариев.
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);
Надеюсь, что это будет экономия времени для кого-то. Ура!
Я выяснил, что вы должны использовать эффект, отделяя темные и светлые цвета или технически все, что меньше 127 (среднее значение R+G+B / 3) в шкале rgb, является черным, а больше 127-белым, поэтому по уровню контраста вы минус значение, скажем, 10 контраста от черных и добавляете то же значение к белым!
Вот пример:: У меня есть два пикселя с цветами RGB, [105,40,200] | [255,200,150] Так что я знаю, что для моего первого пикселя 105 + 40 + 200 = 345, 345/3 = Сто пятнадцать и 115 меньше моей половины 255, которая равна 127, поэтому я считаю пиксель ближе к [0,0,0], поэтому если я хочу минус 10 контраста, то я забираю 10 от каждого цвета в среднем Таким образом, я должен разделить значение каждого цвета на среднее значение итога, которое было 115 для этого случая, и умножить его на мой контраст и вычеркнуть конечное значение из этого конкретного цвета:
Например, я возьму 105 (красный) из моего пикселя, поэтому я делю его на общее среднее значение RGB. что составляет 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; } }
Вы можете взглянуть на документы 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.
С помощью vintaging я предполагаю, что вы пытаетесь применить LUTS..В последнее время я пытаюсь добавить цветовые решения к окнам холста. Если вы действительно хотите применить "LUTS" к окну canvas, я считаю, что вам нужно на самом деле сопоставить массив, который imageData возвращает в массив RGB LUT.
(от иллюзии света) В качестве примера начало 1D LUT может выглядеть примерно так: Примечание: строго говоря, это 3x 1D LUTs, так как каждый цвет (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
Что странно, но вы видите, что для заданного значения R, G или B на входе есть заданное значение R, G и B на выходе.
Таким образом, если бы пиксель имел входное значение 3, 1, 0 для RGB, выходной пиксель был бы 9, 2, 0.
Во время этого я также понял после игры с imageData, что он возвращает Uint8Array и что значения в этом массиве являются десятичными. Большинство 3D-лут являются шестнадцатеричными. Таким образом, вы сначала должны сделать некоторый тип преобразования hex в dec на всем массиве, прежде чем все это сопоставление.
Эта реализация 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; }
Это формула, которую вы ищете ...
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 } }
Надеюсь, это поможет!