2013-11-25 22:00:19 +04:00
|
|
|
(function () {
|
|
|
|
var ns = $.namespace('pskl.utils');
|
|
|
|
|
|
|
|
ns.ImageResizer = {
|
|
|
|
resize : function (image, targetWidth, targetHeight, smoothingEnabled) {
|
|
|
|
var canvas = pskl.CanvasUtils.createCanvas(targetWidth, targetHeight);
|
|
|
|
var context = canvas.getContext('2d');
|
|
|
|
context.save();
|
|
|
|
|
|
|
|
if (!smoothingEnabled) {
|
|
|
|
pskl.CanvasUtils.disableImageSmoothing(canvas);
|
|
|
|
}
|
|
|
|
|
|
|
|
context.translate(canvas.width / 2, canvas.height / 2);
|
|
|
|
context.scale(targetWidth / image.width, targetHeight / image.height);
|
|
|
|
context.drawImage(image, -image.width / 2, -image.height / 2);
|
|
|
|
context.restore();
|
|
|
|
|
|
|
|
return canvas;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Manual implementation of resize using a nearest neighbour algorithm
|
|
|
|
* It is slower than relying on the native 'disabledImageSmoothing' available on CanvasRenderingContext2d.
|
|
|
|
* But it can be useful if :
|
|
|
|
* - IE < 11 (doesn't support msDisableImageSmoothing)
|
|
|
|
* - need to display a gap between pixel
|
|
|
|
*
|
|
|
|
* @param {Canvas2d} source original image to be resized, as a 2d canvas
|
|
|
|
* @param {Number} zoom ratio between desired dim / source dim
|
|
|
|
* @param {Number} margin gap to be displayed between pixels
|
2014-03-30 05:12:56 +04:00
|
|
|
* @param {String} color or the margin (will be transparent if not provided)
|
2013-11-25 22:00:19 +04:00
|
|
|
* @return {Canvas2d} the resized canvas
|
|
|
|
*/
|
2014-03-30 05:12:56 +04:00
|
|
|
resizeNearestNeighbour : function (source, zoom, margin, marginColor) {
|
2013-11-25 22:00:19 +04:00
|
|
|
margin = margin || 0;
|
|
|
|
var canvas = pskl.CanvasUtils.createCanvas(zoom*source.width, zoom*source.height);
|
|
|
|
var context = canvas.getContext('2d');
|
|
|
|
|
|
|
|
var imgData = pskl.CanvasUtils.getImageDataFromCanvas(source);
|
|
|
|
|
|
|
|
var yRanges = {},
|
|
|
|
xOffset = 0,
|
|
|
|
yOffset = 0,
|
|
|
|
xRange,
|
|
|
|
yRange;
|
|
|
|
// Draw the zoomed-up pixels to a different canvas context
|
|
|
|
for (var x = 0; x < source.width; x++) {
|
|
|
|
// Calculate X Range
|
|
|
|
xRange = Math.floor((x + 1) * zoom) - xOffset;
|
|
|
|
|
|
|
|
for (var y = 0; y < source.height; y++) {
|
|
|
|
// Calculate Y Range
|
|
|
|
if (!yRanges[y + ""]) {
|
|
|
|
// Cache Y Range
|
|
|
|
yRanges[y + ""] = Math.floor((y + 1) * zoom) - yOffset;
|
|
|
|
}
|
|
|
|
yRange = yRanges[y + ""];
|
|
|
|
|
|
|
|
var i = (y * source.width + x) * 4;
|
|
|
|
var r = imgData[i];
|
|
|
|
var g = imgData[i + 1];
|
|
|
|
var b = imgData[i + 2];
|
|
|
|
var a = imgData[i + 3];
|
|
|
|
|
|
|
|
context.fillStyle = "rgba(" + r + "," + g + "," + b + "," + (a / 255) + ")";
|
|
|
|
context.fillRect(xOffset, yOffset, xRange-margin, yRange-margin);
|
|
|
|
|
2014-03-30 05:12:56 +04:00
|
|
|
if (margin && marginColor) {
|
|
|
|
context.fillStyle = marginColor;
|
|
|
|
context.fillRect(xOffset + xRange - margin, yOffset, margin, yRange);
|
|
|
|
context.fillRect(xOffset, yOffset + yRange - margin, xRange, margin);
|
|
|
|
}
|
|
|
|
|
2013-11-25 22:00:19 +04:00
|
|
|
yOffset += yRange;
|
|
|
|
}
|
|
|
|
yOffset = 0;
|
|
|
|
xOffset += xRange;
|
|
|
|
}
|
|
|
|
return canvas;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
})();
|