Darken Image Slider/Button -- Javascript - javascript
I have a webpage in which users upload an image of some hand written work. Sometimes it's scanned pencil which can be very difficult to read.
Is it possible to possible to have a slider/button that I could use to darken or maybe even sharpen a particular image? I would need a slider/button per image as the page I view contains several user uploaded images.
Thanks.
Yes, there are two ways, one is css filters (see posit labs answer), the other one is with canvas, here is a nice tutorial for that, and here is my demo.
For the demo, you would have to use an image in your own domain (otherwise the canvas becomes tainted and you can't access the pixels), that's why you see the Data URI src in the image, is the only way to make the image origin from the fiddle.
HTML
<img id="myImage" src="mydomain/img.png">
<button class="filter-btn" data-filter="darken" data-img="#myImage">Darken</button>
<button class="filter-btn" data-filter="sharpen" data-img="#myImage">Sharpen</button>
If you copy and paste the JavaScript, the only thing you have to do is use this markup for it to work, the image can be configured however you want, the buttons are the important part.
Each button has a filter-btn class, to indicate that it's intended to apply a filter, then, you specify the filter via the data-filter attribute (in this case it can be sharpen or darken), and finally you link the button to the image via the data-img attribute, where you can specify any css selector to get to the image.
JavaScript
Remember, you don't have to touch any of these if you follow the HTML markup, but if you have any questions about the code, shoot!
ImageFilter = {}
ImageFilter.init = function () {
var buttons = document.querySelectorAll(".filter-btn");
for (var i = 0; i < buttons.length; i++) {
var btn = buttons[i];
var filter = btn.dataset.filter;
var img = btn.dataset.img;
img.crossOrigin = "Anonymous";
(function (filter, img) {
btn.addEventListener("click", function () {
ImageFilter.doFilter(filter, img);
});
})(filter, img);
}
}
window.addEventListener("load", ImageFilter.init);
ImageFilter.getImage = function (selector) {
return document.querySelector(selector);
}
ImageFilter.createData = function (canvas, w, h) {
var context = canvas.getContext("2d");
return context.createImageData(w, h);
}
ImageFilter.doFilter = function (type, image) {
var image = ImageFilter.getImage(image);
switch (type) {
case "darken":
var adjustment = -5;
var canvas = ImageFilter.newCanvas(image);
var data = ImageFilter.getData(canvas);
var actualData = data.data;
for (var i = 0; i < actualData.length; i++) {
actualData[i] += adjustment;
actualData[i + 1] += adjustment;
actualData[i + 2] += adjustment;
}
ImageFilter.putData(data, canvas);
var newImg = image.cloneNode(true);
newImg.src = ImageFilter.getSource(canvas);
newImg.id = image.id;
replaceNode(image, newImg);
break;
case "sharpen":
var weights = [0, -1, 0, -1, 5, -1,
0, -1, 0];
var canvas = ImageFilter.newCanvas(image);
var data = ImageFilter.getData(canvas);
var side = Math.round(Math.sqrt(weights.length));
var halfSide = Math.floor(side / 2);
var src = data.data;
var sw = data.width;
var sh = data.height;
var w = sw;
var h = sh;
var output = ImageFilter.createData(canvas, w, h);
var dst = output.data;
var alphaFac = 1;
for (var y = 0; y < h; y++) {
for (var x = 0; x < w; x++) {
var sy = y;
var sx = x;
var dstOff = (y * w + x) * 4;
var r = 0,
g = 0,
b = 0,
a = 0;
for (var cy = 0; cy < side; cy++) {
for (var cx = 0; cx < side; cx++) {
var scy = sy + cy - halfSide;
var scx = sx + cx - halfSide;
if (scy >= 0 && scy < sh && scx >= 0 && scx < sw) {
var srcOff = (scy * sw + scx) * 4;
var wt = weights[cy * side + cx];
r += src[srcOff] * wt;
g += src[srcOff + 1] * wt;
b += src[srcOff + 2] * wt;
a += src[srcOff + 3] * wt;
}
}
}
dst[dstOff] = r;
dst[dstOff + 1] = g;
dst[dstOff + 2] = b;
dst[dstOff + 3] = a + alphaFac * (255 - a);
}
}
ImageFilter.putData(output, canvas);
var newImg = image.cloneNode(true);
newImg.src = ImageFilter.getSource(canvas);
replaceNode(image, newImg);
break;
}
}
ImageFilter.newCanvas = function (image) {
var canvas = document.createElement("canvas");
canvas.width = image.width;
canvas.height = image.height;
var context = canvas.getContext("2d");
context.drawImage(image, 0, 0, image.width, image.height);
return canvas;
}
ImageFilter.getData = function (canvas) {
var context = canvas.getContext("2d");
return context.getImageData(0, 0, canvas.width, canvas.height);
}
ImageFilter.putData = function (data, canvas) {
var context = canvas.getContext("2d");
context.putImageData(data, 0, 0);
}
ImageFilter.getSource = function (canvas) {
return canvas.toDataURL();
}
function replaceNode(node1, node2) {
var parent = node1.parentNode;
var next = node1.nextSibling;
if (next) parent.insertBefore(node2, next);
else parent.appendChild(node2);
parent.removeChild(node1);
}
That's it, see the demo, hope it helps!
Updates
Firefox fix: creating a new image and replacing the old one each time seems to fix the firefox bug where it doesn't update the image's src. (29/01/15 2:07a.m)
Short answer: yes.
The easiest way to do this would be with CSS Filters, but they aren't supported on old browsers (support table). The example below applies a 200% contrast filter.
filter: contrast(2);
Another option would be to use HTML Canvas to draw the images and manually manipulate the pixels. It's not very fast, and it's much more complicated than CSS Filters. I won't go into depth, but here is an article about filtering images with canvas.
In my opinion, the users should be responsible for uploading quality images. It seems silly to correct their mistake by adding extra controls to your site.
Related
How to divide image in tiles?
I have to achieve the following task: divides the image into tiles, computes the average color of each tile, fetches a tile from the server for that color, and composites the results into a photomosaic of the original image. What would be the best strategy? the first solution coming to my mind is using canvas.
A simple way to get pixel data and finding the means of tiles. The code will need more checks for images that do not have dimensions that can be divided by the number of tiles. var image = new Image(); image.src = ??? // the URL if the image is not from your domain you will have to move it to your server first // wait for image to load image.onload = function(){ // create a canvas var canvas = document.createElement("canvas"); //set its size to match the image canvas.width = this.width; canvas.height = this.height; var ctx = canvas.getContext("2d"); // get the 2d interface // draw the image on the canvas ctx.drawImage(this,0,0); // get the tile size var tileSizeX = Math.floor(this.width / 10); var tileSizeY = Math.floor(this.height / 10); var x,y; // array to hold tile colours var tileColours = []; // for each tile for(y = 0; y < this.height; y += tileSizeY){ for(x = 0; x < this.width; x += tileSizeX){ // get the pixel data var imgData = ctx.getImageData(x,y,tileSizeX,tileSizeY); var r,g,b,ind; var i = tileSizeY * tileSizeX; // get pixel count ind = r = g = b = 0; // for each pixel (rgba 8 bits each) while(i > 0){ // sum the channels r += imgData.data[ind++]; g += imgData.data[ind++]; b += imgData.data[ind++]; ind ++; i --; } i = ind /4; // get the count again // calculate channel means r /= i; g /= i; b /= i; //store the tile coords and colour tileColours[tileColours.length] = { rgb : [r,g,b], x : x, y : y, } } // all done now fetch the images for the found tiles. }
I created a solution for this (I am not getting the tile images from back end) // first function call to create photomosaic function photomosaic(image) { // Dimensions of each tile var tileWidth = TILE_WIDTH; var tileHeight = TILE_HEIGHT; //creating the canvas for photomosaic var canvas = document.createElement('canvas'); var context = canvas.getContext("2d"); canvas.height = image.height; canvas.width = image.width; var imageData = context.getImageData(0, 0, image.width, image.height); var pixels = imageData.data; // Number of mosaic tiles var numTileRows = image.width / tileWidth; var numTileCols = image.height / tileHeight; //canvas copy of image var imageCanvas = document.createElement('canvas'); var imageCanvasContext = canvas.getContext('2d'); imageCanvas.height = image.height; imageCanvas.width = image.width; imageCanvasContext.drawImage(image, 0, 0); //function for finding the average color function averageColor(row, column) { var blockSize = 1, // we can set how many pixels to skip data, width, height, i = -4, length, rgb = { r: 0, g: 0, b: 0 }, count = 0; try { data = imageCanvasContext.getImageData(column * TILE_WIDTH, row * TILE_HEIGHT, TILE_HEIGHT, TILE_WIDTH); } catch (e) { alert('Not happening this time!'); return rgb; } length = data.data.length; while ((i += blockSize * 4) < length) { ++count; rgb.r += data.data[i]; rgb.g += data.data[i + 1]; rgb.b += data.data[i + 2]; } // ~~ used to floor values rgb.r = ~~(rgb.r / count); rgb.g = ~~(rgb.g / count); rgb.b = ~~(rgb.b / count); return rgb; } // Loop through each tile for (var r = 0; r < numTileRows; r++) { for (var c = 0; c < numTileCols; c++) { // Set the pixel values for each tile var rgb = averageColor(r, c) var red = rgb.r; var green = rgb.g; var blue = rgb.b; // Loop through each tile pixel for (var tr = 0; tr < tileHeight; tr++) { for (var tc = 0; tc < tileWidth; tc++) { // Calculate the true position of the tile pixel var trueRow = (r * tileHeight) + tr; var trueCol = (c * tileWidth) + tc; // Calculate the position of the current pixel in the array var pos = (trueRow * (imageData.width * 4)) + (trueCol * 4); // Assign the colour to each pixel pixels[pos + 0] = red; pixels[pos + 1] = green; pixels[pos + 2] = blue; pixels[pos + 3] = 255; }; }; }; }; // Draw image data to the canvas context.putImageData(imageData, 0, 0); return canvas; } function create() { var image = document.getElementById('image'); var canvas = photomosaic(image); document.getElementById("output").appendChild(canvas); }; DEMO:https://jsfiddle.net/gurinderiitr/sx735L5n/
Try using the JIMP javascript library to read the pixel color and use invert, normalize or similar property for modifying the image. Have a look on the jimp library https://github.com/oliver-moran/jimp
emulate div behavior in canvas
I have a dom editor which a user can insert textbox and images. One of my requirements involve saving a snapshot of what is in the editor into an image. I did some research and there are some solutions, but they don't seem 100% foolproof. I've tried implementing a solution myself, clobbering code here and there: function measureText(text, size, font) { var lDiv = document.createElement('lDiv'); document.body.appendChild(lDiv); lDiv.style.fontSize = size; lDiv.style.fontFamily = font; lDiv.style.position = "absolute"; lDiv.style.left = -1000; lDiv.style.top = -1000; lDiv.innerHTML = text; var metrics = font.measureText(text, size.slice(0, size.length - 2)); var lResult = { width: lDiv.clientWidth, height: metrics.height + lDiv.clientHeight }; document.body.removeChild(lDiv); lDiv = null; return lResult; } function wrapText(context, item) { var words = item.text.split(' '); var line = ''; var x = parseInt(item.x); var y = parseInt(item.y); var width = parseInt(item.width.slice(0, item.width.length - 2)); var height = parseInt(item.height.slice(0, item.height.length - 2)); var fontsize = parseInt(item.size.slice(0, item.size.length - 2)); var font = new Font(); font.onload = function () { context.save(); context.beginPath(); context.rect(x, y, width, height); context.clip(); context.font = item.size + " " + item.font; context.textBaseline = "top"; for (var n = 0; n < words.length; n++) { var testLine = line + words[n] + ' '; var metrics = measureText(testLine, item.size, font); var testWidth = metrics.width; if (testWidth > width && n > 0) { console.log("Drawing '" + line + "' to " + x + " " + y); context.fillText(line, x, y); line = words[n] + ' '; y += metrics.height } else { line = testLine; } } context.fillText(line, x, y); context.restore(); } font.fontFamily = item.font; font.src = font.fontFamily; } this.toImage = function () { console.log("testing"); var canvas = document.getElementById("testcanvas"); canvas.width = 400; canvas.height = 400; var ctx = canvas.getContext("2d"); var imageObj = new Image(); var thisService = this; imageObj.onload = function () { ctx.drawImage(imageObj, 0, 0, 400, 400); for (var i = 0; i < thisService.canvasItems.length; i++) { var component = thisService.canvasItems[i]; if (component.type == "textbox") { var x = component.x.slice(0, component.x.length - 2); var y = component.y.slice(0, component.y.length - 2); var w = component.width.slice(0, component.width.length - 2); var h = component.height.slice(0, component.height.length - 2); wrapText(ctx, component); } } }; imageObj.src = this.base.front_image; } Somehow I believe I almost made it, however from the , There seems to be some positioning/font placement issues, just a few pixels lower. The top 1 a div with no padding, (its model can be seem on the panel on the left), while the bottom one is the canvas. I wish to have a 1 to 1 accurate mapping here, can anyone enlighten what might be the problem?
As far as I know its not possible to draw HTML into a canvas with 100% accuracy due to the obvious "security" reasons. You can still get pretty close using the rasterizeHTML.js library. It uses a SVG image containing the content you want to render. To draw HTML content, you'd use a element containing the HTML, then draw that SVG image into your canvas.
Automatically contrasting text color based on background image color
I'm looking for a way to change the color of text to either #000 or #fff depending on the main color of a background image within a div called "banner". The background images are chosen at random on each page so I need to be able to do this automatically. I came across JavaScript color contraster but I'm struggling to understand how to use it properly. I notice the link I've posted gives a solution in javascript and I've also read about a possible solution in jquery. I'm clueless with functions so if anyone could explain clearly how I could achieve this, where I place functions and how I "call them" (if that's the right term!) to use it I'd be really grateful. Thanks for any help.
You could do something like this. (using Colours.js and this answer) Note, this will only work with images on the same domain and in browsers that support HTML5 canvas. 'use strict'; var getAverageRGB = function(imgEl) { var rgb = { b: 0, g: 0, r: 0 }; var canvas = document.createElement('canvas'); var context = canvas.getContext && canvas.getContext('2d'); if (Boolean(context) === false) { return rgb; } var height = imgEl.naturalHeight || imgEl.offsetHeight || imgEl.height; var width = imgEl.naturalWidth || imgEl.offsetWidth || imgEl.width; canvas.height = height; canvas.width = width; context.drawImage(imgEl, 0, 0); var data; try { data = context.getImageData(0, 0, width, height).data; } catch (e) { console.error('security error, img on diff domain'); return rgb; } var count = 0; var length = data.length; // only visit every 5 pixels var blockSize = 5; var step = (blockSize * 4) - 4; for (var i = step; i < length; i += step) { count += 1; rgb.r += data[i]; rgb.g += data[i + 1]; rgb.b += data[i + 2]; } rgb.r = Math.floor(rgb.r / count); rgb.g = Math.floor(rgb.g / count); rgb.b = Math.floor(rgb.b / count); return rgb; }; var rgb = getAverageRGB(document.getElementById('image')); var avgComplement = Colors.complement(rgb.r, rgb.b, rgb.g); var avgComplementHex = Colors.rgb2hex.apply(null, avgComplement.a); var compliment = parseInt(avgComplementHex.slice(1), 16); document.body.style.backgroundColor = 'rgb(' + [ rgb.r, rgb.g, rgb.b ].join(',') + ')'; var maxColors = 0xFFFFFF; var midPoint = Math.floor(maxColors / 2); document.getElementById('text').style.color = compliment > midPoint ? '#000' : '#fff'; <script src="https://cdnjs.cloudflare.com/ajax/libs/Colors.js/1.2.3/colors.min.js"></script> <div id="text">Setting the BODY's background to the average color in the following image and this text to a complimentary colour of black or white:</div> <img id="image" src="" />
Javascript Image Transform? [scale and rotate]
I'm looking to build an image transform tool with Javascript. Something that utilizes handles around the image similar to Photoshop and allows the user to scale and rotate. I'm looking to make this work in IE 6 and up and Firefox 3+ and Safari 3+. Does anyone know of a library or tool that could help with this? I've seen a lot of tools that utilize the Canvas element but that leaves out IT. I've also seen the Raphael library which might work. Any other options out there?
Have a look at this rotorzoom. It rotates it zooms it's fast and i can do with a few more hits. http://codepen.io/hex2bin/pen/tHwhF var requestId = 0; var animationStartTime = 0; var img = new Image(); initimg(img); dst = document.getElementById("dst").getContext("2d"); dst.drawImage(img, 0, 0, 256, 256); // read the width and height of the canvas i = 0; var imageDataDst = dst.getImageData(0, 0, 1024, 512); var bufDst = new ArrayBuffer(imageDataDst.data.length); var buf8Dst = new Uint8ClampedArray(bufDst); var data32Dst = new Uint32Array(bufDst); var data32Src = new Uint32Array(256*256); var scan1=0; var scan4=0 // fill the source array with the image for (var y = 0; y < 256; ++y) { scan4=y*1024*4; for (var x = 0; x < 256; ++x) { data32Src[scan1++] = (255 << 24) + // alpha (imageDataDst.data[scan4+2] << 16) + // blue (imageDataDst.data[scan4+1] << 8) + // green imageDataDst.data[scan4]; // red scan4=scan4+4; } } animationStartTime = window.performance.now(); requestId = window.requestAnimationFrame(animate); var j=0; function animate(time) { var height=512; var width=1024; j=j+1; var timestamp = j / 100; var pos; var startY = 128; var startX = 128; var i=0; var scaledHeight = 512 + Math.sin(timestamp*1.4) * 500; var scaledWidth = 512 + Math.sin(timestamp*1.4) * 500 var angleRadians = timestamp; var deltaX1 = Math.cos(angleRadians) * scaledHeight / 256; var deltaY1 = Math.sin(angleRadians) * scaledHeight / 256; var deltaY1x256=deltaY1*256; var deltaX2 =Math.sin(angleRadians ) * scaledWidth / 256; var deltaY2 =- Math.cos(angleRadians ) * scaledWidth / 256; var h = height; while (h--) { var x =262144+ startX+deltaX1*-512+deltaX2*-256; var y =262144+ startY+deltaY1*-512+deltaY2*-256; var y256=y*256; var w = width; while (w--) { //Optimised inner loop. Can it be done better? pos =(y256>>0&0xff00)+(x>>0&0xff); data32Dst[i++] =data32Src[pos]; x += deltaX1; y256 += deltaY1x256; //end of inner loop } startX += deltaX2 startY += deltaY2; } imageDataDst.data.set(buf8Dst); dst.putImageData(imageDataDst, 0, 0); requestId = window.requestAnimationFrame(animate); } function initimg(image1) { image1.src = ''; } <body> <canvas id="dst" width="1024", height="512"> Random Canvas </canvas> </body>
Check out Raphael library on Github: https://github.com/DmitryBaranovskiy/raphael
Resizing an image in an HTML5 canvas
I'm trying to create a thumbnail image on the client side using javascript and a canvas element, but when I shrink the image down, it looks terrible. It looks as if it was downsized in photoshop with the resampling set to 'Nearest Neighbor' instead of Bicubic. I know its possible to get this to look right, because this site can do it just fine using a canvas as well. I've tried using the same code they do as shown in the "[Source]" link, but it still looks terrible. Is there something I'm missing, some setting that needs to be set or something? EDIT: I'm trying to resize a jpg. I have tried resizing the same jpg on the linked site and in photoshop, and it looks fine when downsized. Here is the relevant code: reader.onloadend = function(e) { var img = new Image(); var ctx = canvas.getContext("2d"); var canvasCopy = document.createElement("canvas"); var copyContext = canvasCopy.getContext("2d"); img.onload = function() { var ratio = 1; if(img.width > maxWidth) ratio = maxWidth / img.width; else if(img.height > maxHeight) ratio = maxHeight / img.height; canvasCopy.width = img.width; canvasCopy.height = img.height; copyContext.drawImage(img, 0, 0); canvas.width = img.width * ratio; canvas.height = img.height * ratio; ctx.drawImage(canvasCopy, 0, 0, canvasCopy.width, canvasCopy.height, 0, 0, canvas.width, canvas.height); }; img.src = reader.result; } EDIT2: Seems I was mistaken, the linked website wasn't doing any better of a job of downsizing the image. I tried the other methods suggested and none of them look any better. This is what the different methods resulted in: Photoshop: Canvas: Image with image-rendering: optimizeQuality set and scaled with width/height: Image with image-rendering: optimizeQuality set and scaled with -moz-transform: Canvas resize on pixastic: I guess this means firefox isn't using bicubic sampling like its supposed to. I'll just have to wait until they actually add it. EDIT3: Original Image
So what do you do if all the browsers (actually, Chrome 5 gave me quite good one) won't give you good enough resampling quality? You implement them yourself then! Oh come on, we're entering the new age of Web 3.0, HTML5 compliant browsers, super optimized JIT javascript compilers, multi-core(†) machines, with tons of memory, what are you afraid of? Hey, there's the word java in javascript, so that should guarantee the performance, right? Behold, the thumbnail generating code: // returns a function that calculates lanczos weight function lanczosCreate(lobes) { return function(x) { if (x > lobes) return 0; x *= Math.PI; if (Math.abs(x) < 1e-16) return 1; var xx = x / lobes; return Math.sin(x) * Math.sin(xx) / x / xx; }; } // elem: canvas element, img: image element, sx: scaled width, lobes: kernel radius function thumbnailer(elem, img, sx, lobes) { this.canvas = elem; elem.width = img.width; elem.height = img.height; elem.style.display = "none"; this.ctx = elem.getContext("2d"); this.ctx.drawImage(img, 0, 0); this.img = img; this.src = this.ctx.getImageData(0, 0, img.width, img.height); this.dest = { width : sx, height : Math.round(img.height * sx / img.width), }; this.dest.data = new Array(this.dest.width * this.dest.height * 3); this.lanczos = lanczosCreate(lobes); this.ratio = img.width / sx; this.rcp_ratio = 2 / this.ratio; this.range2 = Math.ceil(this.ratio * lobes / 2); this.cacheLanc = {}; this.center = {}; this.icenter = {}; setTimeout(this.process1, 0, this, 0); } thumbnailer.prototype.process1 = function(self, u) { self.center.x = (u + 0.5) * self.ratio; self.icenter.x = Math.floor(self.center.x); for (var v = 0; v < self.dest.height; v++) { self.center.y = (v + 0.5) * self.ratio; self.icenter.y = Math.floor(self.center.y); var a, r, g, b; a = r = g = b = 0; for (var i = self.icenter.x - self.range2; i <= self.icenter.x + self.range2; i++) { if (i < 0 || i >= self.src.width) continue; var f_x = Math.floor(1000 * Math.abs(i - self.center.x)); if (!self.cacheLanc[f_x]) self.cacheLanc[f_x] = {}; for (var j = self.icenter.y - self.range2; j <= self.icenter.y + self.range2; j++) { if (j < 0 || j >= self.src.height) continue; var f_y = Math.floor(1000 * Math.abs(j - self.center.y)); if (self.cacheLanc[f_x][f_y] == undefined) self.cacheLanc[f_x][f_y] = self.lanczos(Math.sqrt(Math.pow(f_x * self.rcp_ratio, 2) + Math.pow(f_y * self.rcp_ratio, 2)) / 1000); weight = self.cacheLanc[f_x][f_y]; if (weight > 0) { var idx = (j * self.src.width + i) * 4; a += weight; r += weight * self.src.data[idx]; g += weight * self.src.data[idx + 1]; b += weight * self.src.data[idx + 2]; } } } var idx = (v * self.dest.width + u) * 3; self.dest.data[idx] = r / a; self.dest.data[idx + 1] = g / a; self.dest.data[idx + 2] = b / a; } if (++u < self.dest.width) setTimeout(self.process1, 0, self, u); else setTimeout(self.process2, 0, self); }; thumbnailer.prototype.process2 = function(self) { self.canvas.width = self.dest.width; self.canvas.height = self.dest.height; self.ctx.drawImage(self.img, 0, 0, self.dest.width, self.dest.height); self.src = self.ctx.getImageData(0, 0, self.dest.width, self.dest.height); var idx, idx2; for (var i = 0; i < self.dest.width; i++) { for (var j = 0; j < self.dest.height; j++) { idx = (j * self.dest.width + i) * 3; idx2 = (j * self.dest.width + i) * 4; self.src.data[idx2] = self.dest.data[idx]; self.src.data[idx2 + 1] = self.dest.data[idx + 1]; self.src.data[idx2 + 2] = self.dest.data[idx + 2]; } } self.ctx.putImageData(self.src, 0, 0); self.canvas.style.display = "block"; }; ...with which you can produce results like these! so anyway, here is a 'fixed' version of your example: img.onload = function() { var canvas = document.createElement("canvas"); new thumbnailer(canvas, img, 188, 3); //this produces lanczos3 // but feel free to raise it up to 8. Your client will appreciate // that the program makes full use of his machine. document.body.appendChild(canvas); }; Now it's time to pit your best browsers out there and see which one will least likely increase your client's blood pressure! Umm, where's my sarcasm tag? (since many parts of the code is based on Anrieff Gallery Generator is it also covered under GPL2? I don't know) † actually due to limitation of javascript, multi-core is not supported.
Fast image resize/resample algorithm using Hermite filter with JavaScript. Support transparency, gives good quality. Preview: Update: version 2.0 added on GitHub (faster, web workers + transferable objects). Finally i got it working! Git: https://github.com/viliusle/Hermite-resize Demo: http://viliusle.github.io/miniPaint/ /** * Hermite resize - fast image resize/resample using Hermite filter. 1 cpu version! * * #param {HtmlElement} canvas * #param {int} width * #param {int} height * #param {boolean} resize_canvas if true, canvas will be resized. Optional. */ function resample_single(canvas, width, height, resize_canvas) { var width_source = canvas.width; var height_source = canvas.height; width = Math.round(width); height = Math.round(height); var ratio_w = width_source / width; var ratio_h = height_source / height; var ratio_w_half = Math.ceil(ratio_w / 2); var ratio_h_half = Math.ceil(ratio_h / 2); var ctx = canvas.getContext("2d"); var img = ctx.getImageData(0, 0, width_source, height_source); var img2 = ctx.createImageData(width, height); var data = img.data; var data2 = img2.data; for (var j = 0; j < height; j++) { for (var i = 0; i < width; i++) { var x2 = (i + j * width) * 4; var weight = 0; var weights = 0; var weights_alpha = 0; var gx_r = 0; var gx_g = 0; var gx_b = 0; var gx_a = 0; var center_y = (j + 0.5) * ratio_h; var yy_start = Math.floor(j * ratio_h); var yy_stop = Math.ceil((j + 1) * ratio_h); for (var yy = yy_start; yy < yy_stop; yy++) { var dy = Math.abs(center_y - (yy + 0.5)) / ratio_h_half; var center_x = (i + 0.5) * ratio_w; var w0 = dy * dy; //pre-calc part of w var xx_start = Math.floor(i * ratio_w); var xx_stop = Math.ceil((i + 1) * ratio_w); for (var xx = xx_start; xx < xx_stop; xx++) { var dx = Math.abs(center_x - (xx + 0.5)) / ratio_w_half; var w = Math.sqrt(w0 + dx * dx); if (w >= 1) { //pixel too far continue; } //hermite filter weight = 2 * w * w * w - 3 * w * w + 1; var pos_x = 4 * (xx + yy * width_source); //alpha gx_a += weight * data[pos_x + 3]; weights_alpha += weight; //colors if (data[pos_x + 3] < 255) weight = weight * data[pos_x + 3] / 250; gx_r += weight * data[pos_x]; gx_g += weight * data[pos_x + 1]; gx_b += weight * data[pos_x + 2]; weights += weight; } } data2[x2] = gx_r / weights; data2[x2 + 1] = gx_g / weights; data2[x2 + 2] = gx_b / weights; data2[x2 + 3] = gx_a / weights_alpha; } } //clear and resize canvas if (resize_canvas === true) { canvas.width = width; canvas.height = height; } else { ctx.clearRect(0, 0, width_source, height_source); } //draw ctx.putImageData(img2, 0, 0); }
Try pica - that's a highly optimized resizer with selectable algorythms. See demo. For example, original image from first post is resized in 120ms with Lanczos filter and 3px window or 60ms with Box filter and 0.5px window. For huge 17mb image 5000x3000px resize takes ~1s on desktop and 3s on mobile. All resize principles were described very well in this thread, and pica does not add rocket science. But it's optimized very well for modern JIT-s, and is ready to use out of box (via npm or bower). Also, it use webworkers when available to avoid interface freezes. I also plan to add unsharp mask support soon, because it's very useful after downscale.
I know this is an old thread but it might be useful for some people such as myself that months after are hitting this issue for the first time. Here is some code that resizes the image every time you reload the image. I am aware this is not optimal at all, but I provide it as a proof of concept. Also, sorry for using jQuery for simple selectors but I just feel too comfortable with the syntax. $(document).on('ready', createImage); $(window).on('resize', createImage); var createImage = function(){ var canvas = document.getElementById('myCanvas'); canvas.width = window.innerWidth || $(window).width(); canvas.height = window.innerHeight || $(window).height(); var ctx = canvas.getContext('2d'); img = new Image(); img.addEventListener('load', function () { ctx.drawImage(this, 0, 0, w, h); }); img.src = 'http://www.ruinvalor.com/Telanor/images/original.jpg'; }; html, body{ height: 100%; width: 100%; margin: 0; padding: 0; background: #000; } canvas{ position: absolute; left: 0; top: 0; z-index: 0; } <script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script> <html> <head> <meta charset="utf-8" /> <title>Canvas Resize</title> </head> <body> <canvas id="myCanvas"></canvas> </body> </html> My createImage function is called once when the document is loaded and after that it is called every time the window receives a resize event. I tested it in Chrome 6 and Firefox 3.6, both on the Mac. This "technique" eats processor as it if was ice cream in the summer, but it does the trick.
I've put up some algorithms to do image interpolation on html canvas pixel arrays that might be useful here: https://web.archive.org/web/20170104190425/http://jsperf.com:80/pixel-interpolation/2 These can be copy/pasted and can be used inside of web workers to resize images (or any other operation that requires interpolation - I'm using them to defish images at the moment). I haven't added the lanczos stuff above, so feel free to add that as a comparison if you'd like.
This is a javascript function adapted from #Telanor's code. When passing a image base64 as first argument to the function, it returns the base64 of the resized image. maxWidth and maxHeight are optional. function thumbnail(base64, maxWidth, maxHeight) { // Max size for thumbnail if(typeof(maxWidth) === 'undefined') var maxWidth = 500; if(typeof(maxHeight) === 'undefined') var maxHeight = 500; // Create and initialize two canvas var canvas = document.createElement("canvas"); var ctx = canvas.getContext("2d"); var canvasCopy = document.createElement("canvas"); var copyContext = canvasCopy.getContext("2d"); // Create original image var img = new Image(); img.src = base64; // Determine new ratio based on max size var ratio = 1; if(img.width > maxWidth) ratio = maxWidth / img.width; else if(img.height > maxHeight) ratio = maxHeight / img.height; // Draw original image in second canvas canvasCopy.width = img.width; canvasCopy.height = img.height; copyContext.drawImage(img, 0, 0); // Copy and resize second canvas to first canvas canvas.width = img.width * ratio; canvas.height = img.height * ratio; ctx.drawImage(canvasCopy, 0, 0, canvasCopy.width, canvasCopy.height, 0, 0, canvas.width, canvas.height); return canvas.toDataURL(); }
I'd highly suggest you check out this link and make sure it is set to true. Controlling image scaling behavior Introduced in Gecko 1.9.2 (Firefox 3.6 / Thunderbird 3.1 / Fennec 1.0) Gecko 1.9.2 introduced the mozImageSmoothingEnabled property to the canvas element; if this Boolean value is false, images won't be smoothed when scaled. This property is true by default. view plainprint? cx.mozImageSmoothingEnabled = false;
If you're simply trying to resize an image, I'd recommend setting width and height of the image with CSS. Here's a quick example: .small-image { width: 100px; height: 100px; } Note that the height and width can also be set using JavaScript. Here's quick code sample: var img = document.getElement("my-image"); img.style.width = 100 + "px"; // Make sure you add the "px" to the end, img.style.height = 100 + "px"; // otherwise you'll confuse IE Also, to ensure that the resized image looks good, add the following css rules to image selector: -ms-interpolation-mode: bicubic: introduce in IE7 image-rendering: optimizeQuality: introduced in FireFox 3.6 As far as I can tell, all browsers except IE using an bicubic algorithm to resize images by default, so your resized images should look good in Firefox and Chrome. If setting the css width and height doesn't work, you may want to play with a css transform: -moz-transform: scale(sx[, sy]) -webkit-transform:scale(sx[, sy]) If for whatever reason you need to use a canvas, please note that there are two ways an image can be resize: by resizing the canvas with css or by drawing the image at a smaller size. See this question for more details.
i got this image by right clicking the canvas element in firefox and saving as. var img = new Image(); img.onload = function () { console.debug(this.width,this.height); var canvas = document.createElement('canvas'), ctx; canvas.width = 188; canvas.height = 150; document.body.appendChild(canvas); ctx = canvas.getContext('2d'); ctx.drawImage(img,0,0,188,150); }; img.src = 'original.jpg'; so anyway, here is a 'fixed' version of your example: var img = new Image(); // added cause it wasnt defined var canvas = document.createElement("canvas"); document.body.appendChild(canvas); var ctx = canvas.getContext("2d"); var canvasCopy = document.createElement("canvas"); // adding it to the body document.body.appendChild(canvasCopy); var copyContext = canvasCopy.getContext("2d"); img.onload = function() { var ratio = 1; // defining cause it wasnt var maxWidth = 188, maxHeight = 150; if(img.width > maxWidth) ratio = maxWidth / img.width; else if(img.height > maxHeight) ratio = maxHeight / img.height; canvasCopy.width = img.width; canvasCopy.height = img.height; copyContext.drawImage(img, 0, 0); canvas.width = img.width * ratio; canvas.height = img.height * ratio; // the line to change // ctx.drawImage(canvasCopy, 0, 0, canvasCopy.width, canvasCopy.height, 0, 0, canvas.width, canvas.height); // the method signature you are using is for slicing ctx.drawImage(canvasCopy, 0, 0, canvas.width, canvas.height); }; // changed for example img.src = 'original.jpg';
For resizing to image with width less that original, i use: function resize2(i) { var cc = document.createElement("canvas"); cc.width = i.width / 2; cc.height = i.height / 2; var ctx = cc.getContext("2d"); ctx.drawImage(i, 0, 0, cc.width, cc.height); return cc; } var cc = img; while (cc.width > 64 * 2) { cc = resize2(cc); } // .. than drawImage(cc, .... ) and it works =).
I have a feeling the module I wrote will produce similar results to photoshop, as it preserves color data by averaging them, not applying an algorithm. It's kind of slow, but to me it is the best, because it preserves all the color data. https://github.com/danschumann/limby-resize/blob/master/lib/canvas_resize.js It doesn't take the nearest neighbor and drop other pixels, or sample a group and take a random average. It takes the exact proportion each source pixel should output into the destination pixel. The average pixel color in the source will be the average pixel color in the destination, which these other formulas, I think they will not be. an example of how to use is at the bottom of https://github.com/danschumann/limby-resize UPDATE OCT 2018: These days my example is more academic than anything else. Webgl is pretty much 100%, so you'd be better off resizing with that to produce similar results, but faster. PICA.js does this, I believe. –
The problem with some of this solutions is that they access directly the pixel data and loop through it to perform the downsampling. Depending on the size of the image this can be very resource intensive, and it would be better to use the browser's internal algorithms. The drawImage() function is using a linear-interpolation, nearest-neighbor resampling method. That works well when you are not resizing down more than half the original size. If you loop to only resize max one half at a time, the results would be quite good, and much faster than accessing pixel data. This function downsample to half at a time until reaching the desired size: function resize_image( src, dst, type, quality ) { var tmp = new Image(), canvas, context, cW, cH; type = type || 'image/jpeg'; quality = quality || 0.92; cW = src.naturalWidth; cH = src.naturalHeight; tmp.src = src.src; tmp.onload = function() { canvas = document.createElement( 'canvas' ); cW /= 2; cH /= 2; if ( cW < src.width ) cW = src.width; if ( cH < src.height ) cH = src.height; canvas.width = cW; canvas.height = cH; context = canvas.getContext( '2d' ); context.drawImage( tmp, 0, 0, cW, cH ); dst.src = canvas.toDataURL( type, quality ); if ( cW <= src.width || cH <= src.height ) return; tmp.src = dst.src; } } // The images sent as parameters can be in the DOM or be image objects resize_image( $( '#original' )[0], $( '#smaller' )[0] ); Credits to this post
So something interesting that I found a while ago while working with canvas that might be helpful: To resize the canvas control on its own, you need to use the height="" and width="" attributes (or canvas.width/canvas.height elements). If you use CSS to resize the canvas, it will actually stretch (i.e.: resize) the content of the canvas to fit the full canvas (rather than simply increasing or decreasing the area of the canvas. It'd be worth a shot to try drawing the image into a canvas control with the height and width attributes set to the size of the image and then using CSS to resize the canvas to the size you're looking for. Perhaps this would use a different resizing algorithm. It should also be noted that canvas has different effects in different browsers (and even different versions of different browsers). The algorithms and techniques used in the browsers is likely to change over time (especially with Firefox 4 and Chrome 6 coming out so soon, which will place heavy emphasis on canvas rendering performance). In addition, you may want to give SVG a shot, too, as it likely uses a different algorithm as well. Best of luck!
Fast and simple Javascript image resizer: https://github.com/calvintwr/blitz-hermite-resize const blitz = Blitz.create() /* Promise */ blitz({ source: DOM Image/DOM Canvas/jQuery/DataURL/File, width: 400, height: 600 }).then(output => { // handle output })catch(error => { // handle error }) /* Await */ let resized = await blizt({...}) /* Old school callback */ const blitz = Blitz.create('callback') blitz({...}, function(output) { // run your callback. }) History This is really after many rounds of research, reading and trying. The resizer algorithm uses #ViliusL's Hermite script (Hermite resizer is really the fastest and gives reasonably good output). Extended with features you need. Forks 1 worker to do the resizing so that it doesn't freeze your browser when resizing, unlike all other JS resizers out there.
I converted #syockit's answer as well as the step-down approach into a reusable Angular service for anyone who's interested: https://gist.github.com/fisch0920/37bac5e741eaec60e983 I included both solutions because they both have their own pros / cons. The lanczos convolution approach is higher quality at the cost of being slower, whereas the step-wise downscaling approach produces reasonably antialiased results and is significantly faster. Example usage: angular.module('demo').controller('ExampleCtrl', function (imageService) { // EXAMPLE USAGE // NOTE: it's bad practice to access the DOM inside a controller, // but this is just to show the example usage. // resize by lanczos-sinc filter imageService.resize($('#myimg')[0], 256, 256) .then(function (resizedImage) { // do something with resized image }) // resize by stepping down image size in increments of 2x imageService.resizeStep($('#myimg')[0], 256, 256) .then(function (resizedImage) { // do something with resized image }) })
Thanks #syockit for an awesome answer. however, I had to reformat a little as follows to make it work. Perhaps due to DOM scanning issues: $(document).ready(function () { $('img').on("load", clickA); function clickA() { var img = this; var canvas = document.createElement("canvas"); new thumbnailer(canvas, img, 50, 3); document.body.appendChild(canvas); } function thumbnailer(elem, img, sx, lobes) { this.canvas = elem; elem.width = img.width; elem.height = img.height; elem.style.display = "none"; this.ctx = elem.getContext("2d"); this.ctx.drawImage(img, 0, 0); this.img = img; this.src = this.ctx.getImageData(0, 0, img.width, img.height); this.dest = { width: sx, height: Math.round(img.height * sx / img.width) }; this.dest.data = new Array(this.dest.width * this.dest.height * 3); this.lanczos = lanczosCreate(lobes); this.ratio = img.width / sx; this.rcp_ratio = 2 / this.ratio; this.range2 = Math.ceil(this.ratio * lobes / 2); this.cacheLanc = {}; this.center = {}; this.icenter = {}; setTimeout(process1, 0, this, 0); } //returns a function that calculates lanczos weight function lanczosCreate(lobes) { return function (x) { if (x > lobes) return 0; x *= Math.PI; if (Math.abs(x) < 1e-16) return 1 var xx = x / lobes; return Math.sin(x) * Math.sin(xx) / x / xx; } } process1 = function (self, u) { self.center.x = (u + 0.5) * self.ratio; self.icenter.x = Math.floor(self.center.x); for (var v = 0; v < self.dest.height; v++) { self.center.y = (v + 0.5) * self.ratio; self.icenter.y = Math.floor(self.center.y); var a, r, g, b; a = r = g = b = 0; for (var i = self.icenter.x - self.range2; i <= self.icenter.x + self.range2; i++) { if (i < 0 || i >= self.src.width) continue; var f_x = Math.floor(1000 * Math.abs(i - self.center.x)); if (!self.cacheLanc[f_x]) self.cacheLanc[f_x] = {}; for (var j = self.icenter.y - self.range2; j <= self.icenter.y + self.range2; j++) { if (j < 0 || j >= self.src.height) continue; var f_y = Math.floor(1000 * Math.abs(j - self.center.y)); if (self.cacheLanc[f_x][f_y] == undefined) self.cacheLanc[f_x][f_y] = self.lanczos(Math.sqrt(Math.pow(f_x * self.rcp_ratio, 2) + Math.pow(f_y * self.rcp_ratio, 2)) / 1000); weight = self.cacheLanc[f_x][f_y]; if (weight > 0) { var idx = (j * self.src.width + i) * 4; a += weight; r += weight * self.src.data[idx]; g += weight * self.src.data[idx + 1]; b += weight * self.src.data[idx + 2]; } } } var idx = (v * self.dest.width + u) * 3; self.dest.data[idx] = r / a; self.dest.data[idx + 1] = g / a; self.dest.data[idx + 2] = b / a; } if (++u < self.dest.width) setTimeout(process1, 0, self, u); else setTimeout(process2, 0, self); }; process2 = function (self) { self.canvas.width = self.dest.width; self.canvas.height = self.dest.height; self.ctx.drawImage(self.img, 0, 0); self.src = self.ctx.getImageData(0, 0, self.dest.width, self.dest.height); var idx, idx2; for (var i = 0; i < self.dest.width; i++) { for (var j = 0; j < self.dest.height; j++) { idx = (j * self.dest.width + i) * 3; idx2 = (j * self.dest.width + i) * 4; self.src.data[idx2] = self.dest.data[idx]; self.src.data[idx2 + 1] = self.dest.data[idx + 1]; self.src.data[idx2 + 2] = self.dest.data[idx + 2]; } } self.ctx.putImageData(self.src, 0, 0); self.canvas.style.display = "block"; } });
I wanted some well defined functions out of answers here so ended up with these which am hoping would be useful for others also, function getImageFromLink(link) { return new Promise(function (resolve) { var image = new Image(); image.onload = function () { resolve(image); }; image.src = link; }); } function resizeImageToBlob(image, width, height, mime) { return new Promise(function (resolve) { var canvas = document.createElement('canvas'); canvas.width = width; canvas.height = height; canvas.getContext('2d').drawImage(image, 0, 0, width, height); return canvas.toBlob(resolve, mime); }); } getImageFromLink(location.href).then(function (image) { // calculate these based on the original size var width = image.width / 4; var height = image.height / 4; return resizeImageToBlob(image, width, height, 'image/jpeg'); }).then(function (blob) { // Do something with the result Blob object document.querySelector('img').src = URL.createObjectURL(blob); }); Just for the sake of testing this run it on a image opened in a tab.
I just ran a page of side by sides comparisons and unless something has changed recently, I could see no better downsizing (scaling) using canvas vs. simple css. I tested in FF6 Mac OSX 10.7. Still slightly soft vs. the original. I did however stumble upon something that did make a huge difference and that was using image filters in browsers that support canvas. You can actually manipulate images much like you can in Photoshop with blur, sharpen, saturation, ripple, grayscale, etc. I then found an awesome jQuery plug-in which makes application of these filters a snap: http://codecanyon.net/item/jsmanipulate-jquery-image-manipulation-plugin/428234 I simply apply the sharpen filter right after resizing the image which should give you the desired effect. I didn't even have to use a canvas element.
Looking for another great simple solution? var img=document.createElement('img'); img.src=canvas.toDataURL(); $(img).css("background", backgroundColor); $(img).width(settings.width); $(img).height(settings.height); This solution will use the resize algorith of browser! :)