whalesong/image/private/kernel.js
2011-07-06 21:28:11 -04:00

1740 lines
45 KiB
JavaScript

var world = {};
world.Kernel = {};
EXPORTS['_kernel'] = world.Kernel;
var types = plt.types;
var worldListeners = [];
var stopped;
var timerInterval = false;
// Inheritance from pg 168: Javascript, the Definitive Guide.
var heir = function(p) {
var f = function() {}
f.prototype = p;
return new f();
}
// clone: object -> object
// Copies an object. The new object should respond like the old
// object, including to things like instanceof
var clone = function(obj) {
var C = function() {}
C.prototype = obj;
var c = new C();
for (property in obj) {
if (obj.hasOwnProperty(property)) {
c[property] = obj[property];
}
}
return c;
};
var announceListeners = [];
world.Kernel.addAnnounceListener = function(listener) {
announceListeners.push(listener);
};
world.Kernel.removeAnnounceListener = function(listener) {
var idx = announceListeners.indexOf(listener);
if (idx != -1) {
announceListeners.splice(idx, 1);
}
};
world.Kernel.announce = function(eventName, vals) {
for (var i = 0; i < announceListeners.length; i++) {
try {
announceListeners[i](eventName, vals);
} catch (e) {}
}
};
// changeWorld: world -> void
// Changes the current world to newWorld.
var changeWorld = function(newWorld) {
world = newWorld;
notifyWorldListeners();
}
// updateWorld: (world -> world) -> void
// Public function: update the world, given the old state of the
// world.
world.Kernel.updateWorld = function(updater) {
var newWorld = updater(world);
changeWorld(newWorld);
}
world.Kernel.shutdownWorld = function() {
stopped = true;
};
// notifyWorldListeners: -> void
// Tells all of the world listeners that the world has changed.
var notifyWorldListeners = function() {
var i;
for (i = 0; i < worldListeners.length; i++) {
worldListeners[i](world);
}
}
// addWorldListener: (world -> void) -> void
// Adds a new world listener: whenever the world is changed, the aListener
// will be called with that new world.
var addWorldListener = function(aListener) {
worldListeners.push(aListener);
}
// getKeyCodeName: keyEvent -> String
// Given an event, try to get the name of the key.
var getKeyCodeName = function(e) {
var code = e.charCode || e.keyCode;
var keyname;
if (code == 37) {
keyname = "left";
} else if (code == 38) {
keyname = "up";
} else if (code == 39) {
keyname = "right";
} else if (code == 40) {
keyname = "down";
} else {
keyname = String.fromCharCode(code);
}
return keyname;
}
// resetWorld: -> void
// Resets all of the world global values.
var resetWorld = function() {
if (timerInterval) {
clearInterval(timerInterval);
timerInterval = false;
}
stopped = false;
worldListeners = [];
}
var getBigBangWindow = function(width, height) {
if (window.document.getElementById("canvas") != undefined) {
return window;
}
var newWindow = window.open(
"big-bang.html",
"big-bang");
//"toolbar=false,location=false,directories=false,status=false,menubar=false,width="+width+",height="+height);
if (newWindow == null) {
throw new Error("Error: Not allowed to create a new window."); }
return newWindow;
}
// scheduleTimerTick: -> void
// Repeatedly schedules an evaluation of the onTick until the program has stopped.
var scheduleTimerTick = function(window, config) {
timerInterval = window.setInterval(
function() {
if (stopped) {
window.clearTimeout(timerInterval);
timerInterval = false;
}
else {
world.Kernel.stimuli.onTick();
}
},
config.lookup('tickDelay'));
}
// Base class for all images.
var BaseImage = function(pinholeX, pinholeY) {
this.pinholeX = pinholeX;
this.pinholeY = pinholeY;
}
world.Kernel.BaseImage = BaseImage;
var isImage = function(thing) {
return (thing !== null &&
thing !== undefined &&
thing instanceof BaseImage);
}
BaseImage.prototype.updatePinhole = function(x, y) {
var aCopy = clone(this);
aCopy.pinholeX = x;
aCopy.pinholeY = y;
return aCopy;
};
// render: context fixnum fixnum: -> void
// Render the image, where the upper-left corner of the image is drawn at
// (x, y).
// NOTE: the rendering should be oblivous to the pinhole.
BaseImage.prototype.render = function(ctx, x, y) {
throw new Error('BaseImage.render unimplemented!');
// plt.Kernel.throwMobyError(
// false,
// "make-moby-error-type:generic-runtime-error",
// "Unimplemented method render");
};
// makeCanvas: number number -> canvas
// Constructs a canvas object of a particular width and height.
world.Kernel.makeCanvas = function(width, height) {
var canvas = document.createElement("canvas");
canvas.width = width;
canvas.height = height;
canvas.style.width = canvas.width + "px";
canvas.style.height = canvas.height + "px";
// KLUDGE: IE compatibility uses /js/excanvas.js, and dynamic
// elements must be marked this way.
if (window && typeof window.G_vmlCanvasManager != 'undefined') {
canvas = window.G_vmlCanvasManager.initElement(canvas);
}
return canvas;
};
var withIeHack = function(canvas, f) {
// canvas.style.display = 'none';
// document.body.appendChild(canvas);
// try {
var result = f(canvas);
// } catch(e) {
// document.body.removeChild(canvas);
// canvas.style.display = '';
// throw e;
// }
// document.body.removeChild(canvas);
// canvas.style.display = '';
return result;
};
BaseImage.prototype.toDomNode = function(cache) {
var that = this;
var width = that.getWidth();
var height = that.getHeight();
var canvas = world.Kernel.makeCanvas(width, height);
// KLUDGE: on IE, the canvas rendering functions depend on a
// context where the canvas is attached to the DOM tree.
// We initialize an afterAttach hook; the client's responsible
// for calling this after the dom node is attached to the
// document.
canvas.afterAttach = function() {
var ctx = canvas.getContext("2d");
that.render(ctx, 0, 0);
};
return canvas;
};
BaseImage.prototype.toWrittenString = function(cache) { return "<image>"; }
BaseImage.prototype.toDisplayedString = function(cache) { return "<image>"; }
BaseImage.prototype.isEqual = function(other, aUnionFind) {
return (this.pinholeX == other.pinholeX &&
this.pinholeY == other.pinholeY);
};
// isScene: any -> boolean
// Produces true when x is a scene.
var isScene = function(x) {
return ((x != undefined) && (x != null) && (x instanceof SceneImage));
};
// SceneImage: primitive-number primitive-number (listof image) -> Scene
var SceneImage = function(width, height, children, withBorder) {
BaseImage.call(this, 0, 0);
this.width = width;
this.height = height;
this.children = children; // arrayof [image, number, number]
this.withBorder = withBorder;
}
SceneImage.prototype = heir(BaseImage.prototype);
// add: image primitive-number primitive-number -> Scene
SceneImage.prototype.add = function(anImage, x, y) {
return new SceneImage(this.width,
this.height,
this.children.concat([[anImage,
x - anImage.pinholeX,
y - anImage.pinholeY]]),
this.withBorder);
};
// render: 2d-context primitive-number primitive-number -> void
SceneImage.prototype.render = function(ctx, x, y) {
var i;
var childImage, childX, childY;
// Clear the scene.
ctx.clearRect(x, y, this.width, this.height);
// Then ask every object to render itself.
for(i = 0; i < this.children.length; i++) {
childImage = this.children[i][0];
childX = this.children[i][1];
childY = this.children[i][2];
ctx.save();
childImage.render(ctx, childX + x, childY + y);
ctx.restore();
}
// Finally, draw the black border if withBorder is true
if (this.withBorder) {
ctx.strokeStyle = 'black';
ctx.strokeRect(x, y, this.width, this.height);
}
};
SceneImage.prototype.getWidth = function() {
return this.width;
};
SceneImage.prototype.getHeight = function() {
return this.height;
};
SceneImage.prototype.isEqual = function(other, aUnionFind) {
if (!(other instanceof SceneImage)) {
return false;
}
if (this.pinholeX != other.pinholeX ||
this.pinholeY != other.pinholeY ||
this.width != other.width ||
this.height != other.height ||
this.children.length != other.children.length) {
return false;
}
for (var i = 0; i < this.children.length; i++) {
var rec1 = this.children[i];
var rec2 = other.children[i];
if (rec1[1] !== rec2[1] ||
rec1[2] !== rec2[2] ||
!plt.baselib.equality.equals(rec1[0],
rec2[0],
aUnionFind)) {
return false;
}
}
return true;
};
//////////////////////////////////////////////////////////////////////
var FileImage = function(src, rawImage) {
BaseImage.call(this, 0, 0);
var self = this;
this.src = src;
this.isLoaded = false;
if (rawImage && rawImage.complete) {
this.img = rawImage;
this.isLoaded = true;
this.pinholeX = self.img.width / 2;
this.pinholeY = self.img.height / 2;
} else {
// fixme: we may want to do something blocking here for
// onload, since we don't know at this time what the file size
// should be, nor will drawImage do the right thing until the
// file is loaded.
this.img = new Image();
this.img.onload = function() {
self.isLoaded = true;
self.pinholeX = self.img.width / 2;
self.pinholeY = self.img.height / 2;
};
this.img.onerror = function(e) {
self.img.onerror = "";
self.img.src = "http://www.wescheme.org/images/broken.png";
}
this.img.src = src;
}
}
FileImage.prototype = heir(BaseImage.prototype);
// world.Kernel.FileImage = FileImage;
var imageCache = {};
FileImage.makeInstance = function(path, rawImage) {
if (! (path in imageCache)) {
imageCache[path] = new FileImage(path, rawImage);
}
return imageCache[path];
};
FileImage.installInstance = function(path, rawImage) {
imageCache[path] = new FileImage(path, rawImage);
};
FileImage.installBrokenImage = function(path) {
imageCache[path] = new TextImage("Unable to load " + path, 10,
colorDb.get("red"));
};
FileImage.prototype.render = function(ctx, x, y) {
ctx.drawImage(this.img, x, y);
};
FileImage.prototype.getWidth = function() {
return this.img.width;
};
FileImage.prototype.getHeight = function() {
return this.img.height;
};
// Override toDomNode: we don't need a full-fledged canvas here.
FileImage.prototype.toDomNode = function(cache) {
return this.img.cloneNode(true);
};
FileImage.prototype.isEqual = function(other, aUnionFind) {
return (other instanceof FileImage &&
this.pinholeX == other.pinholeX &&
this.pinholeY == other.pinholeY &&
this.src == other.src);
// types.isEqual(this.img, other.img, aUnionFind));
};
//////////////////////////////////////////////////////////////////////
// OverlayImage: image image -> image
// Creates an image that overlays img1 on top of the
// other image. shiftX and shiftY are deltas off the first
// image's pinhole.
var OverlayImage = function(img1, img2, shiftX, shiftY) {
var deltaX = img1.pinholeX - img2.pinholeX + shiftX;
var deltaY = img1.pinholeY - img2.pinholeY + shiftY;
var left = Math.min(0, deltaX);
var top = Math.min(0, deltaY);
var right = Math.max(deltaX + img2.getWidth(),
img1.getWidth());
var bottom = Math.max(deltaY + img2.getHeight(),
img1.getHeight());
BaseImage.call(this,
Math.floor((right-left) / 2),
Math.floor((bottom-top) / 2));
this.img1 = img1;
this.img2 = img2;
this.width = right - left;
this.height = bottom - top;
this.img1Dx = -left;
this.img1Dy = -top;
this.img2Dx = deltaX - left;
this.img2Dy = deltaY - top;
};
OverlayImage.prototype = heir(BaseImage.prototype);
OverlayImage.prototype.render = function(ctx, x, y) {
this.img2.render(ctx, x + this.img2Dx, y + this.img2Dy);
this.img1.render(ctx, x + this.img1Dx, y + this.img1Dy);
};
OverlayImage.prototype.getWidth = function() {
return this.width;
};
OverlayImage.prototype.getHeight = function() {
return this.height;
};
OverlayImage.prototype.isEqual = function(other, aUnionFind) {
return ( other instanceof OverlayImage &&
this.pinholeX == other.pinholeX &&
this.pinholeY == other.pinholeY &&
this.width == other.width &&
this.height == other.height &&
this.img1Dx == other.img1Dx &&
this.img1Dy == other.img1Dy &&
this.img2Dx == other.img2Dx &&
this.img2Dy == other.img2Dy &&
plt.baselib.equality.equals(this.img1, other.img1, aUnionFind) &&
plt.baselib.equality.equals(this.img2, other.img2, aUnionFind) );
};
//////////////////////////////////////////////////////////////////////
// rotate: angle image -> image
// Rotates image by angle degrees in a counter-clockwise direction.
// based on http://stackoverflow.com/questions/3276467/adjusting-div-width-and-height-after-rotated
var RotateImage = function(angle, img) {
var sin = Math.sin(angle * Math.PI / 180),
cos = Math.cos(angle * Math.PI / 180);
// (w,0) rotation
var x1 = Math.floor(cos * img.getWidth()),
y1 = Math.floor(sin * img.getWidth());
// (0,h) rotation
var x2 = Math.floor(-sin * img.getHeight()),
y2 = Math.floor( cos * img.getHeight());
// (w,h) rotation
var x3 = Math.floor(cos * img.getWidth() - sin * img.getHeight()),
y3 = Math.floor(sin * img.getWidth() + cos * img.getHeight());
var minX = Math.min(0, x1, x2, x3),
maxX = Math.max(0, x1, x2, x3),
minY = Math.min(0, y1, y2, y3),
maxY = Math.max(0, y1, y2, y3);
var rotatedWidth = maxX - minX,
rotatedHeight = maxY - minY;
// resize the image
BaseImage.call(this,
Math.floor(rotatedWidth / 2),
Math.floor(rotatedHeight / 2));
this.img = img;
this.width = rotatedWidth;
this.height = rotatedHeight;
this.angle = angle;
this.translateX = -minX;
this.translateY = -minY;
};
RotateImage.prototype = heir(BaseImage.prototype);
// translate drawing point, so that this.img appears in the UL corner. Then rotate and render this.img.
RotateImage.prototype.render = function(ctx, x, y) {
ctx.translate(this.translateX, this.translateY);
ctx.rotate(this.angle * Math.PI / 180);
this.img.render(ctx, x, y);
ctx.restore();
};
RotateImage.prototype.getWidth = function() {
return this.width;
};
RotateImage.prototype.getHeight = function() {
return this.height;
};
RotateImage.prototype.isEqual = function(other, aUnionFind) {
return ( other instanceof RotateImage &&
this.pinholeX == other.pinholeX &&
this.pinholeY == other.pinholeY &&
this.width == other.width &&
this.height == other.height &&
this.angle == other.angle &&
this.translateX == other.translateX &&
this.translateY == other.translateY &&
plt.baselib.equality.equals(this.img, other.img, aUnionFind) );
};
//////////////////////////////////////////////////////////////////////
// ScaleImage: factor factor image -> image
// Scale an image
var ScaleImage = function(xFactor, yFactor, img) {
// resize the image
BaseImage.call(this,
Math.floor((img.getWidth() * xFactor) / 2),
Math.floor((img.getHeight() * yFactor) / 2));
this.img = img;
this.width = img.getWidth() * xFactor;
this.height = img.getHeight() * yFactor;
this.xFactor = xFactor;
this.yFactor = yFactor;
};
ScaleImage.prototype = heir(BaseImage.prototype);
// scale the context, and pass it to the image's render function
ScaleImage.prototype.render = function(ctx, x, y) {
ctx.save();
ctx.scale(this.xFactor, this.yFactor);
this.img.render(ctx, x, y);
ctx.restore();
};
ScaleImage.prototype.getWidth = function() {
return this.width;
};
ScaleImage.prototype.getHeight = function() {
return this.height;
};
ScaleImage.prototype.isEqual = function(other, aUnionFind) {
return ( other instanceof ScaleImage &&
this.pinholeX == other.pinholeX &&
this.pinholeY == other.pinholeY &&
this.width == other.width &&
this.height == other.height &&
this.xFactor == other.xFactor &&
this.yFactor == other.yFactor &&
plt.baselib.equality.equals(this.img, other.img, aUnionFind) );
};
//////////////////////////////////////////////////////////////////////
var colorString = function(aColor) {
return ("rgb(" +
types.colorRed(aColor) + "," +
types.colorGreen(aColor) + ", " +
types.colorBlue(aColor) + ")");
};
var RectangleImage = function(width, height, style, color) {
BaseImage.call(this, width/2, height/2);
this.width = width;
this.height = height;
this.style = style;
this.color = color;
};
RectangleImage.prototype = heir(BaseImage.prototype);
RectangleImage.prototype.render = function(ctx, x, y) {
if (this.style.toString().toLowerCase() == "outline") {
ctx.save();
ctx.beginPath();
ctx.strokeStyle = colorString(this.color);
ctx.strokeRect(x, y, this.width, this.height);
ctx.closePath();
ctx.restore();
} else {
ctx.save();
ctx.beginPath();
ctx.fillStyle = colorString(this.color);
ctx.fillRect(x, y, this.width, this.height);
ctx.closePath();
ctx.restore();
}
};
RectangleImage.prototype.getWidth = function() {
return this.width;
};
RectangleImage.prototype.getHeight = function() {
return this.height;
};
RectangleImage.prototype.isEqual = function(other, aUnionFind) {
return (other instanceof RectangleImage &&
this.pinholeX == other.pinholeX &&
this.pinholeY == other.pinholeY &&
this.width == other.width &&
this.height == other.height &&
this.style == other.style &&
plt.baselib.equality.equals(this.color, other.color, aUnionFind));
};
//////////////////////////////////////////////////////////////////////
var TextImage = function(msg, size, color) {
BaseImage.call(this, 0, 0);
this.msg = msg;
this.size = size;
this.color = color;
this.font = this.size + "px Optimer";
var canvas = world.Kernel.makeCanvas(0, 0);
var ctx = canvas.getContext("2d");
ctx.font = this.font;
var metrics = ctx.measureText(msg);
this.width = metrics.width;
// KLUDGE: I don't know how to get at the height.
this.height = ctx.measureText("m").width + 20;
}
TextImage.prototype = heir(BaseImage.prototype);
TextImage.prototype.render = function(ctx, x, y) {
ctx.save();
ctx.font = this.font;
ctx.textAlign = 'left';
ctx.textBaseline = 'top';
ctx.fillStyle = colorString(this.color);
ctx.fillText(this.msg, x, y);
ctx.restore();
};
TextImage.prototype.getWidth = function() {
return this.width;
};
TextImage.prototype.getHeight = function() {
return this.height;
};
TextImage.prototype.isEqual = function(other, aUnionFind) {
return (other instanceof TextImage &&
this.pinholeX == other.pinholeX &&
this.pinholeY == other.pinholeY &&
this.msg == other.msg &&
this.size == other.size &&
plt.baselib.equality.equals(this.color, other.color, aUnionFind) &&
this.font == other.font);
};
//////////////////////////////////////////////////////////////////////
var CircleImage = function(radius, style, color) {
BaseImage.call(this, radius, radius);
this.radius = radius;
this.style = style;
this.color = color;
}
CircleImage.prototype = heir(BaseImage.prototype);
CircleImage.prototype.render = function(ctx, x, y) {
ctx.save();
ctx.beginPath();
ctx.arc(x + this.radius,
y + this.radius,
this.radius, 0, 2*Math.PI, false);
ctx.closePath();
if (this.style.toString().toLowerCase() == "outline") {
ctx.strokeStyle = colorString(this.color);
ctx.stroke();
} else {
ctx.fillStyle = colorString(this.color);
ctx.fill();
}
ctx.restore();
};
CircleImage.prototype.getWidth = function() {
return this.radius * 2;
};
CircleImage.prototype.getHeight = function() {
return this.radius * 2;
};
CircleImage.prototype.isEqual = function(other, aUnionFind) {
return (other instanceof CircleImage &&
this.pinholeX == other.pinholeX &&
this.pinholeY == other.pinholeY &&
this.radius == other.radius &&
this.style == other.style &&
plt.baselib.equality.equals(this.color, other.color, aUnionFind));
};
//////////////////////////////////////////////////////////////////////
// StarImage: fixnum fixnum fixnum color -> image
var StarImage = function(points, outer, inner, style, color) {
BaseImage.call(this,
Math.max(outer, inner),
Math.max(outer, inner));
this.points = points;
this.outer = outer;
this.inner = inner;
this.style = style;
this.color = color;
this.radius = Math.max(this.inner, this.outer);
};
StarImage.prototype = heir(BaseImage.prototype);
var oneDegreeAsRadian = Math.PI / 180;
// render: context fixnum fixnum -> void
// Draws a star on the given context.
// Most of this code here adapted from the Canvas tutorial at:
// http://developer.apple.com/safari/articles/makinggraphicswithcanvas.html
StarImage.prototype.render = function(ctx, x, y) {
ctx.save();
ctx.beginPath();
for( var pt = 0; pt < (this.points * 2) + 1; pt++ ) {
var rads = ( ( 360 / (2 * this.points) ) * pt ) * oneDegreeAsRadian - 0.5;
var radius = ( pt % 2 == 1 ) ? this.outer : this.inner;
ctx.lineTo(x + this.radius + ( Math.sin( rads ) * radius ),
y + this.radius + ( Math.cos( rads ) * radius ) );
}
ctx.closePath();
if (this.style.toString().toLowerCase() == "outline") {
ctx.strokeStyle = colorString(this.color);
ctx.stroke();
} else {
ctx.fillStyle = colorString(this.color);
ctx.fill();
}
ctx.restore();
};
// getWidth: -> fixnum
StarImage.prototype.getWidth = function() {
return this.radius * 2;
};
// getHeight: -> fixnum
StarImage.prototype.getHeight = function() {
return this.radius * 2;
};
StarImage.prototype.isEqual = function(other, aUnionFind) {
return (other instanceof StarImage &&
this.pinholeX == other.pinholeX &&
this.pinholeY == other.pinholeY &&
this.points == other.points &&
this.outer == other.outer &&
this.inner == other.inner &&
this.style == other.style &&
plt.baselib.equality.equals(this.color, other.color, aUnionFind));
};
//////////////////////////////////////////////////////////////////////
//Triangle
///////
var TriangleImage = function(side, style, color) {
this.width = side;
this.height = Math.ceil(side * Math.sqrt(3) / 2);
BaseImage.call(this, Math.floor(this.width/2), Math.floor(this.height/2));
this.side = side;
this.style = style;
this.color = color;
}
TriangleImage.prototype = heir(BaseImage.prototype);
TriangleImage.prototype.render = function(ctx, x, y) {
var width = this.getWidth();
var height = this.getHeight();
ctx.save();
ctx.beginPath();
ctx.moveTo(x + this.side/2, y);
ctx.lineTo(x + width, y + height);
ctx.lineTo(x, y + height);
ctx.closePath();
if (this.style.toString().toLowerCase() == "outline") {
ctx.strokeStyle = colorString(this.color);
ctx.stroke();
}
else {
ctx.fillStyle = colorString(this.color);
ctx.fill();
}
ctx.restore();
};
TriangleImage.prototype.getWidth = function() {
return this.width;
};
TriangleImage.prototype.getHeight = function() {
return this.height;
};
TriangleImage.prototype.isEqual = function(other, aUnionFind) {
return (other instanceof TriangleImage &&
this.pinholeX == other.pinholeX &&
this.pinholeY == other.pinholeY &&
this.side == other.side &&
this.style == other.style &&
plt.baselib.equality.equals(this.color, other.color, aUnionFind));
};
//////////////////////////////////////////////////////////////////////
//Ellipse
var EllipseImage = function(width, height, style, color) {
BaseImage.call(this, Math.floor(width/2), Math.floor(height/2));
this.width = width;
this.height = height;
this.style = style;
this.color = color;
};
EllipseImage.prototype = heir(BaseImage.prototype);
EllipseImage.prototype.render = function(ctx, aX, aY) {
ctx.save();
ctx.beginPath();
// Most of this code is taken from:
// http://webreflection.blogspot.com/2009/01/ellipse-and-circle-for-canvas-2d.html
var hB = (this.width / 2) * .5522848,
vB = (this.height / 2) * .5522848,
eX = aX + this.width,
eY = aY + this.height,
mX = aX + this.width / 2,
mY = aY + this.height / 2;
ctx.moveTo(aX, mY);
ctx.bezierCurveTo(aX, mY - vB, mX - hB, aY, mX, aY);
ctx.bezierCurveTo(mX + hB, aY, eX, mY - vB, eX, mY);
ctx.bezierCurveTo(eX, mY + vB, mX + hB, eY, mX, eY);
ctx.bezierCurveTo(mX - hB, eY, aX, mY + vB, aX, mY);
ctx.closePath();
if (this.style.toString().toLowerCase() == "outline") {
ctx.strokeStyle = colorString(this.color);
ctx.stroke();
}
else {
ctx.fillStyle = colorString(this.color);
ctx.fill();
}
ctx.restore();
};
EllipseImage.prototype.getWidth = function() {
return this.width;
};
EllipseImage.prototype.getHeight = function() {
return this.height;
};
EllipseImage.prototype.isEqual = function(other, aUnionFind) {
return (other instanceof EllipseImage &&
this.pinholeX == other.pinholeX &&
this.pinholeY == other.pinholeY &&
this.width == other.width &&
this.height == other.height &&
this.style == other.style &&
plt.baselib.equality.equals(this.color, other.color, aUnionFind));
};
//////////////////////////////////////////////////////////////////////
//Line
var LineImage = function(x, y, color) {
if (x >= 0) {
if (y >= 0) {
BaseImage.call(this, 0, 0);
} else {
BaseImage.call(this, 0, -y);
}
} else {
if (y >= 0) {
BaseImage.call(this, -x, 0);
} else {
BaseImage.call(this, -x, -y);
}
}
this.x = x;
this.y = y;
this.color = color;
this.width = Math.abs(x) + 1;
this.height = Math.abs(y) + 1;
}
LineImage.prototype = heir(BaseImage.prototype);
LineImage.prototype.render = function(ctx, xstart, ystart) {
ctx.save();
if (this.x >= 0) {
if (this.y >= 0) {
ctx.moveTo(xstart, ystart);
ctx.lineTo((xstart + this.x),
(ystart + this.y));
} else {
ctx.moveTo(xstart, ystart + (-this.y));
ctx.lineTo(xstart + this.x, ystart);
}
} else {
if (this.y >= 0) {
ctx.moveTo(xstart + (-this.x), ystart);
ctx.lineTo(xstart,
(ystart + this.y));
} else {
ctx.moveTo(xstart + (-this.x), ystart + (-this.y));
ctx.lineTo(xstart, ystart);
}
}
ctx.strokeStyle = colorString(this.color);
ctx.stroke();
ctx.restore();
};
LineImage.prototype.getWidth = function() {
return this.width;
};
LineImage.prototype.getHeight = function() {
return this.height;
};
LineImage.prototype.isEqual = function(other, aUnionFind) {
return (other instanceof LineImage &&
this.pinholeX == other.pinholeX &&
this.pinholeY == other.pinholeY &&
this.x == other.x &&
this.y == other.y &&
plt.baselib.equality.equals(this.color, other.color, aUnionFind));
};
//////////////////////////////////////////////////////////////////////
// Effects
/**
* applyEffect: compound-effect -> (arrayof (world -> world))
applyEffect applies all of the effects
@param aCompEffect a compound effect is either a scheme list of
compound effects or a single primitive effect */
world.Kernel.applyEffect = function(aCompEffect) {
if ( plt.baselib.lists.isEmpty(aCompEffect) ) {
// Do Nothing
} else if ( plt.baselib.lists.isPair(aCompEffect) ) {
var results = world.Kernel.applyEffect(aCompEffect.first());
return results.concat(world.Kernel.applyEffect(aCompEffect.rest()));
} else {
var newResult = aCompEffect.run();
if (newResult) {
return newResult;
}
}
return [];
}
//////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////
// Exports
world.Kernel.isImage = isImage;
world.Kernel.isScene = isScene;
world.Kernel.isColor = function(thing) {
return (types.isColor(thing) ||
((types.isString(thing) || plt.baselib.symbols.isSymbol(thing)) &&
typeof(colorDb.get(thing)) != 'undefined'));
};
world.Kernel.colorDb = colorDb;
world.Kernel.sceneImage = function(width, height, children, withBorder) {
return new SceneImage(width, height, children, withBorder);
};
world.Kernel.circleImage = function(radius, style, color) {
return new CircleImage(radius, style, color);
};
world.Kernel.starImage = function(points, outer, inner, style, color) {
return new StarImage(points, outer, inner, style, color);
};
world.Kernel.rectangleImage = function(width, height, style, color) {
return new RectangleImage(width, height, style, color);
};
world.Kernel.triangleImage = function(side, style, color) {
return new TriangleImage(side, style, color);
};
world.Kernel.ellipseImage = function(width, height, style, color) {
return new EllipseImage(width, height, style, color);
};
world.Kernel.lineImage = function(x, y, color) {
return new LineImage(x, y, color);
};
world.Kernel.overlayImage = function(img1, img2, shiftX, shiftY) {
return new OverlayImage(img1, img2, shiftX, shiftY);
};
world.Kernel.rotateImage = function(angle, img) {
return new RotateImage(angle, img);
};
world.Kernel.scaleImage = function(xFactor, yFactor, img) {
return new ScaleImage(xFactor, yFactor, img);
};
world.Kernel.textImage = function(msg, size, color) {
return new TextImage(msg, size, color);
};
world.Kernel.fileImage = function(path, rawImage) {
return FileImage.makeInstance(path, rawImage);
};
world.Kernel.isSceneImage = function(x) { return x instanceof SceneImage; };
world.Kernel.isCircleImage = function(x) { return x instanceof CircleImage; };
world.Kernel.isStarImage = function(x) { return x instanceof StarImage; };
world.Kernel.isRectangleImage = function(x) { return x instanceof RectangleImage; };
world.Kernel.isTriangleImage = function(x) { return x instanceof TriangleImage; };
world.Kernel.isEllipseImage = function(x) { return x instanceof EllipseImage; };
world.Kernel.isLineImage = function(x) { return x instanceof LineImage; };
world.Kernel.isOverlayImage = function(x) { return x instanceof OverlayImage; };
world.Kernel.isRotateImage = function(x) { return x instanceof RotateImage; };
world.Kernel.isTextImage = function(x) { return x instanceof TextImage; };
world.Kernel.isFileImage = function(x) { return x instanceof FileImage; };
//////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////
// Feeds stimuli inputs into the world. The functions here
// are responsible for converting to Scheme values.
//
// NOTE and WARNING: make sure to really do the coersions, even for
// strings. Bad things happen otherwise, as in the sms stuff, where
// we're getting string-like values that aren't actually strings.
world.stimuli = {};
world.Kernel.stimuli = world.stimuli;
(function() {
var handlers = [];
var doNothing = function() {};
var StimuliHandler = function(config, caller, restarter) {
this.config = config;
this.caller = caller;
this.restarter = restarter;
handlers.push(this);
};
// StimuliHandler.prototype.failHandler = function(e) {
// this.onShutdown();
// this.restarter(e);
// };
// doStimuli: CPS( (world -> effect) (world -> world) -> void )
//
// Processes a stimuli by compute the effect and applying it, and
// computing a new world to replace the old.
StimuliHandler.prototype.doStimuli = function(computeEffectF, computeWorldF, restArgs, k) {
var effectUpdaters = [];
var that = this;
try {
that.change(function(w, k2) {
var args = [w].concat(restArgs);
var doStimuliHelper = function() {
if (computeWorldF) {
that.caller(computeWorldF, args, k2);
} else {
k2(w);
}
};
doStimuliHelper();
}, k);
// if (computeEffectF) {
// that.caller(computeEffectF, [args],
// function(effect) {
// effectUpdaters = applyEffect(effect);
// doStimuliHelper();
// },
// this.failHandler);
// }
// else { doStimuliHelper(); }
// },
// function() {
// helpers.forEachK(effectUpdaters,
// function(effect, k2) { that.change(effect, k2); },
// function(e) { throw e; },
// k);
// });
} catch (e) {
// if (console && console.log && e.stack) {
// console.log(e.stack);
// }
this.onShutdown();
}
}
// Orientation change
// args: [azimuth, pitch, roll]
StimuliHandler.prototype.onTilt = function(args, k) {
var onTilt = this.lookup("onTilt");
var onTiltEffect = this.lookup("onTiltEffect");
this.doStimuli(onTiltEffect, onTilt, helpers.map(flt, args), k);
};
// Accelerations
// args: [x, y, z]
StimuliHandler.prototype.onAcceleration = function(args, k) {
var onAcceleration = this.lookup('onAcceleration');
var onAccelerationEffect = this.lookup('onAccelerationEffect');
this.doStimuli(onAccelerationEffect, onAcceleration, helpers.map(flt, args), k);
};
// Shakes
// args: []
StimuliHandler.prototype.onShake = function(args, k) {
var onShake = this.lookup('onShake');
var onShakeEffect = this.lookup('onShakeEffect');
this.doStimuli(onShakeEffect, onShake, [], k);
};
// Sms receiving
// args: [sender, message]
StimuliHandler.prototype.onSmsReceive = function(args, k) {
var onSmsReceive = this.lookup('onSmsReceive');
var onSmsReceiveEffect = this.lookup('onSmsReceiveEffect');
// IMPORTANT: must coerse to string by using x+"". Do not use
// toString(): it's not safe.
this.doStimuli(onSmsReceiveEffect, onSmsReceive, [args[0]+"", args[1]+""], k);
};
// Locations
// args: [lat, lng]
StimuliHandler.prototype.onLocation = function(args, k) {
var onLocationChange = this.lookup('onLocationChange');
var onLocationChangeEffect = this.lookup('onLocationChangeEffect');
this.doStimuli(onLocationChangeEffect, onLocationChange, helpers.map(flt, args), k);
};
// Keystrokes
// args: [e]
StimuliHandler.prototype.onKey = function(args, k) {
// getKeyCodeName: keyEvent -> String
// Given an event, try to get the name of the key.
var getKeyCodeName = function(e) {
var code = e.charCode || e.keyCode;
var keyname;
switch(code) {
case 16: keyname = "shift"; break;
case 17: keyname = "control"; break;
case 19: keyname = "pause"; break;
case 27: keyname = "escape"; break;
case 33: keyname = "prior"; break;
case 34: keyname = "next"; break;
case 35: keyname = "end"; break;
case 36: keyname = "home"; break;
case 37: keyname = "left"; break;
case 38: keyname = "up"; break;
case 39: keyname = "right"; break;
case 40: keyname = "down"; break;
case 42: keyname = "print"; break;
case 45: keyname = "insert"; break;
case 46: keyname = String.fromCharCode(127); break;
case 106: keyname = "*"; break;
case 107: keyname = "+"; break;
case 109: keyname = "-"; break;
case 110: keyname = "."; break;
case 111: keyname = "/"; break;
case 144: keyname = "numlock"; break;
case 145: keyname = "scroll"; break;
case 186: keyname = ";"; break;
case 187: keyname = "="; break;
case 188: keyname = ","; break;
case 189: keyname = "-"; break;
case 190: keyname = "."; break;
case 191: keyname = "/"; break;
case 192: keyname = "`"; break;
case 219: keyname = "["; break;
case 220: keyname = "\\"; break;
case 221: keyname = "]"; break;
case 222: keyname = "'"; break;
default: if (code >= 96 && code <= 105) {
keyname = (code - 96).toString();
}
else if (code >= 112 && code <= 123) {
keyname = "f" + (code - 111);
}
else {
keyname = String.fromCharCode(code).toLowerCase();
}
break;
}
return keyname;
}
var keyname = getKeyCodeName(args[0]);
var onKey = this.lookup('onKey');
var onKeyEffect = this.lookup('onKeyEffect');
this.doStimuli(onKeyEffect, onKey, [keyname], k);
};
// // Time ticks
// // args: []
// StimuliHandler.prototype.onTick = function(args, k) {
// var onTick = this.lookup('onTick');
// var onTickEffect = this.lookup('onTickEffect');
// this.doStimuli(onTickEffect, onTick, [], k);
// };
// Announcements
// args: [eventName, vals]
StimuliHandler.prototype.onAnnounce = function(args, k) {
var vals = args[1];
var valsList = plt.baselib.lists.EMPTY;
for (var i = 0; i < vals.length; i++) {
valsList = plt.baselib.lists.makeCons(vals[vals.length - i - 1], valsList);
}
var onAnnounce = this.lookup('onAnnounce');
var onAnnounceEffect = this.lookup('onAnnounceEffect');
this.doStimuli(onAnnounce, onAnnounceEffect, [args[0], valsList], k);
};
// The shutdown stimuli: special case that forces a world computation to quit.
// Also removes this instance from the list of handlers
StimuliHandler.prototype.onShutdown = function() {
var index = handlers.indexOf(this);
if (index != -1) {
handlers.splice(index, 1);
}
var shutdownWorld = this.lookup('shutdownWorld');
if (shutdownWorld) {
shutdownWorld();
}
};
//////////////////////////////////////////////////////////////////////
// Helpers
var flt = plt.baselib.numbers.makeFloat;
StimuliHandler.prototype.lookup = function(s) {
return this.config.lookup(s);
};
StimuliHandler.prototype.change = function(f, k) {
if (this.lookup('changeWorld')) {
this.lookup('changeWorld')(f, k);
}
else { k(); }
};
// applyEffect: compound-effect: (arrayof (world -> world))
var applyEffect = function(e) {
return world.Kernel.applyEffect(e);
};
var makeStimulusHandler = function(funName) {
return function() {
var args = arguments;
for (var i = 0; i < handlers.length; i++) {
(handlers[i])[funName](args, doNothing);
}
// helpers.forEachK(handlers,
// function(h, k) { h[funName](args, k); },
// function(e) { throw e; },
// doNothing);
}
};
//////////////////////////////////////////////////////////////////////
// Exports
world.stimuli.StimuliHandler = StimuliHandler;
world.stimuli.onTilt = makeStimulusHandler('onTilt');
world.stimuli.onAcceleration = makeStimulusHandler('onAcceleration');
world.stimuli.onShake = makeStimulusHandler('onShake');
world.stimuli.onSmsReceive = makeStimulusHandler('onSmsReceive');
world.stimuli.onLocation = makeStimulusHandler('onLocation');
world.stimuli.onKey = makeStimulusHandler('onKey');
// world.stimuli.onTick = makeStimulusHandler('onTick');
world.stimuli.onAnnounce = makeStimulusHandler('onAnnounce');
world.stimuli.massShutdown = function() {
for (var i = 0; i < handlers.length; i++) {
var shutdownWorld = handlers[i].lookup('shutdownWorld');
if (shutdownWorld) {
shutdownWorld();
}
}
handlers = [];
};
})();
//////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////
(function() {
// var make_dash_effect_colon_none =
// (plt.Kernel.invokeModule("moby/runtime/effect-struct")
// .EXPORTS['make-effect:none']);
world.config = {};
world.Kernel.config = world.config;
// augment: hash hash -> hash
// Functionally extend a hashtable with another one.
var augment = function(o, a) {
var oo = {};
for (var e in o) {
if (o.hasOwnProperty(e)) {
oo[e] = o[e];
}
}
for (var e in a) {
if (a.hasOwnProperty(e)) {
oo[e] = a[e];
}
}
return oo;
}
var WorldConfig = function() {
// The following handler values are initially false until they're updated
// by configuration.
// A handler is a function:
// handler: world X Y ... -> Z
this.vals = {
// changeWorld: (world -> world) -> void
// When called, this will update the world based on the
// updater passed to it.
changeWorld: false,
// shutdownWorld: -> void
// When called, this will shut down the world computation.
shutdownWorld: false,
// initialEffect: effect
// The initial effect to invoke when the world computation
// begins.
initialEffect: false,
// onRedraw: world -> scene
onRedraw: false,
// onDraw: world -> (sexpof dom)
onDraw: false,
// onDrawCss: world -> (sexpof css-style)
onDrawCss: false,
// tickDelay: number
tickDelay: false,
// onTick: world -> world
onTick: false,
// onTickEffect: world -> effect
onTickEffect: false,
// onKey: world key -> world
onKey: false,
// onKeyEffect: world key -> effect
onKeyEffect : false,
// onTilt: world number number number -> world
onTilt: false,
// onTiltEffect: world number number number -> effect
onTiltEffect: false,
// onAcceleration: world number number number -> world
onAcceleration: false,
// onAccelerationEffect: world number number number -> effect
onAccelerationEffect: false,
// onShake: world -> world
onShake: false,
// onShakeEffect: world -> effect
onShakeEffect: false,
// onSmsReceive: world -> world
onSmsReceive: false,
// onSmsReceiveEffect: world -> effect
onSmsReceiveEffect: false,
// onLocationChange: world number number -> world
onLocationChange : false,
// onLocationChangeEffect: world number number -> effect
onLocationChangeEffect: false,
// onAnnounce: world string X ... -> world
onAnnounce: false,
// onAnnounce: world string X ... -> effect
onAnnounceEffect: false,
// stopWhen: world -> boolean
stopWhen: false,
// stopWhenEffect: world -> effect
stopWhenEffect: false,
//////////////////////////////////////////////////////////////////////
// For universe game playing
// connectToGame: string
// Registers with some universe, given an identifier
// which is a URL to a Universe server.
connectToGame: false,
onGameStart: false,
onOpponentTurn: false,
onMyTurn: false,
afterMyTurn: false,
onGameFinish: false
};
}
// WorldConfig.lookup: string -> handler
// Looks up a value in the configuration.
WorldConfig.prototype.lookup = function(key) {
// plt.Kernel.check(key, plt.Kernel.isString, "WorldConfig.lookup", "string", 1);
if (key in this.vals) {
return this.vals[key];
} else {
throw Error("Can't find " + key + " in the configuration");
}
}
// WorldConfig.updateAll: (hashof string handler) -> WorldConfig
WorldConfig.prototype.updateAll = function(aHash) {
var result = new WorldConfig();
result.vals = augment(this.vals, aHash);
return result;
}
world.config.WorldConfig = WorldConfig;
// The following global variable CONFIG is mutated by either
// big-bang from the regular world or the one in jsworld.
world.config.CONFIG = new WorldConfig();
// A handler is a function that consumes a config and produces a
// config.
//////////////////////////////////////////////////////////////////////
var getNoneEffect = function() {
throw new Error("getNoneEffect: We should not be calling effects!");
// return make_dash_effect_colon_none();
}
//////////////////////////////////////////////////////////////////////
world.config.Kernel = world.config.Kernel || {};
world.config.Kernel.getNoneEffect = getNoneEffect;
/*
// makeSimplePropertyUpdater: (string (X -> boolean) string string) -> (X -> handler)
var makeSimplePropertyUpdater = function(propertyName,
propertyPredicate,
propertyTypeName,
updaterName) {
return function(val) {
plt.Kernel.check(val, propertyPredicate, updaterName, propertyTypeName, 1);
return addStringMethods(
function(config) {
return config.updateAll({propertyName: val });
}, updaterName);
}
};
// connects to the game
world.config.Kernel.connect_dash_to_dash_game =
makeSimplePropertyUpdater('connectToGame',
plt.Kernel.isString,
"string",
"connect-to-game");
// Registers a handler for game-start events.
world.config.Kernel.on_dash_game_dash_start =
makeSimplePropertyUpdater('onGameStart',
plt.Kernel.isFunction,
"function",
"on-game-start");
// Registers a handler for opponent-turn events.
world.config.Kernel.on_dash_opponent_dash_turn =
makeSimplePropertyUpdater('onOpponentTurn',
plt.Kernel.isFunction,
"function",
"on-opponent-turn");
// Registers a handler for my turn.
world.config.Kernel.on_dash_my_dash_turn =
makeSimplePropertyUpdater('onMyTurn',
plt.Kernel.isFunction,
"function",
"on-my-turn");
// Register a handler after I make a move.
world.config.Kernel.after_dash_my_dash_turn =
makeSimplePropertyUpdater('afterMyTurn',
plt.Kernel.isFunction,
"function",
"after-my-turn");
world.config.Kernel.on_dash_game_dash_finish =
makeSimplePropertyUpdater('onGameFinish',
plt.Kernel.isFunction,
"function",
"on-game-finish");
*/
})();