Согласование разрешения холста с разрешением видео - наложение

У меня есть холст и видео, определенные в div:

<div id="videos">
    <canvas id="my-canvas"></canvas>
    <video id="remote-video" autoplay></video>
</div>

Вот CSS:

#my-canvas {
    display: block;
    height: 100%;
    max-height: 100%;
    max-width: 100%;
    object-fit: cover;
    /* no letterboxing */
    opacity: 1;
    z-index: 2;
    position: absolute;
    -moz-transform: rotateY(180deg);
    -ms-transform: rotateY(180deg);
    -o-transform: rotateY(180deg);
    -webkit-transform: rotateY(180deg);
    transform: rotateY(180deg);
    transition: opacity 1s;
    width: 100%;
}

#remote-video {
    display: block;
    height: 100%;
    max-height: 100%;
    max-width: 100%;
    object-fit: cover;
    /* no letterboxing */
    opacity: 0;
    -moz-transform: rotateY(180deg);
    -ms-transform: rotateY(180deg);
    -o-transform: rotateY(180deg);
    -webkit-transform: rotateY(180deg);
    transform: rotateY(180deg);
    transition: opacity 1s;
    width: 100%;
}

Когда я рисую холст, разрешение, кажется, не соответствует разрешению видео. Вот код JavaScript:

var my_canvas = (function() {

var options;
var video;
var canvas;
var context;
var renderTimer;

function initVideoStream() {
    if (options.videoElement) {
        video = document.getElementById(options.videoElement);
        video.setAttribute('width', options.width);
        video.setAttribute('height', options.height);

        initCanvas();
    } else {
        options.onNotSupported();
    }
}

function initCanvas() {
    canvas = document.createElement("canvas");
    canvas.setAttribute('width', options.width);
    canvas.setAttribute('height', options.height);

    context = canvas.getContext('2d');

    startCapture();
}

function startCapture() {
    renderTimer = setInterval(function() {
        try {
            context.drawImage(video, 0, 0, video.width, video.height);
            options.onFrame(canvas);
        } catch (e) {
            // TODO
        }
    }, Math.round(1000 / options.fps));
}

function stopCapture() {
    if (renderTimer) {
        clearInterval(renderTimer);
    }

    options.onClear();
}

return {
    init: function(captureOptions) {
        var doNothing = function(){};

        options = captureOptions || {};

        options.fps = options.fps || 30;
        options.width = options.width || 640;
        options.height = options.height || 480;
        options.videoElement = options.videoElement || null;

        options.onSuccess = options.onSuccess || doNothing;
        options.onError = options.onError || doNothing;
        options.onNotSupported = options.onNotSupported || doNothing;
        options.onFrame = options.onFrame || doNothing;
        options.onClear = options.onClear || doNothing;

        initVideoStream();
    },

    start: startCapture,

    stop: stopCapture
};
})();

Замечания:

1) options.videoElement = document.getElementById ('my-canvas');

2) Я играл с разными значениями для options.width и height. Это не исправляет.

Как полностью наложить холст на видео?

1 ответ

Решение

Этот код изменения размера решил проблему:

    function startCapture() {
        window.requestAnimFrame = (function() {
            return window.requestAnimationFrame || window.webkitRequestAnimationFrame || window.mozRequestAnimationFrame || window.oRequestAnimationFrame || window.msRequestAnimationFrame || function( /* function */ callback, /* DOMElement */ element) {
                window.setTimeout(callback, 1000 / options.fps);
            };
        })();

        animate();
    }

    function stopCapture() {
        requestAnimFrame = null;
        options.onClear();
    }

    function animate() {
        if (requestAnimFrame != null) {
            requestAnimFrame(animate);
            render();
        }
    }

    function render() {
        try {
            var video_width = video.offsetWidth;
            var video_height = video.offsetHeight;
            var ratio = video_width / video_height;
            var target_width;
            var target_height;
            var y_of_video = 0;
            var x_of_video = 0
            if (video_width > video_height) {
                target_width = canvas.width;
                target_height = canvas.width / ratio;
                y_of_video = (canvas.height - target_height) / 2;
            } else {
                target_width = canvas.height;
                target_height = canvas.height * ratio;
                x_of_video = (canvas.width - target_width) / 2;
            }
            context.drawImage(video, x_of_video, y_of_video, target_width, target_height);
            options.onFrame(canvas);
        } catch (e) {
            console.log("Failed to render canvas");
        }
    }

Примечание. Ответ отредактирован в соответствии с предложением 350D.

Другие вопросы по тегам