HTML5 Canvas Image Contrast

HTML5 Canvas image contrast

A faster option (based on Escher's approach) is:

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

Derivation similar to the below; this version is mathematically the same, but runs much faster.


Original answer

Here is a simplified version with explanation of an approach already discussed (which was based on this article):

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

Notes

  1. I have chosen to use a contrast range of +/- 100 instead of the original +/- 255. A percentage value seems more intuitive for users, or programmers who don't understand the underlying concepts. Also, my usage is always tied to UI controls; a range from -100% to +100% allows me to label and bind the control value directly instead of adjusting or explaining it.

  2. This algorithm doesn't include range checking, even though the calculated values can far exceed the allowable range - this is because the array underlying the ImageData object is a Uint8ClampedArray. As MSDN explains, with a Uint8ClampedArray the range checking is handled for you:

"if you specified a value that is out of the range of [0,255], 0 or 255 will be set instead."

Usage

Note that while the underlying formula is fairly symmetric (allows round-tripping), data is lost at high levels of filtering because pixels only allow integer values. For example, by the time you de-saturate an image to extreme levels (>95% or so), all the pixels are basically a uniform medium gray (within a few digits of the average possible value of 128). Turning the contrast back up again results in a flattened color range.

Also, order of operations is important when applying multiple contrast adjustments - saturated values "blow out" (exceed the clamped max value of 255) quickly, meaning highly saturating and then de-saturating will result in a darker image overall. De-saturating and then saturating however doesn't have as much data loss, because the highlight and shadow values get muted, instead of clipped (see explanation below).

Generally speaking, when applying multiple filters it's better to start each operation with the original data and re-apply each adjustment in turn, rather than trying to reverse a previous change - at least for image quality. Performance speed or other demands may dictate differently for each situation.

Mandrill contrast examples

Code Example:

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>

HTML5 Canvas contrast filter

Since you said you already adjusted the brightness and it sounds like all you need is the algorithm for contrast adjusting here's a thread for a C# method (Look at Update 2), the logic of the code is sound and will work in any program:

Adjust the contrast of an image in C# efficiently - Stack Overflow

Brightness and Contrast for a canvas image with javascript

There's at least one javascript library I know of which can do this, Pixastic

Usage might look like this.

Pixastic.process(canvas, 'brightness',
{
'brightness': 60,
'contrast': 0.5,
'leaveDOM': true
},
function(img) {
ctx.drawImage(img, 0, 0);
}
);

The library is kind of intended to work with images within your page and it replaces them with canvas elements which contain the rendered result.

But in the code above I've passed in a canvas element rather than an image and included the 'leaveDOM' property to prevent the pixastic library from swapping your canvas in the DOM for the one it creates.

To display the results I've included the callback function which just does ctx.drawImage to put the contents into your original canvas.

Hope that makes sense.

You can check the documentation for more examples. Pixastic Documentation

html5 - how to do the brightness, contrast, saturation, hue up and down?

OK - this almost works for brightness up+ and down-

<script src="https://code.jquery.com/jquery-1.9.1.min.js"></script>
<script>
function greyImages() {
var img = document.getElementById("cvs-src");
var imageData;
var px;
var length;
var i = 0;
var grey;
var can = document.getElementsByTagName("canvas")[0].getContext('2d');
can.drawImage(img, 0, 0);
imageData = can.getImageData(0, 0, 1024, 768);
px = imageData.data;
length = px.length;

// Grey
// for ( ; i < length; i+= 4 ) {
// grey = px[i] * .3 + px[i+1] * .59 + px[i+2] * .11;
// px[i] = grey;
// grey = px[i] * .1 + px[i+1] * .1 + px[i+2] * .1;
// px[i] = px[i+1] = px[i+2] = grey;
// }

// Bright up / down
for ( ; i < length; i+= 4 ) {
px[i] -= 40 ;
px[i+1] -= 40 ;
px[i+2] -= 40 ;
}

// Threshold
// for ( ; i < length; i+= 4) {
// var r = px[i];
// var g = px[i+1];
// var b = px[i+2];
// grey= (0.2126*r + 0.7152*g + 0.0722*b >= 150) ? 255 : 0;
// px[i] = px[i+1] = px[i+2] = grey;
// }

can.putImageData(imageData, 0, 0);
}

$(document).ready(function() {

});
</script>

<h1>Effects</h1>
<table>
<tr>
<td style="vertical-align:text-top;">
<h2>Original</h2>
<button id="take" onclick="greyImages();" />
<video id="cvs-src" autoplay="autoplay" src="/images/1.webm"
type="video/webm" width=640 height=480></video>
</td>

<td>
<canvas width=1024 height=768></canvas>
</td>
</tr>
</table>

how to change the color of an image in a HTML5 Canvas without changing its pattern

Luma preservation

At the risk of looking similar to the existing answer, I would like to point out a small but important difference using a slightly different approach.

The key is to preserve the luma component in an image (ie. shadow details, wrinkles etc. in this case) so two steps are needed to control the look using blending modes via globalCompositeOperation (or alternatively, a manual approach using conversion between RGB and the HSL color-space if older browsers must be supported):

  • "saturation": will alter the chroma (intensity, saturation) from the next drawn element and apply it to the existing content on the canvas, but preserve luma and hue.
  • "hue": will grab the chroma and luma from the source but alter the hue, or color if you will, based on the next drawn element.

As these are blending modes (ignoring the alpha channel) we will also need to clip the result using composition as a last step.

The color blending mode can be used too but it will alter luma which may or may not be desirable. The difference can be subtle in many cases, but also very obvious depending on target chroma and hue where luma/shadow definition is lost.

So, to achieve a good quality result preserving both luma and chroma, these are more or less the main steps (assumes an empty canvas):

// step 1: draw in original image
ctx.globalCompositeOperation = "source-over";
ctx.drawImage(img, 0, 0);

// step 2: adjust saturation (chroma, intensity)
ctx.globalCompositeOperation = "saturation";
ctx.fillStyle = "hsl(0," + sat + "%, 50%)"; // hue doesn't matter here
ctx.fillRect(0, 0);

// step 3: adjust hue, preserve luma and chroma
ctx.globalCompositeOperation = "hue";
ctx.fillStyle = "hsl(" + hue + ",1%, 50%)"; // sat must be > 0, otherwise won't matter
ctx.fillRect(0, 0, c.width, c.height);

// step 4: in our case, we need to clip as we filled the entire area
ctx.globalCompositeOperation = "destination-in";
ctx.drawImage(img, 0, 0);

// step 5: reset comp mode to default
ctx.globalCompositeOperation = "source-over";

50% lightness (L) will keep the original luma value.

Live Example

Click the checkbox to see the effect on the result. Then test with different chroma and hue settings.

var ctx = c.getContext("2d");
var img = new Image(); img.onload = demo; img.src = "//i.stack.imgur.com/Kk1qd.png";
function demo() {c.width = this.width>>1; c.height = this.height>>1; render()}

function render() {
var hue = +rHue.value, sat = +rSat.value, l = +rL.value;

ctx.clearRect(0, 0, c.width, c.height);
ctx.globalCompositeOperation = "source-over";
ctx.drawImage(img, 0, 0, c.width, c.height);

if (!!cColor.checked) {
// use color blending mode
ctx.globalCompositeOperation = "color";
ctx.fillStyle = "hsl(" + hue + "," + sat + "%, 50%)";
ctx.fillRect(0, 0, c.width, c.height);
}
else {
// adjust "lightness"
ctx.globalCompositeOperation = l < 100 ? "color-burn" : "color-dodge";
// for common slider, to produce a valid value for both directions
l = l >= 100 ? l - 100 : 100 - (100 - l);
ctx.fillStyle = "hsl(0, 50%, " + l + "%)";
ctx.fillRect(0, 0, c.width, c.height);

// adjust saturation
ctx.globalCompositeOperation = "saturation";
ctx.fillStyle = "hsl(0," + sat + "%, 50%)";
ctx.fillRect(0, 0, c.width, c.height);

// adjust hue
ctx.globalCompositeOperation = "hue";
ctx.fillStyle = "hsl(" + hue + ",1%, 50%)";
ctx.fillRect(0, 0, c.width, c.height);
}

// clip
ctx.globalCompositeOperation = "destination-in";
ctx.drawImage(img, 0, 0, c.width, c.height);

// reset comp. mode to default
ctx.globalCompositeOperation = "source-over";
}

rHue.oninput = rSat.oninput = rL.oninput = cColor.onchange = render;
body {font:16px sans-serif}
<div>
<label>Hue: <input type=range id=rHue max=359 value=0></label>
<label>Saturation: <input type=range id=rSat value=100></label>
<label>Lightness: <input type=range id=rL max=200 value=100></label>
<label>Use "color" instead: <input type=checkbox id=cColor></label>
</div>
<canvas id=c></canvas>


Related Topics



Leave a reply



Submit