Donovan P
Donovan P

Reputation: 641

How to copy PIXI's view into another canvas

In theory it should be as easy as :

//targetCanvas is a canvas node
//app is instance of PIXI.Application
targetCanvas.drawImage(app.view, 0, 0);

But, trust me, It doesn't work!

Sometime (yes, I said sometime) it does work if I add some delay before copying the image into the targetCanvas. ... So, I wonder if there is an asynchronous process happening when PIXI is rendering... but I can not find this information in PIXI's documentation.

var loadImage = async function(imgPath) {
    return new Promise((resolve, reject) => {
        var img = document.createElement("img");
        img.onload = ()=> {
            resolve(img);
        }
        img.src = imgPath;
    })
}



var canvasTag = document.querySelector("#canvasVideo");
var canvasCtx = canvasTag.getContext("2d");

var canvasResult = document.querySelector("#canvasResult");
var canvasResultCtx = canvasResult.getContext("2d");

var canvasWidth = canvasTag.width;
var canvasHeight = canvasTag.height;


var canvasWidth = 320;
var canvasHeight = 240;
const app = new PIXI.Application({
    view: document.querySelector("#glCanvas"),
    transparent: true,
    width: canvasWidth,
    height: canvasHeight
});

const container = new PIXI.Container();
const renderer = PIXI.autoDetectRenderer();
app.stage.addChild(container);

(async ()=> {
    var imgTag = await loadImage("");
    canvasCtx.drawImage(imgTag, 0, 0, canvasWidth, canvasHeight);

    var bgSprite = PIXI.Sprite.from(imgTag);

    app.stage.addChild(bgSprite);

    canvasResultCtx.drawImage(document.querySelector("#glCanvas"), 0, 0);
})()
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <script src="https://pixijs.download/release/pixi.js"></script>
    <script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.6.1/jquery.min.js"></script>
    <title>Document</title>
</head>
<body>
    <canvas id="canvasVideo" width="320" height="240" autoplay></canvas>
    <canvas id="glCanvas" width="320" height="240" autoplay></canvas>
    <canvas id="canvasResult" width="320" height="240" autoplay></canvas>
</body>
</html>

canvasVideo (left) is normal canvas with 2d context. glCanvas (middle) is a PIXI's view canvasResult (right) is a normal canvas where a PIXI view should be copied into.

We don't have any problem copying canvas into PIXI. (except the CORS prevents the webGL renders the image correctly) But, as you can see, the third canvas is blank. I wonder if there is any solution to this issue... without using PIXI's ticker or setTimeout?

Upvotes: 1

Views: 488

Answers (2)

Eduardo P&#225;ez Rubio
Eduardo P&#225;ez Rubio

Reputation: 1152

You don't need to use the PIXI ticker or a custom timer (requestAnimationFrame as in your own response) to render to the canvas. There's the possibility to simply call app.render() in a synchronous way after the desired elements are added to the stage.

On top of that, if you want to prevent having a running application that renders on each frame, you may want to add autoStart: false to your Application config.

var loadImage = async function(imgPath) {
    return new Promise((resolve, reject) => {
        var img = document.createElement("img");
        img.onload = ()=> {
            resolve(img);
        }
        img.src = imgPath;
    })
}

var canvasWidth = 320;
var canvasHeight = 240;
const app = new PIXI.Application({
    view: document.querySelector("#glCanvas"),
    transparent: true,
    width: canvasWidth,
    height: canvasHeight,
    autoStart: false,
});

(async () => {
    var imgTag = await loadImage("");
    var bgSprite = PIXI.Sprite.from(imgTag);

    app.stage.addChild(bgSprite);

    app.render();
})()
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <script src="https://pixijs.download/release/pixi.js"></script>
    <title>Document</title>
</head>
<body>
    <canvas id="glCanvas" width="320" height="240"></canvas>
</body>
</html>

Upvotes: 2

Kokodoko
Kokodoko

Reputation: 28128

The ticker is not used for non-animated sprites, but apparently pixi still needs ONE frame to draw everything before you can copy it.

I was able to solve your issue with one requestAnimationFrame

(async ()=> {
    var imgTag = await loadImage("");
    canvasCtx.drawImage(imgTag, 0, 0, canvasWidth, canvasHeight);

    var bgSprite = PIXI.Sprite.from(imgTag);
    app.stage.addChild(bgSprite);

    requestAnimationFrame(() => {
        canvasResultCtx.drawImage(document.querySelector("#glCanvas"), 0, 0);
    });
})()

Upvotes: 2

Related Questions