Play-With-MPV

通过MPV播放网页上的视频(详细安装过程见:https://github.com/LuckyPuppy514/Play-With-MPV)

当前为 2022-05-22 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Play-With-MPV
  3. // @name:en Play-With-MPV
  4. // @name:en-US Play-With-MPV
  5. // @name:zh 使用 MPV 播放
  6. // @name:zh-CN 使用 MPV 播放
  7. // @namespace https://github.com/LuckyPuppy514
  8. // @version 1.4.2
  9. // @commit v1.2.1 新增 powershell 脚本升级提醒功能
  10. // @commit:en-US v1.2.1 add powershell scripts update remind
  11. // @commit v1.2.2 修复 youtube 标题带 | 导致错误脚本升级提醒
  12. // @commit:en-US v1.2.2 fix when youtube title has | cause error scripts update remind
  13. // @commit v1.2.3 修改 imomoe 域名
  14. // @commit:en-US v1.2.3 modify imomoe domain
  15. // @commit v1.3.0 新增域名:www.6dm.cc, www.dmla.cc(第一线路:大部分支持,其他线路:小部分支持)
  16. // @commit v1.3.0 新增域名:www.dm233.me(线路III:大部分支持,其他线路:大部分不支持)
  17. // @commit v1.3.0 代码重构,使用继承方便后续添加网站支持
  18. // @commit v1.4.0 b站bug修复:标题带数字,解析出错,修复并优化了获取视频链接的速度
  19. // @commit v1.4.0 新增对plex支持(本地:*://*/web/index.html*,远程:https://app.plex.tv/desktop/*)
  20. // @commit v1.4.1 修复b站番剧播放目录为列表时,无法获取正确集数的bug
  21. // @commit v1.4.2 修复b站番剧播放的bug
  22. // @description 通过MPV播放网页上的视频(详细安装过程见:https://github.com/LuckyPuppy514/Play-With-MPV)
  23. // @description:en play website video using MPV (setup: https://github.com/LuckyPuppy514/Play-With-MPV)
  24. //
  25. // @support-list 支持列表 - support list
  26. // @support-list 大部分支持 - most :www.youtube.com, www.bilibili.com, ddrk.me
  27. //
  28. // @support-list 番剧推荐:https://www.6dm.cc 第一线路, 支持1080p
  29. // @support-list 部分 支持 - part :www.6dm.cc, www.dmla.cc(第一线路:大部分支持,其他线路:小部分支持), www.dm233.me(线路III:大部分支持,其他线路:大部分不支持)
  30. //
  31. // @support-list 不推荐以下网址, 极少部分支持(大部分解析出来都是yun.66dm.net, 返回.jpg)
  32. // @support-list 小部分支持 - little part :www.dmh8.com, www.yhdmp.net(不推荐,现在很多看不了)
  33. //
  34. // @homepage https://github.com/LuckyPuppy514/Play-With-MPV
  35. // @author LuckyPuppy514
  36. // @copyright 2022, Grant LuckyPuppy514 (https://github.com/LuckyPuppy514)
  37. // @license MIT
  38. // @icon 
  39. // @match *://www.youtube.com/*
  40. // @include https://www.youtube.com/watch/*
  41. // @include https://www.bilibili.com/bangumi/play/*
  42. // @include https://www.bilibili.com/video/*
  43. // @connect api.bilibili.com
  44. // @include https://ddrk.me/*
  45. // @include https://www.6dm.cc/play/*
  46. // @include http://www.dmla.cc/play/*
  47. // @include https://danmu.yhdmjx.com/*
  48. // @include https://www.dm233.me/play/*
  49. // @include http://www.dmh8.com/player/*
  50. // @include https://www.yhdmp.net/vp/*
  51. // @match *://*/web/index.html*
  52. // @include https://app.plex.tv/desktop/*
  53. // @run-at document-end
  54. // @require https://cdn.jsdelivr.net/npm/js-base64@3.6.1/base64.min.js
  55. // @require https://cdn.jsdelivr.net/npm/jquery@3.2.1/dist/jquery.min.js
  56. // ==/UserScript==
  57.  
  58. 'use strict';
  59.  
  60. // using for dev
  61. function debug(data) {
  62. // console.log(data);
  63. // alert(data);
  64. }
  65.  
  66. // playwithmpv.ps1 version
  67. const CURRENT_VERSION = "v1.2.3";
  68.  
  69. // Play With MPV CSS
  70. const PWM_CSS = `
  71. #play-with-mpv-button {
  72. width: 50px;
  73. height: 50px;
  74. border: 0px;
  75. border-radius: 50%;
  76. background-size: 50px;
  77. overflow: hidden;
  78. background-size: cover;
  79. background-image: url();
  80. background-repeat: no-repeat;
  81. cursor: pointer;
  82. z-index: 999
  83. }
  84.  
  85. #play-with-mpv-div {
  86. position: fixed;
  87. left: 15px;
  88. bottom: 15px;
  89. }
  90. `;
  91.  
  92. const PWM_DIV_ID = "play-with-mpv-div";
  93. const PWM_BUTTON_ID = "play-with-mpv-button";
  94.  
  95. const STYLE_VISIABLE = "display: block";
  96. const STYLE_INVISIABLE = "display: none";
  97.  
  98. // add play with mpv div
  99. function addPlayWithMPVDiv() {
  100. let pwmCss = document.createElement("style");
  101. pwmCss.innerHTML = PWM_CSS.trim();
  102. document.head.appendChild(pwmCss);
  103.  
  104. let pwmButton = document.createElement("button");
  105. pwmButton.id = PWM_BUTTON_ID;
  106. pwmButton.style = STYLE_INVISIABLE;
  107. // add event listener
  108. pwmButton.onclick = function () {
  109. debug("pwm button click");
  110. handler.playCurrentVideoWithMPV();
  111. handler.pauseCurrentVideo();
  112. }
  113.  
  114. let pwmDiv = document.createElement("div");
  115. pwmDiv.id = PWM_DIV_ID;
  116. pwmDiv.appendChild(pwmButton);
  117. document.body.appendChild(pwmDiv);
  118. debug("add div success");
  119. }
  120.  
  121. function setVisiable() {
  122. debug("set visiable: " + currentVideoUrl);
  123. if (checkVideoUrl(currentVideoUrl)) {
  124. document.getElementById(PWM_BUTTON_ID).style = STYLE_VISIABLE;
  125. }
  126. }
  127. function setInvisiable() {
  128. document.getElementById(PWM_BUTTON_ID).style = STYLE_INVISIABLE;
  129. }
  130.  
  131. // support domain
  132. const YOUTUBE = "www.youtube.com";
  133. const BILIBILI = "www.bilibili.com";
  134. const DDRK = "ddrk.me";
  135. const DM6CC = "www.6dm.cc";
  136. const DMLACC = "www.dmla.cc";
  137. const YHDMJX = "danmu.yhdmjx.com";
  138. const DM233 = "www.dm233.me";
  139. const DMH8 = "www.dmh8.com";
  140. const YHDMP = "www.yhdmp.net";
  141. const PLEX_LOCAL = "/web/index.html";
  142. const PLEX = "app.plex.tv";
  143.  
  144. // api
  145. const BILIBILI_API = 'https://api.bilibili.com'
  146.  
  147. // playwithmpv protocol
  148. const PWM_PROTOCOL = "PlayWithMPV://";
  149. // split char
  150. const PWM_PT_SPLIT_CHAR = "|";
  151.  
  152. // video url need play
  153. var currentVideoUrl;
  154. // currentPage info
  155. var currentUrl;
  156. var currentDomain;
  157.  
  158. var handler;
  159. var ddrkPlayStatus;
  160.  
  161. class Handler {
  162. getProtocolLink() {
  163. return PWM_PROTOCOL + Base64.encode(
  164. currentDomain + PWM_PT_SPLIT_CHAR +
  165. currentVideoUrl + PWM_PT_SPLIT_CHAR +
  166. document.title.replaceAll(PWM_PT_SPLIT_CHAR, " ") + PWM_PT_SPLIT_CHAR +
  167. CURRENT_VERSION
  168. );
  169. }
  170.  
  171. getCurrentVideoUrl() { }
  172. playCurrentVideoWithMPV() { }
  173. pauseCurrentVideo() { }
  174.  
  175. addPlayWithMPVButton() {
  176. addPlayWithMPVDiv();
  177. }
  178. addTimer() {
  179. // first try to get video url after 600ms(wait page load)
  180. setTimeout(refreshCurrentVideoUrl, 600);
  181. // try to refresh video url every 2s(avoid get video url fail)
  182. setInterval(refreshCurrentVideoUrl, 2000);
  183. // page change listener
  184. setInterval(pageChangeListener, 500);
  185.  
  186. function refreshCurrentVideoUrl() {
  187. debug("refresh current video url: " + currentVideoUrl);
  188. debug("current url: " + currentUrl);
  189. if (!checkVideoUrl(currentVideoUrl)) {
  190. handler.getCurrentVideoUrl();
  191. }
  192. }
  193. function pageChangeListener() {
  194. let newCurrentUrl = window.location.href;
  195. if (currentUrl != newCurrentUrl) {
  196. setInvisiable();
  197. init();
  198. }
  199. }
  200. }
  201. }
  202.  
  203. class YoutubeHandler extends Handler {
  204. getCurrentVideoUrl() {
  205. currentVideoUrl = currentUrl;
  206. setVisiable();
  207. }
  208. playCurrentVideoWithMPV() {
  209. window.open(this.getProtocolLink(), "_self");
  210. }
  211. pauseCurrentVideo() {
  212. document.getElementsByTagName("video")[0].pause();
  213. }
  214. }
  215.  
  216. class BilibiliHandler extends Handler {
  217. getCurrentVideoUrl() {
  218. // video
  219. let bvIndex = currentUrl.indexOf('/video/BV');
  220. if (bvIndex != -1) {
  221. let bvid = currentUrl.substring(bvIndex + 9, bvIndex + 19);
  222. debug("bvid: " + bvid);
  223. this.getBilibiliVideoUrlByBvid(bvid);
  224. return;
  225. }
  226.  
  227. // bangumi
  228. // get bilibili video epid
  229. let aElement = document.getElementsByClassName('ep-item cursor visited')[0];
  230. if (!aElement) {
  231. aElement = document.getElementsByClassName('ep-item cursor')[0];
  232. }
  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.  
  238. let eno = document.getElementsByClassName("ep-list-progress")[0];
  239. if(!eno){
  240. return;
  241. }
  242. eno = eno.innerHTML;
  243. eno = eno.substring(0, eno.indexOf('/'));
  244. debug('eno: ' + eno);
  245. this.getBilibiliVideoUrlByEpid(epid, eno);
  246. }
  247.  
  248. playCurrentVideoWithMPV() {
  249. if (currentUrl.indexOf(BILIBILI + "/video") != -1) {
  250. window.open(this.getProtocolLink(), "_blank");
  251. } else {
  252. window.open(this.getProtocolLink(), "_self");
  253. }
  254. }
  255.  
  256. pauseCurrentVideo() {
  257. if (currentUrl.indexOf(BILIBILI + "/video") != -1) {
  258. document.getElementsByClassName("bilibili-player-iconfont")[0].click();
  259. } else {
  260. document.getElementsByTagName("video")[0].pause();
  261. }
  262.  
  263. }
  264.  
  265. getBilibiliVideoUrlByBvid(bvid) {
  266. $.ajax({
  267. type: "GET",
  268. url: BILIBILI_API + "/x/web-interface/view?bvid=" + bvid,
  269. xhrFields: {
  270. // add cookie (CORS ignore cookie)
  271. withCredentials: true
  272. },
  273. success: function (res) {
  274. debug("get acid and cid by bvid result: ");
  275. debug(res);
  276. let avid = res.data.aid;
  277. let cid = res.data.cid;
  278. let index = currentUrl.indexOf("?p=");
  279. if (index != -1) {
  280. let p = currentUrl.substring(index + 3);
  281. let endIndex = p.indexOf("&");
  282. if (endIndex != -1) {
  283. p = p.substring(0, endIndex);
  284. }
  285. cid = res.data.pages[p - 1].cid;
  286. }
  287.  
  288. debug("avid: " + avid);
  289. debug("cid: " + cid);
  290.  
  291. let queryBilibiliVideoUrl = "/x/player/playurl?"
  292. + "qn=120&otype=json&fourk=1&fnver=0&fnval=0"
  293. + "&avid=" + avid
  294. + "&cid=" + cid;
  295. $.ajax({
  296. type: "GET",
  297. url: BILIBILI_API + queryBilibiliVideoUrl,
  298. xhrFields: {
  299. // add cookie (CORS ignore cookie)
  300. withCredentials: true
  301. },
  302. success: function (res) {
  303. debug("get video url by bvid result: ");
  304. debug(res);
  305. currentVideoUrl = res.data.durl[0].url;
  306. setVisiable();
  307. }
  308. })
  309. }
  310. })
  311. }
  312. getBilibiliVideoUrlByEpid(epid, eno) {
  313. if (!epid || !eno) {
  314. return;
  315. }
  316.  
  317. $.ajax({
  318. type: "GET",
  319. url: BILIBILI_API + "/pgc/view/web/season?ep_id=" + epid,
  320. xhrFields: {
  321. // add cookie (CORS ignore cookie)
  322. withCredentials: true
  323. },
  324. success: function (res) {
  325. debug("get acid and cid by epid result: ");
  326. debug(res);
  327. var episodes = res.result.episodes;
  328. if (eno.indexOf('PV') != -1 || eno.indexOf('OP') != -1 || eno.indexOf('ED') != -1) {
  329. return;
  330. }
  331.  
  332. // get avid and cid
  333. var episode = episodes[eno - 1];
  334. var avid = episode.aid;
  335. var cid = episode.cid;
  336. debug("avid: " + avid);
  337. debug("cid: " + cid);
  338.  
  339. let queryBilibiliVideoUrl = "/pgc/player/web/playurl?"
  340. + "qn=120&otype=json&fourk=1&fnver=0&fnval=0"
  341. + "&avid=" + avid
  342. + "&cid=" + cid;
  343. $.ajax({
  344. type: "GET",
  345. url: BILIBILI_API + queryBilibiliVideoUrl,
  346. xhrFields: {
  347. // add cookie (CORS ignore cookie)
  348. withCredentials: true
  349. },
  350. success: function (res) {
  351. debug("get video url by epid result: ");
  352. debug(res);
  353. currentVideoUrl = res.result.durl[0].url;
  354. setVisiable();
  355. }
  356. });
  357. }
  358. })
  359. }
  360. }
  361.  
  362. class DdrkHandler extends Handler {
  363. getCurrentVideoUrl() {
  364. // click play to load video element
  365. if (ddrkPlayStatus == 0) {
  366. // alert("start play");
  367. var playButton = document.getElementsByClassName('vjs-big-play-button')[0];
  368. if (!playButton) {
  369. debug("ddrk get play button fail");
  370. return;
  371. }
  372. playButton.click();
  373. ddrkPlayStatus = 1;
  374. }
  375.  
  376. currentVideoUrl = document.getElementById('vjsp_html5_api').src;
  377. setVisiable();
  378. }
  379. playCurrentVideoWithMPV() {
  380. window.open(this.getProtocolLink(), "_self");
  381. }
  382. pauseCurrentVideo() {
  383. document.getElementsByTagName("video")[0].pause();
  384. }
  385. }
  386.  
  387. class Dm6ccHandler extends Handler {
  388. constructor() {
  389. super();
  390. window.addEventListener('message', function (event) {
  391. currentVideoUrl = event.data;
  392. setVisiable();
  393. window.removeEventListener("message", () => { });
  394. }, false);
  395. }
  396. getCurrentVideoUrl() { }
  397. playCurrentVideoWithMPV() {
  398. window.open(this.getProtocolLink(), "_self");
  399. }
  400. pauseCurrentVideo() {
  401. document.getElementsByTagName("iframe")[2].contentWindow.postMessage("pause", "https://" + YHDMJX);
  402. }
  403. }
  404.  
  405. class DmlaccHandler extends Handler {
  406. constructor() {
  407. super();
  408. window.addEventListener('message', function (event) {
  409. currentVideoUrl = event.data;
  410. setVisiable();
  411. window.removeEventListener("message", () => { });
  412. }, false);
  413. }
  414. getCurrentVideoUrl() { }
  415. playCurrentVideoWithMPV() {
  416. window.open(this.getProtocolLink(), "_self");
  417. }
  418. pauseCurrentVideo() {
  419. document.getElementsByTagName("iframe")[2].contentWindow.postMessage("pause", "https://" + YHDMJX);
  420. }
  421. }
  422.  
  423. class YhdmjxHandler extends Handler {
  424. constructor() {
  425. super();
  426. // listen to pause
  427. window.addEventListener("message", function (event) {
  428. if (event.data == "pause") {
  429. document.getElementsByTagName('video')[0].pause();
  430. }
  431. }, false);
  432. }
  433. getCurrentVideoUrl() {
  434. // send current video url to parent iframe
  435. currentVideoUrl = document.getElementsByTagName('video')[0].src;
  436. if (checkVideoUrl(currentVideoUrl)) {
  437. window.parent.postMessage(currentVideoUrl, "*");
  438. }
  439. }
  440. playCurrentVideoWithMPV() {
  441. window.open(this.getProtocolLink(), "_self");
  442. }
  443. pauseCurrentVideo() { }
  444. }
  445.  
  446. class Dm233Handler extends Handler {
  447. constructor() {
  448. super();
  449. this.videoElement = null;
  450. }
  451. getCurrentVideoUrl() {
  452. let iframe = document.getElementById('id_main_playiframe');
  453. this.videoElement = iframe.contentWindow.document.getElementsByTagName("video")[0];
  454.  
  455. let videoUrl = this.videoElement.src;
  456. if (videoUrl.startsWith("blob:")) {
  457. videoUrl = iframe.src;
  458. let startIndex = videoUrl.indexOf('url=http') + 4;
  459. let endIndex = videoUrl.indexOf('&getplay_url=');
  460. videoUrl = decodeURIComponent(videoUrl.substring(startIndex, endIndex));
  461. }
  462.  
  463. currentVideoUrl = videoUrl;
  464. setVisiable();
  465. }
  466. playCurrentVideoWithMPV() {
  467. window.open(this.getProtocolLink(), "_self");
  468. }
  469. pauseCurrentVideo() {
  470. this.videoElement.pause();
  471. }
  472. }
  473.  
  474. class Dmh8Handler extends Handler {
  475. constructor() {
  476. super();
  477. window.addEventListener('message', function (event) {
  478. currentVideoUrl = event.data;
  479. setVisiable();
  480. window.removeEventListener("message", () => { });
  481. }, false);
  482. }
  483. getCurrentVideoUrl() {
  484. let iframe = document.getElementsByTagName('iframe')[2];
  485. let videoUrl = iframe.src;
  486. let startIndex = videoUrl.indexOf('url=http') + 4;
  487. let endIndex = videoUrl.indexOf('m3u8') + 4;
  488. currentVideoUrl = decodeURIComponent(videoUrl.substring(startIndex, endIndex));
  489. setVisiable();
  490. }
  491. playCurrentVideoWithMPV() {
  492. window.open(this.getProtocolLink(), "_self");
  493. }
  494. pauseCurrentVideo() { }
  495. }
  496.  
  497. class YhdmpHandler extends Handler {
  498. constructor() {
  499. super();
  500. this.videoElement = null;
  501. }
  502. getCurrentVideoUrl() {
  503. let iframe = document.getElementById('yh_playfram');
  504. this.videoElement = iframe.contentWindow.document.getElementsByTagName("video")[0];
  505.  
  506. let videoUrl = iframe.src;
  507. let startIndex = videoUrl.indexOf('url=http') + 4;
  508. let endIndex = videoUrl.indexOf('&getplay_url=');
  509. currentVideoUrl = decodeURIComponent(videoUrl.substring(startIndex, endIndex));
  510. setVisiable();
  511. }
  512. playCurrentVideoWithMPV() {
  513. window.open(this.getProtocolLink(), "_self");
  514. }
  515. pauseCurrentVideo() {
  516. this.videoElement.pause();
  517. }
  518. }
  519.  
  520. class PlexHandler extends Handler {
  521. constructor() {
  522. super();
  523. }
  524. getCurrentVideoUrl() {
  525. let as = document.getElementsByTagName('a');
  526. let pwmButton = document.getElementById(PWM_BUTTON_ID);
  527. if (pwmButton) {
  528. return;
  529. }
  530. for (let a of as) {
  531. if (a) {
  532. if (a.target == "downloadFileFrame") {
  533. pwmButton = document.createElement("button");
  534. pwmButton.id = PWM_BUTTON_ID;
  535. // add event listener
  536. pwmButton.onclick = function () {
  537. debug("pwm button click");
  538. handler.playCurrentVideoWithMPV();
  539. }
  540. pwmButton.innerText = "Play With MPV";
  541. pwmButton.className = a.className;
  542. a.before(pwmButton);
  543. currentVideoUrl = a.href;
  544. break;
  545. }
  546. }
  547. }
  548. }
  549. playCurrentVideoWithMPV() {
  550. window.open(this.getProtocolLink(), "_self");
  551. }
  552. addPlayWithMPVButton() {
  553. }
  554. addTimer() {
  555. setInterval(this.getCurrentVideoUrl, 500);
  556. }
  557. }
  558.  
  559. // check video url valid or not
  560. function checkVideoUrl(videoUrl) {
  561. if (videoUrl == null || videoUrl == undefined || !videoUrl.startsWith("http")) {
  562. return false;
  563. }
  564. if (YOUTUBE == currentDomain && currentUrl.indexOf("/watch") == -1) {
  565. debug("not in youtube/watch: " + currentUrl);
  566. return false;
  567. }
  568. // yun.66dm.net return m3u8 as .jpg, mpv play fail
  569. if (videoUrl.indexOf("yun.66dm.net") != -1) {
  570. debug("yun.66dm.net: " + videoUrl);
  571. return false;
  572. }
  573. return true;
  574. }
  575.  
  576. // init
  577. function init() {
  578. debug("init ......");
  579. currentUrl = window.location.href;
  580. currentDomain = window.location.host;
  581. currentVideoUrl = "";
  582. ddrkPlayStatus = 0;
  583.  
  584. debug("start create handler");
  585. switch (currentDomain) {
  586. case YOUTUBE:
  587. handler = new YoutubeHandler();
  588. break;
  589. case BILIBILI:
  590. handler = new BilibiliHandler();
  591. break;
  592. case DDRK:
  593. handler = new DdrkHandler();
  594. break;
  595. case DM6CC:
  596. handler = new Dm6ccHandler();
  597. break;
  598. case DMLACC:
  599. handler = new DmlaccHandler();
  600. break;
  601. case YHDMJX:
  602. handler = new YhdmjxHandler();
  603. break;
  604. case DM233:
  605. handler = new Dm233Handler();
  606. break;
  607. case DMH8:
  608. handler = new Dmh8Handler();
  609. break;
  610. case YHDMP:
  611. handler = new YhdmpHandler();
  612. break;
  613. default:
  614. if (currentUrl.indexOf(PLEX_LOCAL) != -1 || currentDomain == PLEX) {
  615. handler = new PlexHandler();
  616. }
  617. }
  618.  
  619. debug("start add button");
  620. handler.addPlayWithMPVButton();
  621. debug("start add timer");
  622. handler.addTimer();
  623. }
  624.  
  625.  
  626. debug("Play With MPV");
  627. init();