/** * Module dependencies. */ var Canvas = require('canvas') , assert = require('assert') , crypto = require('crypto') , fs = require('fs'); function hash(val) { return crypto.createHash('md5').update(val).digest('hex'); } function assertChecksum(canvas, path, checksum, msg) { canvas.savePNG(path); fs.readFile(path, function(err, buf){ assert.equal(hash(buf), checksum, msg); }); } module.exports = { 'test .version': function(assert){ assert.match(Canvas.version, /^\d+\.\d+\.\d+$/); }, 'test .parseColor()': function(assert){ assert.equal(null, Canvas.parseColor()); assert.equal(null, Canvas.parseColor('')); // rgb() assert.eql([255,165,0,1], Canvas.parseColor('rgb(255,165,0)')); assert.eql([255,165,0,1], Canvas.parseColor('rgb(255, 165, 0)')); assert.eql([255,165,0,1], Canvas.parseColor('rgb(255 , 165 , 0)')); assert.equal(null, Canvas.parseColor('rgb()')); // rgba() assert.eql([255,165,0,1], Canvas.parseColor('rgba(255,165,0,1)')); assert.eql([255,165,0,1], Canvas.parseColor('rgba(255,165,0,1)')); assert.eql([255,165,0,.6], Canvas.parseColor('rgba(255,165,0,0.6)')); assert.eql([255,165,0,.6], Canvas.parseColor('rgba(255,165, 0, 0.6)')); assert.eql([255,165,0,.6], Canvas.parseColor('rgba(255,165 , 0 ,.6)')); assert.equal(null, Canvas.parseColor('rgba(2554,165 , 0 ,.6)')); assert.equal(null, Canvas.parseColor('rgba()')); // hex assert.eql([165,89,89,1], Canvas.parseColor('#A55959')); assert.eql([255,255,255,1], Canvas.parseColor('#FFFFFF')); assert.eql([255,255,255,1], Canvas.parseColor('#ffffff')); assert.eql([255,255,255,1], Canvas.parseColor('#FFF')); assert.eql([255,255,255,1], Canvas.parseColor('#fff')); // name assert.eql([255,255,255,1], Canvas.parseColor('white')); assert.eql([0,0,0,1], Canvas.parseColor('black')); }, 'test Canvas#getContext("2d")': function(assert){ var canvas = new Canvas(200, 300) , ctx = canvas.getContext('2d'); assert.ok('object' == typeof ctx); assert.equal(canvas, ctx.canvas, 'context.canvas is not canvas'); assert.equal(ctx, canvas.context, 'canvas.context is not context'); }, 'test Canvas#getContext("invalid")': function(assert){ assert.equal(null, new Canvas(200, 300).getContext('invalid')); }, 'test Context2d#clearRect()': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/clearRect.png'; ctx.fillRect(25,25,100,100); ctx.clearRect(45,45,60,60); ctx.fillRect(50,50,50,50); assertChecksum( canvas , path , 'e21404e97142a76c0c8d14cf0fab400f' , 'Context2d#clearRect() failed'); }, 'test Context2d#strokeRect()': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/strokeRect.png'; ctx.fillRect(25,25,100,100); ctx.clearRect(45,45,60,60); ctx.strokeRect(50,50,50,50); assertChecksum( canvas , path , '130c5457e19da9d35b46970c2c3e035f' , 'Context2d#strokeRect() failed'); }, 'test Context2d#lineTo()': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/lineTo.png'; // Filled triangle ctx.beginPath(); ctx.moveTo(25.5,25); ctx.lineTo(105,25); ctx.lineTo(25,105); ctx.fill(); // Stroked triangle ctx.beginPath(); ctx.moveTo(125,125); ctx.lineTo(125,45); ctx.lineTo(45,125); ctx.closePath(); ctx.stroke(); assertChecksum( canvas , path , '3d500105083716f798cf3de3ef6a4a56' , 'Context2d#lineTo() failed' ); }, 'test Context2d#arc()': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/arc.png'; ctx.beginPath(); ctx.arc(75,75,50,0,Math.PI*2,true); // Outer circle ctx.moveTo(110,75); ctx.arc(75,75,35,0,Math.PI,false); // Mouth ctx.moveTo(65,65); ctx.arc(60,65,5,0,Math.PI*2,true); // Left eye ctx.moveTo(95,65); ctx.arc(90,65,5,0,Math.PI*2,true); // Right eye ctx.stroke(); assertChecksum( canvas , path , '82997bc57c3941afea72ba571d713160' , 'Context2d#arc() failed'); }, 'test Context2d#bezierCurveTo()': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/bezierCurveTo.png'; ctx.beginPath(); ctx.moveTo(75,40); ctx.bezierCurveTo(75,37,70,25,50,25); ctx.bezierCurveTo(20,25,20,62.5,20,62.5); ctx.bezierCurveTo(20,80,40,102,75,120); ctx.bezierCurveTo(110,102,130,80,130,62.5); ctx.bezierCurveTo(130,62.5,130,25,100,25); ctx.bezierCurveTo(85,25,75,37,75,40); ctx.fill(); assertChecksum( canvas , path , '5626a53780d77aecc490ec807ee0bc63' , 'Context2d#bezierCurveTo() failed'); }, 'test Context2d#rotate()': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/rotate.png'; ctx.rotate(0.4); ctx.translate(30,0); ctx.rect(0,0,50,50); ctx.stroke(); assertChecksum( canvas , path , 'b364d4572f8b4fe03e1290235dcf2e55' , 'Context2d#rotate() failed'); }, 'test fill with stroke': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/fillWithStroke.png'; ctx.beginPath(); ctx.arc(75,75,50,0,Math.PI*2,true); ctx.fill(); ctx.closePath(); ctx.beginPath(); ctx.fillStyle = 'red'; ctx.strokeStyle = 'yellow'; ctx.arc(75,75,30,0,Math.PI*2,true); ctx.fill(); ctx.stroke(); assertChecksum( canvas , path , '603b1e1c8a4bc0048c9a0944c83e82f4' , 'fill with stroke failed'); }, 'test Context2d#rect()': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/rect.png'; ctx.rect(5,5,50,50); ctx.strokeStyle = 'yellow'; ctx.fill(); ctx.stroke(); assertChecksum( canvas , path , 'a670979e566eafa07e3938aec9e2b7a3' , 'Context2d#rect() failed'); }, 'test Context2d#fillStyle=': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/fillStyle.png'; ctx.fillStyle = 'rgb(0,55,0)'; ctx.fillRect(10, 10, 50, 50); assert.equal('rgba(0,55,0,1)', ctx.fillStyle); ctx.fillStyle = 'rgba(0,0,0,0.1)'; ctx.fillRect(60, 60, 50, 50); assert.equal('rgba(0,0,0,0.1)', ctx.fillStyle); ctx.fillStyle = '#000'; ctx.fillRect(110, 110, 50, 50); assert.equal('rgba(0,0,0,1)', ctx.fillStyle); assertChecksum( canvas , path , '804d2494db2eeda204ccbd6d0b48265a' , 'Context2d#fillStyle= failed'); }, 'test Context2d#lineWidth=': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/lineWidth.png'; ctx.beginPath(); ctx.lineWidth = 10.0; assert.equal(10, ctx.lineWidth); // ctx.lineWidth = Infinity; assert.equal(10, ctx.lineWidth); ctx.lineWidth = -5; assert.equal(10, ctx.lineWidth); ctx.lineWidth = 0; assert.equal(10, ctx.lineWidth); ctx.moveTo(50, 50); ctx.lineTo(50, 100); ctx.lineTo(80, 120); ctx.stroke(); assertChecksum( canvas , path , '0bc6f64d58f326ca7ad3ade4426fb90f' , 'Context2d#lineWidth= failed'); }, 'test Context2d#lineCap=': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/lineCap.png'; ctx.beginPath(); ctx.lineWidth = 10.0; ctx.lineCap = 'round'; assert.equal('round', ctx.lineCap); ctx.moveTo(50, 50); ctx.lineTo(50, 100); ctx.lineTo(80, 120); ctx.stroke(); assertChecksum( canvas , path , 'd5b84ea10a3e6df723b702a32329ed43' , 'Context2d#lineCap= failed'); }, 'test Context2d#lineJoin=': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/lineJoin.png'; ctx.beginPath(); ctx.lineWidth = 10.0; ctx.lineJoin = 'round'; assert.equal('round', ctx.lineJoin); ctx.moveTo(50, 50); ctx.lineTo(50, 100); ctx.lineTo(80, 120); ctx.stroke(); assertChecksum( canvas , path , 'bf97d882a0e99595109fb4f564fa41bf' , 'Context2d#lineJoin= failed'); }, 'test Context2d states': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/states.png'; ctx.arc(5,5,-1) ctx.save(); ctx.rect(50, 50, 100, 100); ctx.stroke(); ctx.restore(); ctx.save(); ctx.translate(50,50); ctx.scale(.5,.5); ctx.strokeRect(51, 51, 100, 100); ctx.restore(); ctx.translate(95,95); ctx.fillRect(0,0,10,10); assertChecksum( canvas , path , 'fc8bbf2cf6ae2d85fcf526103200e844' , 'Context2d#save() / resetore() failed'); }, 'test Context2d#globalAlpha=': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/globalAlpha.png'; ctx.fillRect(0,0,50,50); ctx.translate(15,15); ctx.globalAlpha = 0.6; assert.equal(0.6, ctx.globalAlpha.toFixed(1)); ctx.fillStyle = 'red'; ctx.fillRect(0,0,20,20); ctx.fillStyle = 'yellow'; ctx.fillRect(0,0,10,10); assertChecksum( canvas , path , 'd8365233f2beb420ba18ff78dc6d7405' , 'Context2d#globalAlpha= failed'); }, 'test Context2d#createLinearGradient()': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/linearGradients.png'; var lingrad = ctx.createLinearGradient(0,0,0,150); lingrad.addColorStop(0, '#00ABEB'); lingrad.addColorStop(0.5, '#fff'); lingrad.addColorStop(0.5, '#26C000'); lingrad.addColorStop(1, '#fff'); var lingrad2 = ctx.createLinearGradient(0,50,0,95); lingrad2.addColorStop(0.5, '#000'); lingrad2.addColorStop(1, 'rgba(0,0,0,0)'); ctx.fillStyle = lingrad; ctx.strokeStyle = lingrad2; ctx.fillRect(10,10,130,130); ctx.strokeRect(50,50,50,50); assertChecksum( canvas , path , '55f80d7d33000904ca366e0cfe363079' , 'Context2d#createLinearGradient() failed'); }, 'test Context2d#createRadialGradient()': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/radialGradients.png'; // Create gradients var radgrad = ctx.createRadialGradient(45,45,10,52,50,30); radgrad.addColorStop(0, '#A7D30C'); radgrad.addColorStop(0.9, '#019F62'); radgrad.addColorStop(1, 'rgba(1,159,98,0)'); var radgrad2 = ctx.createRadialGradient(105,105,20,112,120,50); radgrad2.addColorStop(0, '#FF5F98'); radgrad2.addColorStop(0.75, '#FF0188'); radgrad2.addColorStop(1, 'rgba(255,1,136,0)'); var radgrad3 = ctx.createRadialGradient(95,15,15,102,20,40); radgrad3.addColorStop(0, '#00C9FF'); radgrad3.addColorStop(0.8, '#00B5E2'); radgrad3.addColorStop(1, 'rgba(0,201,255,0)'); var radgrad4 = ctx.createRadialGradient(0,150,50,0,140,90); radgrad4.addColorStop(0, '#F4F201'); radgrad4.addColorStop(0.8, '#E4C700'); radgrad4.addColorStop(1, 'rgba(228,199,0,0)'); // draw shapes ctx.fillStyle = radgrad4; ctx.fillRect(0,0,150,150); ctx.fillStyle = radgrad3; ctx.fillRect(0,0,150,150); ctx.fillStyle = radgrad2; ctx.fillRect(0,0,150,150); ctx.fillStyle = radgrad; ctx.fillRect(0,0,150,150); assertChecksum( canvas , path , 'd078f5993eb962a5b3fdde5ca0864179' , 'Context2d#createRadialGradient() failed'); }, 'test invalid {fill,stroke}Style': function(assert){ var canvas = new Canvas(200, 200) , ctx = canvas.getContext('2d') , path = __dirname + '/images/invalidStyle.png'; ctx.fillStyle = 'red'; ctx.strokeStyle = 'yellow'; ctx.rect(50,50,50,50); ctx.fill(); ctx.stroke(); ctx.beginPath(); ctx.fillStyle = 'asdf'; ctx.strokeStyle = 'asdf'; ctx.rect(100,80,15,15); ctx.fill(); ctx.stroke(); assertChecksum( canvas , path , '2ba95ccadd5c38949a5ea493dbc78e08' , 'Context2d invalid fillStyle did not retain previous value'); } }