// ==UserScript== // @name Play-With-MPV // @namespace https://github.com/LuckyPuppy514 // @version 1.1.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/js-base64@3.6.1/base64.min.js // @grant GM_xmlhttpRequest // @downloadURL none // ==/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) { GM_xmlhttpRequest({ method: "get", url: BILIBILI_API + "/x/web-interface/view?bvid=" + bvid, onload: (res) => { debug("get acid and cid by bvid result: "); debug(res); let json = JSON.parse(res.response); let avid = json.data.aid; let cid = json.data.cid; let index = currentUrl.indexOf("?p="); if (index != -1) { let p = currentUrl.substring(index + 3, currentUrl.length); cid = json.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; GM_xmlhttpRequest({ method: "get", url: BILIBILI_API + queryBilibiliVideoUrl, onload: (res) => { debug("get video url by bvid result: "); debug(res); let json = JSON.parse(res.response); currentVideoUrl = json.data.durl[0].url; setVisiable(); } }) } }) } function getBilibiliVideoUrlByEpid(epid) { GM_xmlhttpRequest({ method: "get", url: BILIBILI_API + "/pgc/view/web/season?ep_id=" + epid, onload: (res) => { debug("get acid and cid by epid result: "); debug(res); let json = JSON.parse(res.response); var episodes = json.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; GM_xmlhttpRequest({ method: "get", url: BILIBILI_API + queryBilibiliVideoUrl, onload: (res) => { debug("get video url by epid result: "); debug(res); let json = JSON.parse(res.response); currentVideoUrl = json.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();