experimental Offscreen canvas support

This commit is contained in:
Anton Lavrenov 2020-03-14 21:07:37 -05:00
parent b7b50f22d8
commit 8f18165273
6 changed files with 332 additions and 1402 deletions

View File

@ -8,6 +8,7 @@ This project adheres to [Semantic Versioning](http://semver.org/).
* Add `rotationSnapTolerance` property to `Konva.Transformer`.
* Add `getActiveAnchor()` method to `Konva.Transformer`
* Fix hit for non-closed `Konva.Path`
* Some fixes for experimental Offscreen canvas support inside a worker
## 4.1.6 - 2020-02-25

1430
konva.js

File diff suppressed because it is too large Load Diff

View File

@ -115,8 +115,8 @@ export class Canvas {
return this.height;
}
setSize(width, height) {
this.setWidth(width);
this.setHeight(height);
this.setWidth(width || 0);
this.setHeight(height || 0);
}
/**
* to data url

View File

@ -164,6 +164,9 @@ export class Stage extends Container<BaseLayer> {
}
_checkVisibility() {
if (!this.content) {
return;
}
const style = this.visible() ? '' : 'none';
this.content.style.display = style;
}
@ -339,28 +342,22 @@ export class Stage extends Container<BaseLayer> {
return null;
}
_resizeDOM() {
var width = this.width();
var height = this.height();
if (this.content) {
var width = this.width(),
height = this.height(),
layers = this.getChildren(),
len = layers.length,
n,
layer;
// set content dimensions
this.content.style.width = width + PX;
this.content.style.height = height + PX;
this.bufferCanvas.setSize(width, height);
this.bufferHitCanvas.setSize(width, height);
// set layer dimensions
for (n = 0; n < len; n++) {
layer = layers[n];
layer.setSize({ width, height });
layer.draw();
}
}
this.bufferCanvas.setSize(width, height);
this.bufferHitCanvas.setSize(width, height);
// set layer dimensions
this.children.each(layer => {
layer.setSize({ width, height });
layer.draw();
});
}
add(layer) {
if (arguments.length > 1) {
@ -997,26 +994,36 @@ export class Stage extends Container<BaseLayer> {
this.setPointersPositions(evt);
}
_getContentPosition() {
var rect = this.content.getBoundingClientRect
? this.content.getBoundingClientRect()
: { top: 0, left: 0, width: 1000, height: 1000 };
if (!this.content || !this.content.getBoundingClientRect) {
return {
top: 0,
left: 0,
scaleX: 1,
scaleY: 1
};
}
var rect = this.content.getBoundingClientRect();
return {
top: rect.top,
left: rect.left,
// sometimes clientWidth can be equals to 0
// i saw it in react-konva test, looks like it is because of hidden testing element
scaleX: rect.width / this.content.clientWidth || 1,
scaleY: rect.height / this.content.clientHeight || 1,
scaleY: rect.height / this.content.clientHeight || 1
};
}
_buildDOM() {
// the buffer canvas pixel ratio must be 1 because it is used as an
// intermediate canvas before copying the result onto a scene canvas.
// not setting it to 1 will result in an over compensation
this.bufferCanvas = new SceneCanvas();
this.bufferHitCanvas = new HitCanvas({ pixelRatio: 1 });
this.bufferCanvas = new SceneCanvas({
width: this.width(),
height: this.height()
});
this.bufferHitCanvas = new HitCanvas({
pixelRatio: 1,
width: this.width(),
height: this.height()
});
if (!Konva.isBrowser) {
return;

View File

@ -2,7 +2,7 @@
<html>
<head>
<meta charset="utf-8" />
<title>Konva Canvas Scrolling Demo</title>
<title>Konva Offscreen Canvas Demo</title>
<style>
body {
margin: 0;
@ -13,15 +13,46 @@
</head>
<body>
<canvas id="canvas"></div>
<canvas id="canvas"></canvas>
<script>
var htmlCanvas = document.getElementById("canvas");
var htmlCanvas = document.getElementById('canvas');
htmlCanvas.width = window.innerWidth;
htmlCanvas.height = window.innerHeight;
var offscreen = htmlCanvas.transferControlToOffscreen();
var w = new Worker('./worker.js');
w.onmessage = function(event) {
document.getElementById('result').innerHTML = event.data;
};
w.postMessage({ canvas: offscreen }, [offscreen]);
var EVENTS = [
'mouseenter',
'mousedown',
'mousemove',
'mouseup',
'mouseout',
'touchstart',
'touchmove',
'touchend',
'mouseover',
'wheel',
'contextmenu',
'pointerdown',
'pointermove',
'pointerup',
'pointercancel',
'lostpointercapture'
];
EVENTS.forEach(eventName => {
htmlCanvas.addEventListener(eventName, e => {
w.postMessage({
eventName,
event: {
clientX: e.clientX,
clientY: e.clientY
}
});
});
});
</script>
</body>
</html>

View File

@ -1,33 +1,194 @@
// Can we start Konva inside worker?
importScripts('../../konva.js');
console.log(Konva);
Konva.Util.createCanvasElement = () => {
const canvas = new OffscreenCanvas(100, 100);
const canvas = new OffscreenCanvas(1, 1);
canvas.style = {};
return canvas;
};
Konva.Canvas.prototype.setSize = function(width, height) {
this.setWidth(width || 1);
this.setHeight(height || 1);
};
Konva.Stage.prototype._checkVisibility = function() {};
var stage = new Konva.Stage({
width: 100,
height: 100
width: 200,
height: 200
});
var layer = new Konva.Layer();
stage.add(layer);
var shape = new Konva.Circle({
var topGroup = new Konva.Group();
layer.add(topGroup);
var counter = new Konva.Text({
x: 5,
y: 35
});
topGroup.add(counter);
var button = new Konva.Label({
x: 5,
y: 5,
opacity: 0.75
});
topGroup.add(button);
button.add(
new Konva.Tag({
fill: 'black'
})
);
button.add(
new Konva.Text({
text: 'Push me to add bunnies',
fontFamily: 'Calibri',
fontSize: 18,
padding: 5,
fill: 'white'
})
);
var circle = new Konva.Circle({
x: stage.width() / 2,
y: stage.height() / 2,
radius: 50,
fill: 'red'
radius: 20,
fill: 'red',
draggable: true
});
layer.add(shape);
topGroup.add(circle);
layer.draw();
onmessage = function(evt) {
if (evt.data.canvas) {
var canvas = evt.data.canvas;
stage.setSize({
width: canvas.width,
height: canvas.height
});
const ctx = canvas.getContext('2d');
setInterval(() => {
ctx.clearRect(0, 0, canvas.width, canvas.height);
ctx.drawImage(layer.getCanvas()._canvas, 0, 0);
}, 16);
}
if (evt.data.eventName === 'mouseup') {
Konva.DD._endDragBefore(evt.data.event);
}
if (evt.data.eventName === 'touchend') {
Konva.DD._endDragBefore(evt.data.event);
}
if (evt.data.eventName === 'mousemove') {
Konva.DD._drag(evt.data.event);
}
if (evt.data.eventName === 'touchmove') {
Konva.DD._drag(evt.data.event);
}
if (evt.data.eventName === 'mouseup') {
Konva.DD._endDragAfter(evt.data.event);
}
if (evt.data.eventName === 'touchend') {
Konva.DD._endDragAfter(evt.data.event);
}
if (evt.data.eventName) {
stage['_' + evt.data.eventName](evt.data.event);
}
};
function requestAnimationFrame(cb) {
setTimeout(cb, 16);
}
async function runBunnies() {
var bunnys = [];
var gravity = 0.75;
var startBunnyCount = 100;
var isAdding = false;
var count = 0;
var amount = 10;
const imgBlob = await fetch('./assets/bunny.png').then(r => r.blob());
const img = await createImageBitmap(imgBlob);
button.on('mousedown', function() {
isAdding = true;
});
button.on('mouseup', function() {
isAdding = false;
});
for (var i = 0; i < startBunnyCount; i++) {
var bunny = new Konva.Image({
image: img,
transformsEnabled: 'position',
hitGraphEnabled: false,
x: 10,
y: 10,
listening: false
});
bunny.speedX = Math.random() * 10;
bunny.speedY = Math.random() * 10 - 5;
bunnys.push(bunny);
counter.text('Bunnies number: ' + bunnys.length);
layer.add(bunny);
}
topGroup.moveToTop();
layer.draw();
function update() {
var maxX = stage.width() - 10;
var minX = 0;
var maxY = stage.height() - 10;
var minY = 0;
if (isAdding) {
// add 10 at a time :)
for (var i = 0; i < amount; i++) {
var bunny = new Konva.Image({
image: img,
transformsEnabled: 'position',
x: 0,
y: 0,
listening: false
});
bunny.speedX = Math.random() * 10;
bunny.speedY = Math.random() * 10 - 5;
bunnys.push(bunny);
layer.add(bunny);
counter.text('Bunnies number: ' + bunnys.length);
count++;
}
topGroup.moveToTop();
}
for (var i = 0; i < bunnys.length; i++) {
var bunny = bunnys[i];
bunny.setX(bunny.getX() + bunny.speedX);
bunny.setY(bunny.getY() + bunny.speedY);
bunny.speedY += gravity;
if (bunny.getX() > maxX - img.width) {
bunny.speedX *= -1;
bunny.setX(maxX - img.width);
} else if (bunny.getX() < minX) {
bunny.speedX *= -1;
bunny.setX(minX);
}
if (bunny.getY() > maxY - img.height) {
bunny.speedY *= -0.85;
bunny.setY(maxY - img.height);
if (Math.random() > 0.5) {
bunny.speedY -= Math.random() * 6;
}
} else if (bunny.getY() < minY) {
bunny.speedY = 0;
bunny.setY(minY);
}
}
layer.drawScene();
requestAnimationFrame(update);
}
update();
}
runBunnies();