Play-With-MPV

通过MPV播放网页上的视频(支持:youtube,bilibili,ddrk;部分支持:imomoe,yhdmp(一小部分,m3u8返回jpg后缀,mpv播放报错)),需要安装powershell脚本以支持浏览器打开mpv,详细说明见github

当前为 2022-04-28 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name                    Play-With-MPV
// @namespace               https://github.com/LuckyPuppy514
// @version                 1.2.2
// @commit                  v1.2.1 新增 powershell 脚本升级提醒功能
// @commit:en-US            v1.2.1 add powershell scripts update remind
// @commit                  v1.2.2 修复 youtube 标题带 | 导致错误脚本升级提醒
// @commit:en-US            v1.2.2 fix when youtube title has | cause error scripts update remind
// @description             通过MPV播放网页上的视频(支持:youtube,bilibili,ddrk;部分支持:imomoe,yhdmp(一小部分,m3u8返回jpg后缀,mpv播放报错)),需要安装powershell脚本以支持浏览器打开mpv,详细说明见github
// @description:en-US       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);
}

const CURRENT_VERSION = "v1.2.2";

// 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.replace("|", " ") + PWM_PT_SPLIT_CHAR +
        CURRENT_VERSION
    );

    // 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();