Restructure BlendMode code, fix issues with color-dodge and color-burn, and create BlendModes.html example.

All modes should be implemented according to specs now.
This commit is contained in:
Jürg Lehni 2013-06-18 08:02:04 -07:00
parent 1b42822c2d
commit b133d8fe2e
2 changed files with 273 additions and 190 deletions

File diff suppressed because one or more lines are too long

View file

@ -10,15 +10,8 @@
* All rights reserved. * All rights reserved.
*/ */
var BlendMode = { var BlendMode = new function() {
process: function(blendMode, srcContext, dstContext, alpha, offset) { var min = Math.min,
var srcCanvas = srcContext.canvas,
dstData = dstContext.getImageData(offset.x, offset.y,
srcCanvas.width, srcCanvas.height),
dst = dstData.data,
src = srcContext.getImageData(0, 0,
srcCanvas.width, srcCanvas.height).data,
min = Math.min,
max = Math.max, max = Math.max,
abs = Math.abs, abs = Math.abs,
sr, sg, sb, sa, // source sr, sg, sb, sa, // source
@ -48,7 +41,8 @@ var BlendMode = {
db = l + (db - l) * l / lmn; db = l + (db - l) * l / lmn;
} }
if (mx > 255) { if (mx > 255) {
var ln = 255 - l, mxl = mx - l; var ln = 255 - l,
mxl = mx - l;
dr = l + (dr - l) * ln / mxl; dr = l + (dr - l) * ln / mxl;
dg = l + (dg - l) * ln / mxl; dg = l + (dg - l) * ln / mxl;
db = l + (db - l) * ln / mxl; db = l + (db - l) * ln / mxl;
@ -65,11 +59,11 @@ var BlendMode = {
mn = min(r, g, b), // min mn = min(r, g, b), // min
md; // mid md; // mid
// Determine indices for min and max in col: // Determine indices for min and max in col:
mn = mn == r ? 0 : mn == g ? 1 : 2; mn = mn === r ? 0 : mn === g ? 1 : 2;
mx = mx == r ? 0 : mx == g ? 1 : 2; mx = mx === r ? 0 : mx === g ? 1 : 2;
// Determine the index in col that is not used yet by min and max, // Determine the index in col that is not used yet by min and max,
// and assign it to mid: // and assign it to mid:
md = min(mn, mx) == 0 ? max(mn, mx) == 1 ? 2 : 1 : 0; md = min(mn, mx) === 0 ? max(mn, mx) === 1 ? 2 : 1 : 0;
// Now perform the actual algorithm // Now perform the actual algorithm
if (col[mx] > col[mn]) { if (col[mx] > col[mn]) {
col[md] = (col[md] - col[mn]) * s / (col[mx] - col[mn]); col[md] = (col[md] - col[mn]) * s / (col[mx] - col[mn]);
@ -85,19 +79,23 @@ var BlendMode = {
} }
var modes = { var modes = {
// B(Cb, Cs) = Cb x Cs
multiply: function() { multiply: function() {
dr = br * sr / 255; dr = br * sr / 255;
dg = bg * sg / 255; dg = bg * sg / 255;
db = bb * sb / 255; db = bb * sb / 255;
}, },
// B(Cb, Cs) = 1 - [(1 - Cb) x (1 - Cs)] = Cb + Cs -(Cb x Cs)
screen: function() { screen: function() {
dr = 255 - (255 - br) * (255 - sr) / 255; dr = br + sr - (br * sr / 255);
dg = 255 - (255 - bg) * (255 - sg) / 255; dg = bg + sg - (bg * sg / 255);
db = 255 - (255 - bb) * (255 - sb) / 255; db = bb + sb - (bb * sb / 255);
}, },
// B(Cb, Cs) = HardLight(Cs, Cb)
overlay: function() { overlay: function() {
// = Reverse of hard-light
dr = br < 128 ? 2 * br * sr / 255 : 255 - 2 * (255 - br) * (255 - sr) / 255; dr = br < 128 ? 2 * br * sr / 255 : 255 - 2 * (255 - br) * (255 - sr) / 255;
dg = bg < 128 ? 2 * bg * sg / 255 : 255 - 2 * (255 - bg) * (255 - sg) / 255; dg = bg < 128 ? 2 * bg * sg / 255 : 255 - 2 * (255 - bg) * (255 - sg) / 255;
db = bb < 128 ? 2 * bb * sb / 255 : 255 - 2 * (255 - bb) * (255 - sb) / 255; db = bb < 128 ? 2 * bb * sb / 255 : 255 - 2 * (255 - bb) * (255 - sb) / 255;
@ -112,37 +110,47 @@ var BlendMode = {
db = t + bb * (255 - (255 - bb) * (255 - sb) / 255 - t) / 255; db = t + bb * (255 - (255 - bb) * (255 - sb) / 255 - t) / 255;
}, },
// if (Cs <= 0.5) B(Cb, Cs) = Multiply(Cb, 2 x Cs)
// else B(Cb, Cs) = Screen(Cb, 2 x Cs -1)
'hard-light': function() { 'hard-light': function() {
// = Reverse of overlay
dr = sr < 128 ? 2 * sr * br / 255 : 255 - 2 * (255 - sr) * (255 - br) / 255; dr = sr < 128 ? 2 * sr * br / 255 : 255 - 2 * (255 - sr) * (255 - br) / 255;
dg = sg < 128 ? 2 * sg * bg / 255 : 255 - 2 * (255 - sg) * (255 - bg) / 255; dg = sg < 128 ? 2 * sg * bg / 255 : 255 - 2 * (255 - sg) * (255 - bg) / 255;
db = sb < 128 ? 2 * sb * bb / 255 : 255 - 2 * (255 - sb) * (255 - bb) / 255; db = sb < 128 ? 2 * sb * bb / 255 : 255 - 2 * (255 - sb) * (255 - bb) / 255;
}, },
// if (Cb == 0) B(Cb, Cs) = 0
// else if (Cs == 1) B(Cb, Cs) = 1
// else B(Cb, Cs) = min(1, Cb / (1 - Cs))
'color-dodge': function() { 'color-dodge': function() {
dr = sr == 255 ? sr : min(255, br * 255 / (255 - sr)); dr = br === 0 ? 0 : sr === 255 ? 255 : min(255, 255 * br / (255 - sr));
dg = sg == 255 ? sg : min(255, bg * 255 / (255 - sg)); dg = bg === 0 ? 0 : sg === 255 ? 255 : min(255, 255 * bg / (255 - sg));
db = sb == 255 ? sb : min(255, bb * 255 / (255 - sb)); db = bb === 0 ? 0 : sb === 255 ? 255 : min(255, 255 * bb / (255 - sb));
}, },
// if (Cb == 1) B(Cb, Cs) = 1
// else if(Cs == 0) B(Cb, Cs) = 0
// else B(Cb, Cs) = 1 - min(1, (1 - Cb) / Cs)
'color-burn': function() { 'color-burn': function() {
dr = sr == 0 ? 0 : max(255 - ((255 - br) * 255) / sr, 0); dr = br === 255 ? 255 : sr === 0 ? 0 : max(0, 255 - (255 - br) * 255 / sr);
dg = sg == 0 ? 0 : max(255 - ((255 - bg) * 255) / sg, 0); dg = bg === 255 ? 255 : sg === 0 ? 0 : max(0, 255 - (255 - bg) * 255 / sg);
db = sb == 0 ? 0 : max(255 - ((255 - bb) * 255) / sb, 0); db = bb === 255 ? 255 : sb === 0 ? 0 : max(0, 255 - (255 - bb) * 255 / sb);
}, },
// B(Cb, Cs) = min(Cb, Cs)
darken: function() { darken: function() {
dr = br < sr ? br : sr; dr = br < sr ? br : sr;
dg = bg < sg ? bg : sg; dg = bg < sg ? bg : sg;
db = bb < sb ? bb : sb; db = bb < sb ? bb : sb;
}, },
// B(Cb, Cs) = max(Cb, Cs)
lighten: function() { lighten: function() {
dr = br > sr ? br : sr; dr = br > sr ? br : sr;
dg = bg > sg ? bg : sg; dg = bg > sg ? bg : sg;
db = bb > sb ? bb : sb; db = bb > sb ? bb : sb;
}, },
// B(Cb, Cs) = | Cb - Cs |
difference: function() { difference: function() {
dr = br - sr; dr = br - sr;
if (dr < 0) if (dr < 0)
@ -155,6 +163,7 @@ var BlendMode = {
db = -db; db = -db;
}, },
// B(Cb, Cs) = Cb + Cs - 2 x Cb x Cs
exclusion: function() { exclusion: function() {
dr = br + sr * (255 - br - br) / 255; dr = br + sr * (255 - br - br) / 255;
dg = bg + sg * (255 - bg - bg) / 255; dg = bg + sg * (255 - bg - bg) / 255;
@ -206,6 +215,15 @@ var BlendMode = {
} }
}; };
this.process = function(blendMode, srcContext, dstContext, alpha, offset) {
var srcCanvas = srcContext.canvas,
dstData = dstContext.getImageData(offset.x, offset.y,
srcCanvas.width, srcCanvas.height),
dst = dstData.data,
src = srcContext.getImageData(0, 0,
srcCanvas.width, srcCanvas.height).data;
var process = modes[blendMode]; var process = modes[blendMode];
if (!process) if (!process)
return; return;
@ -228,5 +246,5 @@ var BlendMode = {
dst[i + 3] = sa * alpha + a2 * ba; dst[i + 3] = sa * alpha + a2 * ba;
} }
dstContext.putImageData(dstData, offset.x, offset.y); dstContext.putImageData(dstData, offset.x, offset.y);
} };
}; };