Writing a unit test for your HTML5 Canvas component without mocking any DOM element
Using browser to render and execute your component's unit test logic directly, outputting image snapshot to compare in pixel-level
Support TypeScript and zero config needed in most cases for JavaScript project, using API like it
and describe
with a few unique API for canvas snapshot
Run npm cmd to install on your project
$ npm i @canvest/canvest-cli --save-dev
Create a script in your package.json
"scripts": {
...
"test": "canvest",
...
},
If you just want to test a canvas, create helloworld.canvest.js
under ./canvest
folder
describe('Test my canvas', () => {
it('should render hello world text in canvas', async () => {
let canvas = document.createElement('canvas');
canvas.width = 800;
canvas.height = 600;
const a = await snapshot(canvas);
const context = canvas.getContext('2d');
context.fillStyle = '#ff0';
context.fillRect(0, 0, 800, 600);
const text = 'Hello, World!';
context.font = 'bold 70pt Menlo';
context.textAlign = 'center';
context.fillStyle = '#fff';
context.fillText(text, 200, 200);
const b = await snapshot(canvas);
a.notEqual(b);// this should pass
await autoShot('test', canvas); // this create a snapshot in local that will check canvas image to test if it is still the same
});
});
Or test the class you create, for example for pixi.js
export class MyClass
{
...
updateRotation(rotation){
this._sprite.rotation = rotation;
}
...
}
Create MyClass.canvest.js
under ./canvest
folder
import * as PIXI from 'pixi.js';
import { MyClass } from './MyClass';
describe('Test my class', () => {
it('should update rotation as expect', async () => {
const app = new PIXI.Application({
width: XXX, height: YYY,
preserveDrawingBuffer: true //<--- important for Canvest to take snapshot, false might lead to a blank image
});
const myClass = new MyClass();
app.stage.addChild(myClass._sprite);
myClass.updateRotation(1.5);
/**
* the second time this autoShot function run,
* it will check the current Canvas's image with the local cached 'rotation_that_expected.png' file,
* if this one is not the same with the cached image,
* the test will fail
**/
await autoShot('rotation_that_expected', app.view);
const rotation15Snapshot = await snapshot(app.view); // 1.5 rotation snapshot
myClass.updateRotation(1.6); // rotate sprite to 1.6 rotation
const rotation16Snapshot = await snapshot(app.view);
rotation16Snapshot.notEqual(rotation15Snapshot); // should pass
/**
* if the number of different pixels between rotation16Snapshot and rotation15Snapshot is below 20%,
* isMatch function will make the test as passed
**/
rotation16Snapshot.isMatch(rotation15Snapshot, 0.2);
});
});
Then using npm cmd
$ npm test
Once you ran npm test
, Canvest-cli will start running unit test that under canvest
folder if files name is after *.canvest.(js|ts)
pattern.
Canvest-cli will start two node servers, to config the port you could change npm script by
"scripts": {
...
"test": "canvest --cachePort XXX --pagePort XXX",
...
},
cachePort
: this is the port that canvest-cli using to start the node server to cache your snapshot, default is45670
.pagePort
: this is the port that canvest-cli using to start the web page to run your unit test withMocha
, runningwebapck-dev-server
under the hood.
If some test case failed, you will see diff comparison under bottom showing in highlight red color.
One thing should notice is the diff for cached snapshot won't show up if cached snapshot has different size with current one.
Canvest framework is using Mocha
with Chai
under the hood, every API Mocha had in browser, Canvest should have as well by accessing mocha
variable but this is not recommended.
You could create file canvest.init.js
(or canvest.init.ts
when using --ts
) under ./canvest
folder and doing all the setup in there.
take a snapshot of current canvas and cached in local, if local snapshot already exists, compare current snapshot with local one automatically
-
name
: a unique name for snapshot to save under./canvest/autoShot/(the-name-you-given).png
-
canvas
: HTML5 canvas dom element -
as long as your local snapshot doesn't get removed,
autoShot
will do the comparison instead of caching it and pass test
take a snapshot of current canvas
canvas
: HTML5 canvas dom elementsnapshot Object
: return a Canvest snapshot that has 4 APIs in belowisEqual( otherSnapshot )
: snapshot should completely equal tootherSnapshot
notEqual( otherSnapshot )
: snapshot should not equal tootherSnapshot
isMatch( otherSnapshot, tolerance )
: snapshot could equal tootherSnapshot
if test ignores number of tolerance percentage of pixelsnotMatch( otherSnapshot, tolerance )
: snapshot could not equal tootherSnapshot
even after test ignores number of tolerance percentage of pixels
set global threshold for snapshot comparision, if you just want one snapshot comparision to be more tolerant, use isMatch
or notMatch
function with tolerance
parameter
number
threshold, a number that can be set to maximum1.0
. Default is0.05
, higher number will ignore more differences between snapshot comparision
Canvest will create test coverage after all test is done, you could find the coverage.json
and/or open index.html
under ./coverage
folder
To support TypeScript, you will need run npm i @canvest/canvest-ts --save-dev
to install canvest-ts
plugin
Then using canvest --ts ./path-to-your-tsconfig.json
to start testing
If you want to run with headless browser in CI, you could use --ci
option. This will make Canvest exit when all tests are done, and add a div inside web page <body>
with className test_end
or test_end_with_failed
.
For example, canvest --ci ./canvest
will create a canvest-test-result
folder under ./canvest
folder, any failed test will create a diff image in result folder.
If diff is a cached snapshot and it is not valid because new snapshot size is different, it will still create an empty file failed-Name.diff.failed
.
Canvest example for pixi.js with TypeScript https://github.com/TyrealGray/canvest-pixi.js-example
Canvest example for three.js https://github.com/TyrealGray/canvest-three.js-example
AFL-3.0