Play-With-MPV

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

当前为 2022-08-16 提交的版本,查看 最新版本

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