Play-With-MPV

play website video using MPV(support:youtube,bilibili,ddrk; partial support: imomoe,yhdmp(a little part, m3u8 return .jpg, mpv play error)), need powershell ps1 to support browser run mpv, details see github

目前为 2022-04-27 提交的版本。查看 最新版本

// ==UserScript==
// @name                    Play-With-MPV
// @namespace               https://github.com/LuckyPuppy514
// @version                 1.2.0
// @description:zh-CN       通过MPV播放网页上的视频(支持:youtube,bilibili,ddrk;部分支持:imomoe,yhdmp(一小部分,m3u8返回jpg后缀,mpv播放报错)),需要安装powershell脚本以支持浏览器打开mpv,详细说明见github
// @description             play website video using MPV(support:youtube,bilibili,ddrk; partial support: imomoe,yhdmp(a little part, m3u8 return .jpg, mpv play error)), need powershell ps1 to support browser run mpv, details see github
// @homepage                https://github.com/LuckyPuppy514/Play-With-MPV
// @author                  LuckyPuppy514
// @copyright               2022, Grant LuckyPuppy514 (https://github.com/LuckyPuppy514)
// @license                 MIT
// @icon                    
// @match                   *://www.youtube.com/*
// @include                 https://www.youtube.com/watch/*
// @include                 https://www.bilibili.com/bangumi/play/*
// @include                 https://www.bilibili.com/video/*
// @connect                 api.bilibili.com
// @include                 http://www.imomoe.live/player/*
// @include                 https://www.yhdmp.net/vp/*
// @include                 https://ddrk.me/*
// @run-at                  document-end
// @require                 https://cdn.jsdelivr.net/npm/[email protected]/base64.min.js
// @require                 https://cdn.jsdelivr.net/npm/[email protected]/dist/jquery.min.js
// ==/UserScript==

'use strict';

// using for dev
function debug(data) {
    // console.log(data);
    // alert(data);
}

// Play With MPV CSS
const PWM_CSS = `
#play-with-mpv-button {
  width: 50px;
  height: 50px;
  border: 0px;
  border-radius: 50%;
  background-size: 50px;
  overflow: hidden;
  background-size: cover;
  background-image: url();
  background-repeat: no-repeat;
  z-index: 999
}

#play-with-mpv-div {
  position: fixed;
  left: 15px;
  bottom: 15px;
}
`;

const PWM_DIV_ID = "play-with-mpv-div";
const PWM_BUTTON_ID = "play-with-mpv-button";

const STYLE_VISIABLE = "display: block";
const STYLE_INVISIABLE = "display: none";

// support domain
const YOUTUBE = "www.youtube.com";
const BILIBILI = "www.bilibili.com";
const IMOMOE = "www.imomoe.live";
const YHDMP = "www.yhdmp.net";
const DDRK = "ddrk.me";

const BILIBILI_API = 'https://api.bilibili.com'

// playwithmpv protocol
const PWM_PROTOCOL = "PlayWithMPV://";
// split char
const PWM_PT_SPLIT_CHAR = "|";

// video url need play
var currentVideoUrl;

// currentPage info
var currentUrl;
var currentDomain;

var ddrkPlayStatus = 0;

// add play with mpv div
function addPlayWithMPVDiv() {
    let pwmCss = document.createElement("style");
    pwmCss.innerHTML = PWM_CSS.trim();
    document.head.appendChild(pwmCss);

    let pwmButton = document.createElement("button");
    pwmButton.id = PWM_BUTTON_ID;
    // set invisiable
    pwmButton.style = STYLE_INVISIABLE;
    // add event listener
    pwmButton.onclick = function () {
        debug("pwm button click");
        playCurrentVideoWithMPV();
        pauseCurrentVideo();
    }

    let pwmDiv = document.createElement("div");
    pwmDiv.id = PWM_DIV_ID;
    pwmDiv.appendChild(pwmButton);
    document.body.appendChild(pwmDiv);
    debug("add div success");
}

function setVisiable() {
    debug("set visiable: " + currentVideoUrl);
    if (checkVideoUrl(currentVideoUrl)) {
        document.getElementById(PWM_BUTTON_ID).style = STYLE_VISIABLE;
    }
}

