var addColorStops = function(row, gradient, sharpness){
var colWidth = 1/row.length, step = 1/(row.length-1);
gradient.addColorStop(0, row[0]);
for(var i=0; i<row.length; ++i){
if(i==0){
if(sharpness > 0)gradient.addColorStop(colWidth*sharpness, row[i]);
}else if(i == row.length-1){
if(sharpness > 0)gradient.addColorStop(1-(colWidth*sharpness), row[i]);
}else{
if(sharpness > 0)gradient.addColorStop(step*i-colWidth/2*sharpness, row[i]);
gradient.addColorStop(step*i, row[i]);
if(sharpness > 0)gradient.addColorStop(step*i+colWidth/2*sharpness, row[i]);
}
}
gradient.addColorStop(1, row[row.length-1]);
}
var drawRow = function(ctx, canvas, row, sharpness){
var gradient = ctx.createLinearGradient(0, 0, canvas.width, 0);
addColorStops(row, gradient, sharpness)
ctx.fillStyle = gradient;
ctx.fillRect(0, 0, canvas.width, canvas.height);
}
var drawMask = function(ctx, canvas, matrix, rowIndex, sharpness){
var mask = ctx.createLinearGradient(0, 0, 0, canvas.height);
addColorStops(matrix.map(function(row,i){return i==rowIndex?'rgba(255,255,255,0)':'rgba(255,255,255,1)'}), mask, sharpness)
ctx.fillStyle = mask;
ctx.globalCompositeOperation = 'destination-out';
ctx.fillRect(0, 0, canvas.width, canvas.height);
ctx.globalCompositeOperation = 'source-over';
}
var drawMatrixGradient = function(matrix, sharpness){
sharpness = Math.min(1, Math.max(0, sharpness||0))
var onscreen = document.getElementById('canvas');
var ctx = onscreen.getContext('2d');
var offScreen = document.createElement('canvas');
offScreen.width = onscreen.width;
offScreen.height = onscreen.height;
var ctx2 = offScreen.getContext('2d');
//Draw the first row
drawRow(ctx, onscreen, matrix[0], sharpness)
//Now draw the rest
for(var i=1; i<matrix.length; ++i){
drawRow(ctx2, offScreen, matrix[i], sharpness)
drawMask(ctx2, offScreen, matrix, i, sharpness);
ctx.drawImage(offScreen, 0, 0, canvas.width, canvas.height);
}
}
drawMatrixGradient([['red','green','blue'],['yellow','purple','orange'],['magenta','aqua','white']], 0.5)
<canvas id="canvas" width="250" height="250"></canvas>