diff --git a/README.md b/README.md index f352e28..dd76c75 100644 --- a/README.md +++ b/README.md @@ -1,6 +1,8 @@ Whalesong ========= +IMPORTANT: Use -j 1 to build Whalesong (this turns off parallel builds) + This fork of Whalesong differs from dyoo/whalesong in the following ways: * Builds on latest release of Racket @@ -11,9 +13,13 @@ This fork of Whalesong differs from dyoo/whalesong in the following ways: (require whalesong/lang/match) * Adds on-release (as a complement to on-key) + Contributed by Darren Cruse * Adds parameters (require whalesong/lang/parameters) - + * Extended whalesong/image and whalesong/images + (more functions, bug fixes, not matches WeScheme) + Contributed by Emmanuel Schanzer + Note: The implementation of parameters works fine, as long as you don't mix parameterize with non-local-exits and reentries (i.e. call/cc and friends) diff --git a/whalesong/image/private/js-impl.js b/whalesong/image/private/js-impl.js index f000d0a..3ca222d 100644 --- a/whalesong/image/private/js-impl.js +++ b/whalesong/image/private/js-impl.js @@ -37,9 +37,12 @@ var isFontWeight = function(x){ || (x === false); // false is also acceptable }; var isMode = function(x) { - return ((isString(x) || isSymbol(x)) && - (x.toString().toLowerCase() == "solid" || - x.toString().toLowerCase() == "outline")); + return ((isString(x) || isSymbol(x)) && + (x.toString().toLowerCase() == "solid" || + x.toString().toLowerCase() == "outline")) || + ((jsnums.isReal(x)) && + (jsnums.greaterThanOrEqual(x, 0) && + jsnums.lessThanOrEqual(x, 255))); }; var isPlaceX = function(x) { @@ -67,6 +70,24 @@ var isStyle = function(x) { +// Useful trigonometric functions based on htdp teachpack + +// excess : compute the Euclidean excess +// Note: If the excess is 0, then C is 90 deg. +// If the excess is negative, then C is obtuse. +// If the excess is positive, then C is acuse. +function excess(sideA, sideB, sideC) { + return sideA*sideA + sideB*sideB - sideC*sideC; +} + +// return c^2 = a^2 + b^2 - 2ab cos(C) +function cosRel(sideA, sideB, angleC) { + return (sideA*sideA) + (sideB*sideB) - (2*sideA*sideB*Math.cos(angleC * Math.PI/180)); +} + +var less = function(lhs, rhs) { + return (rhs - lhs) > 0.00001; +} @@ -137,7 +158,7 @@ var checkAngle = plt.baselib.check.makeCheckArgumentType( var checkMode = plt.baselib.check.makeCheckArgumentType( isMode, - 'solid or outline'); + 'solid or outline or [0-255]'); var checkSideCount = plt.baselib.check.makeCheckArgumentType( @@ -161,10 +182,18 @@ var checkListofColor = plt.baselib.check.makeCheckListofArgumentType( - - ////////////////////////////////////////////////////////////////////// +EXPORTS['image=?'] = + makePrimitiveProcedure( + 'image=?', + 2, + function(MACHINE) { + var img1 = checkImageOrScene(MACHINE,'image=?', 0); + var img2 = checkImageOrScene(MACHINE,'image=?', 1); + return img1.equals(img2); + }); + EXPORTS['image-color?'] = makePrimitiveProcedure( @@ -322,6 +351,40 @@ EXPORTS['image-url'] = 'image-url'); +EXPORTS['video/url'] = + makeClosure( + 'video/url', + 1, + function(MACHINE) { + var path = checkString(MACHINE, 'video/url', 0); + PAUSE( + function(restart) { + var rawVideo = document.createElement('video'); + rawVideo.src = path.toString(); + rawVideo.addEventListener('canplay', function() { + restart(function(MACHINE) { + function pause(){ rawVideo.pause(); return true;}; + finalizeClosureCall( + MACHINE, + makeFileVideo(path.toString(), rawVideo)); +// aState.addBreakRequestedListener(pause); + }); + }); + rawVideo.addEventListener('error', function(e) { + restart(function(MACHINE) { + plt.baselib.exceptions.raiseFailure( + MACHINE, + plt.baselib.format.format( + "unable to load ~a: ~a", + [url, + e.message])); + }); + }); + rawVideo.src = path.toString(); + } + ); + }); + EXPORTS['overlay'] = @@ -580,6 +643,28 @@ EXPORTS['empty-scene'] = true); }); +EXPORTS['put-image'] = + makePrimitiveProcedure( + 'put-image', + 4, + function(MACHINE) { + var picture = checkImage(MACHINE, "put-image", 0); + var x = checkReal(MACHINE, "put-image", 1); + var y = checkReal(MACHINE, "put-image", 2); + var background = checkImageOrScene(MACHINE, "place-image", 3); + if (isScene(background)) { + return background.add(picture, jsnums.toFixnum(x), background.getHeight() - jsnums.toFixnum(y)); + } else { + var newScene = makeSceneImage(background.getWidth(), + background.getHeight(), + [], + false); + newScene = newScene.add(background, background.getWidth()/2, background.getHeight()/2); + newScene = newScene.add(picture, jsnums.toFixnum(x), background.getHeight() - jsnums.toFixnum(y)); + return newScene; + } + + }); EXPORTS['place-image'] = @@ -598,8 +683,8 @@ EXPORTS['place-image'] = background.getHeight(), [], false); - newScene = newScene.add(background.updatePinhole(0, 0), 0, 0); - newScene = newScene.add(picture, jsnums.toFixnum(x), jsnums.toFixnum(y)); + newScene = newScene.add(background, background.getWidth()/2, background.getHeight()/2); + newScene = newScene.add(picture, jsnums.toFixnum(x), jsnums.toFixnum(y)); return newScene; } @@ -632,8 +717,8 @@ EXPORTS['place-image/align'] = background.getHeight(), [], false); - newScene = newScene.add(background.updatePinhole(0, 0), 0, 0); - newScene = newScene.add(img, jsnums.toFixnum(x), jsnums.toFixnum(y)); + newScene = newScene.add(background, background.getWidth()/2, background.getHeight()/2); + newScene = newScene.add(img, jsnums.toFixnum(x), jsnums.toFixnum(y)); return newScene; } }); @@ -784,17 +869,19 @@ EXPORTS['scene+line'] = var c = checkColor(MACHINE, "scene+line", 5); // make a scene containing the image var newScene = makeSceneImage(jsnums.toFixnum(img.getWidth()), - jsnums.toFixnum(img.getHeight()), - [], - true); - newScene = newScene.add(img.updatePinhole(0, 0), 0, 0); + jsnums.toFixnum(img.getHeight()), + [], + false); + newScene = newScene.add(img, img.getWidth()/2, img.getHeight()/2); // make an image containing the line var line = makeLineImage(jsnums.toFixnum(x2-x1), - jsnums.toFixnum(y2-y1), - c, - false); + jsnums.toFixnum(y2-y1), + c, + false), + leftMost = Math.min(x1,x2), + topMost = Math.min(y1,y2); // add the line to scene, offset by the original amount - return newScene.add(line, jsnums.toFixnum(x1), jsnums.toFixnum(y1)); + return newScene.add(line, line.getWidth()/2+leftMost, line.getHeight()/2+topMost); }); @@ -836,9 +923,27 @@ EXPORTS['rectangle'] = s.toString(), c); }); +/* + need to port over checks for isListofPosns and isListOfLength + +EXPORTS['polygon'] = + makePrimitiveProcedure( + 'polygon', + 3, + function(MACHINE) { + function isPosnList(lst){ return isListOf(lst, types.isPosn); } + var points = checkListOfLength(MACHINE, "polygon", 0); + var points = checkListOfPosns(MACHINE, "polygon", 0); + var s = checkMode(MACHINE, "polygon", 2); + var c = checkColor(MACHINE, "polygon", 3); + return makePosnImage(points, + s.toString(), + c); + }); -EXPORTS['regular-polygon'] = +*/ +EXPORTS['regular-polygon'] = makePrimitiveProcedure( 'regular-polygon', 4, @@ -880,14 +985,219 @@ EXPORTS['triangle'] = var s = checkNonNegativeReal(MACHINE, "triangle", 0); var m = checkMode(MACHINE, "triangle", 1); var c = checkColor(MACHINE, "triangle", 2); - return makeTriangleImage(jsnums.toFixnum(s), - 60, - m.toString(), - c); + return makeTriangleImage(jsnums.toFixnum(s), + jsnums.toFixnum(360-60), + jsnums.toFixnum(s), + m.toString(), + c); }); -EXPORTS['right-triangle'] = +EXPORTS['triangle/sas'] = + makePrimitiveProcedure( + 'triangle/sas', + 5, + function(MACHINE) { + var sideA = checkNonNegativeReal(MACHINE, "triangle/sas", 0); + var angleB = checkAngle(MACHINE, "triangle/sas", 1); + var sideC = checkNonNegativeReal(MACHINE, "triangle/sas", 2); + var style = checkMode(MACHINE, "triangle/sas", 3); + var color = checkColor(MACHINE, "triangle/sas", 4); + // cast to fixnums + sideA = jsnums.toFixnum(sideA); angleB = jsnums.toFixnum(angleB); sideC = jsnums.toFixnum(sideC); + var sideB2 = cosRel(sideA, sideC, angleB); + var sideB = Math.sqrt(sideB2); + + if (sideB2 <= 0) { + raise( types.incompleteExn(types.exnFailContract, "The given side, angle and side will not form a triangle: " + + sideA + ", " + angleB + ", " + sideC, []) ); + } else { + if (less(sideA + sideC, sideB) || + less(sideB + sideC, sideA) || + less(sideA + sideB, sideC)) { + raise( types.incompleteExn(types.exnFailContract, "The given side, angle and side will not form a triangle: " + + sideA + ", " + angleB + ", " + sideC, []) ); + } + } + + var angleA = Math.acos(excess(sideB, sideC, sideA) / (2 * sideB * sideC)) * (180 / Math.PI); + + return makeTriangleImage(jsnums.toFixnum(sideC), + jsnums.toFixnum(angleA), + jsnums.toFixnum(sideB), + style.toString(), + color); + }); + +EXPORTS['triangle/sss'] = + makePrimitiveProcedure( + 'triangle/sss', + 5, + function(MACHINE) { + var sideA = checkNonNegativeReal(MACHINE, "triangle/sss", 0); + var sideB = checkNonNegativeReal(MACHINE, "triangle/sss", 1); + var sideC = checkNonNegativeReal(MACHINE, "triangle/sss", 2); + var style = checkMode(MACHINE, "triangle/sss", 3); + var color = checkColor(MACHINE, "triangle/sss", 4); + // cast to fixnums + sideA = jsnums.toFixnum(sideA); sideB = jsnums.toFixnum(sideB); sideC = jsnums.toFixnum(sideC); + if (less(sideA + sideB, sideC) || + less(sideC + sideB, sideA) || + less(sideA + sideC, sideB)) { + raise( types.incompleteExn(types.exnFailContract, "The given sides will not form a triangle: " + + sideA+", "+sideB+", "+sideC, []) ); + } + + var angleA = Math.acos(excess(sideB, sideC, sideA) / (2 * sideB * sideC)) * (180 / Math.PI); + return makeTriangleImage(jsnums.toFixnum(sideC), + jsnums.toFixnum(angleA), + jsnums.toFixnum(sideB), + style.toString(), + color); + }); + +EXPORTS['triangle/ass'] = + makePrimitiveProcedure( + 'triangle/ass', + 5, + function(MACHINE) { + var angleA = checkAngle(MACHINE, "triangle/ass", 0); + var sideB = checkNonNegativeReal(MACHINE, "triangle/ass", 1); + var sideC = checkNonNegativeReal(MACHINE, "triangle/ass", 2); + var style = checkMode(MACHINE, "triangle/ass", 3); + var color = checkColor(MACHINE, "triangle/ass", 4); + // cast to fixnums + angleA = jsnums.toFixnum(angleA); sideB = jsnums.toFixnum(sideB); sideC = jsnums.toFixnum(sideC); + if (colorDb.get(color)) { color = colorDb.get(color); } + if (less(180, angleA)) { + raise( types.incompleteExn(types.exnFailContract, "The given angle, side and side will not form a triangle: " + + angleA + ", " + sideB + ", " + sideC, []) ); + } + return makeTriangleImage(jsnums.toFixnum(sideC), + jsnums.toFixnum(angleA), + jsnums.toFixnum(sideB), + style.toString(), + color); + }); + +EXPORTS['triangle/ssa'] = + makePrimitiveProcedure( + 'triangle/ssa', + 5, + function(MACHINE) { + var sideA = checkNonNegativeReal(MACHINE, "triangle/ssa", 0); + var sideB = checkNonNegativeReal(MACHINE, "triangle/ssa", 1); + var angleC = checkAngle(MACHINE, "triangle/ssa", 2); + var style = checkMode(MACHINE, "triangle/ssa", 3); + var color = checkColor(MACHINE, "triangle/ssa", 4); + // cast to fixnums + sideA = jsnums.toFixnum(sideA); sideB = jsnums.toFixnum(sideB); angleC = jsnums.toFixnum(angleC); + if (less(180, angleC)) { + raise( types.incompleteExn(types.exnFailContract, "The given side, side and angle will not form a triangle: " + + sideA + ", " + sideB + ", " + angleC, []) ); + } + var sideC2 = cosRel(sideA, sideB, angleC); + var sideC = Math.sqrt(sideC2); + + if (sideC2 <= 0) { + raise( types.incompleteExn(types.exnFailContract, "The given side, side and angle will not form a triangle: " + + sideA + ", " + sideB + ", " + angleC, []) ); + } else { + if (less(sideA + sideB, sideC) || + less(sideC + sideB, sideA) || + less(sideA + sideC, sideB)) { + raise( types.incompleteExn(types.exnFailContract, "The given side, side and angle will not form a triangle: " + + sideA + ", " + sideB + ", " + angleC, []) ); + } + } + + var angleA = Math.acos(excess(sideB, sideC, sideA) / (2 * sideB * sideC)) * (180 / Math.PI); + return makeTriangleImage(jsnums.toFixnum(sideC), + jsnums.toFixnum(angleA), + jsnums.toFixnum(sideB), + style.toString(), + color); + }); + +EXPORTS['triangle/aas'] = + makePrimitiveProcedure( + 'triangle/aas', + 5, + function(MACHINE) { + var angleA = checkAngle(MACHINE, "triangle/aas", 0); + var angleB = checkAngle(MACHINE, "triangle/aas", 1); + var sideC = checkNonNegativeReal(MACHINE, "triangle/aas", 2); + var style = checkMode(MACHINE, "triangle/aas", 3); + var color = checkColor(MACHINE, "triangle/aas", 4); + // cast to fixnums + var angleA = jsnums.toFixnum(angleA); angleB = jsnums.toFixnum(angleB); sideC = jsnums.toFixnum(sideC); + var angleC = (180 - angleA - angleB); + if (less(angleC, 0)) { + raise( types.incompleteExn(types.exnFailContract, "The given angle, angle and side will not form a triangle: " + + angleA + ", " + angleB + ", " + sideC, []) ); + } + var hypotenuse = sideC / (Math.sin(angleC*Math.PI/180)) + var sideB = hypotenuse * Math.sin(angleB*Math.PI/180); + return makeTriangleImage(jsnums.toFixnum(sideC), + jsnums.toFixnum(angleA), + jsnums.toFixnum(sideB), + style.toString(), + color); + }); + + +EXPORTS['triangle/asa'] = + makePrimitiveProcedure( + 'triangle/asa', + 5, + function(MACHINE) { + var angleA = checkAngle(MACHINE, "triangle/asa", 0); + var sideB = checkNonNegativeReal(MACHINE, "triangle/asa", 1); + var angleC = checkAngle(MACHINE, "triangle/asa", 2); + var style = checkMode(MACHINE, "triangle/asa", 3); + var color = checkColor(MACHINE, "triangle/asa", 4); + // cast to fixnums + var angleA = jsnums.toFixnum(angleA); sideB = jsnums.toFixnum(sideB); angleC = jsnums.toFixnum(angleC); + var angleB = (180 - angleA - angleC); + if (less(angleB, 0)) { + raise( types.incompleteExn(types.exnFailContract, "The given angle, side and angle will not form a triangle: " + + angleA + ", " + sideB + ", " + angleC, []) ); + } + var base = (sideB * Math.sin(angleA*Math.PI/180)) / (Math.sin(angleB*Math.PI/180)); + var sideC = (sideB * Math.sin(angleC*Math.PI/180)) / (Math.sin(angleB*Math.PI/180)); + return makeTriangleImage(jsnums.toFixnum(sideC), + jsnums.toFixnum(angleA), + jsnums.toFixnum(sideB), + style.toString(), + color); + }); + +EXPORTS['triangle/saa'] = + makePrimitiveProcedure( + 'triangle/saa', + 5, + function(MACHINE) { + var sideA = checkNonNegativeReal(MACHINE, "triangle/saa", 0); + var angleB = checkAngle(MACHINE, "triangle/saa", 1); + var angleC = checkAngle(MACHINE, "triangle/saa", 2); + var style = checkMode(MACHINE, "triangle/saa", 3); + var color = checkColor(MACHINE, "triangle/saa", 4); + // cast to fixnums + var sideA = jsnums.toFixnum(sideA); angleB = jsnums.toFixnum(angleB); angleC = jsnums.toFixnum(angleC); + var angleA = (180 - angleC - angleB); + var hypotenuse = sideA / (Math.sin(angleA*Math.PI/180)); + var sideC = hypotenuse * Math.sin(angleC*Math.PI/180); + var sideB = hypotenuse * Math.sin(angleB*Math.PI/180); + return makeTriangleImage(jsnums.toFixnum(sideC), + jsnums.toFixnum(angleA), + jsnums.toFixnum(sideB), + style.toString(), + color); + }); + + + +EXPORTS['right-triangle'] = makePrimitiveProcedure( 'right-triangle', 4, @@ -896,10 +1206,11 @@ EXPORTS['right-triangle'] = var side2 = checkNonNegativeReal(MACHINE, "right-triangle", 1); var s = checkMode(MACHINE, "right-triangle", 2); var c = checkColor(MACHINE, "right-triangle", 3); - return makeRightTriangleImage(jsnums.toFixnum(side1), - jsnums.toFixnum(side2), - s.toString(), - c); + return makeTriangleImage(jsnums.toFixnum(side1), + jsnums.toFixnum(360-90), + jsnums.toFixnum(side2), + s.toString(), + c); }); @@ -909,13 +1220,18 @@ EXPORTS['isosceles-triangle'] = 4, function(MACHINE) { var side = checkNonNegativeReal(MACHINE, "isosceles-triangle", 0); - var angle = checkAngle(MACHINE, "isosceles-triangle", 1); + var angleC = checkAngle(MACHINE, "isosceles-triangle", 1); var s = checkMode(MACHINE, "isosceles-triangle", 2); var c = checkColor(MACHINE, "isosceles-triangle", 3); - return makeTriangleImage(jsnums.toFixnum(side), - jsnums.toFixnum(angle), - s.toString(), - c); + // cast to fixnums + side = jsnums.toFixnum(side); angleC = jsnums.toFixnum(angleC); + var angleAB = (180-angleC)/2; + var base = 2*side*Math.sin((angleC*Math.PI/180)/2); + return makeTriangleImage(jsnums.toFixnum(base), + jsnums.toFixnum(360-angleAB),// add 180 to make the triangle point up + jsnums.toFixnum(side), + s.toString(), + c); }); diff --git a/whalesong/image/private/kernel.js b/whalesong/image/private/kernel.js index 7e5c8e6..7796ebb 100644 --- a/whalesong/image/private/kernel.js +++ b/whalesong/image/private/kernel.js @@ -12,6 +12,7 @@ var colorRed = function(c) { return colorStruct.accessor(c, 0); }; var colorGreen = function(c) { return colorStruct.accessor(c, 1); }; var colorBlue = function(c) { return colorStruct.accessor(c, 2); }; var colorAlpha = function(c) { return colorStruct.accessor(c, 3); }; +var equals = plt.baselib.equality.equals; ////////////////////////////////////////////////////////////////////// var heir = plt.baselib.heir; @@ -27,32 +28,30 @@ var isAngle = function(x) { }; - - // Produces true if the value is a color or a color string. // On the Racket side of things, this is exposed as image-color?. var isColorOrColorString = function(thing) { return (isColor(thing) || - ((plt.baselib.strings.isString(thing) || + ((plt.baselib.strings.isString(thing) || plt.baselib.symbols.isSymbol(thing)) && - typeof(colorDb.get(thing)) != 'undefined')); -} - - - - - - -var colorString = function(aColor) { - return ("rgb(" + - colorRed(aColor) + "," + - colorGreen(aColor) + ", " + - colorBlue(aColor) + ")"); + typeof(colorDb.get(thing)) != 'undefined')); }; +////////////////////////////////////////////////////////////////////// +// colorString : hexColor Style -> rgba +// Style can be "solid" (1.0), "outline" (1.0), a number (0-1.0) or null (1.0) +var colorString = function(aColor, aStyle) { + var alpha = isNaN(aStyle)? 1.0 : aStyle/255; + return "rgba(" + colorRed(aColor) + "," + + colorGreen(aColor) + ", " + + colorBlue(aColor) + ", " + + alpha + ")"; +}; + + var isSideCount = function(x) { @@ -87,13 +86,27 @@ var isImage = function(thing) { return true; }; +// given two arrays of {x,y} structs, determine their equivalence +var verticesEqual = function(v1, v2){ + if(v1.length !== v2.length){ return false; } + for(var i=0; i< v1.length; i++){ + if(v1[i].x !== v2[i].x || v1[i].y !== v2[i].y){ return false; } + } + return true; +}; +// given two arrays of xs and ys, zip them into a vertex array +var zipVertices = function(xs, ys){ + if(xs.length !== ys.length){throw new Error('failure in zipVertices');} + var vertices = []; + for(var i=0; i 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. +// If the image isn't vertex-based, throw an error +// Otherwise, stroke and fill the vertices. BaseImage.prototype.render = function(ctx, x, y) { - throw new Error('BaseImage.render unimplemented!'); + if(!this.vertices){ + throw new Error('BaseImage.render is not implemented for this type!'); + } + ctx.save(); + ctx.beginPath(); + ctx.moveTo(x+this.vertices[0].x, y+this.vertices[0].y); + for(var i=1; i < this.vertices.length; i++){ + ctx.lineTo(x+this.vertices[i].x, y+this.vertices[i].y); + } + ctx.closePath(); + + if (this.style.toString().toLowerCase() === "outline") { + ctx.strokeStyle = colorString(this.color); + ctx.stroke(); + } else { + ctx.fillStyle = colorString(this.color, this.style); + ctx.fill(); + } + ctx.restore(); }; @@ -186,8 +226,8 @@ BaseImage.prototype.toDomNode = function(params) { // document. var onAfterAttach = function(event) { // jQuery(canvas).unbind('afterAttach', onAfterAttach); - var ctx = this.getContext("2d"); - that.render(ctx, 0, 0); + var ctx = this.getContext("2d"); + that.render(ctx, 0, 0); }; jQuery(canvas).bind('afterAttach', onAfterAttach); @@ -203,9 +243,38 @@ BaseImage.prototype.toDomNode = function(params) { BaseImage.prototype.toWrittenString = function(cache) { return ""; } BaseImage.prototype.toDisplayedString = function(cache) { return ""; } +// Best-Guess equivalence for images. If they're vertex-based we're in luck, +// otherwise we go pixel-by-pixel. It's up to exotic image types to provide +// more efficient ways of comparing one another BaseImage.prototype.equals = function(other, aUnionFind) { - return (this.pinholeX == other.pinholeX && - this.pinholeY == other.pinholeY); + if(this.width !== other.getWidth() || + this.height !== other.getHeight()){ return false; } + // if they're both vertex-based images, all we need to compare are + // their styles, vertices and color + if(this.vertices && other.vertices){ + return (this.style === other.style && + verticesEqual(this.vertices, other.vertices) && + equals(this.color, other.color, aUnionFind)); + } + // if it's something more sophisticated, render both images to canvases + // First check canvas dimensions, then go pixel-by-pixel + var c1 = this.toDomNode(), c2 = other.toDomNode(); + if(c1.width !== c2.width || c1.height !== c2.height){ return false;} + try{ + var ctx1 = c1.getContext('2d'), ctx2 = c2.getContext('2d'), + data1 = ctx1.getImageData(0, 0, c1.width, c1.height), + data2 = ctx2.getImageData(0, 0, c2.width, c2.height); + var pixels1 = data1.data, + pixels2 = data2.data; + for(var i = 0; i < pixels1.length; i++){ + if(pixels1[i] !== pixels2[i]){ return false; } + } + } catch(e){ + // if we violate CORS, just bail + return false; + } + // if, after all this, we're still good...then they're equal! + return true; }; @@ -221,183 +290,181 @@ var isScene = function(x) { ////////////////////////////////////////////////////////////////////// // SceneImage: primitive-number primitive-number (listof image) -> Scene var SceneImage = function(width, height, children, withBorder) { - BaseImage.call(this, Math.floor(width/2), Math.floor(height/2)); - this.width = width; - this.height = height; - this.children = children; // arrayof [image, number, number] - this.withBorder = withBorder; -} + BaseImage.call(this); + 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); + return new SceneImage(this.width, + this.height, + this.children.concat([[anImage, + x - anImage.getWidth()/2, + y - anImage.getHeight()/2]]), + 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); - } + var i; + var childImage, childX, childY; + // create a clipping region around the boundaries of the Scene + ctx.save(); + ctx.fillStyle = "rgba(0,0,0,0)"; + ctx.fillRect(x, y, this.width, this.height); + ctx.restore(); + // save the context, reset the path, and clip to the path around the scene edge + ctx.save(); + ctx.beginPath(); + ctx.rect(x, y, this.width, this.height); + ctx.clip(); + // Ask every object to render itself inside the region + for(i = 0; i < this.children.length; i++) { + // then, render the child images + childImage = this.children[i][0]; + childX = this.children[i][1]; + childY = this.children[i][2]; + childImage.render(ctx, childX + x, childY + y); + } + // unclip + ctx.restore(); + + if (this.withBorder) { + ctx.strokeStyle = 'black'; + ctx.strokeRect(x, y, this.width, this.height); + } }; SceneImage.prototype.equals = 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; + if (!(other instanceof SceneImage)) { + return BaseImage.prototype.equals.call(this, other, aUnionFind); + } + if (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] || + !equals(rec1[0], rec2[0], aUnionFind)) { + 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; + } + return true; }; - ////////////////////////////////////////////////////////////////////// // FileImage: string node -> Image var FileImage = function(src, rawImage) { - BaseImage.call(this, 0, 0); - var self = this; - this.src = src; - this.isLoaded = false; - - // animationHack: see installHackToSupportAnimatedGifs() for details. - this.animationHackImg = undefined; - - 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() { + BaseImage.call(this); + var self = this; + this.src = src; + this.isLoaded = false; + + // animationHack: see installHackToSupportAnimatedGifs() for details. + this.animationHackImg = undefined; + + if (rawImage && rawImage.complete) { + this.img = rawImage; + this.isLoaded = true; + self.width = self.img.width; + self.height = self.img.height; + } 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.width = self.img.width; + self.height = self.img.height; + }; + this.img.onerror = function(e) { self.img.onerror = ""; self.img.src = "http://www.wescheme.org/images/broken.png"; - } - this.img.src = src; } + this.img.src = src; + } } FileImage.prototype = heir(BaseImage.prototype); var imageCache = {}; FileImage.makeInstance = function(path, rawImage) { - if (! (path in imageCache)) { - imageCache[path] = new FileImage(path, rawImage); - } - return imageCache[path]; + if (! (path in imageCache)) { + imageCache[path] = new FileImage(path, rawImage); + } + return imageCache[path]; }; FileImage.installInstance = function(path, rawImage) { - imageCache[path] = new FileImage(path, rawImage); + imageCache[path] = new FileImage(path, rawImage); }; FileImage.installBrokenImage = function(path) { - imageCache[path] = new TextImage("Unable to load " + path, 10, colorDb.get("red"), - "normal", "Optimer","","",false); + imageCache[path] = new TextImage("Unable to load " + path, 10, colorDb.get("red"), + "normal", "Optimer","","",false); }; FileImage.prototype.render = function(ctx, x, y) { - this.installHackToSupportAnimatedGifs(); - ctx.drawImage(this.animationHackImg, x, y); + this.installHackToSupportAnimatedGifs(); + ctx.drawImage(this.animationHackImg, x, y); }; // The following is a hack that we use to allow animated gifs to show // as animating on the canvas. FileImage.prototype.installHackToSupportAnimatedGifs = function() { - if (this.animationHackImg) { return; } - this.animationHackImg = this.img.cloneNode(true); - document.body.appendChild(this.animationHackImg); - this.animationHackImg.width = 0; - this.animationHackImg.height = 0; + if (this.animationHackImg) { return; } + this.animationHackImg = this.img.cloneNode(true); + document.body.appendChild(this.animationHackImg); + this.animationHackImg.style.position = 'absolute'; + this.animationHackImg.style.top = '-2000px'; }; FileImage.prototype.getWidth = function() { - return this.img.width; + return this.img.width; }; FileImage.prototype.getHeight = function() { - return this.img.height; + return this.img.height; }; // Override toDomNode: we don't need a full-fledged canvas here. FileImage.prototype.toDomNode = function(params) { - return this.img.cloneNode(true); + return this.img.cloneNode(true); }; FileImage.prototype.equals = function(other, aUnionFind) { - return (other instanceof FileImage && - this.pinholeX == other.pinholeX && - this.pinholeY == other.pinholeY && - this.src == other.src); + if (!(other instanceof FileImage)) { + return BaseImage.prototype.equals.call(this, other, aUnionFind); + } + return (this.src === other.src); }; ////////////////////////////////////////////////////////////////////// -// VideoImage: String Node -> Video -var VideoImage = function(src, rawVideo) { - BaseImage.call(this, 0, 0); +// FileVideoe: String Node -> Video +var FileVideo = function(src, rawVideo) { + BaseImage.call(this); var self = this; this.src = src; - if (rawVideo) { + if (rawVideo) { this.video = rawVideo; this.width = self.video.videoWidth; this.height = self.video.videoHeight; - this.pinholeX = self.width / 2; - this.pinholeY = self.height / 2; this.video.volume = 1; this.video.poster = "http://www.wescheme.org/images/broken.png"; this.video.autoplay = true; @@ -414,8 +481,6 @@ var VideoImage = function(src, rawVideo) { this.video.addEventListener('canplay', function() { this.width = self.video.videoWidth; this.height = self.video.videoHeight; - this.pinholeX = self.width / 2; - this.pinholeY = self.height / 2; this.video.poster = "http://www.wescheme.org/images/broken.png"; this.video.autoplay = true; this.video.autobuffer=true; @@ -428,497 +493,481 @@ var VideoImage = function(src, rawVideo) { }); } } -VideoImage.prototype = heir(BaseImage.prototype); - +FileVideo.prototype = heir(BaseImage.prototype); var videos = {}; -VideoImage.makeInstance = function(path, rawVideo) { - if (! (path in VideoImage)) { - videos[path] = new VideoImage(path, rawVideo); +FileVideo.makeInstance = function(path, rawVideo) { + if (! (path in FileVideo)) { + videos[path] = new FileVideo(path, rawVideo); } return videos[path]; }; -VideoImage.prototype.render = function(ctx, x, y) { +FileVideo.prototype.render = function(ctx, x, y) { ctx.drawImage(this.video, x, y); }; +FileVideo.prototype.equals = function(other, aUnionFind) { + return (other instanceof FileVideo) && (this.src === other.src); +}; -VideoImage.prototype.equals = function(other, aUnionFind) { - return (other instanceof VideoImage && - this.pinholeX == other.pinholeX && - this.pinholeY == other.pinholeY && - this.src == other.src); +////////////////////////////////////////////////////////////////////// +// FileAudio: String Node -> Video +var FileAudio = function(src, loop, rawAudio) { + this.src = src; + var that = this; + if (rawAudio && (rawAudio.readyState===4)) { + that.audio = rawAudio; + that.audio.autoplay = true; + that.audio.autobuffer = true; + that.audio.currentTime = 0; + that.audio.loop = loop; + that.audio.play(); + } 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. + that.audio = document.createElement('audio'); + that.audio.src = src; + that.audio.addEventListener('canplay', function() { + that.audio.autoplay = true; + that.audio.autobuffer = true; + that.audio.currentTime = 0; + that.audio.loop = loop; + that.audio.play(); + }); + } + return true; +}; +var audioCache = {}; +FileAudio.makeInstance = function(path, loop, rawAudio) { + /* if (! (path in audioCache)) { + audioCache[path] = new FileAudio(path, loop, rawAudio, afterInit); + return audioCache[path]; + } else { + audioCache[path].audio.play(); + afterInit(audioCache[path]); + return audioCache[path]; + } + */ + return new FileAudio(path, loop, rawAudio); +}; + +////////////////////////////////////////////////////////////////////// +// ImageDataImage: imageData -> image +// Given an array of pixel data, create an image +var ImageDataImage = function(imageData) { + BaseImage.call(this); + this.imageData= imageData; + this.width = imageData.width; + this.height = imageData.height; +}; + +ImageDataImage.prototype = heir(BaseImage.prototype); + +ImageDataImage.prototype.render = function(ctx, x, y) { + ctx.putImageData(this.imageData, x, y); }; ////////////////////////////////////////////////////////////////////// // OverlayImage: image image placeX placeY -> image // Creates an image that overlays img1 on top of the -// other image. +// other image img2. var OverlayImage = function(img1, img2, placeX, placeY) { - // calculate centers using width/height, so we are scene/image agnostic - var c1x = img1.getWidth()/2; - var c1y = img1.getHeight()/2; - var c2x = img2.getWidth()/2; - var c2y = img2.getHeight()/2; + BaseImage.call(this); + + // An overlay image consists of width, height, x1, y1, x2, and + // y2. We need to compute these based on the inputs img1, + // img2, placex, and placey. + + // placeX and placeY may be non-numbers, in which case their values + // depend on the img1 and img2 geometry. - // calculate absolute offset of 2nd image's *CENTER* - // convert relative X,Y to center offsets, - // if placeX and placeY are UL corner offsets, convert to center offsets - if (placeX == "left" ) var xOffset = img2.getWidth()-(c1x+c2x); - else if (placeX == "right" ) var xOffset = img1.getWidth()-(c1x+c2x); - else if (placeX == "beside") var xOffset = c1x+c2x; - else if (placeX == "middle") var xOffset = 0; - else if (placeX == "center") var xOffset = 0; - else var xOffset = placeX - (c1x-c2x); + var x1, y1, x2, y2; + + if (placeX === "left") { + x1 = 0; + x2 = 0; + } else if (placeX === "right") { + x1 = Math.max(img1.getWidth(), img2.getWidth()) - img1.getWidth(); + x2 = Math.max(img1.getWidth(), img2.getWidth()) - img2.getWidth(); + } else if (placeX === "beside") { + x1 = 0; + x2 = img1.getWidth(); + } else if (placeX === "middle" || placeX === "center") { + x1 = Math.max(img1.getWidth(), img2.getWidth())/2 - img1.getWidth()/2; + x2 = Math.max(img1.getWidth(), img2.getWidth())/2 - img2.getWidth()/2; + } else { + x1 = Math.max(placeX, 0) - placeX; + x2 = Math.max(placeX, 0); + } + + if (placeY === "top") { + y1 = 0; + y2 = 0; + } else if (placeY === "bottom") { + y1 = Math.max(img1.getHeight(), img2.getHeight()) - img1.getHeight(); + y2 = Math.max(img1.getHeight(), img2.getHeight()) - img2.getHeight(); + } else if (placeY === "above") { + y1 = 0; + y2 = img1.getHeight(); + } else if (placeY === "baseline") { + y1 = Math.max(img1.getBaseline(), img2.getBaseline()) - img1.getBaseline(); + y2 = Math.max(img1.getBaseline(), img2.getBaseline()) - img2.getBaseline(); + } else if (placeY === "middle" || placeY === "center") { + y1 = Math.max(img1.getHeight(), img2.getHeight())/2 - img1.getHeight()/2; + y2 = Math.max(img1.getHeight(), img2.getHeight())/2 - img2.getHeight()/2; + } else { + y1 = Math.max(placeY, 0) - placeY; + y2 = Math.max(placeY, 0); + } + + // calculate the vertices of this image by translating the verticies of the sub-images + var i, v1 = img1.getVertices(), v2 = img2.getVertices(), xs = [], ys = []; - if (placeY == "bottom") var yOffset = img1.getHeight()-(c1y+c2y); - else if (placeY == "top") var yOffset = img2.getHeight()-(c1y+c2y); - else if (placeY == "above" ) var yOffset = c1y+c2y; - else if (placeY == "middle") var yOffset = 0; - else if (placeY == "center") var yOffset = 0; - else if (placeY == "baseline") var yOffset= img1.getBaseline()-img2.getBaseline(); - else var yOffset = placeY - (c1y-c2y); - - // Correct offsets when dealing with Scenes instead of images, - // by adding the center values - if(isScene(img1)){xOffset =+c1x; yOffset =+c1y;} - if(isScene(img2)){xOffset =+c2x; yOffset =+c2y;} - - // The *center* of the 2nd image, once overlaid, changes by the original difference in centers, - // plus the size of the offsets. Calculate this delta for X and Y. - var deltaX = c1x - c2x + xOffset; - var deltaY = c1y - c2y + yOffset; - - // Each edge of the new, combined image may be grown or shrunk, depending on deltaX or deltaY - 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()); - - // Calculate the new width, height and center based on edge lengths - this.width = right - left; - this.height = bottom - top; - BaseImage.call(this, - Math.floor((right-left) / 2), - Math.floor((bottom-top) / 2)); - - // store the overlaid images, and the offsets for each - this.img1 = img1; - this.img2 = img2; - this.img1Dx = -left; - this.img1Dy = -top; - this.img2Dx = deltaX - left; - this.img2Dy = deltaY - top; + for(i=0; i 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 +// TODO: special case for ellipse? var RotateImage = function(angle, img) { - var sin = Math.sin(angle * Math.PI / 180), - cos = Math.cos(angle * Math.PI / 180); - var width = img.getWidth(); - var height = img.getHeight(); - - // (w,0) rotation - var x1 = (cos * width), - y1 = (sin * width); - - // (0,h) rotation - var x2 = (-sin * height), - y2 = ( cos * height); - - // (w,h) rotation - var x3 = (cos * width - sin * height), - y3 = (sin * width + cos * height); - - 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 = Math.floor(rotatedWidth); - this.height = Math.floor(rotatedHeight); - this.angle = angle; - this.translateX = Math.floor(-minX); - this.translateY = Math.floor(-minY); + BaseImage.call(this); + var sin = Math.sin(angle * Math.PI / 180); + var cos = Math.cos(angle * Math.PI / 180); + var width = img.getWidth(); + var height= img.getHeight(); + + // rotate each point as if it were rotated about (0,0) + var vertices = img.getVertices(), xs = [], ys = []; + for(var i=0; i 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 = Math.floor(img.getWidth() * xFactor); - this.height = Math.floor(img.getHeight() * yFactor); - this.xFactor = xFactor; - this.yFactor = yFactor; + BaseImage.call(this); + var vertices = img.getVertices(); + var xs = [], ys = []; + for(var i=0; i image // Crop an image var CropImage = function(x, y, width, height, img) { - - BaseImage.call(this, - Math.floor(width / 2), - Math.floor(height / 2)); - - this.x = x; - this.y = y; - this.width = width; - this.height = height; - this.img = img; + BaseImage.call(this); + this.x = x; + this.y = y; + this.width = width; + this.height = height; + this.img = img; }; CropImage.prototype = heir(BaseImage.prototype); - CropImage.prototype.render = function(ctx, x, y) { - ctx.save(); - ctx.translate(-this.x, -this.y); - this.img.render(ctx, x, y); - ctx.restore(); + ctx.save(); + ctx.beginPath(); + ctx.rect(x, y, this.width, this.height); + ctx.clip(); + ctx.translate(-this.x, -this.y); + this.img.render(ctx, x, y); + ctx.restore(); }; CropImage.prototype.equals = function(other, aUnionFind) { - return ( other instanceof CropImage && - this.pinholeX == other.pinholeX && - this.pinholeY == other.pinholeY && - this.width == other.width && - this.height == other.height && - this.x == other.x && - this.y == other.y && - plt.baselib.equality.equals(this.img, other.img, aUnionFind) ); + if (!(other instanceof CropImage)) { + return BaseImage.prototype.equals.call(this, other, aUnionFind); + } + return (this.width === other.width && + this.height === other.height && + this.x === other.x && + this.y === other.y && + equals(this.img, other.img, aUnionFind) ); }; ////////////////////////////////////////////////////////////////////// // FrameImage: factor factor image -> image // Stick a frame around the image var FrameImage = function(img) { - - BaseImage.call(this, - Math.floor(img.getWidth()/ 2), - Math.floor(img.getHeight()/ 2)); - - this.img = img; - this.width = img.getWidth(); - this.height = img.getHeight(); + BaseImage.call(this); + this.img = img; + this.width = img.getWidth(); + this.height = img.getHeight(); }; FrameImage.prototype = heir(BaseImage.prototype); - // scale the context, and pass it to the image's render function FrameImage.prototype.render = function(ctx, x, y) { - ctx.save(); - this.img.render(ctx, x, y); - ctx.beginPath(); - ctx.strokeStyle = "black"; - ctx.strokeRect(x, y, this.width, this.height); - ctx.closePath(); - ctx.restore(); + ctx.save(); + this.img.render(ctx, x, y); + ctx.beginPath(); + ctx.strokeStyle = "black"; + ctx.strokeRect(x, y, this.width, this.height); + ctx.closePath(); + ctx.restore(); }; FrameImage.prototype.equals = function(other, aUnionFind) { - return ( other instanceof FrameImage && - this.pinholeX == other.pinholeX && - this.pinholeY == other.pinholeY && - plt.baselib.equality.equals(this.img, other.img, aUnionFind) ); + if (!(other instanceof FrameImage)) { + return BaseImage.prototype.equals.call(this, other, aUnionFind); + } + return equals(this.img, other.img, aUnionFind); }; ////////////////////////////////////////////////////////////////////// // FlipImage: image string -> image // Flip an image either horizontally or vertically var FlipImage = function(img, direction) { - this.img = img; - this.width = img.getWidth(); - this.height = img.getHeight(); - this.direction = direction; - BaseImage.call(this, - img.pinholeX, - img.pinholeY); + BaseImage.call(this); + this.img = img; + this.width = img.getWidth(); + this.height = img.getHeight(); + this.direction = direction; }; FlipImage.prototype = heir(BaseImage.prototype); - FlipImage.prototype.render = function(ctx, x, y) { - // when flipping an image of dimension M and offset by N across an axis, - // we need to translate the canvas by M+2N in the opposite direction - ctx.save(); - if(this.direction == "horizontal"){ - ctx.scale(-1, 1); - ctx.translate(-(this.width+2*x), 0); - this.img.render(ctx, x, y); - } - if (this.direction == "vertical"){ - ctx.scale(1, -1); - ctx.translate(0, -(this.height+2*y)); - this.img.render(ctx, x, y); - } - ctx.restore(); + // when flipping an image of dimension M and offset by N across an axis, + // we need to translate the canvas by M+2N in the opposite direction + ctx.save(); + if(this.direction === "horizontal"){ + ctx.scale(-1, 1); + ctx.translate(-(this.width+2*x), 0); + this.img.render(ctx, x, y); + } + if (this.direction === "vertical"){ + ctx.scale(1, -1); + ctx.translate(0, -(this.height+2*y)); + this.img.render(ctx, x, y); + } + ctx.restore(); }; - FlipImage.prototype.getWidth = function() { - return this.width; + return this.width; }; FlipImage.prototype.getHeight = function() { - return this.height; + return this.height; }; FlipImage.prototype.equals = function(other, aUnionFind) { - return ( other instanceof FlipImage && - this.pinholeX == other.pinholeX && - this.pinholeY == other.pinholeY && - this.width == other.width && - this.height == other.height && - this.direction == other.direction && - plt.baselib.equality.equals(this.img, other.img, aUnionFind) ); + if (!(other instanceof FlipImage)) { + return BaseImage.prototype.equals.call(this, other, aUnionFind); + } + return (this.width === other.width && + this.height === other.height && + this.direction === other.direction && + equals(this.img, other.img, aUnionFind) ); }; - ////////////////////////////////////////////////////////////////////// // RectangleImage: Number Number Mode Color -> Image 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; + BaseImage.call(this); + this.width = width; + this.height = height; + this.style = style; + this.color = color; + this.vertices = [{x:0,y:height},{x:0,y:0},{x:width,y:0},{x:width,y:height}]; }; 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; + return this.width; }; - RectangleImage.prototype.getHeight = function() { - return this.height; + return this.height; }; -RectangleImage.prototype.equals = 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)); -}; - - ////////////////////////////////////////////////////////////////////// // RhombusImage: Number Number Mode Color -> Image var RhombusImage = function(side, angle, style, color) { - // sin(angle/2-in-radians) * side = half of base - this.width = Math.sin(angle/2 * Math.PI / 180) * side * 2; - // cos(angle/2-in-radians) * side = half of height - this.height = Math.abs(Math.cos(angle/2 * Math.PI / 180)) * side * 2; - BaseImage.call(this, this.width/2, this.height/2); - this.side = side; - this.angle = angle; - this.style = style; - this.color = color; + BaseImage.call(this); + // sin(angle/2-in-radians) * side = half of base + // cos(angle/2-in-radians) * side = half of height + this.width = Math.sin(angle/2 * Math.PI / 180) * side * 2; + this.height = Math.abs(Math.cos(angle/2 * Math.PI / 180)) * side * 2; + this.side = side; + this.angle = angle; + this.style = style; + this.color = color; + this.vertices = [{x:this.width/2, y:0}, + {x:this.width, y:this.height/2}, + {x:this.width/2, y:this.height}, + {x:0, y:this.height/2}]; + }; RhombusImage.prototype = heir(BaseImage.prototype); - -RhombusImage.prototype.render = function(ctx, x, y) { - ctx.save(); - ctx.beginPath(); - // if angle < 180 start at the top of the canvas, otherwise start at the bottom - ctx.moveTo(x+this.getWidth()/2, y); - ctx.lineTo(x+this.getWidth(), y+this.getHeight()/2); - ctx.lineTo(x+this.getWidth()/2, y+this.getHeight()); - ctx.lineTo(x, y+this.getHeight()/2); - 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(); -}; - RhombusImage.prototype.getWidth = function() { - return this.width; + return this.width; }; - RhombusImage.prototype.getHeight = function() { - return this.height; + return this.height; }; -RhombusImage.prototype.equals = function(other, aUnionFind) { - return (other instanceof RhombusImage && - this.pinholeX == other.pinholeX && - this.pinholeY == other.pinholeY && - this.side == other.side && - this.angle == other.angle && - this.style == other.style && - plt.baselib.equality.equals(this.color, other.color, aUnionFind)); -}; - - ////////////////////////////////////////////////////////////////////// - - -var ImageDataImage = function(imageData) { - BaseImage.call(this, 0, 0); - this.imageData = imageData; - this.width = imageData.width; - this.height = imageData.height; +// PosnImage: Vertices Mode Color -> Image +// +var PosnImage = function(vertices, style, color) { + BaseImage.call(this); + var xs = vertices.map(function(v){return types.posnX(v);}), + ys = vertices.map(function(v){return types.posnY(v);}), + vertices = zipVertices(xs, ys); + + this.width = Math.max.apply(Math, xs) - Math.min.apply(Math, xs); + this.height = Math.max.apply(Math, ys) - Math.min.apply(Math, ys); + this.style = style; + this.color = color; + // shift the vertices by the calculated offsets, now that we know the width + var xOffset = Math.min.apply(Math, xs); + var yOffset = Math.min.apply(Math, ys); + for(var i=0; i Image @@ -928,494 +977,286 @@ ImageDataImage.prototype.equals = function(other, aUnionFind) { // another circle is inscribed in the polygon, whose radius is length/2tan(pi/count) // rotate a 3/4 quarter turn plus half the angle length to keep bottom base level var PolygonImage = function(length, count, step, style, color) { - this.aVertices = []; - var xMax = 0; - var yMax = 0; - var xMin = 0; - var yMin = 0; - - this.outerRadius = Math.floor(length/(2*Math.sin(Math.PI/count))); - this.innerRadius = Math.floor(length/(2*Math.tan(Math.PI/count))); - var adjust = (3*Math.PI/2)+Math.PI/count; - - // rotate around outer circle, storing x,y pairs as vertices - // keep track of mins and maxs - var radians = 0; - for(var i = 0; i < count; i++) { - // rotate to the next vertex (skipping by this.step) - radians = radians + (step*2*Math.PI/count); - - var v = { x: this.outerRadius*Math.cos(radians-adjust), - y: this.outerRadius*Math.sin(radians-adjust) }; - if(v.x < xMin) xMin = v.x; - if(v.x > xMax) xMax = v.y; - if(v.y < yMin) yMin = v.x; - if(v.y > yMax) yMax = v.y; - this.aVertices.push(v); - } - // HACK: try to work around handling of non-integer coordinates in CANVAS - // by ensuring that the boundaries of the canvas are outside of the vertices - for(var i=0; i xMax) xMax = this.aVertices[i].x+1; - if(this.aVertices[i].y < yMin) yMin = this.aVertices[i].y-1; - if(this.aVertices[i].y > yMax) yMax = this.aVertices[i].y+1; - } - - this.width = Math.floor(xMax-xMin); - this.height = Math.floor(yMax-yMin); - this.length = length; - this.count = count; - this.step = step; - this.style = style; - this.color = color; - BaseImage.call(this, Math.floor(this.width/2), Math.floor(this.height/2)); + BaseImage.call(this); + this.outerRadius = Math.floor(length/(2*Math.sin(Math.PI/count))); + this.innerRadius = Math.floor(length/(2*Math.tan(Math.PI/count))); + var adjust = (3*Math.PI/2)+Math.PI/count; + + // rotate around outer circle, storing x and y coordinates + var radians = 0, xs = [], ys = []; + for(var i = 0; i < count; i++) { + radians = radians + (step*2*Math.PI/count); + xs.push(Math.round(this.outerRadius*Math.cos(radians-adjust))); + ys.push(Math.round(this.outerRadius*Math.sin(radians-adjust))); + } + var vertices = zipVertices(xs, ys); + + this.width = Math.max.apply(Math, xs) - Math.min.apply(Math, xs); + this.height = Math.max.apply(Math, ys) - Math.min.apply(Math, ys); + this.length = length; + this.count = count; + this.step = step; + this.style = style; + this.color = color; + + // shift the vertices by the calculated offsets, now that we know the width + var xOffset = Math.round(this.width/2); + var yOffset = ((this.count % 2)? this.outerRadius : this.innerRadius); + for(i=0; i Image -////////////////////////////////////////////////////////////////////// -// TextImage: String Number Color String String String String any/c -> Image -var TextImage = function(msg, size, color, face, family, style, weight, underline) { - var metrics; - this.msg = msg; - this.size = size; - this.color = color; - this.face = face; - this.family = family; - this.style = (style == "slant")? "oblique" : style; // Racket's "slant" -> CSS's "oblique" - this.weight = (weight== "light")? "lighter" : weight; // Racket's "light" -> CSS's "lighter" - this.underline = underline; - // example: "bold italic 20px 'Times', sans-serif". - // Default weight is "normal", face is "Optimer" - var canvas = makeCanvas(0, 0); - var ctx = canvas.getContext("2d"); - - this.font = (this.weight + " " + - this.style + " " + - this.size + "px " + - maybeQuote(this.face) + " " + - maybeQuote(this.family)); - try { - ctx.font = this.font; - } catch (e) { - this.fallbackOnFont(); - ctx.font = this.font; - } - - // Defensive: on IE, this can break. - try { - metrics = ctx.measureText(msg); - this.width = metrics.width; - this.height = Number(this.size); - } catch(e) { - this.fallbackOnFont(); - } - BaseImage.call(this, Math.round(this.width/2), 0);// weird pinhole settings needed for "baseline" alignment -} +var TextImage = function(msg, size, color, face, family, style, weight, underline) { + BaseImage.call(this); + var metrics; + this.msg = msg; + this.size = size; // 18 + this.color = color; // red + this.face = face; // Gill Sans + this.family = family; // 'swiss + this.style = (style === "slant")? "oblique" : style; // Racket's "slant" -> CSS's "oblique" + this.weight = (weight=== "light")? "lighter" : weight; // Racket's "light" -> CSS's "lighter" + this.underline = underline; + // example: "bold italic 20px 'Times', sans-serif". + // Default weight is "normal", face is "Arial" + + // NOTE: we *ignore* font-family, as it causes a number of font bugs due the browser inconsistencies + var canvas = makeCanvas(0, 0), + ctx = canvas.getContext("2d"); + + this.font = (this.style + " " + + this.weight + " " + + this.size + "px " + + '"'+this.face+'", '+ + this.family); + + try { + ctx.font = this.font; + } catch (e) { + this.fallbackOnFont(); + ctx.font = this.font; + } + + // Defensive: on IE, this can break. + try { + metrics = ctx.measureText(msg); + this.width = metrics.width; + this.height = Number(this.size); + } catch(e) { + this.fallbackOnFont(); + } +}; TextImage.prototype = heir(BaseImage.prototype); TextImage.prototype.fallbackOnFont = function() { - // Defensive: if the browser doesn't support certain features, we - // reduce to a smaller feature set and try again. - this.font = this.size + "px " + maybeQuote(this.family); - var canvas = makeCanvas(0, 0); - var ctx = canvas.getContext("2d"); - ctx.font = this.font; - var metrics = ctx.measureText(this.msg); - this.width = metrics.width; - // KLUDGE: I don't know how to get at the height. - this.height = Number(this.size);//ctx.measureText("m").width + 20; + // Defensive: if the browser doesn't support certain features, we + // reduce to a smaller feature set and try again. + this.font = this.size + "px " + maybeQuote(this.family); + var canvas = makeCanvas(0, 0); + var ctx = canvas.getContext("2d"); + ctx.font = this.font; + var metrics = ctx.measureText(this.msg); + this.width = metrics.width; + // KLUDGE: I don't know how to get at the height. + this.height = Number(this.size);//ctx.measureText("m").width + 20; }; TextImage.prototype.render = function(ctx, x, y) { - ctx.save(); - - ctx.textAlign = 'left'; - ctx.textBaseline= 'top'; - ctx.fillStyle = colorString(this.color); - ctx.font = this.font; - try { - ctx.fillText(this.msg, x, y); - } catch (e) { - this.fallbackOnFont(); - ctx.font = this.font; - ctx.fillText(this.msg, x, y); - } - if(this.underline){ - ctx.beginPath(); - ctx.moveTo(x, y+this.size); - // we use this.size, as it is more accurate for underlining than this.height - ctx.lineTo(x+this.width, y+this.size); - ctx.closePath(); - ctx.strokeStyle = colorString(this.color); - ctx.stroke(); - } - ctx.restore(); + ctx.save(); + ctx.textAlign = 'left'; + ctx.textBaseline= 'top'; + ctx.fillStyle = colorString(this.color); + ctx.font = this.font; + try { + ctx.fillText(this.msg, x, y); + } catch (e) { + this.fallbackOnFont(); + ctx.font = this.font; + ctx.fillText(this.msg, x, y); + } + if(this.underline){ + ctx.beginPath(); + ctx.moveTo(x, y+this.size); + // we use this.size, as it is more accurate for underlining than this.height + ctx.lineTo(x+this.width, y+this.size); + ctx.closePath(); + ctx.strokeStyle = colorString(this.color); + ctx.stroke(); + } + ctx.restore(); }; - TextImage.prototype.getBaseline = function() { - return this.size; + return this.size; }; TextImage.prototype.equals = function(other, aUnionFind) { - return (other instanceof TextImage && - this.pinholeX == other.pinholeX && - this.pinholeY == other.pinholeY && - this.msg == other.msg && - this.size == other.size && - this.face == other.face && - this.family == other.family && - this.style == other.style && - this.weight == other.weight && - this.underline == other.underline && - plt.baselib.equality.equals(this.color, other.color, aUnionFind) && - this.font == other.font); + if (!(other instanceof TextImage)) { + return BaseImage.prototype.equals.call(this, other, aUnionFind); + } + return (this.msg === other.msg && + this.size === other.size && + this.face === other.face && + this.family === other.family && + this.style === other.style && + this.weight === other.weight && + this.underline === other.underline && + equals(this.color, other.color, aUnionFind) && + this.font === other.font); }; ////////////////////////////////////////////////////////////////////// // StarImage: fixnum fixnum fixnum color -> image +// Most of this code here adapted from the Canvas tutorial at: +// http://developer.apple.com/safari/articles/makinggraphicswithcanvas.html 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); - this.width = this.radius*2; - this.height = this.radius*2; + BaseImage.call(this); + this.points = points; + this.outer = outer; + this.inner = inner; + this.style = style; + this.color = color; + this.radius = Math.max(this.inner, this.outer); + this.width = this.radius*2; + this.height = this.radius*2; + var vertices = []; + + var oneDegreeAsRadian = Math.PI / 180; + 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; + vertices.push({x:this.radius + ( Math.sin( rads ) * radius ), + y:this.radius + ( Math.cos( rads ) * radius )} ); + } + this.vertices = vertices; }; 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(); -}; - -StarImage.prototype.equals = 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)); -}; - - - ///////////////////////////////////////////////////////////////////// -//TriangleImage: Number Number Mode Color -> Image -var TriangleImage = function(side, angle, style, color) { - // sin(angle/2-in-radians) * side = half of base - this.width = Math.sin(angle/2 * Math.PI / 180) * side * 2; - // cos(angle/2-in-radians) * side = height of altitude - this.height = Math.floor(Math.abs(Math.cos(angle/2 * Math.PI / 180)) * side); - - BaseImage.call(this, Math.floor(this.width/2), Math.floor(this.height/2)); - this.side = side; - this.angle = angle; - this.style = style; - this.color = color; -} +//TriangleImage: Number Number Number Mode Color -> Image +// Draws a triangle with the base = sideC, and the angle between sideC +// and sideB being angleA +// See http://docs.racket-lang.org/teachpack/2htdpimage.html#(def._((lib._2htdp/image..rkt)._triangle)) +var TriangleImage = function(sideC, angleA, sideB, style, color) { + BaseImage.call(this); + this.width = sideC; + this.height = Math.sqrt(Math.pow(sideB,2) - Math.pow(0.5*sideC,2)); + + var vertices = []; + // if angle < 180 start at the top of the canvas, otherwise start at the bottom + if(angleA < 180){ + vertices.push({x: 0, y: 0}); + vertices.push({x: sideC, y: 0}); + vertices.push({x: sideB*Math.cos(angleA*Math.PI/180), + y: this.height}); + } else { + vertices.push({x: 0, y: this.height - 0}); + vertices.push({x: sideC, y: this.height - 0}); + vertices.push({x: Math.abs(sideB*Math.cos(angleA*Math.PI/180)), + y: 0}); + } + this.vertices = vertices; + + 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(); - // if angle < 180 start at the top of the canvas, otherwise start at the bottom - if(this.angle < 180){ - ctx.moveTo(x+width/2, y); - ctx.lineTo(x, y+height); - ctx.lineTo(x+width, y+height); - } else { - ctx.moveTo(x+width/2, y+height); - ctx.lineTo(x, y); - ctx.lineTo(x+width, y); - } - 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.equals = function(other, aUnionFind) { - return (other instanceof TriangleImage && - this.pinholeX == other.pinholeX && - this.pinholeY == other.pinholeY && - this.side == other.side && - this.angle == other.angle && - this.style == other.style && - plt.baselib.equality.equals(this.color, other.color, aUnionFind)); -}; - -///////////////////////////////////////////////////////////////////// -//RightTriangleImage: Number Number Mode Color -> Image -var RightTriangleImage = function(side1, side2, style, color) { - this.width = side1; - this.height = side2; - - BaseImage.call(this, Math.floor(this.width/2), Math.floor(this.height/2)); - this.side1 = side1; - this.side2 = side2; - this.style = style; - this.color = color; -} -RightTriangleImage.prototype = heir(BaseImage.prototype); - - -RightTriangleImage.prototype.render = function(ctx, x, y) { - var width = this.getWidth(); - var height = this.getHeight(); - ctx.save(); - ctx.beginPath(); - ctx.moveTo(x, y+this.side2); - ctx.lineTo(x+this.side1, y+this.side2); - ctx.lineTo(x, y); - 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(); -}; - -RightTriangleImage.prototype.equals = function(other, aUnionFind) { - return (other instanceof RightTriangleImage && - this.pinholeX == other.pinholeX && - this.pinholeY == other.pinholeY && - this.side1 == other.side1 && - this.side2 == other.side2 && - this.style == other.style && - plt.baselib.equality.equals(this.color, other.color, aUnionFind)); -}; - ////////////////////////////////////////////////////////////////////// //Ellipse : Number Number Mode Color -> Image 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; + BaseImage.call(this); + 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(); + 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) * 0.5522848, + vB = (this.height / 2) * 0.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, this.style); + ctx.fill(); + } + + ctx.restore(); }; EllipseImage.prototype.equals = 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)); + if (!(other instanceof EllipseImage)) { + return BaseImage.prototype.equals.call(this, other, aUnionFind); + } + return (this.width === other.width && + this.height === other.height && + this.style === other.style && + equals(this.color, other.color, aUnionFind)); }; ////////////////////////////////////////////////////////////////////// -//Line: Number Number Color Boolean -> Image -var LineImage = function(x, y, color, normalPinhole) { - 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; - - // put the pinhle in the center of the image - if(normalPinhole){ - this.pinholeX = this.width/2; - this.pinholeY = this.height/2; - } -} +// Line: Number Number Color Boolean -> Image +var LineImage = function(x, y, color) { + BaseImage.call(this); + var vertices; + if (x >= 0) { + if (y >= 0) { vertices = [{x: 0, y: 0}, {x: x, y: y}]; } + else { vertices = [{x: 0, y: -y}, {x: x, y: 0}]; } + } else { + if (y >= 0) { vertices = [{x: -x, y: 0}, {x: 0, y: y}]; } + else { vertices = [{x: -x, y: -y}, {x: 0, y: 0}]; } + } + // preserve the invariant that all vertex-based images have a style + this.style = "outline"; + this.color = color; + this.width = Math.abs(x); + this.height = Math.abs(y); + this.vertices = vertices; +}; LineImage.prototype = heir(BaseImage.prototype); -LineImage.prototype.render = function(ctx, xstart, ystart) { - ctx.save(); - ctx.beginPath(); - ctx.strokeStyle = colorString(this.color); - 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.closePath(); - ctx.stroke(); - ctx.restore(); -}; - - -LineImage.prototype.equals = 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)); -}; - - - - - - var imageToColorList = function(img) { @@ -1495,17 +1336,17 @@ var makeRectangleImage = function(width, height, style, color) { var makeRhombusImage = function(side, angle, style, color) { return new RhombusImage(side, angle, style, color); }; +var makePosnImage = function(posns, style, color) { + return new PosnImage(posns, style, color); +}; var makePolygonImage = function(length, count, step, style, color) { return new PolygonImage(length, count, step, style, color); }; var makeSquareImage = function(length, style, color) { return new RectangleImage(length, length, style, color); }; -var makeRightTriangleImage = function(side1, side2, style, color) { - return new RightTriangleImage(side1, side2, style, color); -}; -var makeTriangleImage = function(side, angle, style, color) { - return new TriangleImage(side, angle, style, color); +var makeTriangleImage = function(side, angle, side2, style, color) { + return new TriangleImage(side, angle, side2, style, color); }; var makeEllipseImage = function(width, height, style, color) { return new EllipseImage(width, height, style, color); @@ -1540,8 +1381,11 @@ var makeImageDataImage = function(imageData) { var makeFileImage = function(path, rawImage) { return FileImage.makeInstance(path, rawImage); }; -var makeVideoImage = function(path, rawVideo) { - return VideoImage.makeInstance(path, rawVideo); +var makeFileVideo = function(path, rawVideo) { + return FileVideo.makeInstance(path, rawVideo); +}; +var makeFileAudio = function(path, rawAudio){ + return FileAudio.makeInstance(path, rawAudio) }; @@ -1554,7 +1398,6 @@ var isPolygonImage = function(x) { return x instanceof PolygonImage; }; var isRhombusImage = function(x) { return x instanceof RhombusImage; }; var isSquareImage = function(x) { return x instanceof SquareImage; }; var isTriangleImage= function(x) { return x instanceof TriangleImage; }; -var isRightTriangleImage = function(x) { return x instanceof RightTriangleImage; }; var isEllipseImage = function(x) { return x instanceof EllipseImage; }; var isLineImage = function(x) { return x instanceof LineImage; }; var isOverlayImage = function(x) { return x instanceof OverlayImage; }; @@ -1584,7 +1427,8 @@ EXPORTS.makeCanvas = makeCanvas; EXPORTS.BaseImage = BaseImage; EXPORTS.SceneImage = SceneImage; EXPORTS.FileImage = FileImage; -EXPORTS.VideoImage = VideoImage; +EXPORTS.FileVideo = FileVideo; +EXPORTS.FileAudio = FileAudio EXPORTS.OverlayImage = OverlayImage; EXPORTS.RotateImage = RotateImage; EXPORTS.ScaleImage = ScaleImage; @@ -1598,7 +1442,6 @@ EXPORTS.PolygonImage = PolygonImage; EXPORTS.TextImage = TextImage; EXPORTS.StarImage = StarImage; EXPORTS.TriangleImage = TriangleImage; -EXPORTS.RightTriangleImage = RightTriangleImage; EXPORTS.EllipseImage = EllipseImage; EXPORTS.LineImage = LineImage; EXPORTS.StarImage = StarImage; @@ -1614,7 +1457,6 @@ EXPORTS.makeRectangleImage = makeRectangleImage; EXPORTS.makeRhombusImage = makeRhombusImage; EXPORTS.makePolygonImage = makePolygonImage; EXPORTS.makeSquareImage = makeSquareImage; -EXPORTS.makeRightTriangleImage = makeRightTriangleImage; EXPORTS.makeTriangleImage = makeTriangleImage; EXPORTS.makeEllipseImage = makeEllipseImage; EXPORTS.makeLineImage = makeLineImage; @@ -1627,8 +1469,8 @@ EXPORTS.makeFlipImage = makeFlipImage; EXPORTS.makeTextImage = makeTextImage; EXPORTS.makeImageDataImage = makeImageDataImage; EXPORTS.makeFileImage = makeFileImage; -EXPORTS.makeVideoImage = makeVideoImage; - +EXPORTS.makeFileVideo = makeFileVideo; +EXPORTS.makeFileAudio = makeFileAudio; EXPORTS.imageToColorList = imageToColorList; EXPORTS.colorListToImage = colorListToImage; @@ -1650,7 +1492,6 @@ EXPORTS.isPolygonImage = isPolygonImage; EXPORTS.isRhombusImage = isRhombusImage; EXPORTS.isSquareImage = isSquareImage; EXPORTS.isTriangleImage = isTriangleImage; -EXPORTS.isRightTriangleImage = isRightTriangleImage; EXPORTS.isEllipseImage = isEllipseImage; EXPORTS.isLineImage = isLineImage; EXPORTS.isOverlayImage = isOverlayImage; @@ -1662,7 +1503,7 @@ EXPORTS.isFlipImage = isFlipImage; EXPORTS.isTextImage = isTextImage; EXPORTS.isFileImage = isFileImage; EXPORTS.isFileVideo = isFileVideo; - +//EXPORTS.isFileAudio = isFileAudio; EXPORTS.makeColor = makeColor; diff --git a/whalesong/image/private/main.rkt b/whalesong/image/private/main.rkt index 6ccc5f8..0e71a16 100644 --- a/whalesong/image/private/main.rkt +++ b/whalesong/image/private/main.rkt @@ -12,11 +12,12 @@ "js-impl.js") #:provided-values (text text/font - + bitmap/url image-url ;; older name for bitmap/url open-image-url ;; older name for bitmap/url - + video/url + play-sound overlay overlay/xy overlay/align @@ -28,6 +29,7 @@ above above/align empty-scene + put-image place-image place-image/align rotate @@ -43,9 +45,17 @@ circle square rectangle + polygon regular-polygon ellipse triangle + triangle/sas + triangle/sss + triangle/ass + triangle/ssa + triangle/aas + triangle/asa + triangle/saa right-triangle isosceles-triangle star @@ -66,6 +76,6 @@ side-count? step-count? image? - + image=? name->color )) diff --git a/whalesong/image/private/racket-impl.rkt b/whalesong/image/private/racket-impl.rkt index 65fa2cd..e3e57b8 100644 --- a/whalesong/image/private/racket-impl.rkt +++ b/whalesong/image/private/racket-impl.rkt @@ -15,6 +15,7 @@ above above/align empty-scene + put-image place-image place-image/align rotate @@ -30,9 +31,17 @@ circle square rectangle + polygon regular-polygon ellipse triangle + triangle/sas + triangle/sss + triangle/ass + triangle/ssa + triangle/aas + triangle/asa + triangle/saa right-triangle isosceles-triangle star @@ -51,19 +60,22 @@ angle? side-count? image? + image=? ;; Something funky is happening on the Racket side of things with regards ;; to step-count? See: http://bugs.racket-lang.org/query/?cmd=view&pr=12031 ;; step-count? - + bitmap/url - + video/url + play-sound + name->color - + step-count? image-url open-image-url color-list->bitmap - + ) @@ -99,6 +111,7 @@ above above/align empty-scene + put-image place-image place-image/align rotate @@ -114,9 +127,17 @@ circle square rectangle + polygon regular-polygon ellipse triangle + triangle/sas + triangle/sss + triangle/ass + triangle/ssa + triangle/aas + triangle/asa + triangle/saa right-triangle isosceles-triangle star @@ -134,12 +155,15 @@ y-place? angle? side-count? - + image? + image=? ;; Something funky is happening on the Racket side of things with regards ;; to step-count? See: http://bugs.racket-lang.org/query/?cmd=view&pr=12031 ;; step-count? bitmap/url + video/url + play-sound name->color step-count? image-url @@ -149,7 +173,7 @@ - + ;(define (my-step-count? x) ; (and (integer? x) @@ -161,9 +185,9 @@ #;(define (name->color n) - (error 'name->color "not implemented yet")) + (error 'name->color "not implemented yet")) #;(provide (rename-out [my-step-count? step-count?] - [bitmap/url image-url] - [bitmap/url open-image-url])) + [bitmap/url image-url] + [bitmap/url open-image-url])) diff --git a/whalesong/info.rkt b/whalesong/info.rkt index 107d657..1106f21 100644 --- a/whalesong/info.rkt +++ b/whalesong/info.rkt @@ -24,6 +24,7 @@ "sandbox" "examples" "experiments" + "selfhost" "simulator" "tmp")) (define can-be-loaded-with 'all) diff --git a/whalesong/tests/test-images.rkt b/whalesong/tests/test-images.rkt new file mode 100644 index 0000000..ef3a8e8 --- /dev/null +++ b/whalesong/tests/test-images.rkt @@ -0,0 +1,897 @@ +#lang whalesong + +(require whalesong/world) +(require whalesong/image) + +;(play-sound "http://www.html5tutorial.info/media/vincent.mp3" true) + +;; "Checking Empty scene" +;; (empty-scene 40 50 "red") +"These three circles (red, green, blue) should be left aligned" +(above/align "left" + (circle 30 "solid" "red") + (above/align "left" (circle 50 'solid 'green) (circle 20 'solid 'blue))) + + +"These three circles (red, green, blue) should be right aligned" +(above/align "right" + (circle 30 "solid" "red") + (above/align "right" (circle 50 'solid 'green) (circle 20 'solid 'blue))) + + +"These three circles (red, green, blue) should be middle aligned, vertically" +(above/align "middle" + (circle 30 "solid" "red") + (above/align "middle" (circle 50 'solid 'green) (circle 20 'solid 'blue))) + + + +"These three circles (red, green, blue) should be top-aligned" +(beside/align "top" + (circle 30 "solid" "red") + (beside/align "top" (circle 50 'solid 'green) (circle 20 'solid 'blue))) + +"These three circles (red, green, blue) should be bottom-aligned" +(beside/align "bottom" + (circle 30 "solid" "red") + (beside/align "bottom" (circle 50 'solid 'green) (circle 20 'solid 'blue))) + +"These three circles (red, green, blue) should be middle-aligned, horizontally" +(beside/align "middle" + (circle 30 "solid" "red") + (beside/align "middle" (circle 50 'solid 'green) (circle 20 'solid 'blue))) + + + + + +"should be a bar graph" +(define (make-stars number) + (cond [(eq? number 1) (star 12 "solid" "purple")] + [true (beside (star 12 "solid" "purple") (make-stars (- number 1)))] )) +(define (bar-graph l1) + (cond [(empty? l1) (circle 0 "outline" "blue")] + [true (above/align "left" (make-stars (car l1)) (bar-graph (cdr l1)))])) +(bar-graph '(1 3 5 3 9 5 3 4 4 3 5 2)) + + + +(check-expect (image? 'blue) #f) +(check-expect (image? (circle 20 "solid" "green")) #t) + +"should be a solid green circle: " (circle 20 "solid" "green") + +"should be an outline turquoise rectangle: " (rectangle 20 30 "outline" "turquoise") + +"should be a solid, mostly-translucent red rectangle: " (rectangle 200 300 10 "red") +"should be an outline red rectangle: " (rectangle 200 300 "outline" "red") +"should be an *invisible* red rectangle: " (rectangle 200 300 0 "red") + +(define halfred (make-color 255 0 0 128)) +(define quarterred (make-color 255 0 0 64)) +"should be a solid red triangle" (triangle 50 "solid" "red") +"should be a solid triangle made from a half-transparent red" (triangle 50 "solid" halfred) +"should be a solid, half-alpha triangle made from a half-transparent red" (triangle 50 128 halfred) +"should be a solid triangle made from a quater-trasparent red" (triangle 50 "solid" quarterred) + +;(check-expect (color? (make-color 3 4 5))) + +(check-expect (color-red (make-color 3 4 5)) 3) +(check-expect (color-green (make-color 3 4 5)) 4) +(check-expect (color-blue (make-color 3 4 5)) 5) + +(check-expect (image? (empty-scene 20 50)) true) + +(check-expect (image? (place-image (circle 50 'solid 'blue) + 50 + 50 + (empty-scene 100 100))) + true) + +"should be a blue circle in a scene with a border: " (place-image (circle 50 'solid 'blue) + 50 + 50 + (empty-scene 100 100)) + +"should be a blue circle in the UR of a scene with a border: " (put-image (circle 50 'solid 'blue) + 100 + 100 + (empty-scene 100 100)) + +(check-expect (image? + (rectangle 20 20 'solid 'green)) + true) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; TEXT & TEXT/FONT +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"simple text functionality" +(text "hello world" 20 'black) +(text (string-copy "hello world") 30 'purple) +(text "hello world" 40 'red) + + +"test font-weight" +(text/font "Hello" 24 "purple" + "Gill Sans" 'swiss 'normal 'bold #f) +(text/font "Hello" 24 "green" + "Gill Sans" 'swiss 'normal 'light #f) + +"test font-style" +(text/font "Goodbye" 48 "indigo" + "Helvetica" 'modern 'italic 'normal #f) +(text/font "Goodbye" 48 "indigo" + "Helvetica" 'modern 'normal 'normal #f) + +"test underline-height calculation" +(text/font "test this!" 80 "purple" + "Helvetica" 'roman 'normal 'normal #t) + +(text/font "low-hanging glyphs" 36 "blue" + "Times" 'roman 'normal 'bold #t) + +(text/font "teeny-tiny text" 8 "black" + "Times" 'roman 'normal 'bold #t) + +(text/font "not really a link" 36 "blue" + "Courier" 'roman 'italic 'normal #t) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; IMAGE-URL & VIDEO-URL +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"importing images and video" +(image-url "http://www.bootstrapworld.org/images/icon.png") +(open-image-url "http://www.bootstrapworld.org/images/icon.png") + +;(video/url "http://www.quirksmode.org/html5/videos/big_buck_bunny.mp4") +#;(overlay (circle 20 "solid" "red") + (video/url "http://www.quirksmode.org/html5/videos/big_buck_bunny.mp4")) +#;(rotate 45 + (video/url "http://www.quirksmode.org/html5/videos/big_buck_bunny.mp4")) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; OVERLAY +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"the next two images should be identical" +(overlay (circle 20 "solid" (make-color 50 50 255)) + (square 40 "solid" (make-color 100 100 255))) + + +(overlay (circle 20 "solid" (make-color 50 50 255)) + (regular-polygon 40 4 "solid" (make-color 100 100 255))) + +(overlay (ellipse 10 10 "solid" "red") + (ellipse 20 20 "solid" "black") + (ellipse 30 30 "solid" "red") + (ellipse 40 40 "solid" "black") + (ellipse 50 50 "solid" "red") + (ellipse 60 60 "solid" "black")) + +"the next two images should be identical" +(overlay (square 20 "solid" (make-color 50 50 255)) + (square 26 "solid" (make-color 100 100 255)) + (square 32 "solid" (make-color 150 150 255)) + (square 38 "solid" (make-color 200 200 255)) + (square 44 "solid" (make-color 250 250 255))) +(overlay (regular-polygon 20 4 "solid" (make-color 50 50 255)) + (regular-polygon 26 4 "solid" (make-color 100 100 255)) + (regular-polygon 32 4 "solid" (make-color 150 150 255)) + (regular-polygon 38 4 "solid" (make-color 200 200 255)) + (regular-polygon 44 4 "solid" (make-color 250 250 255))) + +"overlay with place-image - target should be centered" + (place-image (overlay (ellipse 10 10 "solid" "white") + (ellipse 20 20 "solid" "black") + (ellipse 30 30 "solid" "white") + (ellipse 40 40 "solid" "black") + (ellipse 50 50 "solid" "white") + (ellipse 60 60 "solid" "black")) + 150 100 + (rectangle 300 200 "solid" "black")) + + + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; OVERLAY/XY +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"should be some overlay/xys" +(overlay/xy (rectangle 20 20 "outline" "black") + 20 0 + (rectangle 20 20 "outline" "black")) +(overlay/xy (rectangle 20 20 "solid" "red") + 20 20 + (rectangle 20 20 "solid" "black")) +(overlay/xy (rectangle 20 20 "solid" "red") + -20 -20 + (rectangle 20 20 "solid" "black")) +(overlay/xy + (overlay/xy (ellipse 40 40 "outline" "black") + 10 + 15 + (ellipse 10 10 "solid" "forestgreen")) + 20 + 15 + (ellipse 10 10 "solid" "forestgreen")) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; OVERLAY/ALIGN +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"some examples of overlay/align" +(overlay/align "middle" "middle" + (ellipse 60 30 "solid" "purple") + (rectangle 30 60 "solid" "orange")) +(overlay/align "right" "top" + (ellipse 60 30 "solid" "purple") + (rectangle 30 60 "solid" "orange")) +(overlay/align "left" "bottom" + (ellipse 60 30 "solid" "purple") + (rectangle 30 60 "solid" "orange")) + +(overlay/align "right" "bottom" + (rectangle 20 20 "solid" "silver") + (rectangle 30 30 "solid" "seagreen") + (rectangle 40 40 "solid" "silver") + (rectangle 50 50 "solid" "seagreen")) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; UNDERLAY +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"some underlays" +(underlay (circle 20 'solid 'green) + (rectangle 10 20 'solid 'blue)) + +(underlay (ellipse 10 60 "solid" "red") + (ellipse 20 50 "solid" "black") + (ellipse 30 40 "solid" "red") + (ellipse 40 30 "solid" "black") + (ellipse 50 20 "solid" "red") + (ellipse 60 10 "solid" "black")) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; UNDERLAY/XY & UNDERLAY/ALIGN +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"examples of underlay and underlay/align" +(underlay/xy (circle 20 'solid 'green) + 30 10 + (rectangle 10 20 'solid 'blue)) + + + +;; color list +"the following should be a blue circle, but by using color-list->image" +(let ([circle-color-list (image->color-list (circle 20 'solid 'blue))]) + ;; fixme: add tests for number of colors + (color-list->image circle-color-list 40 40 0 0)) + + + + +(underlay/align "middle" "middle" + (ellipse 60 30 "solid" "purple") + (rectangle 30 60 "solid" "orange")) +(underlay/align "right" "top" + (ellipse 60 30 "solid" "purple") + (rectangle 30 60 "solid" "orange")) +(underlay/align "left" "bottom" + (ellipse 60 30 "solid" "purple") + (rectangle 30 60 "solid" "orange")) + +(underlay/align "right" "bottom" + (rectangle 50 50 "solid" "silver") + (rectangle 40 40 "solid" "seagreen") + (rectangle 30 30 "solid" "silver") + (rectangle 20 20 "solid" "seagreen")) + +"This is issue 40 https://github.com/dyoo/WeScheme/issues/40" +(underlay/align "left" "middle" + (rectangle 30 60 "solid" "orange") + (ellipse 60 30 "solid" "purple")) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; BESIDE & BESIDE/ALIGN +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"examples of beside and beside/align" +(beside (ellipse 20 70 "solid" "gray") + (ellipse 20 50 "solid" "darkgray") + (ellipse 20 30 "solid" "dimgray") + (ellipse 20 10 "solid" "black")) + +(beside/align "bottom" + (ellipse 20 70 "solid" "lightsteelblue") + (ellipse 20 50 "solid" "mediumslateblue") + (ellipse 20 30 "solid" "slateblue") + (ellipse 20 10 "solid" "navy")) + +(beside/align "top" + (ellipse 20 70 "solid" "mediumorchid") + (ellipse 20 50 "solid" "darkorchid") + (ellipse 20 30 "solid" "purple") + (ellipse 20 10 "solid" "indigo")) + +"align these text images on their baselines" +(beside/align "baseline" + (text "ijy" 18 "black") + (text "ijy" 24 "black")) + + +"issue 25 https://github.com/dyoo/WeScheme/issues/25" +(beside/align "top" + (rectangle 20 100 "solid" "black") + (rectangle 20 120 "solid" "black") + (rectangle 20 80 "solid" "black")) + + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; ABOVE & ABOVE/ALIGN +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"should be some examples of above and above/align" +(above (ellipse 70 20 "solid" "gray") + (ellipse 50 20 "solid" "darkgray") + (ellipse 30 20 "solid" "dimgray") + (ellipse 10 20 "solid" "black")) + +(above/align "right" + (ellipse 70 20 "solid" "gold") + (ellipse 50 20 "solid" "goldenrod") + (ellipse 30 20 "solid" "darkgoldenrod") + (ellipse 10 20 "solid" "sienna")) +(above/align "left" + (ellipse 70 20 "solid" "yellowgreen") + (ellipse 50 20 "solid" "olivedrab") + (ellipse 30 20 "solid" "darkolivegreen") + (ellipse 10 20 "solid" "darkgreen")) + + + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; PLACE-IMAGE/ALIGN +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"should be right in the center" +(place-image/align (circle 16 "solid" "yellow") + 32 32 "center" "center" + (rectangle 64 64 "solid" "goldenrod")) + +"should be at the bottom-right corner" +(place-image/align (circle 16 "solid" "yellow") + 32 32 "left" "top" + (rectangle 64 64 "solid" "goldenrod")) + +"should be at the upper-left corner" +(place-image/align (circle 16 "solid" "yellow") + 32 32 "right" "bottom" + (rectangle 64 64 "solid" "goldenrod")) + +"test 'beside' with scenes -- from the DrRacket documentation" +(beside (place-image/align (circle 8 "solid" "tomato") + 0 0 "center" "center" + (rectangle 32 32 "outline" "black")) + (place-image/align (circle 8 "solid" "tomato") + 8 8 "center" "center" + (rectangle 32 32 "outline" "black")) + (place-image/align (circle 8 "solid" "tomato") + 16 16 "center" "center" + (rectangle 32 32 "outline" "black")) + (place-image/align (circle 8 "solid" "tomato") + 24 24 "center" "center" + (rectangle 32 32 "outline" "black")) + (place-image/align (circle 8 "solid" "tomato") + 32 32 "center" "center" + (rectangle 32 32 "outline" "black"))) + +"some overlay and place-image stress tests" +(define flag2 + (place-image + (rotate 90 + (underlay/align + "center" "center" + (rectangle 50 450 "solid" "white") + (rotate 90 + (rectangle 50 450 "solid" "white")) + (rotate 90 + (rectangle 30 450 "solid" "red")) + (rotate 180 + (rectangle 30 450 "solid" "red")))) + + 200 100 + (place-image + (rotate 65 + (underlay/align + "center" "center" + (rectangle 15 450 "solid" "red") + (rotate 50 + (rectangle 15 450 "solid" "red")))) + 200 100 + (place-image + (rotate 65 + (underlay/align + "center" "center" + (rectangle 40 450 "solid" "white") + (rotate 50 + (rectangle 40 450 "solid" "white")))) + 200 100 + (rectangle 400 200 "solid" "navy"))))) + + +(define Australia2 + (place-image + flag2 + 200 100 + (place-image + (star-polygon 30 7 3 "solid" "white") + 650 60 + (place-image + (star-polygon 50 7 3 "solid" "white") + 200 300 + (place-image + (star-polygon 40 7 3 "solid" "white") + 60 20 + (place-image + (star-polygon 40 7 3 "solid" "white") + 68 124 + (rectangle 900 400 "solid" "navy"))))))) + flag2 +Australia2 + + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; TRIANGLE, RIGHT TRIANGLE & ISOSCELES-TRIANGLE +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"Three triangles of various sizes and fills" +(triangle 36 "solid" "darkslategray") +(triangle 4 "solid" "purple") +(triangle 30 "outline" "cornflowerblue") + +"Triangles side by side" +(beside (triangle 36 "solid" "darkslategray") + (triangle 30 "solid" "cornflowerblue")) + +"Triangles above." +(above (triangle 36 "solid" "darkslategray") + (triangle 30 "solid" "cornflowerblue")) + +"Three right triangles of various sizes and fills" +(right-triangle 36 48 "solid" "darkslategray") +(right-triangle 4 60 "solid" "purple") +(right-triangle 30 40 "solid" "cornflowerblue") + +"Three isosceles triangles of various sizes and fills" + +(isosceles-triangle 60 30 "solid" "aquamarine") +(isosceles-triangle 200 170 "outline" "seagreen") +(isosceles-triangle 60 330 "solid" "lightseagreen") + +"Trying ASA triangle (30 40 60)" +(triangle/asa 30 40 60 "solid" "blue") + +"Trying AAS triangle (30 60 40)" +(triangle/aas 30 60 40 "outline" "green") + +"Trying SAA triangle (100 30 90)" +(triangle/saa 100 30 90 "solid" "red") + +"Trying SSA triangle (60 60 40)" +(triangle/ass 60 60 40 "outline" "turquoise") + +"Trying ASS triangle (60 80 90)" +(triangle/ass 60 80 90 "solid" "maroon") + +"Trying SSS triangle (60 60 60)" +(triangle/sss 60 60 60 "outline" "red") + +"Trying SAS triangle (60 30 60)" +(triangle/sas 60 30 60 "solid" "brown") + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; STAR, RADIAL-STAR & STAR-POLYGON +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"old star implementation" + +(star 5 8 4 "solid" "darkslategray") +(star 5 30 15 "outline" "black") +(star 5 20 10 "solid" "red") + +"new star implementation" +(star 8 "solid" "darkslategray") +(star 30 "outline" "black") +(star 20 "solid" "red") + +"radial star" +(radial-star 8 8 64 "solid" "darkslategray") +(radial-star 32 30 40 "outline" "black") +(radial-star 5 20 40 "solid" "red") + +"star-polygon" +(star-polygon 40 5 2 "solid" "seagreen") +(star-polygon 40 7 3 "outline" "darkred") +(star-polygon 20 10 3 "solid" "cornflowerblue") +"should look like a pentagon" +(star-polygon 20 5 1 "solid" "darkblue") + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; SQUARE +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"Three squares of various sizes and fills" +(square 60 "outline" "black") +(square 200 "solid" "seagreen") +(square 100 "outline" "blue") + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; RHOMBUS +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"Three rhombuses of various sizes and fills" +(rhombus 40 45 "solid" "magenta") +(rhombus 100 200 "solid" "orange") +(rhombus 80 330 "outline" "seagreen") + +"rhombuses beside each other" +(beside (rhombus 40 45 "solid" "magenta") + (rhombus 100 200 "solid" "orange") + (rhombus 80 330 "outline" "seagreen")) + + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; REGULAR-POLYGON +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"Some regular polygons..." +"A triangle of side-length 20: should be 20x17" +(regular-polygon 20 3 "solid" "purple") +"A square of side-length 40: should be 40x40" +(regular-polygon 40 4 "solid" "aquamarine") +"A pentagon of side-length 30: should be 49x46" +(regular-polygon 30 5 "solid" "pink") +"A hexagon of side-length 20: should be 40x35" +(regular-polygon 20 6 "solid" "gold") +"A septagon of side-length 40: should be 90x88" +(regular-polygon 40 7 "solid" "goldenrod") +"An octagon of side-length 30: should be 72x72" +(regular-polygon 30 8 "solid" "darkgoldenrod") +"A nonagon of side-length 20: should be 58x57" +(regular-polygon 20 9 "solid" "sienna") + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; POLYGON +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"Some polygons defined with posns..." +#;(polygon (list (make-posn 0 0) + (make-posn -10 20) + (make-posn 60 0) + (make-posn -10 -20)) + "solid" + "burlywood") + +#;(polygon (list (make-posn 0 0) + (make-posn 0 40) + (make-posn 20 40) + (make-posn 20 60) + (make-posn 40 60) + (make-posn 40 20) + (make-posn 20 20) + (make-posn 20 0)) + "solid" + "plum") + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; ROTATE +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"Three images at 30, 60, 90 degree rotation:" +(rotate 30 (image-url "http://www.bootstrapworld.org/images/icon.png")) +(rotate 60 (image-url "http://www.bootstrapworld.org/images/icon.png")) +(rotate 90 (image-url "http://www.bootstrapworld.org/images/icon.png")) + +"Rotated, huge image" +(rotate 30 (scale 3 (image-url "http://www.bootstrapworld.org/images/icon.png"))) + +"From the Racket documentation:" +(rotate 45 (ellipse 60 20 "solid" "olivedrab")) +(rotate 5 (rectangle 50 50 "outline" "black")) +"unrotated T" +(beside/align + "center" + (rectangle 40 20 "solid" "darkseagreen") + (rectangle 20 100 "solid" "darkseagreen")) +"rotate 45 degrees" +(rotate 45 + (beside/align + "center" + (rectangle 40 20 "solid" "darkseagreen") + (rectangle 20 100 "solid" "darkseagreen"))) + +(beside + (rotate 30 (square 50 "solid" "red")) + (flip-horizontal + (rotate 30 (square 50 "solid" "blue")))) + +"A solid blue triangle, rotated 30 degrees. Should be flush left" +(rotate 30 (triangle 100 "solid" "blue")) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; SCALE & SCALE/XY +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"scaling small and large" +(scale 1/2 (image-url "http://www.bootstrapworld.org/images/icon.png")) +(scale 2 (image-url "http://www.bootstrapworld.org/images/icon.png")) + +(scale/xy 1 2 (image-url "http://www.bootstrapworld.org/images/icon.png")) +(scale/xy 2 1 (image-url "http://www.bootstrapworld.org/images/icon.png")) + +"This should be the normal image" +(scale/xy 1 1 (image-url "http://www.bootstrapworld.org/images/icon.png")) + +"From the Racket documentation: two identical ellipses, and a circle" +(scale 2 (ellipse 20 30 "solid" "blue")) +(ellipse 40 60 "solid" "blue") +(scale/xy 3 + 2 + (ellipse 20 30 "solid" "blue")) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; FRAME AND CROP +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"frame and crop examples from DrRacket documentation" +(frame (ellipse 20 20 "outline" "black")) + +(beside + (ellipse 20 70 "solid" "lightsteelblue") + (frame (ellipse 20 50 "solid" "mediumslateblue")) + (ellipse 20 30 "solid" "slateblue") + (ellipse 20 10 "solid" "navy")) + +(crop 0 0 40 40 (circle 40 "solid" "chocolate")) +(crop 40 60 40 60 (ellipse 80 120 "solid" "dodgerblue")) +(above + (beside (crop 40 40 40 40 (circle 40 "solid" "palevioletred")) + (crop 0 40 40 40 (circle 40 "solid" "lightcoral"))) + (beside (crop 40 0 40 40 (circle 40 "solid" "lightcoral")) + (crop 0 0 40 40 (circle 40 "solid" "palevioletred")))) + +"should be a quarter of a circle, inscribed in a square" +(place-image + (crop 0 0 20 20 (circle 20 "solid" "Magenta")) + 10 10 + (rectangle 40 40 "solid" "blue")) +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; LINE, ADD-LINE & SCENE+LINE +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"Three tests for line" +(line 30 30 "black") + +(line -30 20 "red") + +(line 30 -20 "red") + +"Three tests for add-line" +(add-line (ellipse 40 40 "outline" "maroon") + 0 40 40 0 "maroon") + +(add-line (rectangle 40 40 "solid" "gray") + -10 50 50 -10 "maroon") + +(add-line + (rectangle 100 100 "solid" "darkolivegreen") + 25 25 100 100 + "goldenrod") + +"Three tests for scene+line: should be identical to above, but cropped around base image" +(scene+line (ellipse 40 40 "outline" "maroon") + 0 40 40 0 "maroon") + +(scene+line (rectangle 40 40 "solid" "gray") + -10 50 50 -10 "maroon") + +(scene+line + (rectangle 100 100 "solid" "darkolivegreen") + 25 25 100 100 + "goldenrod") + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; FLIP-VERTICAL & FLIP-HORIZONTAL +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"a red triangle, a blue one flippled horizontally and a green one flippled vertically" +(right-triangle 30 40 "solid" "red") +(flip-horizontal (right-triangle 30 40 "solid" "blue")) +(flip-vertical (right-triangle 30 40 "solid" "green")) + +"those three triangles beside each other" +(beside (right-triangle 30 40 "solid" "red") + (flip-horizontal (right-triangle 30 40 "solid" "blue")) + (flip-vertical (right-triangle 30 40 "solid" "green"))) + + + +"one image flipped vertically, and one flipped horizontally" +(flip-vertical (image-url "http://www.bootstrapworld.org/images/icon.png")) +(flip-horizontal (image-url "http://www.bootstrapworld.org/images/icon.png")) + +"BESIDE: reference image" +(beside (square 20 "solid" (make-color 50 50 255)) + (square 34 "solid" (make-color 150 150 255))) + +"flip the second one horizontally" +(beside (square 20 "solid" (make-color 50 50 255)) + (flip-horizontal (square 34 "solid" (make-color 150 150 255)))) + +"flip the second one vertically" +(beside (square 20 "solid" (make-color 50 50 255)) + (flip-vertical (square 34 "solid" (make-color 150 150 255)))) + +"flip the first one horizontally" +(beside (flip-horizontal (square 20 "solid" (make-color 50 50 255))) + (square 34 "solid" (make-color 150 150 255))) + +"flip the first one vertically" +(beside (flip-vertical (square 20 "solid" (make-color 50 50 255))) + (square 34 "solid" (make-color 150 150 255))) + +"ABOVE: reference image" +(above (square 20 "solid" (make-color 50 50 255)) + (square 34 "solid" (make-color 150 150 255))) + +"flip the second one horizontally" +(above (square 20 "solid" (make-color 50 50 255)) + (flip-horizontal (square 34 "solid" (make-color 150 150 255)))) + +"flip the second one vertically" +(above (square 20 "solid" (make-color 50 50 255)) + (flip-vertical (square 34 "solid" (make-color 150 150 255)))) + +"flip the first one horizontally" +(above (flip-horizontal (square 20 "solid" (make-color 50 50 255))) + (square 34 "solid" (make-color 150 150 255))) + +"flip the first one vertically" +(above (flip-vertical (square 20 "solid" (make-color 50 50 255))) + (square 34 "solid" (make-color 150 150 255))) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; IMAGE EQUALITY +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"checking a circle against a rectangle" +(check-expect (image=? (circle 50 "solid" "blue") + (rectangle 20 30 "outline" "turquoise")) + #f) +(check-expect (image=? (empty-scene 20 50) (empty-scene 20 50)) true) + +"checking a circle against a different one" +(check-expect (image=? (circle 50 "solid" "blue") + (circle 50 "solid" "turquoise")) + #f) + +"checking a triangle against a different one" +(check-expect (image=? (triangle 50 "solid" "blue") + (triangle 50 "outline" "blue")) + #f) + +"checking a circle against a different one" +(check-expect (image=? (circle 50 "solid" "blue") + (circle 50 "solid" "turquoise")) + #f) + +"checking a textImage against a different one" +(check-expect (image=? (text "purple" 50 "blue") + (text "purple" 50 "red")) + #f) + +"checking a textImage against itself" +(check-expect (image=? (text "purple" 50 "blue") + (text "purple" 50 "blue")) + #t) + +"checking a bitmap against itself" +(check-expect (image=? (bitmap/url "http://www.bootstrapworld.org/images/icon.gif") + (bitmap/url "http://www.bootstrapworld.org/images/icon.gif")) + #t) + +"checking a bitmap against a shape of the same size" +(check-expect (image=? (bitmap/url "http://www.bootstrapworld.org/images/icon.gif") + (rectangle 150 150 "solid" "pink")) + #f) + +"checking a bitmap against a shape of a different size" +(check-expect (image=? (bitmap/url "http://www.bootstrapworld.org/images/icon.gif") + (rectangle 100 100 "solid" "pink")) + #f) + +"checking a bitmap against a different one" +(check-expect (image=? (bitmap/url "http://www.bootstrapworld.org/images/icon.gif") + (bitmap/url "http://www.bootstrapworld.org/images/icon.png")) + #f) + +"checking a rectangle against itself" +(check-expect (image=? (rectangle 100 50 "solid" "blue") + (rectangle 100 50 "solid" "blue")) + #t) + +"checking a rhombus against itself" +(check-expect (image=? (rhombus 100 50 "solid" "blue") + (rhombus 100 50 "solid" "blue")) + #t) + +"checking a square against a 2x larger one that's been scaled by 1/2" +(check-expect (image=? (scale 1/2 (square 100 "solid" "blue")) + (square 50 "solid" "blue")) + #t) + +"checking a square against a 2x larger one that's been scaled by 1/3" +(check-expect (image=? (scale 1/3 (square 100 "solid" "blue")) + (square 50 "solid" "blue")) + #f) + +"checking a rectangle against its equivalent polygon" +(check-expect (image=? (regular-polygon 40 4 "solid" "black") + (rectangle 40 40 "solid" "black")) + #t) + +"checking a circle against its equivalent ellipse" +(check-expect (image=? (circle 50 90 "orange") + (ellipse 100 100 90 "orange")) + #t) +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; IMAGE PROPERTIES +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +"(image-width (ellipse 30 40 'solid' 'orange'))" +(image-width (ellipse 30 40 "solid" "orange")) + +"(image-width (circle 30 'solid' 'orange'))" +(image-width (circle 30 "solid" "orange")) + +"(image-width (beside (circle 20 'solid' 'orange') (circle 20 'solid' 'purple')))" +(image-width (beside (circle 20 "solid" "orange") (circle 20 "solid" "purple"))) + +"(image-height (overlay (circle 20 'solid' 'orange') (circle 30 'solid' 'purple')))" +(image-height (overlay (circle 20 "solid" "orange") (circle 30 "solid" "purple"))) + +"(image-height (rectangle 10 0 'solid' 'purple'))" +(image-height (rectangle 10 0 "solid" "purple")) + +"(image-baseline (text 'Hello' 24 'black'))" +(image-baseline (text "Hello" 24 "black")) + +"(image-baseline (text/font 'Goodbye' 48 'indigo' 'Helvetica' 'modern 'normal 'normal #f))" +(image-baseline (text/font "Goodbye" 48 "indigo" "Helvetica" 'modern 'normal 'normal #f)) + + +"(image-height (text/font 'Goodbye' 48 'indigo' 'Helvetica' 'modern 'normal 'normal #f))" +(image-height (text/font "Goodbye" 48 "indigo" "Helvetica" 'modern 'normal 'normal #f)) + +"(image-baseline (rectangle 100 100 'solid' 'black'))" +(image-baseline (rectangle 100 100 "solid" "black")) + +"(image-height (rectangle 100 100 'solid' 'black'))" +(image-height (rectangle 100 100 "solid" "black")) + + +"(mode? 'outline')" +(mode? "outline") + +"(mode? 'checkered')" +(mode? "checkered") + +"(image-color? 'pink')" +(image-color? "pink") + +"(image-color? 'puke')" +(image-color? "puke") + +"(y-place? 'middle')" +(y-place? "middle") + +"(x-place? 'up-top')" +(x-place? "up-top") + +"(angle? 290)" +(angle? 290) + +"(angle? -290)" +(angle? -290) + +"(side-count? 20)" +(side-count? 20) + +"(side-count? 2)" +(side-count? 2) + +"(step-count? 2)" +(step-count? 2) + +"(step-count? 0)" +(step-count? 0)