From 0d502baccd1ab102b43625623a1e76f9fb3eb9ec Mon Sep 17 00:00:00 2001 From: Anton Lavrenov Date: Wed, 17 Jan 2024 19:12:49 -0500 Subject: [PATCH] fix buffer export --- CHANGELOG.md | 5 +++++ src/Container.ts | 8 ++++---- src/Node.ts | 10 ++++++++-- src/Shape.ts | 21 +++++---------------- test/unit/Shape-test.ts | 35 +++++++++++++++++++++++++++++++++++ 5 files changed, 57 insertions(+), 22 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 73d6115b..d750190b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -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 diff --git a/src/Container.ts b/src/Container.ts index 10f99292..167831be 100644 --- a/src/Container.ts +++ b/src/Container.ts @@ -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(); diff --git a/src/Node.ts b/src/Node.ts index 85deed31..0aa9f59d 100644 --- a/src/Node.ts +++ b/src/Node.ts @@ -449,7 +449,7 @@ export abstract class Node { 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 { }), 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 { context.translate(-1 * x, -1 * y); } - this.drawScene(canvas); + this.drawScene(canvas, undefined, bufferCanvas); context.restore(); return canvas; diff --git a/src/Shape.ts b/src/Shape.ts index c327dc12..6f403878 100644 --- a/src/Shape.ts +++ b/src/Shape.ts @@ -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); diff --git a/test/unit/Shape-test.ts b/test/unit/Shape-test.ts index d3035562..0415825f 100644 --- a/test/unit/Shape-test.ts +++ b/test/unit/Shape-test.ts @@ -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();