HTML5 Canvas Library Render Image Performance (v22)

Revision 22 of this benchmark created by Jare on


Description

This test compares the image force re-rendering speeds of popular HTML5 Canvas libraries, such as KineticJS, Fabric.js, Paper.js, EaselJS, Raphael.js, and a baseline native canvas rendering.

Note: These tests don't compare apples to apples. Some libraries are much simpler in nature, and mostly provide an easier API for canvas drawing, while other libraries are much more complex and provide additional functionality, ultimately adding overhead.

Preparation HTML

<!-- image to render -->
<img width="104" height="75" id="sheep" style="display: none;" src="" />
<!-- native canvas -->
<canvas id="natCanvas" width="512" height="512"></canvas>
<!-- KineticJS -->
<div id="kinContainer"></div>
<script src="https://cdnjs.cloudflare.com/ajax/libs/kineticjs/5.0.1/kinetic.min.js"></script>
<!-- Fabric.js -->
<div id="fabContainer"></div>
<script src="https://cdnjs.cloudflare.com/ajax/libs/fabric.js/1.4.0/fabric.min.js"></script>
<!-- EaselJS -->
<canvas id="easCanvas" width="512" height="512"></canvas>
<script src="https://cdnjs.cloudflare.com/ajax/libs/EaselJS/0.7.1/easeljs.min.js"></script>
<!-- Paper.js -->
<canvas id="papCanvas" width="512" height="512"></canvas>
<script src="https://cdn.bootcss.com/paper.js/0.9.18/paper-core.min.js"></script>
<!-- Raphael.js -->
<div id="rapContainer"></div>
<script src="https://cdnjs.cloudflare.com/ajax/libs/raphael/2.1.2/raphael-min.js"></script>

Setup

var imgEL = document.getElementById('sheep');
    
    // =================== native canvas
    var natCanvas = document.getElementById('natCanvas');
    var natContext = natCanvas.getContext('2d');
    
    // =================== KineticJS
    var kinStage = new Kinetic.Stage({
      container: 'kinContainer',
      width: 512,
      height: 512
    });
    var kinLayer = new Kinetic.Layer();
    kinStage.add(kinLayer);
    
    // =================== Fabric.js
    var fabCanvasEL = document.createElement('canvas');
    fabCanvasEL.id = 'fabCanvas';
    fabCanvasEL.width = 512;
    fabCanvasEL.height = 512;
    document.getElementById('fabContainer').appendChild(fabCanvasEL);
    var fCanvas = new fabric.Canvas('fabCanvas');
    
    // =================== EaselJS
    var easStage = new createjs.Stage('easCanvas');
    
    // =================== Paper.js
    var papCanvas = document.getElementById('papCanvas');
    paper.setup(papCanvas);
    
    // =================== Raphael.js
    var rapPaper = Raphael('rapContainer', 512, 512);

Teardown


    // =================== native canvas
    natContext.save();
    natContext.setTransform(1, 0, 0, 1, 0, 0);
    natContext.clearRect(0, 0, 512, 512);
    natContext.restore();
    // =================== KineticJS
    kinStage.destroy();
    //document.getElementById('kinContainer').innerHTML = '';
    // =================== EaselJS
    easStage.removeAllChildren();
    easStage.enableMouseOver(0);
    easStage.enableDOMEvents(false);
    easStage.canvas = null;
    // =================== Fabric.js
    var canvasContainer = fCanvas.getElement().parentNode;
    fCanvas.clear();
    fCanvas.dispose();
    if (canvasContainer.parentNode) {
      canvasContainer.parentNode.removeChild(canvasContainer);
    }
    // =================== Paper.js
    paper.project.remove();
    // =================== Raphael.js
    //document.getElementById('rapContainer').innerHTML = '';
    rapPaper.clear();
    rapPaper.remove();
  

Test runner

Ready to run.

Testing in
TestOps/sec
KineticJS
kinLayer.clear();
var kinImgEL = new Kinetic.Image({
  x: 256,
  y: 256,
  image: imgEL,
  width: imgEL.width,
  height: imgEL.height
});
kinLayer.add(kinImgEL);
kinLayer.draw();
ready
Fabric.js
fCanvas.clear();
var fabImg = new fabric.Image(imgEL, {
  left: 256,
  top: 256
});
fCanvas.add(fabImg);
fCanvas.renderAll();
ready
Native Canvas
natContext.save();
natContext.setTransform(1, 0, 0, 1, 0, 0);
natContext.clearRect(0, 0, 512, 512);
natContext.restore();
natContext.drawImage(imgEL, 256, 256, imgEL.width, imgEL.height);
ready
EaselJS
easStage.removeAllChildren();
var easelBitmap = new createjs.Bitmap(imgEL.src);
easelBitmap.x = 256;
easelBitmap.y = 256;
easStage.addChild(easelBitmap);
easStage.update();
ready
Paper.js
paper.project.clear();
var raster = new paper.Raster('sheep');
raster.position = new paper.Point(256, 256);
paper.view.draw();
ready
Raphael.js
rapPaper.clear();
var rapEL = rapPaper.image(imgEL.src, 256, 256, imgEL.width, imgEL.height);
ready

Revisions

You can edit these tests or add more tests to this page by appending /edit to the URL.