2018-09-18 20:17:34 +00:00
CanvasKit - Skia + WebAssembly
2018-09-20 20:30:21 +00:00
==============================
2018-09-18 20:17:34 +00:00
2018-09-20 20:30:21 +00:00
Skia now offers a WebAssembly build for easy deployment of our graphics APIs on
the web.
2018-09-18 20:17:34 +00:00
2018-09-20 20:30:21 +00:00
CanvasKit provides a playground for testing new Canvas and SVG platform APIs,
enabling fast-paced development on the web platform.
It can also be used as a deployment mechanism for custom web apps requiring
cutting-edge features, like Skia's [Lottie
2018-09-20 21:39:31 +00:00
animation](https://skia.org/user/modules/skottie) support.
2018-09-20 20:30:21 +00:00
Features
--------
- WebGL context encapsulated as an SkSurface, allowing for direct drawing to
an HTML canvas
- Core set of Skia canvas/paint/path/text APIs available, see bindings
- Draws to a hardware-accelerated backend
- Security tested with Skia's fuzzers
Samples
-------
2018-09-18 20:17:34 +00:00
< style >
#demo canvas {
border: 1px dashed #AAA ;
2018-09-20 21:39:31 +00:00
margin: 2px;
2018-09-18 20:17:34 +00:00
}
2019-12-04 20:25:51 +00:00
#patheffect , #ink , #shaping {
2018-09-20 21:39:31 +00:00
width: 400px;
2018-09-18 20:17:34 +00:00
height: 400px;
}
2018-09-21 14:14:30 +00:00
#sk_legos , #sk_drinks , #sk_party , #sk_onboarding {
2018-09-18 20:17:34 +00:00
width: 300px;
height: 300px;
}
2018-09-21 14:14:30 +00:00
figure {
display: inline-block;
margin: 0;
}
figcaption > a {
margin: 2px 10px;
}
2018-09-18 20:17:34 +00:00
< / style >
< div id = demo >
2019-02-27 13:42:33 +00:00
< h3 > Go beyond the HTML Canvas2D< / h3 >
2018-09-21 14:14:30 +00:00
< figure >
< canvas id = patheffect width = 400 height = 400 > < / canvas >
< figcaption >
2019-02-27 13:42:33 +00:00
< a href = "https://jsfiddle.skia.org/canvaskit/ea89749ae8c90bce807ea2e7e34fb7b09b950cee70d9db0a9cdfd2d67bd48ef0"
2018-09-21 14:14:30 +00:00
target=_blank rel=noopener>
Star JSFiddle< / a >
< / figcaption >
< / figure >
< figure >
< canvas id = ink width = 400 height = 400 > < / canvas >
< figcaption >
2018-11-07 15:43:42 +00:00
< a href = "https://jsfiddle.skia.org/canvaskit/43475699d6d7d3d7dad1004c29f84015752a6a6dee2bb90f2e891b53e31d45cc"
2018-09-21 14:14:30 +00:00
target=_blank rel=noopener>
Ink JSFiddle< / a >
< / figcaption >
< / figure >
< h3 > Skottie (click for fiddles)< / h3 >
2018-10-30 19:05:04 +00:00
< a href = "https://jsfiddle.skia.org/canvaskit/092690b273b41076d2f00f0d43d004893d6bb9992c387c0385efa8e6f6bc83d7"
2018-09-21 14:14:30 +00:00
target=_blank rel=noopener>
< canvas id = sk_legos width = 300 height = 300 > < / canvas >
< / a >
2018-10-30 19:05:04 +00:00
< a href = "https://jsfiddle.skia.org/canvaskit/e7ac983d9859f89aff1b6d385190919202c2eb53d028a79992892cacceffd209"
2018-09-21 14:14:30 +00:00
target=_blank rel=noopener>
< canvas id = sk_drinks width = 500 height = 500 > < / canvas >
< / a >
2018-10-30 19:05:04 +00:00
< a href = "https://jsfiddle.skia.org/canvaskit/0e06547181759731e7369d3e3613222a0826692f48c41b16504ed68d671583e1"
2018-09-21 14:14:30 +00:00
target=_blank rel=noopener>
< canvas id = sk_party width = 500 height = 500 > < / canvas >
< / a >
2018-10-30 19:05:04 +00:00
< a href = "https://jsfiddle.skia.org/canvaskit/be3fc1c5c351e7f43cc2840033f80b44feb3475925264808f321bb9e2a21174a"
2018-09-21 14:14:30 +00:00
target=_blank rel=noopener>
< canvas id = sk_onboarding width = 500 height = 500 > < / canvas >
< / a >
2019-12-04 20:25:51 +00:00
< h3 > SkParagraph (using ICU and Harfbuzz)< / h3 >
2019-02-27 13:42:33 +00:00
< figure >
2019-12-04 20:25:51 +00:00
< canvas id = shaping width = 500 height = 500 > < / canvas >
2019-02-27 13:42:33 +00:00
< figcaption >
2019-12-04 20:25:51 +00:00
< a href = "https://jsfiddle.skia.org/canvaskit/56cb197c724dfdfad0c3d8133d4fcab587e4c4e7f31576e62c17251637d3745c"
2019-02-27 13:42:33 +00:00
target=_blank rel=noopener>
2019-12-04 20:25:51 +00:00
SkParagraph JSFiddle< / a >
2019-02-27 13:42:33 +00:00
< / figcaption >
< / figure >
2018-09-18 20:17:34 +00:00
< / div >
< script type = "text/javascript" charset = "utf-8" >
(function() {
// Tries to load the WASM version if supported, shows error otherwise
let s = document.createElement('script');
var locate_file = '';
if (window.WebAssembly & & typeof window.WebAssembly.compile === 'function') {
console.log('WebAssembly is supported!');
2019-12-04 20:25:51 +00:00
locate_file = 'https://unpkg.com/canvaskit-wasm@0.9.0/bin/';
2018-09-18 20:17:34 +00:00
} else {
console.log('WebAssembly is not supported (yet) on this browser.');
document.getElementById('demo').innerHTML = "< div > WASM not supported by your browser. Try a recent version of Chrome, Firefox, Edge, or Safari.< / div > ";
return;
}
2018-10-09 13:36:35 +00:00
s.src = locate_file + 'canvaskit.js';
2018-09-18 20:17:34 +00:00
s.onload = () => {
var CanvasKit = null;
var legoJSON = null;
var drinksJSON = null;
var confettiJSON = null;
2018-09-21 14:14:30 +00:00
var onboardingJSON = null;
var fullBounds = {fLeft: 0, fTop: 0, fRight: 500, fBottom: 500};
2018-09-18 20:17:34 +00:00
CanvasKitInit({
locateFile: (file) => locate_file + file,
2019-02-25 21:04:02 +00:00
}).ready().then((CK) => {
2018-09-18 20:17:34 +00:00
CanvasKit = CK;
DrawingExample(CanvasKit);
2018-09-20 21:39:31 +00:00
InkExample(CanvasKit);
2019-02-27 13:42:33 +00:00
ShapingExample(CanvasKit);
2018-09-20 21:39:31 +00:00
// Set bounds to fix the 4:3 resolution of the legos
SkottieExample(CanvasKit, 'sk_legos', legoJSON, {fLeft: -50, fTop: 0, fRight: 350, fBottom: 300});
2018-09-21 14:14:30 +00:00
// Re-size to fit
SkottieExample(CanvasKit, 'sk_drinks', drinksJSON, fullBounds);
SkottieExample(CanvasKit, 'sk_party', confettiJSON, fullBounds);
SkottieExample(CanvasKit, 'sk_onboarding', onboardingJSON, fullBounds);
2018-09-18 20:17:34 +00:00
});
fetch('https://storage.googleapis.com/skia-cdn/misc/lego_loader.json').then((resp) => {
resp.text().then((str) => {
legoJSON = str;
2018-09-21 14:14:30 +00:00
SkottieExample(CanvasKit, 'sk_legos', legoJSON, {fLeft: -50, fTop: 0, fRight: 350, fBottom: 300});
2018-09-18 20:17:34 +00:00
});
});
fetch('https://storage.googleapis.com/skia-cdn/misc/drinks.json').then((resp) => {
resp.text().then((str) => {
drinksJSON = str;
2018-09-21 14:14:30 +00:00
SkottieExample(CanvasKit, 'sk_drinks', drinksJSON, fullBounds);
2018-09-18 20:17:34 +00:00
});
});
fetch('https://storage.googleapis.com/skia-cdn/misc/confetti.json').then((resp) => {
resp.text().then((str) => {
confettiJSON = str;
2018-09-21 14:14:30 +00:00
SkottieExample(CanvasKit, 'sk_party', confettiJSON, fullBounds);
});
});
fetch('https://storage.googleapis.com/skia-cdn/misc/onboarding.json').then((resp) => {
resp.text().then((str) => {
onboardingJSON = str;
SkottieExample(CanvasKit, 'sk_onboarding', onboardingJSON, fullBounds);
2018-09-18 20:17:34 +00:00
});
});
2018-10-02 15:33:52 +00:00
function preventScrolling(canvas) {
canvas.addEventListener('touchmove', (e) => {
// Prevents touch events in the canvas from scrolling the canvas.
e.preventDefault();
e.stopPropagation();
});
}
2018-09-18 20:17:34 +00:00
function DrawingExample(CanvasKit) {
2018-10-30 19:05:04 +00:00
const surface = CanvasKit.MakeCanvasSurface('patheffect');
2018-09-18 20:17:34 +00:00
if (!surface) {
console.log('Could not make surface');
}
const context = CanvasKit.currentContext();
const canvas = surface.getCanvas();
const paint = new CanvasKit.SkPaint();
2018-10-02 15:33:52 +00:00
const textPaint = new CanvasKit.SkPaint();
textPaint.setColor(CanvasKit.Color(40, 0, 0, 1.0));
textPaint.setAntiAlias(true);
2019-02-27 13:42:33 +00:00
const textFont = new CanvasKit.SkFont(null, 30);
2018-09-18 20:17:34 +00:00
let i = 0;
2018-09-21 14:14:30 +00:00
let X = 200;
let Y = 200;
2018-09-18 20:17:34 +00:00
function drawFrame() {
const path = starPath(CanvasKit, X, Y);
CanvasKit.setCurrentContext(context);
const dpe = CanvasKit.MakeSkDashPathEffect([15, 5, 5, 10], i/5);
i++;
paint.setPathEffect(dpe);
2018-11-07 15:43:42 +00:00
paint.setStyle(CanvasKit.PaintStyle.Stroke);
2018-09-18 20:17:34 +00:00
paint.setStrokeWidth(5.0 + -3 * Math.cos(i/30));
paint.setAntiAlias(true);
paint.setColor(CanvasKit.Color(66, 129, 164, 1.0));
canvas.clear(CanvasKit.Color(255, 255, 255, 1.0));
canvas.drawPath(path, paint);
2019-02-27 13:42:33 +00:00
canvas.drawText('Try Clicking!', 10, 380, textPaint, textFont);
2018-09-18 20:17:34 +00:00
canvas.flush();
dpe.delete();
path.delete();
window.requestAnimationFrame(drawFrame);
}
window.requestAnimationFrame(drawFrame);
// Make animation interactive
2018-10-02 15:33:52 +00:00
let interact = (e) => {
2018-09-21 14:14:30 +00:00
if (!e.buttons) {
return;
}
2018-09-18 20:17:34 +00:00
X = e.offsetX;
Y = e.offsetY;
2018-10-02 15:33:52 +00:00
};
document.getElementById('patheffect').addEventListener('pointermove', interact);
document.getElementById('patheffect').addEventListener('pointerdown', interact);
preventScrolling(document.getElementById('patheffect'));
2018-09-18 20:17:34 +00:00
2019-02-27 13:42:33 +00:00
// A client would need to delete this if it didn't go on forever.
// font.delete();
// paint.delete();
2018-09-18 20:17:34 +00:00
}
2018-09-20 21:39:31 +00:00
function InkExample(CanvasKit) {
2018-10-30 19:05:04 +00:00
const surface = CanvasKit.MakeCanvasSurface('ink');
2018-09-20 21:39:31 +00:00
if (!surface) {
console.log('Could not make surface');
}
const context = CanvasKit.currentContext();
const canvas = surface.getCanvas();
let paint = new CanvasKit.SkPaint();
paint.setAntiAlias(true);
paint.setColor(CanvasKit.Color(0, 0, 0, 1.0));
2018-11-07 15:43:42 +00:00
paint.setStyle(CanvasKit.PaintStyle.Stroke);
2018-09-20 21:39:31 +00:00
paint.setStrokeWidth(4.0);
// This effect smooths out the drawn lines a bit.
paint.setPathEffect(CanvasKit.MakeSkCornerPathEffect(50));
2018-09-21 14:14:30 +00:00
// Draw I N K
2018-09-20 21:39:31 +00:00
let path = new CanvasKit.SkPath();
2018-09-21 14:14:30 +00:00
path.moveTo(80, 30);
path.lineTo(80, 80);
path.moveTo(100, 80);
path.lineTo(100, 15);
path.lineTo(130, 95);
path.lineTo(130, 30);
path.moveTo(150, 30);
path.lineTo(150, 80);
path.moveTo(170, 30);
path.lineTo(150, 55);
path.lineTo(170, 80);
2018-09-20 21:39:31 +00:00
let paths = [path];
let paints = [paint];
function drawFrame() {
CanvasKit.setCurrentContext(context);
for (let i = 0; i < paints.length & & i < paths . length ; i + + ) {
canvas.drawPath(paths[i], paints[i]);
}
canvas.flush();
window.requestAnimationFrame(drawFrame);
}
let hold = false;
2018-10-02 15:33:52 +00:00
let interact = (e) => {
let type = e.type;
if (type === 'lostpointercapture' || type === 'pointerup' || !e.pressure ) {
2018-09-20 21:39:31 +00:00
hold = false;
return;
}
if (hold) {
path.lineTo(e.offsetX, e.offsetY);
} else {
paint = paint.copy();
paint.setColor(CanvasKit.Color(Math.random() * 255, Math.random() * 255, Math.random() * 255, Math.random() + .2));
paints.push(paint);
path = new CanvasKit.SkPath();
paths.push(path);
path.moveTo(e.offsetX, e.offsetY);
}
hold = true;
2018-10-02 15:33:52 +00:00
};
document.getElementById('ink').addEventListener('pointermove', interact);
document.getElementById('ink').addEventListener('pointerdown', interact);
document.getElementById('ink').addEventListener('lostpointercapture', interact);
document.getElementById('ink').addEventListener('pointerup', interact);
preventScrolling(document.getElementById('ink'));
2018-09-20 21:39:31 +00:00
window.requestAnimationFrame(drawFrame);
}
2019-02-27 13:42:33 +00:00
function ShapingExample(CanvasKit) {
const surface = CanvasKit.MakeCanvasSurface('shaping');
if (!surface) {
console.log('Could not make surface');
return;
}
2019-12-04 20:25:51 +00:00
let robotoData = null;
fetch('https://storage.googleapis.com/skia-cdn/google-web-fonts/Roboto-Regular.ttf').then((resp) => {
resp.arrayBuffer().then((buffer) => {
robotoData = buffer;
requestAnimationFrame(drawFrame);
});
});
let emojiData = null;
fetch('https://storage.googleapis.com/skia-cdn/misc/NotoColorEmoji.ttf').then((resp) => {
resp.arrayBuffer().then((buffer) => {
emojiData = buffer;
requestAnimationFrame(drawFrame);
});
});
2019-02-27 13:42:33 +00:00
const skcanvas = surface.getCanvas();
2019-12-04 20:25:51 +00:00
const font = new CanvasKit.SkFont(null, 18);
const fontPaint = new CanvasKit.SkPaint();
fontPaint.setStyle(CanvasKit.PaintStyle.Fill);
fontPaint.setAntiAlias(true);
2019-02-27 13:42:33 +00:00
2019-12-04 20:25:51 +00:00
skcanvas.drawText(`Fetching Font data...`, 5, 450, fontPaint, font);
surface.flush();
2019-02-27 13:42:33 +00:00
2019-12-04 20:25:51 +00:00
const context = CanvasKit.currentContext();
let paragraph = null;
let X = 10;
let Y = 10;
const str = 'The quick brown fox 🦊 ate a zesty hamburgerfons 🍔.\nThe 👩👩👧👧 laughed.';
2019-02-27 13:42:33 +00:00
function drawFrame() {
2019-12-04 20:25:51 +00:00
if (robotoData & & emojiData & & !paragraph) {
const fontMgr = CanvasKit.SkFontMgr.FromData([robotoData, emojiData]);
const paraStyle = new CanvasKit.ParagraphStyle({
textStyle: {
color: CanvasKit.BLACK,
fontFamilies: ['Roboto', 'Noto Color Emoji'],
fontSize: 50,
},
textAlign: CanvasKit.TextAlign.Left,
maxLines: 7,
ellipsis: '...',
});
const builder = CanvasKit.ParagraphBuilder.Make(paraStyle, fontMgr);
builder.addText(str);
paragraph = builder.build();
}
if (!paragraph) {
requestAnimationFrame(drawFrame);
return;
}
2019-02-27 13:42:33 +00:00
CanvasKit.setCurrentContext(context);
2019-12-04 20:25:51 +00:00
skcanvas.clear(CanvasKit.WHITE);
const wrapTo = 350 + 150 * Math.sin(Date.now() / 2000);
paragraph.layout(wrapTo);
skcanvas.drawParagraph(paragraph, 0, 0);
skcanvas.drawLine(wrapTo, 0, wrapTo, 400, fontPaint);
let posA = paragraph.getGlyphPositionAtCoordinate(X, Y);
const cp = str.codePointAt(posA.pos);
if (cp) {
const glyph = String.fromCodePoint(cp);
skcanvas.drawText(`At (${X.toFixed(2)}, ${Y.toFixed(2)}) glyph is '${glyph}'`, 5, 450, fontPaint, font);
}
2019-02-27 13:42:33 +00:00
surface.flush();
2019-12-04 20:25:51 +00:00
requestAnimationFrame(drawFrame);
2019-02-27 13:42:33 +00:00
}
// Make animation interactive
let interact = (e) => {
2019-12-04 20:25:51 +00:00
// multiply by 4/5 to account for the difference in the canvas width and the CSS width.
// The 10 accounts for where the mouse actually is compared to where it is drawn.
X = (e.offsetX * 4/5) - 10;
Y = e.offsetY * 4/5;
2019-02-27 13:42:33 +00:00
};
document.getElementById('shaping').addEventListener('pointermove', interact);
document.getElementById('shaping').addEventListener('pointerdown', interact);
document.getElementById('shaping').addEventListener('lostpointercapture', interact);
document.getElementById('shaping').addEventListener('pointerup', interact);
preventScrolling(document.getElementById('shaping'));
window.requestAnimationFrame(drawFrame);
}
2018-09-18 20:17:34 +00:00
function starPath(CanvasKit, X=128, Y=128, R=116) {
let p = new CanvasKit.SkPath();
p.moveTo(X + R, Y);
for (let i = 1; i < 8 ; i + + ) {
let a = 2.6927937 * i;
p.lineTo(X + R * Math.cos(a), Y + R * Math.sin(a));
}
return p;
}
2018-09-20 21:39:31 +00:00
function SkottieExample(CanvasKit, id, jsonStr, bounds) {
2018-09-18 20:17:34 +00:00
if (!CanvasKit || !jsonStr) {
return;
}
2018-09-20 21:39:31 +00:00
const animation = CanvasKit.MakeAnimation(jsonStr);
const duration = animation.duration() * 1000;
const size = animation.size();
let c = document.getElementById(id);
bounds = bounds || {fLeft: 0, fTop: 0, fRight: size.w, fBottom: size.h};
2018-10-30 19:05:04 +00:00
const surface = CanvasKit.MakeCanvasSurface(id);
2018-09-18 20:17:34 +00:00
if (!surface) {
console.log('Could not make surface');
}
const context = CanvasKit.currentContext();
const canvas = surface.getCanvas();
2018-09-20 21:39:31 +00:00
let firstFrame = new Date().getTime();
2018-09-18 20:17:34 +00:00
function drawFrame() {
2018-09-20 21:39:31 +00:00
let now = new Date().getTime();
let seek = ((now - firstFrame) / duration) % 1.0;
2018-09-18 20:17:34 +00:00
CanvasKit.setCurrentContext(context);
2018-09-20 21:39:31 +00:00
animation.seek(seek);
animation.render(canvas, bounds);
2018-09-18 20:17:34 +00:00
canvas.flush();
window.requestAnimationFrame(drawFrame);
}
window.requestAnimationFrame(drawFrame);
//animation.delete();
}
}
document.head.appendChild(s);
})();
< / script >
2018-09-20 20:30:21 +00:00
Lottie files courtesy of the lottiefiles.com community:
2018-10-30 19:05:04 +00:00
[Lego Loader ](https://www.lottiefiles.com/410-lego-loader ),
[I'm thirsty ](https://www.lottiefiles.com/77-im-thirsty ),
2018-09-21 14:14:30 +00:00
[Confetti ](https://www.lottiefiles.com/1370-confetti ),
[Onboarding ](https://www.lottiefiles.com/1134-onboarding-1 )
2018-09-18 20:17:34 +00:00
2018-09-20 20:30:21 +00:00
Test server
-----------
Test your code on our [CanvasKit Fiddle ](https://jsfiddle.skia.org/canvaskit )
2018-09-18 20:17:34 +00:00
2018-09-20 20:30:21 +00:00
Download
--------
2018-12-18 13:28:59 +00:00
Get [CanvasKit on NPM ](https://www.npmjs.com/package/canvaskit-wasm )