Play-With-MPV

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

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

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