fix buffer export

This commit is contained in:
Anton Lavrenov 2024-01-17 19:12:49 -05:00
parent a1660e1ccb
commit 0d502baccd
5 changed files with 57 additions and 22 deletions

View File

@ -3,6 +3,11 @@
All notable changes to this project will be documented in this file.
This project adheres to [Semantic Versioning](http://semver.org/).
### 9.3.1 (2024-01-17)
- Fix Pixelate filter work/fix caching size
- Fix node export when large buffer canvas is used
### 9.3.0 (2023-12-20)
- New attribute `rotateLineVisible` for `Konva.Transformer` to show/hide rotate line

View File

@ -342,7 +342,7 @@ export abstract class Container<
});
this._requestDraw();
}
drawScene(can?: SceneCanvas, top?: Node) {
drawScene(can?: SceneCanvas, top?: Node, bufferCanvas?: SceneCanvas) {
var layer = this.getLayer()!,
canvas = can || (layer && layer.getCanvas()),
context = canvas && canvas.getContext(),
@ -361,7 +361,7 @@ export abstract class Container<
this._drawCachedSceneCanvas(context);
context.restore();
} else {
this._drawChildren('drawScene', canvas, top);
this._drawChildren('drawScene', canvas, top, bufferCanvas);
}
return this;
}
@ -387,7 +387,7 @@ export abstract class Container<
}
return this;
}
_drawChildren(drawMethod, canvas, top) {
_drawChildren(drawMethod, canvas, top, bufferCanvas?) {
var context = canvas && canvas.getContext(),
clipWidth = this.clipWidth(),
clipHeight = this.clipHeight(),
@ -426,7 +426,7 @@ export abstract class Container<
}
this.children?.forEach(function (child) {
child[drawMethod](canvas, top);
child[drawMethod](canvas, top, bufferCanvas);
});
if (hasComposition) {
context.restore();

View File

@ -449,7 +449,7 @@ export abstract class Node<Config extends NodeConfig = NodeConfig> {
return this._cache.has(CANVAS);
}
abstract drawScene(canvas?: Canvas, top?: Node): void;
abstract drawScene(canvas?: Canvas, top?: Node, bufferCanvas?: Canvas): void;
abstract drawHit(canvas?: Canvas, top?: Node): void;
/**
* Return client rectangle {x, y, width, height} of node. This rectangle also include all styling (strokes, shadows, etc).
@ -1932,6 +1932,12 @@ export abstract class Node<Config extends NodeConfig = NodeConfig> {
}),
context = canvas.getContext();
const bufferCanvas = new SceneCanvas({
width: canvas.width,
height: canvas.height,
pixelRatio: canvas.pixelRatio,
});
if (config.imageSmoothingEnabled === false) {
context._context.imageSmoothingEnabled = false;
}
@ -1941,7 +1947,7 @@ export abstract class Node<Config extends NodeConfig = NodeConfig> {
context.translate(-1 * x, -1 * y);
}
this.drawScene(canvas);
this.drawScene(canvas, undefined, bufferCanvas);
context.restore();
return canvas;

View File

@ -469,10 +469,6 @@ export class Shape<
// so they use that method with forced fill
// it probably will be simpler, then copy/paste the code
// buffer canvas is available only inside the stage
if (!this.getStage()) {
return false;
}
// force skip buffer canvas
const perfectDrawEnabled = this.attrs.perfectDrawEnabled ?? true;
if (!perfectDrawEnabled) {
@ -573,7 +569,7 @@ export class Shape<
}
return rect;
}
drawScene(can?: SceneCanvas, top?: Node) {
drawScene(can?: SceneCanvas, top?: Node, bufferCanvas?: SceneCanvas) {
// basically there are 3 drawing modes
// 1 - simple drawing when nothing is cached.
// 2 - when we are caching current
@ -586,7 +582,6 @@ export class Shape<
drawFunc = this.getSceneFunc(),
hasShadow = this.hasShadow(),
stage,
bufferCanvas,
bufferContext;
var skipBuffer = canvas.isCache;
@ -614,8 +609,8 @@ export class Shape<
// if buffer canvas is needed
if (this._useBufferCanvas() && !skipBuffer) {
stage = this.getStage();
bufferCanvas = stage.bufferCanvas;
bufferContext = bufferCanvas.getContext();
const bc = bufferCanvas || stage.bufferCanvas;
bufferContext = bc.getContext();
bufferContext.clear();
bufferContext.save();
bufferContext._applyLineJoin(this);
@ -626,20 +621,14 @@ export class Shape<
drawFunc.call(this, bufferContext, this);
bufferContext.restore();
var ratio = bufferCanvas.pixelRatio;
var ratio = bc.pixelRatio;
if (hasShadow) {
context._applyShadow(this);
}
context._applyOpacity(this);
context._applyGlobalCompositeOperation(this);
context.drawImage(
bufferCanvas._canvas,
0,
0,
bufferCanvas.width / ratio,
bufferCanvas.height / ratio
);
context.drawImage(bc._canvas, 0, 0, bc.width / ratio, bc.height / ratio);
} else {
context._applyLineJoin(this);

View File

@ -12,6 +12,7 @@ import {
compareLayers,
loadImage,
Konva,
compareCanvases,
} from './test-utils';
describe('Shape', function () {
@ -1479,6 +1480,40 @@ describe('Shape', function () {
}
});
it('export when buffer canvas is used should handle scaling correctly', async function () {
var stage = addStage();
var layer = new Konva.Layer();
stage.add(layer);
var group = new Konva.Group();
layer.add(group);
var text = new Konva.Text({
text: 'hello',
fontSize: 300,
fill: 'green',
shadowColor: 'black',
});
group.add(text);
const canvas1 = group.toCanvas({
x: group.x(),
y: group.y(),
width: text.width(),
height: text.height(),
});
text.stroke('transparent');
const canvas2 = group.toCanvas({
x: group.x(),
y: group.y(),
width: text.width(),
height: text.height(),
});
compareCanvases(canvas2, canvas1, 255, 10);
});
// ======================================================
it('optional disable shadow for stroke', function () {
var stage = addStage();