paper.js/src/item/Raster.js

271 lines
7.1 KiB
JavaScript
Raw Normal View History

2011-03-06 19:50:44 -05:00
/*
* Paper.js
*
* This file is part of Paper.js, a JavaScript Vector Graphics Library,
* based on Scriptographer.org and designed to be largely API compatible.
2011-03-07 20:41:50 -05:00
* http://paperjs.org/
2011-03-06 19:50:44 -05:00
* http://scriptographer.org/
*
2011-03-07 20:41:50 -05:00
* Distributed under the MIT license. See LICENSE file for details.
*
2011-03-06 19:50:44 -05:00
* Copyright (c) 2011, Juerg Lehni & Jonathan Puckey
* http://lehni.org/ & http://jonathanpuckey.com/
*
2011-03-07 20:41:50 -05:00
* All rights reserved.
2011-03-06 19:50:44 -05:00
*/
var Raster = this.Raster = Item.extend({
2011-02-21 12:43:56 -05:00
beans: true,
// TODO: implement url / type, width, height
// TODO: have PlacedSymbol & Raster inherit from a shared class?
initialize: function(object) {
2011-02-21 12:43:56 -05:00
this.base();
if (object.getContext) {
this.setCanvas(object);
} else {
// If it's a string, get the element with this id first.
if (typeof object == 'string')
object = document.getElementById(object);
this.setImage(object);
2011-02-21 12:43:56 -05:00
}
this.matrix = new Matrix();
2011-02-21 12:43:56 -05:00
},
/**
* The size of the raster in pixels.
*/
getSize: function() {
return this._size;
},
setSize: function() {
var size = Size.read(arguments),
// Get reference to image before changing canvas
image = this.getImage();
2011-03-04 21:40:38 -05:00
// Setting canvas internally sets _size
this.setCanvas(CanvasProvider.getCanvas(size));
2011-03-04 21:40:38 -05:00
// Draw image back onto new canvas
2011-03-06 09:08:41 -05:00
this.getContext().drawImage(image, 0, 0, size.width, size.height);
},
2011-02-21 12:43:56 -05:00
/**
* The width of the raster in pixels.
*/
getWidth: function() {
return this._size.width;
2011-02-21 12:43:56 -05:00
},
2011-02-21 12:43:56 -05:00
/**
* The height of the raster in pixels.
*/
getHeight: function() {
return this._size.height;
2011-02-21 12:43:56 -05:00
},
/**
* Pixels per inch of the raster at it's current size.
*/
getPpi: function() {
var matrix = this.matrix;
orig = new Point(0, 0).transform(matrix),
u = new Point(1, 0).transform(matrix).subtract(orig),
v = new Point(0, 1).transform(matrix).subtract(orig);
return new Size(
2011-03-04 21:40:38 -05:00
72 / u.getLength(),
72 / v.getLength()
);
},
2011-03-04 21:40:38 -05:00
getContext: function() {
if (!this._context) {
this._context = this.getCanvas().getContext('2d');
}
return this._context;
},
setContext: function(context) {
this._context = context;
},
getCanvas: function() {
if (!this._canvas) {
this._canvas = CanvasProvider.getCanvas(this._size);
if (this._image)
this.getContext().drawImage(this._image, 0, 0);
}
return this._canvas;
},
setCanvas: function(canvas) {
if (this._canvas)
CanvasProvider.returnCanvas(this._canvas);
this._canvas = canvas;
this._size = new Size(canvas.width, canvas.height);
this._image = null;
this._context = null;
this._bounds = null;
},
getImage: function() {
return this._image || this.getCanvas();
},
setImage: function(image) {
if (this._canvas)
CanvasProvider.returnCanvas(this._canvas);
this._image = image;
// TODO: cross browser compatible?
this._size = new Size(image.naturalWidth, image.naturalHeight);
this._canvas = null;
this._context = null;
this._bounds = null;
},
getSubImage: function(/* rectangle */) {
var rectangle = Rectangle.read(arguments),
canvas = CanvasProvider.getCanvas(rectangle.getSize()),
context = canvas.getContext('2d');
context.drawImage(this.getCanvas(), rectangle.x, rectangle.y,
canvas.width, canvas.height, 0, 0, canvas.width, canvas.height);
return canvas;
},
drawImage: function(image, point) {
var point = Point.read(arguments, 1);
this.getContext().drawImage(image, point.x, point.y);
},
/**
* {@grouptitle Pixels}
*
* Gets the color of a pixel in the raster.
* @param x
* @param y
*/
getPixel: function() {
var point = Point.read(arguments),
ctx = this.getContext(),
pixels = ctx.getImageData(point.x, point.y, 1, 1).data,
channels = new Array(4);
for (var i = 0; i < 4; i++)
channels[i] = pixels[i] / 255;
return RGBColor.read(channels);
2011-02-21 12:43:56 -05:00
},
2011-02-21 12:43:56 -05:00
// TODO: setPixel(point, color)
setPixel: function(x, y, color) {
color = Color.read(arguments, 2);
var ctx = this.getContext(),
imageData = ctx.getImageData(x, y, 1, 1),
alpha = color.getAlpha();
imageData.data[0] = color.getRed() * 255;
imageData.data[1] = color.getGreen() * 255;
imageData.data[2] = color.getBlue() * 255;
imageData.data[3] = alpha != null ? alpha * 255 : 255;
ctx.putImageData(imageData, x, y);
},
_transform: function(matrix, flags) {
// In order to set the right context transformation when drawing the
// raster, simply preconcatenate the internal matrix with the provided
// one.
this.matrix.preConcatenate(matrix);
this._bounds = null;
2011-02-21 12:43:56 -05:00
},
2011-02-21 12:43:56 -05:00
getBounds: function() {
if (!this._bounds) {
this._bounds = this.matrix.transformBounds(
new Rectangle(this._size).setCenter(0, 0));
}
2011-02-21 12:43:56 -05:00
return this._bounds;
},
draw: function(ctx, param) {
ctx.save();
this.matrix.applyToContext(ctx);
ctx.drawImage(this._canvas || this._image,
-this._size.width / 2, -this._size.height / 2);
ctx.restore();
2011-02-21 12:43:56 -05:00
}
}, new function() {
function getAverageColor(pixels) {
var channels = [0, 0, 0],
total = 0;
2011-03-02 02:52:39 -05:00
for (var i = 0, l = pixels.length; i < l; i += 4) {
var alpha = pixels[i + 3];
total += alpha;
alpha /= 255;
2011-03-02 02:52:39 -05:00
channels[0] += pixels[i] * alpha;
channels[1] += pixels[i + 1] * alpha;
channels[2] += pixels[i + 2] * alpha;
}
for (var i = 0; i < 3; i++)
channels[i] /= total;
return total ? Color.read(channels) : null;
}
return {
/**
* {@grouptitle Average Color}
* Calculates the average color of the image within the given path,
* rectangle or point. This can be used for creating raster image
* effects.
*
* @param object
* @return the average color contained in the area covered by the
* specified path, rectangle or point.
*/
getAverageColor: function(object) {
var image;
if (object) {
var bounds, path;
if (object instanceof Path) {
// TODO: what if the path is smaller than 1 px?
// TODO: how about rounding of bounds.size?
// TODO: test with compound paths.
path = object;
bounds = object.getBounds();
} else if (object.width) {
bounds = new Rectangle(object);
} else if (object.x) {
2011-02-28 14:24:15 -05:00
bounds = new Rectangle(object.x - 0.5, object.y - 0.5,
1, 1);
}
var canvas = CanvasProvider.getCanvas(bounds.getSize()),
ctx = canvas.getContext('2d'),
delta = bounds.getTopLeft().multiply(-1);
ctx.translate(delta.x, delta.y);
if (path) {
2011-03-04 19:23:16 -05:00
var style = object.getStyle();
path.draw(ctx);
ctx.clip();
2011-03-04 19:23:16 -05:00
path.setStyle(style);
}
var matrix = this.matrix.clone(),
transMatrix = Matrix.getTranslateInstance(delta);
matrix.preConcatenate(transMatrix);
matrix.applyToContext(ctx);
ctx.drawImage(this._canvas || this._image,
-this._size.width / 2, -this._size.height / 2);
image = canvas;
} else {
image = this.image;
}
var size = new Size(32),
sampleCanvas = CanvasProvider.getCanvas(size),
ctx = sampleCanvas.getContext('2d');
2011-02-23 20:39:33 -05:00
ctx.drawImage(image, 0, 0, size.width, size.height);
2011-02-28 14:24:15 -05:00
var pixels = ctx.getImageData(0.5, 0.5,
size.width, size.height).data,
color = getAverageColor(pixels);
2011-02-23 19:51:49 -05:00
CanvasProvider.returnCanvas(sampleCanvas);
if (image instanceof HTMLCanvasElement)
CanvasProvider.returnCanvas(image);
return color;
}
}
2011-03-03 11:32:55 -05:00
});