// pause current video
function pauseCurrentVideo() {
    debug("pause current video");

    // bilibili/video
    if (currentUrl.indexOf(BILIBILI + "/video") != -1) {
        let playButton = document.getElementsByClassName("bilibili-player-iconfont")[0];
        playButton.click();
        return;
    }

    // youtube or bilibili/bangumi: get video element to pause
    if (currentDomain == YOUTUBE || currentDomain == BILIBILI || currentDomain == DDRK) {
        let videoElement = document.getElementsByTagName("video")[0];
        if (videoElement) {
            videoElement.pause();
        }
        return;
    }

    // yhdmp: key space to pause
    if (currentDomain == YHDMP || currentDomain == IMOMOE) {
        let keySpace = new KeyboardEvent('keydown', { bubbles: true, cancelable: true, keyCode: 32 });
        document.body.dispatchEvent(keySpace);
        return;
    }
}

// play current video with mpv
function playCurrentVideoWithMPV() {
    debug("play current video with mpv");
    if (!checkVideoUrl(currentVideoUrl)) {
        alert("视频链接错误, 请刷新页面或稍后再试: video url invalid");
        return;
    }
    let protocolLink = PWM_PROTOCOL + Base64.encode(
        currentDomain + PWM_PT_SPLIT_CHAR +
        currentVideoUrl + PWM_PT_SPLIT_CHAR +
        document.title
    );

    // bilibili/video pause will cause the page error(need to refresh), open in another page is ok.
    if (currentUrl.indexOf(BILIBILI + "/video") != -1) {
        window.open(protocolLink, "_blank");
    } else {
        window.open(protocolLink, "_self");
    }
}

// check video url valid or not
function checkVideoUrl(videoUrl) {
    let newCurrentUrl = window.location.href;
    if (currentUrl != newCurrentUrl) {
        changePage();
        return false;
    }
    if (YOUTUBE == currentDomain && currentUrl.indexOf("/watch") == -1) {
        debug("not in youtube/watch: " + currentUrl);
        return false;
    }

    if (videoUrl == null || videoUrl == undefined || !videoUrl.startsWith("http")) {
        return false;
    }
    return true;
}

function getCurrentVideoUrl() {
    debug("get current video url: " + currentUrl);
    // youtube
    if (YOUTUBE == currentDomain) {
        getYoutubeVideoUrl();
        return;
    }

    // bilibili
    if (BILIBILI == currentDomain) {
        getBilibiliVideoUrl();
        return;
    }

    // imomoe
    if (IMOMOE == currentDomain) {
        getImomoeVideoUrl()
        return;
    }

    // yhdmp
    if (YHDMP == currentDomain) {
        getYhdmpVideoUrl();
        return;
    }

    // ddrk
    if (DDRK == currentDomain) {
        getDdrkVideoUrl();
        return;
    }
}

function getYoutubeVideoUrl() {
    currentVideoUrl = currentUrl;
    setVisiable();
}

function getBilibiliVideoUrl() {
    // video
    let bvIndex = currentUrl.indexOf('/video/BV');
    if (bvIndex != -1) {
        let bvid = currentUrl.substring(bvIndex + 9, bvIndex + 19);
        debug("bvid: " + bvid);
        getBilibiliVideoUrlByBvid(bvid);
        return;
    }

    // bangumi
    // get bilibili video epid
    let aElement = document.getElementsByClassName('ep-item cursor visited')[0];
    let epid = aElement.getElementsByTagName('a')[0].href;
    epid = epid.substring(epid.indexOf('/ep') + 3);
    epid = epid.substring(0, epid.indexOf('/'));
    debug('epid: ' + epid);
    getBilibiliVideoUrlByEpid(epid);
}

function getImomoeVideoUrl() {
    let videoUrlElement = document.getElementsByTagName('iframe')[2];
    debug(videoUrlElement);
    let videoUrl = videoUrlElement.src;
    let startIndex = videoUrl.indexOf('url=http') + 4;
    let endIndex = videoUrl.indexOf('m3u8') + 4;
    currentVideoUrl = decodeURIComponent(videoUrl.substring(startIndex, endIndex));
    setVisiable();
}

function getYhdmpVideoUrl() {
    let videoUrlElement = document.getElementById('yh_playfram');
    let videoUrl = videoUrlElement.src;
    let startIndex = videoUrl.indexOf('url=http') + 4;
    let endIndex = videoUrl.indexOf('&getplay_url=');
    currentVideoUrl = decodeURIComponent(videoUrl.substring(startIndex, endIndex));
    setVisiable();
}

function getDdrkVideoUrl() {
    // click play to load video element
    if (ddrkPlayStatus == 0) {
        // alert("start play");
        var playButton = document.getElementsByClassName('vjs-big-play-button')[0];
        if (!playButton) {
            debug("ddrk get play button fail");
            return "";
        }
        playButton.click();
        ddrkPlayStatus = 1;
    }

    currentVideoUrl = document.getElementById('vjsp_html5_api').src;
    setVisiable();
}

