哔哩哔哩画中画弹幕

哔哩哔哩画中画支持显示弹幕

目前为 2022-07-11 提交的版本。查看 最新版本

// ==UserScript==
// @name         哔哩哔哩画中画弹幕
// @namespace    qwq0
// @version      0.7
// @description  哔哩哔哩画中画支持显示弹幕
// @author       QwQ~
// @match        https://www.bilibili.com/video/*
// @match        https://www.bilibili.com/medialist/play/*
// @match        https://www.bilibili.com/bangumi/play/*
// @icon         
// @grant        none
// ==/UserScript==

setTimeout(function() {
    'use strict';
    var videoHolder = document.getElementsByClassName("bilibili-player-video")[0] || document.getElementsByClassName("bpx-player-video-wrap")[0];
    var video = videoHolder.children[0];

    var width = video.videoWidth;
    var height = video.videoHeight;

    var canvas = document.createElement("canvas");
    var context = canvas.getContext("2d");
    var canvasWidth = canvas.width = Math.floor(width / 2);
    var canvasHeight = canvas.height = Math.floor(height / 2);
    var nVideo = document.createElement("video");

    var danmuList = [];
    var danmuLine = [];
    var danmuCount = 0;

    var danmuHolder = document.getElementsByClassName("bilibili-player-video-danmaku")[0] || document.getElementsByClassName("bpx-player-row-dm-wrap")[0];
    function addDanmu(text, color)
    {
        danmuCount++;
        var lineNum = 0;
        for(var i=0;i < 8 && danmuLine[lineNum] + 3 >= danmuCount;i++)
        {
            lineNum = Math.floor(Math.random() * 11);
        }
        danmuLine[lineNum] = danmuCount;
        danmuList.push({text: text, color: (color ? color : "rgb(255, 255, 255)"), x: canvasWidth, y: lineNum * 36 });
    }
    var danmuObserver = new MutationObserver(e=>{
        e.forEach(o=>{
            // console.log("danmu(all)", o);
            if(o.type == "childList")
            {
                o.addedNodes.forEach(ele =>{
                    if(ele.innerText)
                    {
                        let text = ele.innerText;
                        addDanmu(text, ele.style.color);
                        console.log("danmu(it)", ele.style.color, text);
                    }
                    else if(ele.textContent)
                    {
                        let text = ele.textContent;
                        addDanmu(text, o.target.style.color);
                        console.log("danmu(ct)", o.target.color, text);
                    }
                });
            }
        });
    });
    danmuObserver.observe(danmuHolder, { childList: true, subtree: true });
    setInterval(()=>{
        var nowDanmuHolder = document.getElementsByClassName("bilibili-player-video-danmaku")[0] || document.getElementsByClassName("bpx-player-row-dm-wrap")[0];
        if(nowDanmuHolder != danmuHolder)
        {
            danmuHolder = nowDanmuHolder;
            danmuObserver.disconnect();
            width = video.videoWidth;
            height = video.videoHeight;
            canvasWidth = canvas.width = Math.floor(width / 2);
            canvasHeight = canvas.height = Math.floor(height / 2);
            nVideo.srcObject = canvas.captureStream(50);
            setTimeout(() => nVideo.play(), 3500);
            danmuObserver.observe(danmuHolder, { childList: true, subtree: true });
        }
    }, 3000);

    var lastTime = 0;
    function draw(nowTime)
    {
        var timeInterval = nowTime - lastTime;
        lastTime = nowTime;
        context.globalAlpha = 1;
        context.drawImage(video, 0, 0, width, height, 0, 0, width/2, height/2);
        context.font = '36px SimHei,"Microsoft JhengHei",Arial,Helvetica,sans-serif';
        context.textBaseline = "top";
        context.fillStyle = "rgb(255, 255, 255)";
        context.shadowColor = "rgb(0, 0, 0)";
        context.globalAlpha = 0.8;
        danmuList.forEach((o, i)=>{
            context.fillStyle = o.color;
            context.shadowBlur = 5;
            context.fillText(o.text, o.x, o.y);
            o.x -= timeInterval * 0.2;
            if(o.x < -1000)
            {
                danmuList.splice(i, 1);
            }
        });
        requestAnimationFrame(draw);
    }
    requestAnimationFrame(draw);
    console.log("[哔哩哔哩画中画弹幕]", "已加载")
    console.log("[哔哩哔哩画中画弹幕]", "视频分辨率", width, height);
    console.log("[哔哩哔哩画中画弹幕]", "视频分辨率", canvasWidth, canvasHeight);
    nVideo.srcObject = canvas.captureStream(60);
    video.addEventListener("enterpictureinpicture",() =>{
        nVideo.requestPictureInPicture();
        nVideo.play();
    });
}, 3500);

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址