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-26 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Play-With-MPV
  3. // @namespace https://github.com/LuckyPuppy514
  4. // @version 1.1.0
  5. // @description:zh-CN 通过MPV播放网页上的视频(支持:youtube,bilibili,ddrk;部分支持:imomoe,yhdmp(一小部分,m3u8返回jpg后缀,mpv播放报错)),需要安装powershell脚本以支持浏览器打开mpv,详细说明见github
  6. // @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
  7. // @homepage https://github.com/LuckyPuppy514/Play-With-MPV
  8. // @author LuckyPuppy514
  9. // @copyright 2022, Grant LuckyPuppy514 (https://github.com/LuckyPuppy514)
  10. // @license MIT
  11. // @icon 
  12. // @match *://www.youtube.com/*
  13. // @include https://www.youtube.com/watch/*
  14. // @include https://www.bilibili.com/bangumi/play/*
  15. // @include https://www.bilibili.com/video/*
  16. // @connect api.bilibili.com
  17. // @include http://www.imomoe.live/player/*
  18. // @include https://www.yhdmp.net/vp/*
  19. // @include https://ddrk.me/*
  20. // @run-at document-end
  21. // @require https://cdn.jsdelivr.net/npm/js-base64@3.6.1/base64.min.js
  22. // @grant GM_xmlhttpRequest
  23. // ==/UserScript==
  24.  
  25. 'use strict';
  26.  
  27. // using for dev
  28. function debug(data) {
  29. // console.log(data);
  30. // alert(data);
  31. }
  32.  
  33. // Play With MPV CSS
  34. const PWM_CSS = `
  35. #play-with-mpv-button {
  36. width: 50px;
  37. height: 50px;
  38. border: 0px;
  39. border-radius: 50%;
  40. background-size: 50px;
  41. overflow: hidden;
  42. background-size: cover;
  43. background-image: url();
  44. background-repeat: no-repeat;
  45. z-index: 999
  46. }
  47.  
  48. #play-with-mpv-div {
  49. position: fixed;
  50. left: 15px;
  51. bottom: 15px;
  52. }
  53. `;
  54.  
  55. const PWM_DIV_ID = "play-with-mpv-div";
  56. const PWM_BUTTON_ID = "play-with-mpv-button";
  57.  
  58. const STYLE_VISIABLE = "display: block";
  59. const STYLE_INVISIABLE = "display: none";
  60.  
  61. // support domain
  62. const YOUTUBE = "www.youtube.com";
  63. const BILIBILI = "www.bilibili.com";
  64. const IMOMOE = "www.imomoe.live";
  65. const YHDMP = "www.yhdmp.net";
  66. const DDRK = "ddrk.me";
  67.  
  68. const BILIBILI_API = 'https://api.bilibili.com'
  69.  
  70. // playwithmpv protocol
  71. const PWM_PROTOCOL = "PlayWithMPV://";
  72. // split char
  73. const PWM_PT_SPLIT_CHAR = "|";
  74.  
  75. // video url need play
  76. var currentVideoUrl;
  77.  
  78. // currentPage info
  79. var currentUrl;
  80. var currentDomain;
  81.  
  82. var ddrkPlayStatus = 0;
  83.  
  84. // add play with mpv div
  85. function addPlayWithMPVDiv() {
  86. let pwmCss = document.createElement("style");
  87. pwmCss.innerHTML = PWM_CSS.trim();
  88. document.head.appendChild(pwmCss);
  89.  
  90. let pwmButton = document.createElement("button");
  91. pwmButton.id = PWM_BUTTON_ID;
  92. // set invisiable
  93. pwmButton.style = STYLE_INVISIABLE;
  94. // add event listener
  95. pwmButton.onclick = function () {
  96. debug("pwm button click");
  97. playCurrentVideoWithMPV();
  98. pauseCurrentVideo();
  99. }
  100.  
  101. let pwmDiv = document.createElement("div");
  102. pwmDiv.id = PWM_DIV_ID;
  103. pwmDiv.appendChild(pwmButton);
  104. document.body.appendChild(pwmDiv);
  105. debug("add div success");
  106. }
  107.  
  108. function setVisiable() {
  109. debug("set visiable: " + currentVideoUrl);
  110. if (checkVideoUrl(currentVideoUrl)) {
  111. document.getElementById(PWM_BUTTON_ID).style = STYLE_VISIABLE;
  112. }
  113. }
  114.  
  115. // pause current video
  116. function pauseCurrentVideo() {
  117. debug("pause current video");
  118.  
  119. // bilibili/video
  120. if (currentUrl.indexOf(BILIBILI + "/video") != -1) {
  121. let playButton = document.getElementsByClassName("bilibili-player-iconfont")[0];
  122. playButton.click();
  123. return;
  124. }
  125.  
  126. // youtube or bilibili/bangumi: get video element to pause
  127. if (currentDomain == YOUTUBE || currentDomain == BILIBILI || currentDomain == DDRK) {
  128. let videoElement = document.getElementsByTagName("video")[0];
  129. if (videoElement) {
  130. videoElement.pause();
  131. }
  132. return;
  133. }
  134.  
  135. // yhdmp: key space to pause
  136. if (currentDomain == YHDMP || currentDomain == IMOMOE) {
  137. let keySpace = new KeyboardEvent('keydown', { bubbles: true, cancelable: true, keyCode: 32 });
  138. document.body.dispatchEvent(keySpace);
  139. return;
  140. }
  141. }
  142.  
  143. // play current video with mpv
  144. function playCurrentVideoWithMPV() {
  145. debug("play current video with mpv");
  146. if (!checkVideoUrl(currentVideoUrl)) {
  147. alert("视频链接错误, 请刷新页面或稍后再试: video url invalid");
  148. return;
  149. }
  150. let protocolLink = PWM_PROTOCOL + Base64.encode(
  151. currentDomain + PWM_PT_SPLIT_CHAR +
  152. currentVideoUrl + PWM_PT_SPLIT_CHAR +
  153. document.title
  154. );
  155.  
  156. // bilibili/video pause will cause the page error(need to refresh), open in another page is ok.
  157. if (currentUrl.indexOf(BILIBILI + "/video") != -1) {
  158. window.open(protocolLink, "_blank");
  159. } else {
  160. window.open(protocolLink, "_self");
  161. }
  162. }
  163.  
  164. // check video url valid or not
  165. function checkVideoUrl(videoUrl) {
  166. let newCurrentUrl = window.location.href;
  167. if (currentUrl != newCurrentUrl) {
  168. changePage();
  169. return false;
  170. }
  171. if (YOUTUBE == currentDomain && currentUrl.indexOf("/watch") == -1) {
  172. debug("not in youtube/watch: " + currentUrl);
  173. return false;
  174. }
  175.  
  176. if (videoUrl == null || videoUrl == undefined || !videoUrl.startsWith("http")) {
  177. return false;
  178. }
  179. return true;
  180. }
  181.  
  182. function getCurrentVideoUrl() {
  183. debug("get current video url: " + currentUrl);
  184. // youtube
  185. if (YOUTUBE == currentDomain) {
  186. getYoutubeVideoUrl();
  187. return;
  188. }
  189.  
  190. // bilibili
  191. if (BILIBILI == currentDomain) {
  192. getBilibiliVideoUrl();
  193. return;
  194. }
  195.  
  196. // imomoe
  197. if (IMOMOE == currentDomain) {
  198. getImomoeVideoUrl()
  199. return;
  200. }
  201.  
  202. // yhdmp
  203. if (YHDMP == currentDomain) {
  204. getYhdmpVideoUrl();
  205. return;
  206. }
  207.  
  208. // ddrk
  209. if (DDRK == currentDomain) {
  210. getDdrkVideoUrl();
  211. return;
  212. }
  213. }
  214.  
  215. function getYoutubeVideoUrl() {
  216. currentVideoUrl = currentUrl;
  217. setVisiable();
  218. }
  219.  
  220. function getBilibiliVideoUrl() {
  221. // video
  222. let bvIndex = currentUrl.indexOf('/video/BV');
  223. if (bvIndex != -1) {
  224. let bvid = currentUrl.substring(bvIndex + 9, bvIndex + 19);
  225. debug("bvid: " + bvid);
  226. getBilibiliVideoUrlByBvid(bvid);
  227. return;
  228. }
  229.  
  230. // bangumi
  231. // get bilibili video epid
  232. let aElement = document.getElementsByClassName('ep-item cursor visited')[0];
  233. let epid = aElement.getElementsByTagName('a')[0].href;
  234. epid = epid.substring(epid.indexOf('/ep') + 3);
  235. epid = epid.substring(0, epid.indexOf('/'));
  236. debug('epid: ' + epid);
  237. getBilibiliVideoUrlByEpid(epid);
  238. }
  239.  
  240. function getImomoeVideoUrl() {
  241. let videoUrlElement = document.getElementsByTagName('iframe')[2];
  242. debug(videoUrlElement);
  243. let videoUrl = videoUrlElement.src;
  244. let startIndex = videoUrl.indexOf('url=http') + 4;
  245. let endIndex = videoUrl.indexOf('m3u8') + 4;
  246. currentVideoUrl = decodeURIComponent(videoUrl.substring(startIndex, endIndex));
  247. setVisiable();
  248. }
  249.  
  250. function getYhdmpVideoUrl() {
  251. let videoUrlElement = document.getElementById('yh_playfram');
  252. let videoUrl = videoUrlElement.src;
  253. let startIndex = videoUrl.indexOf('url=http') + 4;
  254. let endIndex = videoUrl.indexOf('&getplay_url=');
  255. currentVideoUrl = decodeURIComponent(videoUrl.substring(startIndex, endIndex));
  256. setVisiable();
  257. }
  258.  
  259. function getDdrkVideoUrl() {
  260. // click play to load video element
  261. if (ddrkPlayStatus == 0) {
  262. // alert("start play");
  263. var playButton = document.getElementsByClassName('vjs-big-play-button')[0];
  264. if (!playButton) {
  265. debug("ddrk get play button fail");
  266. return "";
  267. }
  268. playButton.click();
  269. ddrkPlayStatus = 1;
  270. }
  271.  
  272. currentVideoUrl = document.getElementById('vjsp_html5_api').src;
  273. setVisiable();
  274. }
  275.  
  276. function getBilibiliVideoUrlByBvid(bvid) {
  277. GM_xmlhttpRequest({
  278. method: "get",
  279. url: BILIBILI_API + "/x/web-interface/view?bvid=" + bvid,
  280. onload: (res) => {
  281. debug("get acid and cid by bvid result: ");
  282. debug(res);
  283. let json = JSON.parse(res.response);
  284. let avid = json.data.aid;
  285. let cid = json.data.cid;
  286. let index = currentUrl.indexOf("?p=");
  287. if (index != -1) {
  288. let p = currentUrl.substring(index + 3, currentUrl.length);
  289. cid = json.data.pages[p - 1].cid;
  290. }
  291.  
  292. debug("avid: " + avid);
  293. debug("cid: " + cid);
  294.  
  295. let queryBilibiliVideoUrl = "/x/player/playurl?"
  296. + "qn=120&otype=json&fourk=1&fnver=0&fnval=0"
  297. + "&avid=" + avid
  298. + "&cid=" + cid;
  299. GM_xmlhttpRequest({
  300. method: "get",
  301. url: BILIBILI_API + queryBilibiliVideoUrl,
  302. onload: (res) => {
  303. debug("get video url by bvid result: ");
  304. debug(res);
  305. let json = JSON.parse(res.response);
  306. currentVideoUrl = json.data.durl[0].url;
  307. setVisiable();
  308. }
  309. })
  310. }
  311. })
  312. }
  313.  
  314. function getBilibiliVideoUrlByEpid(epid) {
  315. GM_xmlhttpRequest({
  316. method: "get",
  317. url: BILIBILI_API + "/pgc/view/web/season?ep_id=" + epid,
  318. onload: (res) => {
  319. debug("get acid and cid by epid result: ");
  320. debug(res);
  321. let json = JSON.parse(res.response);
  322. var episodes = json.result.episodes;
  323. var num;
  324. // get episode num from title
  325. var playerTitle = document.getElementById('player-title');
  326. num = playerTitle.innerHTML;
  327. debug("bilibili player title: " + num);
  328. if (num.indexOf('PV') != -1 || num.indexOf('OP') != -1 || num.indexOf('ED') != -1) {
  329. return;
  330. }
  331.  
  332. // only single episode
  333. if (episodes.length == 1) {
  334. num = 1;
  335.  
  336. } else {
  337. num = num.replace(/[^0-9]/ig, "");
  338. }
  339. if (num.length < 1) {
  340. return;
  341. }
  342.  
  343. // get avid and cid
  344. var episode = episodes[num - 1];
  345. var avid = episode.aid;
  346. var cid = episode.cid;
  347. debug("avid: " + avid);
  348. debug("cid: " + cid);
  349.  
  350. let queryBilibiliVideoUrl = "/pgc/player/web/playurl?"
  351. + "qn=120&otype=json&fourk=1&fnver=0&fnval=0"
  352. + "&avid=" + avid
  353. + "&cid=" + cid;
  354. GM_xmlhttpRequest({
  355. method: "get",
  356. url: BILIBILI_API + queryBilibiliVideoUrl,
  357. onload: (res) => {
  358. debug("get video url by epid result: ");
  359. debug(res);
  360. let json = JSON.parse(res.response);
  361. currentVideoUrl = json.result.durl[0].url;
  362. setVisiable();
  363. }
  364. });
  365. }
  366. })
  367. }
  368.  
  369. // init
  370. function init() {
  371. debug("init ......");
  372. currentUrl = window.location.href;
  373. currentDomain = window.location.host;
  374.  
  375. // first try to get video url after 1s(wait page load)
  376. setTimeout(refreshCurrentVideoUrl, 1000);
  377. // try to refresh video url every 2s(avoid get video url fail)
  378. setInterval(refreshCurrentVideoUrl, 2000);
  379. // page change listener
  380. setInterval(pageChangeListener, 500);
  381. }
  382. function refreshCurrentVideoUrl() {
  383. debug("refresh current video url: " + currentVideoUrl);
  384. debug("current url: " + currentUrl);
  385. if (!checkVideoUrl(currentVideoUrl)) {
  386. getCurrentVideoUrl();
  387. }
  388. }
  389. function pageChangeListener() {
  390. let newCurrentUrl = window.location.href;
  391. if (currentUrl != newCurrentUrl) {
  392. changePage(newCurrentUrl);
  393. }
  394. }
  395. function changePage() {
  396. debug("page change");
  397. document.getElementById(PWM_BUTTON_ID).style = STYLE_INVISIABLE;
  398. currentVideoUrl = "";
  399. currentUrl = window.location.href;
  400. currentDomain = window.location.host;
  401. ddrkPlayStatus = 0;
  402. }
  403.  
  404. debug("Play With MPV");
  405. addPlayWithMPVDiv();
  406. init();
  407.  
  408.