function getBilibiliVideoUrlByBvid(bvid) {
    $.ajax({
        type: "GET",
        url: BILIBILI_API + "/x/web-interface/view?bvid=" + bvid,
        xhrFields: {
            // add cookie (CORS ignore cookie)
            withCredentials: true
        },
        success: function (res) {
            debug("get acid and cid by bvid result: ");
            debug(res);
            let avid = res.data.aid;
            let cid = res.data.cid;
            let index = currentUrl.indexOf("?p=");
            if (index != -1) {
                let p = currentUrl.substring(index + 3);
                let endIndex = p.indexOf("&");
                if(endIndex != -1){
                    p = p.substring(0, endIndex);
                }
                cid = res.data.pages[p - 1].cid;
            }

            debug("avid: " + avid);
            debug("cid: " + cid);

            let queryBilibiliVideoUrl = "/x/player/playurl?"
                + "qn=120&otype=json&fourk=1&fnver=0&fnval=0"
                + "&avid=" + avid
                + "&cid=" + cid;
            $.ajax({
                type: "GET",
                url: BILIBILI_API + queryBilibiliVideoUrl,
                xhrFields: {
                    // add cookie (CORS ignore cookie)
                    withCredentials: true
                },
                success: function (res) {
                    debug("get video url by bvid result: ");
                    debug(res);
                    currentVideoUrl = res.data.durl[0].url;
                    setVisiable();
                }
            })
        }
    })
}

function getBilibiliVideoUrlByEpid(epid) {
    $.ajax({
        type: "GET",
        url: BILIBILI_API + "/pgc/view/web/season?ep_id=" + epid,
        xhrFields: {
            // add cookie (CORS ignore cookie)
            withCredentials: true
        },
        success: function (res) {
            debug("get acid and cid by epid result: ");
            debug(res);
            var episodes = res.result.episodes;
            var num;
            // get episode num from title
            var playerTitle = document.getElementById('player-title');
            num = playerTitle.innerHTML;
            debug("bilibili player title: " + num);
            if (num.indexOf('PV') != -1 || num.indexOf('OP') != -1 || num.indexOf('ED') != -1) {
                return;
            }

            // only single episode
            if (episodes.length == 1) {
                num = 1;

            } else {
                num = num.replace(/[^0-9]/ig, "");
            }
            if (num.length < 1) {
                return;
            }

            // get avid and cid
            var episode = episodes[num - 1];
            var avid = episode.aid;
            var cid = episode.cid;
            debug("avid: " + avid);
            debug("cid: " + cid);

            let queryBilibiliVideoUrl = "/pgc/player/web/playurl?"
                + "qn=120&otype=json&fourk=1&fnver=0&fnval=0"
                + "&avid=" + avid
                + "&cid=" + cid;
            $.ajax({
                type: "GET",
                url: BILIBILI_API + queryBilibiliVideoUrl,
                xhrFields: {
                    // add cookie (CORS ignore cookie)
                    withCredentials: true
                },
                success: function (res) {
                    debug("get video url by epid result: ");
                    debug(res);
                    currentVideoUrl = res.result.durl[0].url;
                    setVisiable();
                }
            });
        }
    })
}

// init
function init() {
    debug("init ......");
    currentUrl = window.location.href;
    currentDomain = window.location.host;

    // first try to get video url after 1s(wait page load)
    setTimeout(refreshCurrentVideoUrl, 1000);
    // try to refresh video url every 2s(avoid get video url fail)
    setInterval(refreshCurrentVideoUrl, 2000);
    // page change listener
    setInterval(pageChangeListener, 500);
}
function refreshCurrentVideoUrl() {
    debug("refresh current video url: " + currentVideoUrl);
    debug("current url: " + currentUrl);
    if (!checkVideoUrl(currentVideoUrl)) {
        getCurrentVideoUrl();
    }
}
function pageChangeListener() {
    let newCurrentUrl = window.location.href;
    if (currentUrl != newCurrentUrl) {
        changePage(newCurrentUrl);
    }
}
function changePage() {
    debug("page change");
    document.getElementById(PWM_BUTTON_ID).style = STYLE_INVISIABLE;
    currentVideoUrl = "";
    currentUrl = window.location.href;
    currentDomain = window.location.host;
    ddrkPlayStatus = 0;
}

debug("Play With MPV");
addPlayWithMPVDiv();
init();

QingJ © 2025

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