2019-10-07 16:21:27 +00:00
Particles
=========
Skia’ s particle module provides a way to quickly generate large numbers of
drawing primitives with dynamic, animated behavior. Particles can be used to
create effects like fireworks, spark trails, ambient “weather”, and much more.
Nearly all properties and behavior are controlled by scripts written in Skia’ s
custom language, SkSL.
Samples
-------
< style >
#demo canvas {
border: 1px dashed #AAA ;
margin: 2px;
}
figure {
display: inline-block;
margin: 0;
}
figcaption > a {
margin: 2px 10px;
}
< / style >
< div id = demo >
2019-10-15 17:35:04 +00:00
< figure >
< canvas id = trail width = 400 height = 400 > < / canvas >
< figcaption >
2019-10-17 19:36:21 +00:00
Trail (Click and Drag!)
< / figcaption >
< / figure >
< figure >
< canvas id = cube width = 400 height = 400 > < / canvas >
< figcaption >
2020-04-21 15:20:29 +00:00
< a href = "https://particles.skia.org/5515ab65a31eab1ce5840a714d322643"
2019-10-17 19:36:21 +00:00
target=_blank rel=noopener>Cuboid< / a >
2019-10-15 17:35:04 +00:00
< / figcaption >
< / figure >
2019-10-07 16:21:27 +00:00
< figure >
< canvas id = confetti width = 400 height = 400 > < / canvas >
< figcaption >
2020-04-21 15:20:29 +00:00
< a href = "https://particles.skia.org/eb484bdbac5952c0184a7f1d25773746"
2019-10-07 16:21:27 +00:00
target=_blank rel=noopener>Confetti< / a >
< / figcaption >
< / figure >
< figure >
< canvas id = curves width = 400 height = 400 > < / canvas >
< figcaption >
2020-04-21 15:20:29 +00:00
< a href = "https://particles.skia.org/632d713dacfa01d8905ffee98bc46acc"
2019-10-07 16:21:27 +00:00
target=_blank rel=noopener>Curves< / a >
< / figcaption >
< / figure >
2019-10-07 17:37:35 +00:00
< figure >
< canvas id = fireworks width = 400 height = 400 > < / canvas >
< figcaption >
2020-04-21 15:20:29 +00:00
< a href = "https://particles.skia.org/4d2befa962190e14575075d5676b98bf"
2019-10-07 17:37:35 +00:00
target=_blank rel=noopener>Fireworks< / a >
< / figcaption >
< / figure >
< figure >
< canvas id = raincloud width = 400 height = 400 > < / canvas >
< figcaption >
2020-04-21 15:20:29 +00:00
< a href = "https://particles.skia.org/47f49494cb9bdfef2691369428c6d672"
2019-10-07 17:37:35 +00:00
target=_blank rel=noopener>Raincloud< / a >
< / figcaption >
< / figure >
2019-10-07 16:21:27 +00:00
< figure >
< canvas id = text width = 400 height = 400 > < / canvas >
< figcaption >
2020-04-21 15:20:29 +00:00
< a href = "https://particles.skia.org/9c18c154a286e7c5d64192c9d6661ce0"
2019-10-07 16:21:27 +00:00
target=_blank rel=noopener>Text< / a >
< / figcaption >
< / figure >
< / 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-10-07 17:37:35 +00:00
locate_file = 'https://particles.skia.org/static/';
2019-10-07 16:21:27 +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;
}
s.src = locate_file + 'canvaskit.js';
s.onload = () => {
var CanvasKit = null;
CanvasKitInit({
locateFile: (file) => locate_file + file,
2020-05-26 23:10:48 +00:00
}).then((CK) => {
2019-10-07 16:21:27 +00:00
CanvasKit = CK;
2019-10-15 17:35:04 +00:00
TrailExample(CanvasKit, 'trail', trail);
2019-10-07 16:21:27 +00:00
ParticleExample(CanvasKit, 'confetti', confetti, 200, 200);
2019-10-08 15:07:59 +00:00
ParticleExample(CanvasKit, 'curves', curves, 200, 300);
2019-10-17 19:36:21 +00:00
ParticleExample(CanvasKit, 'cube', cube, 200, 200);
2019-10-07 17:37:35 +00:00
ParticleExample(CanvasKit, 'fireworks', fireworks, 200, 300);
ParticleExample(CanvasKit, 'raincloud', raincloud, 200, 100);
2019-10-07 16:21:27 +00:00
ParticleExample(CanvasKit, 'text', text, 75, 250);
});
function ParticleExample(CanvasKit, id, jsonData, cx, cy) {
if (!CanvasKit || !jsonData) {
return;
}
const surface = CanvasKit.MakeCanvasSurface(id);
if (!surface) {
console.error('Could not make surface');
return;
}
const context = CanvasKit.currentContext();
const canvas = surface.getCanvas();
canvas.translate(cx, cy);
const particles = CanvasKit.MakeParticles(JSON.stringify(jsonData));
particles.start(Date.now() / 1000.0, true);
function drawFrame(canvas) {
particles.update(Date.now() / 1000.0);
canvas.clear(CanvasKit.WHITE);
particles.draw(canvas);
surface.requestAnimationFrame(drawFrame);
}
surface.requestAnimationFrame(drawFrame);
}
const confetti ={
"MaxCount": 200,
"Drawable": {
"Type": "SkCircleDrawable",
"Radius": 8
},
"EffectCode": [
"void effectSpawn(inout Effect effect) {",
" effect.lifetime = 2;",
"}",
"",
"void effectUpdate(inout Effect effect) {",
" if (effect.age < 0.25 | | effect . age > 0.75) { effect.rate = 0; }",
" else { effect.rate = 200; }",
"}",
""
],
"Code": [
"void spawn(inout Particle p) {",
" float3 colors[4];",
" colors[0] = float3(0.87, 0.24, 0.11);",
" colors[1] = float3(1, 0.9, 0.2);",
" colors[2] = float3(0.44, 0.73, 0.24);",
" colors[3] = float3(0.38, 0.54, 0.95);",
2020-04-21 15:20:29 +00:00
" int idx = int(rand(p.seed) * 4);",
2019-10-07 16:21:27 +00:00
" p.color.rgb = colors[idx];",
"",
" p.lifetime = (1 - effect.age) * effect.lifetime;",
2020-04-21 15:20:29 +00:00
" p.scale = mix(0.6, 1, rand(p.seed));",
2019-10-07 16:21:27 +00:00
"}",
"",
"void update(inout Particle p) {",
" p.color.a = 1 - p.age;",
"",
2020-04-21 15:20:29 +00:00
" float a = radians(rand(p.seed) * 360);",
2019-10-07 16:21:27 +00:00
" float invAge = 1 - p.age;",
2020-04-21 15:20:29 +00:00
" p.vel = float2(cos(a), sin(a)) * mix(250, 550, rand(p.seed)) * invAge * invAge;",
2019-10-07 16:21:27 +00:00
"}",
""
],
"Bindings": []
};
2019-10-17 19:36:21 +00:00
const cube = {
"MaxCount": 2000,
"Drawable": {
"Type": "SkCircleDrawable",
"Radius": 4
},
"EffectCode": [
"void effectSpawn(inout Effect effect) {",
" effect.lifetime = 2;",
" effect.rate = 200;",
"}",
""
],
"Code": [
"void spawn(inout Particle p) {",
" p.lifetime = 10;",
"}",
"",
"float4x4 rx(float rad) {",
" float c = cos(rad);",
" float s = sin(rad);",
" return float4x4(1, 0, 0, 0,",
" 0, c, -s, 0,",
" 0, s, c, 0,",
" 0, 0, 0, 1);",
"}",
"",
"float4x4 ry(float rad) {",
" float c = cos(rad);",
" float s = sin(rad);",
" return float4x4(c, 0, -s, 0,",
" 0, 1, 0, 0,",
" s, 0, c, 0,",
" 0, 0, 0, 1);",
"}",
"",
"float4x4 rz(float rad) {",
" float c = cos(rad);",
" float s = sin(rad);",
" return float4x4( c, s, 0, 0,",
" -s, c, 0, 0,",
" 0, 0, 1, 0,",
" 0, 0, 0, 1);",
"}",
"",
"void update(inout Particle p) {",
2020-04-21 15:20:29 +00:00
" float3 pos = float3(rand(p.seed), rand(p.seed), rand(p.seed));",
" if (rand(p.seed) < 0.33 ) { " ,
2019-10-17 19:36:21 +00:00
" if (pos.x > 0.5) {",
" pos.x = 1;",
" p.color.rgb = float3(1, 0.2, 0.2);",
" } else {",
" pos.x = 0;",
" p.color.rgb = float3(0.2, 1, 1);",
" }",
2020-04-21 15:20:29 +00:00
" } else if (rand(p.seed) < 0.5 ) { " ,
2019-10-17 19:36:21 +00:00
" if (pos.y > 0.5) {",
" pos.y = 1;",
" p.color.rgb = float3(0.2, 0.2, 1);",
" } else {",
" pos.y = 0;",
" p.color.rgb = float3(1, 1, 0.2);",
" }",
" } else {",
" if (pos.z > 0.5) {",
" pos.z = 1;",
" p.color.rgb = float3(0.2, 1, 0.2);",
" } else {",
" pos.z = 0;",
" p.color.rgb = float3(1, 0.2, 1);",
" }",
" }",
"",
" float s = effect.age * 2 - 1;",
" s = s < 0 ? -s : s ; " ,
"",
" pos = pos * 2 - 1;",
" pos = mix(pos, normalize(pos), s);",
" pos = pos * 100;",
"",
" float age = effect.loop + effect.age;",
" float4x4 mat = rx(age * radians(60))",
" * ry(age * radians(70))",
" * rz(age * radians(80));",
" pos = (mat * float4(pos, 1)).xyz;",
"",
" p.pos.x = pos.x;",
" p.pos.y = pos.y;",
" p.scale = ((pos.z + 50) / 100 + 0.5) / 2;",
"}",
""
],
"Bindings": []
};
2019-10-07 16:21:27 +00:00
const curves = {
"MaxCount": 1000,
"Drawable": {
"Type": "SkCircleDrawable",
"Radius": 2
},
"EffectCode": [
"void effectSpawn(inout Effect effect) {",
2019-10-08 15:07:59 +00:00
" effect.rate = 200;",
" effect.color = float4(1, 0, 0, 1);",
2019-10-07 16:21:27 +00:00
"}",
""
],
"Code": [
"void spawn(inout Particle p) {",
2020-04-21 15:20:29 +00:00
" p.lifetime = 3 + rand(p.seed);",
2019-10-07 16:21:27 +00:00
" p.vel.y = -50;",
"}",
"",
"void update(inout Particle p) {",
2019-10-08 15:07:59 +00:00
" float w = mix(15, 3, p.age);",
2020-04-21 15:20:29 +00:00
" p.pos.x = sin(radians(p.age * 320)) * mix(25, 10, p.age) + mix(-w, w, rand(p.seed));",
" if (rand(p.seed) < 0.5 ) { p . pos . x = -p.pos.x; } " ,
2019-10-08 15:07:59 +00:00
"",
2020-04-21 15:20:29 +00:00
" p.color.g = (mix(75, 220, p.age) + mix(-30, 30, rand(p.seed))) / 255;",
2019-10-07 16:21:27 +00:00
"}",
""
],
2019-10-08 15:07:59 +00:00
"Bindings": []
2019-10-07 16:21:27 +00:00
};
2019-10-07 17:37:35 +00:00
const fireworks = {
"MaxCount": 1000,
"Drawable": {
"Type": "SkCircleDrawable",
"Radius": 1
},
"EffectCode": [
"void effectSpawn(inout Effect effect) {",
" effect.lifetime = 2;",
" effect.rate = 120;",
2020-04-21 15:20:29 +00:00
" float a = radians(mix(-20, 20, rand(effect.seed)) - 90);",
" float s = mix(200, 220, rand(effect.seed));",
2019-10-07 17:37:35 +00:00
" effect.vel.x = cos(a) * s;",
" effect.vel.y = sin(a) * s;",
2020-04-21 15:20:29 +00:00
" effect.color.rgb = float3(rand(effect.seed), rand(effect.seed), rand(effect.seed));",
2019-10-07 17:37:35 +00:00
" effect.pos.x = 0;",
" effect.pos.y = 0;",
"}",
"",
"void effectUpdate(inout Effect effect) {",
" effect.vel.y += dt * 90;",
"}",
"",
"void effectDeath(inout Effect effect) {",
" explode(false);",
"}",
""
],
"Code": [
"void spawn(inout Particle p) {",
" p.lifetime = 0.5;",
2020-04-21 15:20:29 +00:00
" float a = radians(rand(p.seed) * 360);",
" float s = mix(5, 10, rand(p.seed));",
2019-10-07 17:37:35 +00:00
" p.vel.x = cos(a) * s;",
" p.vel.y = sin(a) * s;",
"}",
"",
"void update(inout Particle p) {",
" p.color.a = 1 - p.age;",
"}",
""
],
"Bindings": [
{
"Type": "SkEffectBinding",
"Name": "explode",
"MaxCount": 50,
"Drawable": {
"Type": "SkCircleDrawable",
"Radius": 3
},
"EffectCode": [
"void effectSpawn(inout Effect effect) {",
" effect.burst = 50;",
" effect.lifetime = 2.5;",
"}",
""
],
"Code": [
"void spawn(inout Particle p) {",
2020-04-21 15:20:29 +00:00
" p.lifetime = 2 + rand(p.seed) * 0.5;",
" float a = radians(rand(p.seed) * 360);",
" float s = mix(90, 100, rand(p.seed));",
2019-10-07 17:37:35 +00:00
" p.vel.x = cos(a) * s;",
" p.vel.y = sin(a) * s;",
"}",
"",
"void update(inout Particle p) {",
" p.color.a = 1 - p.age;",
" p.vel.y += dt * 50;",
"}",
""
],
"Bindings": []
}
]
};
2019-10-07 16:21:27 +00:00
const raincloud = {
"MaxCount": 128,
"Drawable": {
"Type": "SkCircleDrawable",
"Radius": 2
},
"EffectCode": [
"void effectSpawn(inout Effect effect) {",
" if (effect.loop == 0) {",
" cloud(true);",
" }",
" effect.color = float4(0.1, 0.1, 1.0, 1.0);",
" effect.rate = 10;",
"}",
""
],
"Code": [
"void spawn(inout Particle p) {",
" p.lifetime = 4;",
2020-04-21 15:20:29 +00:00
" p.pos.x = mix(-50, 50, rand(p.seed));",
2019-10-07 16:21:27 +00:00
" p.vel.y = 50;",
"}",
"",
"bool once(bool cond, inout uint flags, uint flag) {",
" bool result = false;",
" if (cond & & (flags & flag) == 0) {",
" flags |= flag;",
" result = true;",
" }",
" return result;",
"}",
"",
"void update(inout Particle p) {",
" p.vel.y += 20 * dt;",
" if (once(p.pos.y > 150, p.flags, 0x1)) {",
" p.scale = 0;",
" splash(false);",
" }",
"}",
""
],
"Bindings": [
{
"Type": "SkEffectBinding",
"Name": "cloud",
"MaxCount": 60,
"Drawable": {
"Type": "SkCircleDrawable",
"Radius": 16
},
"EffectCode": [
"void effectSpawn(inout Effect effect) {",
2019-10-07 17:37:35 +00:00
" effect.color = float4(0.8, 0.8, 0.8, 1);",
2019-10-07 16:21:27 +00:00
" effect.rate = 30;",
"}",
""
],
"Code": [
2020-04-21 15:20:29 +00:00
"float2 circle(inout uint seed) {",
2019-10-07 16:21:27 +00:00
" float2 xy;",
" do {",
2020-04-21 15:20:29 +00:00
" xy.x = 2 * rand(seed) - 1;",
" xy.y = 2 * rand(seed) - 1;",
2019-10-07 16:21:27 +00:00
" } while (dot(xy, xy) > 1);",
" return xy;",
"}",
"",
"void spawn(inout Particle p) {",
" p.lifetime = 2.5;",
2020-04-21 15:20:29 +00:00
" p.pos = circle(p.seed) * float2(50, 10);",
" p.vel.x = mix(-10, 10, rand(p.seed));",
" p.vel.y = mix(-10, 10, rand(p.seed));",
2019-10-07 16:21:27 +00:00
"}",
"",
"void update(inout Particle p) {",
" p.color.a = 1 - (length(p.pos) / 150);",
"}",
""
],
"Bindings": []
},
{
"Type": "SkEffectBinding",
"Name": "splash",
"MaxCount": 8,
"Drawable": {
"Type": "SkCircleDrawable",
"Radius": 1
},
"EffectCode": [
"void effectSpawn(inout Effect effect) {",
" effect.burst = 8;",
" effect.scale = 1;",
"}",
""
],
"Code": [
"void spawn(inout Particle p) {",
2020-04-21 15:20:29 +00:00
" p.lifetime = rand(p.seed);",
" float a = radians(mix(-80, 80, rand(p.seed)) - 90);",
2019-10-07 16:21:27 +00:00
" p.vel.x = cos(a) * 20;",
" p.vel.y = sin(a) * 20;",
"}",
"",
"void update(inout Particle p) {",
" p.vel.y += dt * 20;",
"}",
""
],
"Bindings": []
}
]
};
const text = {
"MaxCount": 2000,
"Drawable": {
"Type": "SkCircleDrawable",
"Radius": 1
},
"EffectCode": [
"void effectSpawn(inout Effect effect) {",
" effect.rate = 1000;",
"}",
""
],
"Code": [
"void spawn(inout Particle p) {",
2020-04-21 15:20:29 +00:00
" p.lifetime = mix(1, 3, rand(p.seed));",
" float a = radians(mix(250, 290, rand(p.seed)));",
" float s = mix(10, 30, rand(p.seed));",
2019-10-07 16:21:27 +00:00
" p.vel.x = cos(a) * s;",
" p.vel.y = sin(a) * s;",
2020-04-21 15:20:29 +00:00
" p.pos = text(rand(p.seed)).xy;",
2019-10-07 16:21:27 +00:00
"}",
"",
"void update(inout Particle p) {",
" float4 startColor = float4(1, 0.196, 0.078, 1);",
" float4 endColor = float4(1, 0.784, 0.078, 1);",
" p.color = mix(startColor, endColor, p.age);",
"}",
""
],
"Bindings": [
{
"Type": "SkTextBinding",
"Name": "text",
"Text": "SKIA",
"FontSize": 96
}
]
};
2019-10-15 17:35:04 +00:00
function preventScrolling(canvas) {
canvas.addEventListener('touchmove', (e) => {
// Prevents touch events in the canvas from scrolling the canvas.
e.preventDefault();
e.stopPropagation();
});
}
function TrailExample(CanvasKit, id, jsonData) {
if (!CanvasKit || !jsonData) {
return;
}
const surface = CanvasKit.MakeCanvasSurface(id);
if (!surface) {
console.error('Could not make surface');
return;
}
const context = CanvasKit.currentContext();
const canvas = surface.getCanvas();
const particles = CanvasKit.MakeParticles(JSON.stringify(jsonData));
particles.start(Date.now() / 1000.0, true);
function drawFrame(canvas) {
particles.update(Date.now() / 1000.0);
canvas.clear(CanvasKit.WHITE);
particles.draw(canvas);
surface.requestAnimationFrame(drawFrame);
}
surface.requestAnimationFrame(drawFrame);
let interact = (e) => {
particles.setPosition([e.offsetX, e.offsetY]);
particles.setRate(e.pressure * 1000);
};
document.getElementById('trail').addEventListener('pointermove', interact);
document.getElementById('trail').addEventListener('pointerdown', interact);
document.getElementById('trail').addEventListener('pointerup', interact);
preventScrolling(document.getElementById('trail'));
}
const trail = {
"MaxCount": 2000,
"Drawable": {
"Type": "SkCircleDrawable",
"Radius": 4
},
"EffectCode": "",
"Code": [
"void spawn(inout Particle p) {",
2020-04-21 15:20:29 +00:00
" p.lifetime = 2 + rand(p.seed);",
" float a = radians(rand(p.seed) * 360);",
" p.vel = float2(cos(a), sin(a)) * mix(5, 15, rand(p.seed));",
" p.scale = mix(0.25, 0.75, rand(p.seed));",
2019-10-15 17:35:04 +00:00
"}",
"",
"void update(inout Particle p) {",
" p.color.r = p.age;",
" p.color.g = 1 - p.age;",
"}",
""
],
"Bindings": []
};
2019-10-07 16:21:27 +00:00
}
document.head.appendChild(s);
})();
< / script >