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 18

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)
}

Примечания

  1. Я решил использовать contrast диапазон +/- 100 вместо оригинал +/- 255. Процентное значение кажется более интуитивным для пользователей или программистов, которые не понимают основных понятий. Кроме того, мое использование всегда привязано к элементам управления пользовательского интерфейса; диапазон от -100% до +100% позволяет мне помечать и связывать значение элемента управления напрямую, а не настраивать или объяснять его.

  2. Этот алгоритм не включает проверку диапазона, хотя вычисленные значения могут значительно превышать допустимый диапазон - это связано с тем, что массив, лежащий в основе Объект 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) . Настройка контраста означает увеличение или уменьшение разницы между текущим значением и средним значением.

Задача, которую решает алгоритм, состоит в том, чтобы:

  1. выбрал постоянный коэффициент для регулировки контраста по
  2. для каждого цветового канала каждого пикселя масштабируйте "контраст" (расстояние от среднего) на этот постоянный коэффициент

Или, как намекается в CSS spec, просто выбирая наклон и перехват линии:

<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 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), и мы знаем, что вторая точка на линии лежит далеко от полного диапазона как 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
    }
}

Надеюсь, это поможет!