Play-With-MPV

使用 mpv 播放网页中的视频,并支持 potplayer 及自定义播放器

当前为 2023-09-03 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Play-With-MPV
  3. // @name:zh 使用 MPV 播放
  4. // @namespace https://github.com/LuckyPuppy514
  5. // @version 3.8.9
  6. // @author LuckyPuppy514
  7. // @copyright 2023, Grant LuckyPuppy514 (https://github.com/LuckyPuppy514)
  8. // @license MIT
  9. // @description 使用 mpv 播放网页中的视频,并支持 potplayer 及自定义播放器
  10. // @homepage https://github.com/LuckyPuppy514/Play-With-MPV
  11. // @icon https://www.lckp.top/gh/LuckyPuppy514/pic-bed/common/mpv.png
  12. // @match https://www.bilibili.com/bangumi/play/*
  13. // @match https://www.bilibili.com/video/*
  14. // @match https://www.bilibili.com/festival/*
  15. // @match https://www.bilibili.com/list/*
  16. // @match https://live.bilibili.com/*
  17. // @match https://www.ixigua.com/*
  18. // @match https://yun.nxflv.com/?url=*
  19. // @match https://ddys.art/*
  20. // @match https://ddys.pro/*
  21. // @include *://*.libvio.*
  22. // @match https://*.chinaeast2.cloudapp.chinacloudapi.cn/*?url=*
  23. // @match https://*.cfnode1.xyz/*?url=*
  24. // @match https://www.nivod.tv/*
  25. // @match https://www.pkmkv.com/py/*
  26. // @match https://www.pkmkv.com/addons/dplayer/?url=*
  27. // @match https://www.btnull.org/py/*
  28. // @match https://www.btnull.to/py/*
  29. // @match https://www.btnull.nu/py/*
  30. // @match https://www.btnull.in/py/*
  31. // @include *://www.*dm.com/play/*
  32. // @match *://www.dmlaa.com/play/*
  33. // @match *://www.qdmsh.com/play/*
  34. // @match *://www.ntdm8.com/play/*
  35. // @match https://danmu.yhdmjx.com/*?url=*
  36. // @match https://dick.xfani.com/watch/*
  37. // @match https://dick.xfani.com/addons/dp/player/*
  38. // @match https://player.moedot.net/player/*
  39. // @match https://m3.moedot.net/muiplayer/?url=*
  40. // @match https://www.mgnacg.com/bangumi/*
  41. // @match https://play.mknacg.top:8585/*
  42. // @match https://www.omofun.top/index.php/vod/play/id/*
  43. // @match https://*.omofun.top/?url=*
  44. // @match https://spdcat.net/vodplay/*
  45. // @match https://spdcat.net/addons/dp/player/*
  46. // @match http://www.dm88.me/player/*
  47. // @match https://jianghu.live2008.com/*?url=*
  48. // @match https://www.kk151.com/play/*
  49. // @match https://jx.m3u8.tv/jiexi/?url=*
  50. // @match https://jx.wolongzywcdn.com:65/m3u8.php?url=*
  51. // @match https://www.m3u8.tv.cdn.8old.cn/jx.php?url=*
  52. // @match https://jx.wujinkk.com/dplayer/?url=*
  53. // @match https://www.ikdmjx.com/?url=*
  54. // @match https://hls.kuaibofang.com/?url=*
  55. // @match https://jx.jxbdzyw.com/m3u8/?url=*
  56. // @match https://hdzyk.com/?m=*
  57. // @match https://1080zyk1.com/?m=*
  58. // @match https://1080zyk2.com/?m=*
  59. // @match https://1080zyk3.com/?m=*
  60. // @match https://1080zyk4.com/?m=*
  61. // @match https://1080zyk5.com/?m=*
  62. // @match https://vip.zykbf.com/?url=*
  63. // @match https://*.yzzy-tv1.com/*
  64. // @match https://*.yzzy-tv-cdn.com/*
  65. // @match https://www.bdys10.com/*
  66. // @match https://www.haitu.tv/*
  67. // @include *://*alist*
  68. // @include *://*:5244*
  69. // @match *://*/*.mp4
  70. // @match *://*/*.mkv
  71. // @match *://*/*.flv
  72. // @match https://www.dora-family.com/Resource:TV
  73. // @match https://www.olehdtv.com/*
  74. // @match *://tkznp.com/vodplay/*
  75. // @match *://www.tkznp.com/vodplay/*
  76. // @match *://www.tkznp1.com/vodplay/*
  77. // @match *://www.tkznp2.com/vodplay/*
  78. // @match *://www.tkznp3.com/vodplay/*
  79. // @match *://www.tkznp4.com/vodplay/*
  80. // @match *://www.tkznp5.com/vodplay/*
  81. // @match *://www.tkznp6.com/vodplay/*
  82. // @match https://vip.ckllk.com/?url=*
  83. // @match https://www.hdmoli.com/*
  84. // @match https://play.qwertwe.top/xplay/?url=*
  85. // @match https://www.anfuns.cc/play/*
  86. // @match https://www.anfuns.cc/vapi/*
  87. // @match https://www.youtube.com/*
  88. // @match https://odysee.com/*
  89. // @match https://rumble.com/*
  90. // @match https://www.bitchute.com/*
  91. // @match https://ani.gamer.com.tw/animeVideo.php?sn=*
  92. // @match https://hanime1.me/watch?v=*
  93. // @match https://jable.tv/videos/*
  94. // @match https://ok.ru/*
  95. // @match https://tver.jp/*
  96. // @match https://www.lckp.top/play-with-mpv/index.html
  97. // @match https://www.douyin.com/
  98. // @match https://www.douyin.com/video/*
  99. // @match https://www.douyin.com/discover?modal_id=*
  100. // @match https://www.mengfan.tv/play/*
  101. // @match https://video1.beijcloud.com/player/?url=*
  102. // @match https://www.tucao.cam/play/*
  103. // @match https://mypikpak.com/drive/*
  104. // @match https://www.icourse163.org/learn/*
  105. // @match https://www.iole.tv/*
  106. // @match https://www.zhihu.com/zvideo/*
  107. // @match *://www.susudm8.com/*
  108. // @match *://susudyy.com/*
  109. // @match *://buding3.com/*
  110. // @match *://buding6.com/*
  111. // @match *://v2.shenjw.com:*/wap.php?url=*
  112. // @match *://u88.xigua88ok.com:*/wap.php?url=*
  113. // @match *://test3.gqyy8.com:*/f/aliplayer.php?url=*
  114. // @match *://v.mksec.cn/*
  115. // @include *://*dsh*.com/*
  116. // @match https://www.twitch.tv/*
  117. // @match https://jojo.bdys.top/watch/*
  118. // @match https://www.agemys.org/play/*
  119. // @include https://vip.sp-flv.com:*?url=*
  120. // @match https://anime.girigirilove.com/*
  121. // @connect api.bilibili.com
  122. // @connect api.live.bilibili.com
  123. // @require https://unpkg.com/jquery@3.2.1/dist/jquery.min.js
  124. // @require https://unpkg.com/md5@2.3.0/dist/md5.min.js
  125. // @grant GM_setValue
  126. // @grant GM_getValue
  127. // @run-at document-end
  128. // ==/UserScript==
  129.  
  130. 'use strict';
  131.  
  132. const INFO = `
  133.  
  134. ▶️🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽◀️
  135. ▶️ ◀️
  136. ▶️ Play-With-MPV ◀️
  137. ▶️ ◀️
  138. ▶️ https://github.com/LuckyPuppy514/Play-With-MPV ◀️
  139. ▶️ ◀️
  140. ▶️ © 2023 LuckyPuppy514 ◀️
  141. ▶️ ◀️
  142. ▶️🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼◀️
  143.  
  144. `;
  145. // gm key
  146. const KEY = {
  147. config: "config"
  148. }
  149. // 默认配置
  150. const DEFAULT_CONFIG = {
  151. player: "mpv",
  152. mpv: {
  153. path: "",
  154. regVersion: "20230514",
  155. },
  156. potplayer: {
  157. path: "",
  158. regVersion: "20230514",
  159. },
  160. proxy: "",
  161. bestQuality: "2160p",
  162. bilibiliCodecs: 12,
  163. playAuto: 0,
  164. closeAuto: 0,
  165. syncStartTime: 0,
  166. subtitlePrefer: "zh-Hans",
  167. customplayer: {
  168. name: "customplayer",
  169. path: "",
  170. params: {
  171. videoUrl: 'iina://weblink?url=${EvideoUrl}',
  172. audioUrl: '',
  173. subtitleUrl: '',
  174. title: '',
  175. startTime: '',
  176. referer: '',
  177. origin: '&mpv_http-header-fields=origin%3A%20${Eorigin}',
  178. proxy: '',
  179. other: ''
  180. }
  181. },
  182. version: "20230902"
  183. };
  184. var currentConfig;
  185. // 视频链接匹配正则
  186. const VIDEO_URL_REGEX = /https?:\/\/(?![^"^']*http)[^"^']+(\.|%2e)(m3u8|m3u|mp4|mkv|flv|avi)(\?[\w&=-]+|)/g;
  187. // 父子页面方法名
  188. const METHOD = {
  189. pause: "PAUSE",
  190. report: "REPORT"
  191. };
  192. // 时间 ms
  193. const TIME = {
  194. out: 3000,
  195. toast: 3500,
  196. refresh: 600,
  197. reportInterval: 600,
  198. pauseInterval: 2000,
  199. showButton: 5000,
  200. }
  201. // 尝试次数
  202. var tryTime = 0;
  203. const TRY_TIME = {
  204. maxPause: 5,
  205. maxParse: 8
  206. };
  207. // 播放器配置
  208. const PLAYER = {
  209. mpv: {
  210. name: "mpv",
  211. params: {
  212. videoUrl: 'mpv://"${videoUrl}"',
  213. audioUrl: ' --audio-file="${audioUrl}"',
  214. subtitleUrl: ' --sub-file="${subtitleUrl}"',
  215. title: ' --force-media-title="${title}"',
  216. startTime: ' --start=${startTime}',
  217. referer: ' --http-header-fields="referer: ${referer}"',
  218. origin: ' --http-header-fields="origin: ${origin}"',
  219. proxy: ' --http-proxy=${proxy} --ytdl-raw-options=proxy=[${proxy}]',
  220. other: ' ${other}'
  221. }
  222. },
  223. potplayer: {
  224. name: "potplayer",
  225. params: {
  226. videoUrl: 'potplayer://${videoUrl} /current',
  227. subtitleUrl: ' /sub="${subtitleUrl}"',
  228. title: ' /title="${title}"',
  229. startTime: ' /seek=${startTime}',
  230. referer: ' /referer="${referer}"',
  231. origin: ' /headers="origin: ${origin}"',
  232. proxy: ' /user_agent="${proxy}"'
  233. }
  234. },
  235. customplayer: {
  236. name: "customplayer",
  237. params: undefined
  238. }
  239. }
  240. // 页面信息
  241. var page = {
  242. host: undefined,
  243. url: undefined,
  244. isFullScreen: false,
  245. };
  246. // 处理器
  247. var handler;
  248. // 前缀
  249. const PREFIX = "pwm";
  250. // 组件 id
  251. const ID = {
  252. loadingDiv: `${PREFIX}-loading-div`,
  253. toastDiv: `${PREFIX}-toast-div`,
  254. buttonDiv: `${PREFIX}-button-div`,
  255. infoButton: `${PREFIX}-info-button`,
  256. infoDiv: `${PREFIX}-info-div`,
  257. infoTable: `${PREFIX}-info-table`,
  258. mpvPlayButton: `${PREFIX}-mpv-play-button`,
  259. potplayerPlayButton: `${PREFIX}-potplayer-play-button`,
  260. customplayerPlayButton: `${PREFIX}-customplayer-play-button`,
  261. settingButton: `${PREFIX}-setting-button`,
  262. settingDiv: `${PREFIX}-setting-div`,
  263. settingTable: `${PREFIX}-setting-table`,
  264. playerRadio: `${PREFIX}-player-radio`,
  265. softwarePathInput: `${PREFIX}-software-path-input`,
  266. proxyInput: `${PREFIX}-proxy-input`,
  267. bestQualityRadio: `${PREFIX}-best-quality-radio`,
  268. bilibiliCodecsRadio: `${PREFIX}-bilibili-codecs-radio`,
  269. saveButton: `${PREFIX}-save-button`,
  270. downloadButton: `${PREFIX}-download-button`,
  271. deleteButton: `${PREFIX}-delete-button`,
  272. playAutoInput: `${PREFIX}-play-auto-input`,
  273. closeAutoInput: `${PREFIX}-close-auto-input`,
  274. syncStartTimeInput: `${PREFIX}-sync-start-time-input`,
  275. syncStartTimeSpan: `${PREFIX}-sync-start-time-span`,
  276. infoDiv: `${PREFIX}-info-div`,
  277. infoTable: `${PREFIX}-info-table`,
  278. subtitlePreferRadio: `${PREFIX}-subtitle-prefer-radio`,
  279. customplayerSettingButton: `${PREFIX}-customplayer-setting-button`,
  280. customplayerSettingTable: `${PREFIX}-customplayer-setting-table`,
  281. videoUrlParamInput: `${PREFIX}-video-url-param-input`,
  282. audioUrlParamInput: `${PREFIX}-audio-url-param-input`,
  283. subtitleUrlParamInput: `${PREFIX}-subtitle-url-param-input`,
  284. titleParamInput: `${PREFIX}-title-param-input`,
  285. startTimeParamInput: `${PREFIX}-start-time-param-input`,
  286. proxyParamInput: `${PREFIX}-proxy-param-input`,
  287. refererParamInput: `${PREFIX}-referer-param-input`,
  288. originParamInput: `${PREFIX}-origin-param-input`,
  289. nxParserIframe: `${PREFIX}-nx-parser-iframe`
  290. }
  291. // 组件 class
  292. const CLASS = {
  293. button: `${PREFIX}-button`,
  294. titleSpan: `${PREFIX}-title-span-class`,
  295. titleTd: `${PREFIX}-title-td-class`,
  296. closeButton: `${PREFIX}-cloase-button-class`,
  297. tipSpan: `${PREFIX}-tip-span-class`,
  298. footerSpan: `${PREFIX}-footer-span-class`,
  299. switchLabel: `${PREFIX}-switch-label-class`,
  300. sliderSpan: `${PREFIX}-slider-span-class`,
  301. roundSpan: `${PREFIX}-round-span-class`,
  302. readOnly: `${PREFIX}-read-only-class`,
  303. footerA: `${PREFIX}-footer-a-class`,
  304. infoInput: `${PREFIX}-info-input-class`,
  305. }
  306. // 消息类型
  307. const TOAST_TYPE = {
  308. info: "info",
  309. warn: "warn",
  310. error: "error"
  311. }
  312. // 图标
  313. const ICON_BASE64 = {
  314. customplayer: "url('')",
  315. back: "url('')",
  316. }
  317. const CSS = `
  318. #${ID.loadingDiv} {
  319. display: none;
  320. position: fixed;
  321. bottom: 50%;
  322. left: 50%;
  323. z-index: 999999;
  324. transform: translate(-50%, -50%);
  325. background-color: rgba(255, 255, 255, 0);
  326. }
  327. #${ID.loadingDiv} .spinner {
  328. width: 40px;
  329. height: 40px;
  330. background-color: rgba(255, 255, 255, 1);
  331. -webkit-animation: sk-rotateplane 1.2s infinite ease-in-out;
  332. animation: sk-rotateplane 1.2s infinite ease-in-out;
  333. }
  334. @-webkit-keyframes sk-rotateplane {
  335. 0% {
  336. -webkit-transform: perspective(120px)
  337. }
  338.  
  339. 50% {
  340. -webkit-transform: perspective(120px) rotateY(180deg)
  341. }
  342.  
  343. 100% {
  344. -webkit-transform: perspective(120px) rotateY(180deg) rotateX(180deg)
  345. }
  346. }
  347.  
  348. @keyframes sk-rotateplane {
  349. 0% {
  350. transform: perspective(120px) rotateX(0deg) rotateY(0deg);
  351. -webkit-transform: perspective(120px) rotateX(0deg) rotateY(0deg)
  352. }
  353.  
  354. 50% {
  355. transform: perspective(120px) rotateX(-180.1deg) rotateY(0deg);
  356. -webkit-transform: perspective(120px) rotateX(-180.1deg) rotateY(0deg)
  357. }
  358.  
  359. 100% {
  360. transform: perspective(120px) rotateX(-180deg) rotateY(-179.9deg);
  361. -webkit-transform: perspective(120px) rotateX(-180deg) rotateY(-179.9deg);
  362. }
  363. }
  364. #${ID.toastDiv} {
  365. display: none;
  366. position: fixed;
  367. bottom: 80%;
  368. left: 50%;
  369. max-width: 60%;
  370. min-width: 150px;
  371. padding: 0 14px;
  372. height: 40px;
  373. color: rgb(255, 255, 255);
  374. line-height: 40px;
  375. text-align: center;
  376. border-radius: 4px;
  377. transform: translate(-50%, -50%);
  378. z-index: 9999999999;
  379. background: rgba(0, 255, 0, .9);
  380. font-size: 14px;
  381. font-weight: blod;
  382. }
  383. /* 按钮 */
  384. #${ID.buttonDiv} {
  385. display: none;
  386. position: fixed;
  387. bottom: 0;
  388. left: 0;
  389. cursor: pointer;
  390. z-index: 99999;
  391. width: 190px;
  392. height: 90px;
  393. }
  394. #${ID.buttonDiv}:hover .${CLASS.button} {
  395. visibility: visible !important;
  396. }
  397. .${CLASS.button} {
  398. position: fixed;
  399. cursor: pointer;
  400. z-index: 99999;
  401. border: none;
  402. border-radius: 50%;
  403. background-size: cover;
  404. background-color: rgba(255, 255, 255, 0);
  405. }
  406. #${ID.potplayerPlayButton} {
  407. bottom: 11px;
  408. left: 16px;
  409. width: 42px;
  410. height: 42px;
  411. background-image: url()
  412. }
  413. #${ID.mpvPlayButton} {
  414. bottom: 6px;
  415. left: 66px;
  416. width: 50px;
  417. height: 50px;
  418. background-image: url();
  419. }
  420. #${ID.customplayerPlayButton} {
  421. bottom: 8px;
  422. left: 123px;
  423. width: 46px;
  424. height: 46px;
  425. background-image: url()
  426. }
  427. #${ID.potplayerPlayButton}:hover {
  428. bottom: 9px;
  429. left: 14px;
  430. width: 46px;
  431. height: 46px;
  432. }
  433. #${ID.mpvPlayButton}:hover {
  434. bottom: 4px;
  435. left: 64px;
  436. width: 54px;
  437. height: 54px;
  438. }
  439. #${ID.customplayerPlayButton}:hover {
  440. bottom: 6px;
  441. left: 121px;
  442. width: 50px;
  443. height: 50px;
  444. }
  445. /* 设置 */
  446. #${ID.settingButton} {
  447. bottom: 50px;
  448. left: 160px;
  449. width: 28px;
  450. height: 28px;
  451. background-image: url();
  452. }
  453. #${ID.settingButton}:hover {
  454. bottom: 48px;
  455. left: 158px;
  456. width: 32px;
  457. height: 32px;
  458. }
  459. #${ID.settingDiv},
  460. #${ID.infoDiv} {
  461. position: fixed;
  462. top: 40%;
  463. left: 50%;
  464. transform: translate(-50%, -50%);
  465. z-index: 999999999;
  466. width: 900px;
  467. height: 520px;
  468. background-color: rgb(65, 146, 247);
  469. display: none;
  470. flex-direction: column;
  471. border-radius: 6px;
  472. align-items: center;
  473. color: rgba(0, 0, 0, .7);
  474. font-family: "微软雅黑";
  475. }
  476. #${ID.infoDiv} {
  477. background-color: rgb(234, 122, 153) !important;
  478. }
  479. #${ID.infoTable},
  480. #${ID.settingTable},
  481. #${ID.customplayerSettingTable} {
  482. width: 800px;
  483. height: 460px;
  484. border-radius: 5px !important;
  485. border: 3px solid rgba(255, 255, 255, 1) !important;
  486. text-align: left;
  487. border-collapse: unset !important;
  488. display: flex;
  489. justify-content: center;
  490. padding-top: 3px;
  491. padding-bottom: 12px;
  492. line-height: 20px !important;
  493. font-weight: normal !important;
  494. }
  495. #${ID.customplayerSettingTable} {
  496. display: none;
  497. }
  498. #${ID.infoTable} td,
  499. #${ID.settingTable} td,
  500. #${ID.customplayerSettingTable} td {
  501. font-size: 14px;
  502. border: 0px solid rgba(255, 255, 255, 0.5);
  503. padding: 19px 0px 0px 0px !important;
  504. }
  505. #${ID.infoTable} td {
  506. padding-top: 16.5px !important;
  507. }
  508. .${CLASS.titleSpan} {
  509. padding-top: 12.5px !important;;
  510. padding-bottom: 12.5px !important;;
  511. font-size: 16px;
  512. font-weight: bold;
  513. color: rgba(255, 255, 255, 1) !important;
  514. }
  515. .${CLASS.closeButton} {
  516. position: absolute;
  517. top: 4px;
  518. right: 8px;
  519. height: 25px;
  520. width: 25px;
  521. border: none;
  522. font-size: 17px !important;
  523. font-weight: normal !important;
  524. background-color: rgba(0, 0, 0, 0);
  525. line-height: 0px;
  526. border-radius: 3px;
  527. transform: scale(1.32, 1);
  528. color: rgba(255, 255, 255, 1);
  529. }
  530. .${CLASS.closeButton}:hover {
  531. font-size: 20px;
  532. background-color: rgba(255, 255, 255, .5);
  533. cursor: pointer;
  534. }
  535. .${CLASS.tipSpan} {
  536. font-size: 12px;
  537. color: yellow;
  538. position: fixed;
  539. }
  540. .${CLASS.titleTd} {
  541. position: relative;
  542. width: 80px;
  543. height: 30px;
  544. border: none;
  545. font-size: 14px;
  546. text-align: center;
  547. color: rgba(255, 255, 255, 1) !important;
  548. cursor: default;
  549. }
  550. #${ID.infoTable} input,
  551. #${ID.settingTable} input,
  552. #${ID.customplayerSettingTable} input {
  553. font-size: 12px !important;
  554. width: 500px;
  555. height: 26px;
  556. border: none;
  557. outline: none;
  558. text-indent: 5px;
  559. padding: 0px !important;
  560. border-radius: 0px !important;
  561. color: rgba(0, 0, 0, 1);
  562. background-color: rgba(255, 255, 255, 1);
  563. cursor: auto;
  564. display: flex !important;
  565. margin-top: 1px !important;
  566. margin-bottom: 1px !important;
  567. border-collapse: unset !important;
  568. }
  569. #${ID.infoTable} input:hover,
  570. #${ID.settingTable} input:hover,
  571. #${ID.customplayerSettingTable} input:hover,
  572. #${ID.infoTable} input:focus-visible,
  573. #${ID.settingTable} input:focus-visible,
  574. #${ID.customplayerSettingTable} input:focus-visible {
  575. box-shadow: none;
  576. }
  577. #${ID.settingTable} input::placeholder,
  578. #${ID.customplayerSettingTable} input::placeholder {
  579. font-size: 12px;
  580. color: rgba(0, 0, 0, .3);
  581. }
  582. #${ID.saveButton} {
  583. font-size: 14px;
  584. margin-left: 83px;
  585. width: 300px;
  586. height: 30px;
  587. border: none;
  588. border-radius: 3px;
  589. color: rgba(255, 255, 255, 1);
  590. background-color: rgba(0, 255, 0, .7);
  591. }
  592. #${ID.downloadButton} {
  593. font-size: x-small;
  594. margin-left: 10px;
  595. width: 100px;
  596. height: 30px;
  597. border: none;
  598. border-radius: 3px;
  599. color: rgba(255, 255, 255, 1);
  600. background-color: rgba(0, 255, 0, .7);
  601. }
  602. #${ID.deleteButton} {
  603. text-decoration: none;
  604. font-size: x-small;
  605. width: 80px;
  606. height: 30px;
  607. border: none;
  608. border-radius: 3px;
  609. color: rgba(255, 255, 255, 1);
  610. background-color: rgba(0,0,0,0);
  611. }
  612. #${ID.saveButton}:hover,
  613. #${ID.downloadButton}:hover {
  614. opacity: .8;
  615. background-color: rgba(0, 255, 0, .8);
  616. cursor: pointer;
  617. }
  618. #${ID.deleteButton}:hover {
  619. opacity: .8;
  620. background-color: rgba(0,0,0,0);
  621. cursor: pointer;
  622. }
  623. .${CLASS.footerSpan} {
  624. margin-top: 8px !important;
  625. margin-bottom: 8px !important;
  626. color: rgba(255, 255, 255, 1);
  627. }
  628. #${ID.infoDiv} a,
  629. .${CLASS.footerSpan} a {
  630. color: rgba(255, 255, 255, 1);
  631. text-decoration: none;
  632. font-size: 14px !important;
  633. font-weight: normal !important;
  634. margin-bottom: 1px;
  635. display: inline-block;
  636. }
  637. /* switch */
  638. .${CLASS.switchLabel} {
  639. position: relative;
  640. display: inline-block;
  641. width: 50px;
  642. height: 21px;
  643. margin-top: 3px;
  644. }
  645. .${CLASS.switchLabel} input {
  646. opacity: 0;
  647. width: 0 !important;
  648. height: 0 !important;
  649. }
  650. .${CLASS.sliderSpan} {
  651. position: absolute;
  652. cursor: pointer;
  653. top: 0;
  654. left: 0;
  655. right: 0;
  656. bottom: 0;
  657. background-color: rgba(255, 255, 255, .6);
  658. -webkit-transition: .4s;
  659. transition: .4s;
  660. }
  661. .${CLASS.sliderSpan}:before {
  662. position: absolute;
  663. content: "";
  664. height: 13px;
  665. width: 13px;
  666. left: 4px;
  667. bottom: 4px;
  668. background-color: rgba(255, 255, 255, 1);
  669. -webkit-transition: .4s;
  670. transition: .4s;
  671. }
  672. #${ID.settingDiv} input:checked + .${CLASS.sliderSpan} {
  673. background-color: rgba(0, 255, 0, .7);
  674. }
  675. #${ID.settingDiv} input:focus + .${CLASS.sliderSpan} {
  676. box-shadow: 0 0 1px rgba(0, 255, 0, .7);
  677. }
  678. #${ID.settingDiv} input:checked + .${CLASS.sliderSpan}:before {
  679. -webkit-transform: translateX(29px);
  680. -ms-transform: translateX(29px);
  681. transform: translateX(29px);
  682. }
  683. .${CLASS.sliderSpan}.${CLASS.roundSpan} {
  684. border-radius: 34px;
  685. }
  686. .${CLASS.sliderSpan}.${CLASS.roundSpan}:before {
  687. border-radius: 50%;
  688. }
  689. .${CLASS.readOnly} {
  690. color: rgba(255, 255, 255, .3) !important;
  691. background-color: rgba(0, 0, 0, .3) !important;
  692. cursor: default !important;
  693. opacity: 1 !important;
  694. }
  695. .${CLASS.readOnly}::placeholder {
  696. color: rgba(255, 255, 255, .3) !important;
  697. }
  698. #${ID.infoButton} {
  699. bottom: 52px;
  700. left: 3px;
  701. width: 25px;
  702. height: 25px;
  703. background-image: url();
  704. }
  705. #${ID.infoButton}:hover {
  706. bottom: 50px;
  707. left: 1px;
  708. width: 29px;
  709. height: 29px;
  710. }
  711. #${ID.settingDiv} .tabs {
  712. display: flex;
  713. width: 501px;
  714. height: 28px;
  715. }
  716. #${ID.settingDiv} .tabs>.tab {
  717. flex: 1;
  718. display: flex;
  719. border: 1px solid rgb(65,146,247);
  720. margin: 0;
  721. }
  722. #${ID.settingDiv} .tabs>.tab:after {
  723. background: rgba(0, 0, 0, 0);
  724. color: rgba(0, 0, 0, 0) !important;
  725. }
  726. #${ID.settingDiv} .tab>.tab-input {
  727. width: 0 !important;
  728. height: 0 !important;
  729. margin: 0 !important;
  730. display: none !important;
  731. }
  732. #${ID.settingDiv} .tab>.tab-box {
  733. padding: 4px 0px 0.5px 0px;
  734. width: 100%;
  735. height: 22px;
  736. text-align: center;
  737. transition: 0.3s;
  738. background: rgba(255, 255, 255, 1);
  739. font-size: 12px;
  740. font-weight: normal !important;
  741. display: table !important;
  742. color: #000000B3;
  743. }
  744. #${ID.settingDiv} .tab>.tab-box:hover {
  745. opacity: .8;
  746. cursor: pointer;
  747. }
  748. #${ID.settingDiv} .tab>.tab-input:checked+.tab-box {
  749. color: rgba(255, 255, 255, 1);
  750. background: rgba(0, 255, 0, .7);
  751. }
  752. #${ID.customplayerSettingButton}:hover:after,
  753. #${ID.settingDiv} .${CLASS.titleTd}:hover:after,
  754. .${CLASS.footerA}:hover:after {
  755. position: absolute;
  756. font-size: 12px;
  757. left: 0px;
  758. top: -3px;
  759. padding: 5px 5px 5px 5px !important;
  760. background-color: rgba(0, 0, 0, 0.6);
  761. color: rgba(255, 255, 255, 1);
  762. content: attr(data-tip);
  763. text-align: center;
  764. z-index: 999999;
  765. width: auto !important;
  766. height: auto !important;
  767. white-space: nowrap;
  768. }
  769. #${ID.customplayerSettingButton}:hover:after {
  770. left: -10px !important;
  771. top: -28px !important;
  772. }
  773. .${CLASS.footerA} {
  774. position: relative;
  775. }
  776. .${CLASS.footerA}:hover:after {
  777. left: 0px !important;
  778. top: -27px !important;
  779. }
  780. #${ID.customplayerSettingButton} {
  781. position: absolute;
  782. right: 122px;
  783. top: 76px;
  784. width: 22px;
  785. height: 22px;
  786. border: none;
  787. cursor: pointer;
  788. background-image: ${ICON_BASE64.customplayer};
  789. z-index: 999999;
  790. }
  791. `;
  792. const HTML = `
  793. <div id="${ID.loadingDiv}">
  794. <div class="spinner"></div>
  795. </div>
  796. <div id="${ID.toastDiv}"></div>
  797.  
  798. <div id="${ID.buttonDiv}">
  799. <button id="${ID.infoButton}" class="${CLASS.button}"></button>
  800. <button id="${ID.potplayerPlayButton}" class="${CLASS.button}"></button>
  801. <button id="${ID.mpvPlayButton}" class="${CLASS.button}"></button>
  802. <button id="${ID.customplayerPlayButton}" class="${CLASS.button}"></button>
  803. <button id="${ID.settingButton}" class="${CLASS.button}"></button>
  804. </div>
  805.  
  806. <div id="${ID.infoDiv}">
  807. <span class="${CLASS.titleSpan}">Play-With-MPV<button class="${CLASS.closeButton}">X</button></span>
  808. <table id="${ID.infoTable}">
  809. <tr>
  810. <td colspan="4" style="text-align: center; color: white; font-size: 16px;">本 息</td>
  811. </tr>
  812. <tr>
  813. <td class="${CLASS.titleTd}">视频标题</td>
  814. <td colspan="8">
  815. <input type="text" readonly class="${CLASS.infoInput}">
  816. </td>
  817. </tr>
  818. <tr>
  819. <td class="${CLASS.titleTd}">视频链接</td>
  820. <td colspan="8">
  821. <input type="text" readonly class="${CLASS.infoInput}">
  822. </td>
  823. </tr>
  824. <tr>
  825. <td class="${CLASS.titleTd}">音频链接</td>
  826. <td colspan="8">
  827. <input type="text" readonly class="${CLASS.infoInput}">
  828. </td>
  829. </tr>
  830. <tr>
  831. <td class="${CLASS.titleTd}">字幕链接</td>
  832. <td colspan="8">
  833. <input type="text" readonly class="${CLASS.infoInput}">
  834. </td>
  835. </tr>
  836. <tr>
  837. <td class="${CLASS.titleTd}">referer</td>
  838. <td colspan="8">
  839. <input type="text" readonly class="${CLASS.infoInput}">
  840. </td>
  841. </tr>
  842. <tr>
  843. <td class="${CLASS.titleTd}">origin</td>
  844. <td colspan="8">
  845. <input type="text" readonly class="${CLASS.infoInput}">
  846. </td>
  847. </tr>
  848. <tr><td></td></tr>
  849. <tr>
  850. <td style="text-align: right;">
  851. <a href="https://greasyfork.org/zh-CN/scripts/444056-play-with-mpv" target="_blank">🆕 升级 🆕</a>
  852. </td>
  853. <td style="text-align: right;">
  854. <a href="https://www.lckp.top/play-with-mpv/index.html" target="_blank">🧭 网站导航 🧭</a>
  855. </td>
  856. <td style="text-align: right;">
  857. <a href="https://github.com/LuckyPuppy514/Play-With-MPV" target="_blank">🌟 项目源码 🌟</a>
  858. </td>
  859. <td style="text-align: right;">
  860. <a href="https://github.com/LuckyPuppy514/Play-With-MPV/issues/new" target="_blank">👻 反馈 👻</a>
  861. </td>
  862. </tr>
  863. </table>
  864. <span class="${CLASS.footerSpan}">
  865. <a href="https://greasyfork.org/zh-CN/scripts/444056-play-with-mpv" target="_blank" class="${CLASS.footerA}" data-tip="版本升级"> 🆕 </a>
  866. <a href="https://github.com/LuckyPuppy514/Play-With-MPV" target="_blank" class="${CLASS.footerA}" data-tip="项目源码"> © 2023 LuckyPuppy514 </a>
  867. <a href="https://github.com/LuckyPuppy514/Play-With-MPV/issues/new" target="_blank" class="${CLASS.footerA}" data-tip="问题反馈"> 👻 </a>
  868. </span>
  869. </div>
  870.  
  871. <div id="${ID.settingDiv}">
  872. <span class="${CLASS.titleSpan}">Play-With-MPV<button class="${CLASS.closeButton}">X</button></span>
  873. <button id="${ID.customplayerSettingButton}" class="${CLASS.button}" data-tip="设置自定义播放器"></button>
  874. <table id="${ID.settingTable}">
  875. <tr>
  876. <td class="${CLASS.titleTd}" data-tip="选择 mpv 以外播放器时,部分功能无效">播放软件</td>
  877. <td colspan="8">
  878. <div class="tabs">
  879. <label class="tab">
  880. <input type="radio" name="${ID.playerRadio}" value="${PLAYER.mpv.name}" class="tab-input">
  881. <div class="tab-box">mpv</div>
  882. </label>
  883. <label class="tab">
  884. <input type="radio" name="${ID.playerRadio}" value="${PLAYER.potplayer.name}" class="tab-input">
  885. <div class="tab-box">potplayer</div>
  886. </label>
  887. <label class="tab">
  888. <input type="radio" name="${ID.playerRadio}" value="${PLAYER.customplayer.name}" class="tab-input">
  889. <div class="tab-box">自定义</div>
  890. </label>
  891. </div>
  892. </td>
  893. </tr>
  894. <tr>
  895. <td class="${CLASS.titleTd}" data-tip="mpv.exe 或 PotPlayerMini64.exe 的完整路径">软件路径</td>
  896. <td colspan="8">
  897. <div>
  898. <input id="${ID.softwarePathInput}" type=text placeholder="请输入软件路径,例如:D://mpvnet//mpvnet.exe">
  899. </div>
  900. </td>
  901. </tr>
  902. <tr>
  903. <td class="${CLASS.titleTd}" data-tip="旁路由网关实现代理一般不需要设置">代理设置</td>
  904. <td colspan="8">
  905. <div>
  906. <input id="${ID.proxyInput}" type=text placeholder="请输入代理地址,例如:http://127.0.0.1:10809">
  907. </div>
  908. </td>
  909. </tr>
  910. <tr>
  911. <td class="${CLASS.titleTd}" data-tip="仅适用于B站或使用 yt-dlp 解析的网站,例如:油管,OK,TVer 等">最高画质</td>
  912. <td colspan="8">
  913. <div class="tabs">
  914. <label class="tab">
  915. <input type="radio" name="${ID.bestQualityRadio}" value="unlimited" class="tab-input">
  916. <div class="tab-box" name="${ID.bestQualityRadio}">无限制</div>
  917. </label>
  918. <label class="tab">
  919. <input type="radio" name="${ID.bestQualityRadio}" value="2160p" class="tab-input">
  920. <div class="tab-box" name="${ID.bestQualityRadio}">2160p</div>
  921. </label>
  922. <label class="tab">
  923. <input type="radio" name="${ID.bestQualityRadio}" value="1440p" class="tab-input">
  924. <div class="tab-box" name="${ID.bestQualityRadio}">1440p</div>
  925. </label>
  926. <label class="tab">
  927. <input type="radio" name="${ID.bestQualityRadio}" value="1080p" class="tab-input">
  928. <div class="tab-box" name="${ID.bestQualityRadio}">1080p</div>
  929. </label>
  930. <label class="tab">
  931. <input type="radio" name="${ID.bestQualityRadio}" value="720p" class="tab-input">
  932. <div class="tab-box" name="${ID.bestQualityRadio}">720p</div>
  933. </label>
  934. </div>
  935. </td>
  936. </tr>
  937. <tr>
  938. <td class="${CLASS.titleTd}" data-tip="仅适用于B站">视频编码</td>
  939. <td colspan="8">
  940. <div class="tabs">
  941. <label class="tab">
  942. <input type="radio" name="${ID.bilibiliCodecsRadio}" value="12" class="tab-input">
  943. <div class="tab-box" name="${ID.bilibiliCodecsRadio}">HEVC</div>
  944. </label>
  945. <label class="tab">
  946. <input type="radio" name="${ID.bilibiliCodecsRadio}" value="13" class="tab-input">
  947. <div class="tab-box" name="${ID.bilibiliCodecsRadio}">AV1</div>
  948. </label>
  949. <label class="tab">
  950. <input type="radio" name="${ID.bilibiliCodecsRadio}" value="7" class="tab-input">
  951. <div class="tab-box" name="${ID.bilibiliCodecsRadio}">AVC</div>
  952. </label>
  953. </div>
  954. </td>
  955. </tr>
  956. <tr>
  957. <td class="${CLASS.titleTd}" data-tip="仅适用于B站">首选字幕</td>
  958. <td colspan="8">
  959. <div class="tabs">
  960. <label class="tab">
  961. <input type="radio" name="${ID.subtitlePreferRadio}" value="zh-Hans" class="tab-input">
  962. <div class="tab-box" name="${ID.subtitlePreferRadio}">简体</div>
  963. </label>
  964. <label class="tab">
  965. <input type="radio" name="${ID.subtitlePreferRadio}" value="zh-Hant" class="tab-input">
  966. <div class="tab-box" name="${ID.subtitlePreferRadio}">繁体</div>
  967. </label>
  968. <label class="tab">
  969. <input type="radio" name="${ID.subtitlePreferRadio}" value="en-US" class="tab-input"=>
  970. <div class="tab-box" name="${ID.subtitlePreferRadio}">英语</div>
  971. </label>
  972. <label class="tab">
  973. <input type="radio" name="${ID.subtitlePreferRadio}" value="off" class="tab-input"=>
  974. <div class="tab-box" name="${ID.subtitlePreferRadio}">关闭</div>
  975. </label>
  976. </div>
  977. </td>
  978. </tr>
  979. <tr>
  980. <td class="${CLASS.titleTd}" data-tip="解析成功自动播放">自动播放</td>
  981. <td colspan="2">
  982. <div>
  983. <label class="${CLASS.switchLabel}">
  984. <input type="checkbox" id="${ID.playAutoInput}">
  985. <span class="${CLASS.sliderSpan} ${CLASS.roundSpan}"></span>
  986. </label>
  987. </div>
  988. </td>
  989. <td class="${CLASS.titleTd}" data-tip="播放时自动关闭页面(和自动播放一起开启时修改配置请前往导航页)">自动关闭</td>
  990. <td colspan="2">
  991. <div>
  992. <label class="${CLASS.switchLabel}">
  993. <input type="checkbox" id="${ID.closeAutoInput}">
  994. <span class="${CLASS.sliderSpan} ${CLASS.roundSpan}"></span>
  995. </label>
  996. </div>
  997. </td>
  998. <td class="${CLASS.titleTd}" data-tip="同步网页播放时间">同步时间</td>
  999. <td colspan="2">
  1000. <div>
  1001. <label class="${CLASS.switchLabel}">
  1002. <input type="checkbox" id="${ID.syncStartTimeInput}">
  1003. <span class="${CLASS.sliderSpan} ${CLASS.roundSpan}" id="${ID.syncStartTimeSpan}"></span>
  1004. </label>
  1005. </div>
  1006. </td>
  1007. </tr>
  1008. <tr>
  1009. <td colspan="9">
  1010. <button id="${ID.saveButton}">保存设置</button>
  1011. <button id="${ID.downloadButton}">下载注册表</button>
  1012. <button id="${ID.deleteButton}">删除注册表</button>
  1013. </td>
  1014. </tr>
  1015. </table>
  1016. <table id="${ID.customplayerSettingTable}">
  1017. <tr>
  1018. <td class="${CLASS.titleTd}" data-tip="必填(视频格式:yt-dlp / m3u8 / flv / m4s / mp4 / mkv ... 播放器不支持则无法播放对应格式视频)">视频参数</td>
  1019. <td colspan="8">
  1020. <div>
  1021. <input id="${ID.videoUrlParamInput}" type=text placeholder='请输入视频参数,例如:mpv://"$\{videoUrl\}"'>
  1022. </div>
  1023. </td>
  1024. </tr>
  1025. <tr>
  1026. <td class="${CLASS.titleTd}" data-tip="选填(为空则不支持最高画质和视频编码)">音频参数</td>
  1027. <td colspan="8">
  1028. <div>
  1029. <input id="${ID.audioUrlParamInput}" type=text placeholder='请输入音频参数,例如: --audio-file="$\{audioUrl\}"'>
  1030. </div>
  1031. </td>
  1032. </tr>
  1033. <tr>
  1034. <td class="${CLASS.titleTd}" data-tip="选填(为空则无法加载B站外挂字幕)">字幕参数</td>
  1035. <td colspan="8">
  1036. <div>
  1037. <input id="${ID.subtitleUrlParamInput}" type=text placeholder='请输入字幕参数,例如: --sub-file="$\{subtitleUrl\}"'>
  1038. </div>
  1039. </td>
  1040. </tr>
  1041. <tr>
  1042. <td class="${CLASS.titleTd}" data-tip="选填(为空则无法传递标题)">标题参数</td>
  1043. <td colspan="8">
  1044. <div>
  1045. <input id="${ID.titleParamInput}" type=text placeholder='请输入标题参数,例如: --force-media-title="$\{title\}"'>
  1046. </div>
  1047. </td>
  1048. </tr>
  1049. <tr>
  1050. <td class="${CLASS.titleTd}" data-tip="选填(为空则不支持同步时间)">时间参数</td>
  1051. <td colspan="8">
  1052. <div>
  1053. <input id="${ID.startTimeParamInput}" type=text placeholder='请输入时间参数,例如: --start=$\{startTime\}'>
  1054. </div>
  1055. </td>
  1056. </tr>
  1057. <tr>
  1058. <td class="${CLASS.titleTd}" data-tip="选填(为空则不支持代理设置)">代理参数</td>
  1059. <td colspan="8">
  1060. <div>
  1061. <input id="${ID.proxyParamInput}" type=text placeholder='请输入代理参数,例如: --http-proxy=$\{proxy\} --ytdl-raw-options=proxy=[$\{proxy\}]'>
  1062. </div>
  1063. </td>
  1064. </tr>
  1065. <tr>
  1066. <td class="${CLASS.titleTd}" data-tip="选填(为空则无法观看B站和橘子动漫)">referer</td>
  1067. <td colspan="8">
  1068. <div>
  1069. <input id="${ID.refererParamInput}" type=text placeholder='请输入 referer,例如: --http-header-fields="referer: $\{referer\}"'>
  1070. </div>
  1071. </td>
  1072. </tr>
  1073. <tr>
  1074. <td class="${CLASS.titleTd}" data-tip="选填(为空则无法观看巴哈姆特)">origin</td>
  1075. <td colspan="8">
  1076. <div>
  1077. <input id="${ID.originParamInput}" type=text placeholder='请输入 origin,例如: --http-header-fields="origin: $\{origin\}" '>
  1078. </div>
  1079. </td>
  1080. </tr>
  1081. </table>
  1082. <span class="${CLASS.footerSpan}">
  1083. <a href="https://greasyfork.org/zh-CN/scripts/444056-play-with-mpv" target="_blank" class="${CLASS.footerA}" data-tip="版本升级"> 🆕 </a>
  1084. <a href="https://github.com/LuckyPuppy514/Play-With-MPV" target="_blank" class="${CLASS.footerA}" data-tip="项目源码"> © 2023 LuckyPuppy514 </a>
  1085. <a href="https://github.com/LuckyPuppy514/Play-With-MPV/issues/new" target="_blank" class="${CLASS.footerA}" data-tip="问题反馈"> 👻 </a>
  1086. </span>
  1087. </div>
  1088. `;
  1089. const REG =
  1090. `Windows Registry Editor Version 5.00
  1091. [HKEY_LOCAL_MACHINE\\SOFTWARE\\Policies\\Google\\Chrome]
  1092. "ExternalProtocolDialogShowAlwaysOpenCheckbox"=dword:00000001
  1093.  
  1094. [HKEY_LOCAL_MACHINE\\SOFTWARE\\Policies\\Microsoft\\Edge]
  1095. "ExternalProtocolDialogShowAlwaysOpenCheckbox"=dword:00000001
  1096.  
  1097. [HKEY_CLASSES_ROOT\\\${PLAYER_NAME}]
  1098. @="\${PLAYER_NAME} Protocol"
  1099. "URL Protocol"=""
  1100.  
  1101. [HKEY_CLASSES_ROOT\\\${PLAYER_NAME}\\DefaultIcon]
  1102. @=""
  1103.  
  1104. [HKEY_CLASSES_ROOT\\\${PLAYER_NAME}\\shell]
  1105. @=""
  1106.  
  1107. [HKEY_CLASSES_ROOT\\\${PLAYER_NAME}\\shell\\open]
  1108. @=""
  1109.  
  1110. [HKEY_CLASSES_ROOT\\\${PLAYER_NAME}\\shell\\open\\command]
  1111. @="C:\\\\Windows\\\\System32\\\\WindowsPowerShell\\\\v1.0\\\\powershell.exe -WindowStyle Hidden -Command \\"& {Add-Type -AssemblyName System.Web;$PARAMS=([System.Web.HTTPUtility]::UrlDecode('%1') -replace '^\${PLAYER_NAME}://'); Start-Process -FilePath \\\\\\\"\${SOFTWARE_PATH}\\\\\\\" -ArgumentList $PARAMS}\\""
  1112. `
  1113. const REG_DELETE =
  1114. `Windows Registry Editor Version 5.00
  1115. [-HKEY_CLASSES_ROOT\\\${PLAYER_NAME}]
  1116. `
  1117. function appendCSS() {
  1118. let css = document.createElement("style");
  1119. css.innerHTML = CSS.trim();
  1120. document.head.appendChild(css);
  1121. }
  1122. function appendHTML() {
  1123. let div = document.createElement("div");
  1124. div.innerHTML = HTML.trim();
  1125. document.body.appendChild(div);
  1126. }
  1127. function loading(visiable) {
  1128. let loadingDiv = document.getElementById(ID.loadingDiv);
  1129. if (visiable) {
  1130. loadingDiv.style.display = "flex";
  1131. setTimeout(() => {
  1132. if (loadingDiv.style.display == "flex") {
  1133. document.getElementById(ID.loadingDiv).style.display = "none";
  1134. toast("超时辣 ...... 😓", TOAST_TYPE.error);
  1135. }
  1136. }, TIME.out);
  1137. } else {
  1138. loadingDiv.style.display = "none";
  1139. }
  1140. }
  1141. function toast(message, type, duration) {
  1142. type = type ? type : TOAST_TYPE.info;
  1143. duration = isNaN(duration) ? TIME.toast : duration;
  1144. let toastDiv = document.getElementById(ID.toastDiv);
  1145. toastDiv.innerHTML = message;
  1146. toastDiv.style.display = "block";
  1147. if (type == TOAST_TYPE.info) {
  1148. toastDiv.style.backgroundColor = "rgba(75, 180 ,54, 1)";
  1149. } else if (type == TOAST_TYPE.warn) {
  1150. toastDiv.style.backgroundColor = "rgba(190, 190, 70, 1)";
  1151. } else if (type == TOAST_TYPE.error) {
  1152. toastDiv.style.backgroundColor = "rgba(210, 51, 35, 1)";
  1153. }
  1154. setTimeout(() => {
  1155. toastDiv.style.display = "none";
  1156. }, duration);
  1157. }
  1158. function playButtonClick(player) {
  1159. if (playDisabled) {
  1160. return;
  1161. }
  1162. if (currentConfig.player != player) {
  1163. let playAuto = currentConfig.playAuto;
  1164. currentConfig.player = player;
  1165. currentConfig.playAuto = 1;
  1166. GM_setValue(KEY.config, currentConfig);
  1167. setTimeout(() => {
  1168. currentConfig.playAuto = playAuto;
  1169. GM_setValue(KEY.config, currentConfig);
  1170. }, TIME.pauseInterval);
  1171. init();
  1172. return;
  1173. }
  1174. if (currentConfig.player == PLAYER.mpv.name || currentConfig.player == PLAYER.potplayer.name) {
  1175. let message = undefined;
  1176. if (!currentConfig[currentConfig.player].path) {
  1177. message = "请先进行设置";
  1178. } else if (!currentConfig[currentConfig.player].regVersion) {
  1179. message = "请先下载注册表";
  1180. } else if (currentConfig[currentConfig.player].regVersion != DEFAULT_CONFIG[currentConfig.player].regVersion) {
  1181. message = "注册表有更新,请重新下载注册表";
  1182. }
  1183. if (message) {
  1184. toast(message, TOAST_TYPE.warn);
  1185. settingDiv.style.display = "none";
  1186. settingButton.click();
  1187. return;
  1188. }
  1189. }
  1190. loading(true);
  1191. try {
  1192. playLimit();
  1193. handler.play();
  1194. if (currentConfig.closeAuto == 1 && page.url !== "https://www.lckp.top/play-with-mpv/index.html") {
  1195. setTimeout(() => {
  1196. if (history.length === 1) {
  1197. window.location.href = "about:blank";
  1198. window.top.close();
  1199. } else {
  1200. history.back();
  1201. }
  1202. }, 1000);
  1203. } else {
  1204. handler.pause();
  1205. }
  1206. } catch (error) {
  1207. toast("出错辣 ...... 😓", TOAST_TYPE.error);
  1208. console.log(error);
  1209. }
  1210. loading(false);
  1211. }
  1212. // 限制播放按钮点击频率
  1213. var playDisabled = false;
  1214. function playLimit() {
  1215. playDisabled = true;
  1216. setTimeout(() => {
  1217. playDisabled = false;
  1218. }, TIME.pauseInterval);
  1219. }
  1220. function addListener() {
  1221. let buttonDiv = document.getElementById(ID.buttonDiv);
  1222. let mpvPlayButton = document.getElementById(ID.mpvPlayButton);
  1223. let potplayerPlayButton = document.getElementById(ID.potplayerPlayButton);
  1224. let customplayerPlayButton = document.getElementById(ID.customplayerPlayButton);
  1225. let settingButton = document.getElementById(ID.settingButton);
  1226. let settingDiv = document.getElementById(ID.settingDiv);
  1227. let settingTable = document.getElementById(ID.settingTable);
  1228. let softwarePathInput = document.getElementById(ID.softwarePathInput);
  1229. let proxyInput = document.getElementById(ID.proxyInput);
  1230. let playAutoInput = document.getElementById(ID.playAutoInput);
  1231. let closeAutoInput = document.getElementById(ID.closeAutoInput);
  1232. let syncStartTimeInput = document.getElementById(ID.syncStartTimeInput);
  1233. let syncStartTimeSpan = document.getElementById(ID.syncStartTimeSpan);
  1234. let downloadButton = document.getElementById(ID.downloadButton);
  1235. let deleteButton = document.getElementById(ID.deleteButton);
  1236. let saveButton = document.getElementById(ID.saveButton);
  1237. let closeButtons = document.getElementsByClassName(CLASS.closeButton);
  1238. let infoButton = document.getElementById(ID.infoButton);
  1239. let infoDiv = document.getElementById(ID.infoDiv);
  1240. let customplayerSettingTable = document.getElementById(ID.customplayerSettingTable);
  1241. let customplayerSettingButton = document.getElementById(ID.customplayerSettingButton);
  1242. let videoUrlParamInput = document.getElementById(ID.videoUrlParamInput);
  1243. let audioUrlParamInput = document.getElementById(ID.audioUrlParamInput);
  1244. let subtitleUrlParamInput = document.getElementById(ID.subtitleUrlParamInput);
  1245. let titleParamInput = document.getElementById(ID.titleParamInput);
  1246. let startTimeParamInput = document.getElementById(ID.startTimeParamInput);
  1247. let proxyParamInput = document.getElementById(ID.proxyParamInput);
  1248. let refererParamInput = document.getElementById(ID.refererParamInput);
  1249. let originParamInput = document.getElementById(ID.originParamInput);
  1250. let infoInputs = document.getElementsByClassName(CLASS.infoInput);
  1251. switchStatus(downloadButton, false);
  1252. // 播放按钮
  1253. mpvPlayButton.onclick = function () {
  1254. playButtonClick(PLAYER.mpv.name);
  1255. };
  1256. potplayerPlayButton.onclick = function () {
  1257. playButtonClick(PLAYER.potplayer.name);
  1258. }
  1259. customplayerPlayButton.onclick = function () {
  1260. playButtonClick(PLAYER.customplayer.name);
  1261. }
  1262. // 播放快捷键 CTRL + P
  1263. window.onkeydown = async function () {
  1264. if (event.ctrlKey && event.keyCode === 80 && !event.shiftKey) {
  1265. event.preventDefault();
  1266. playButtonClick(currentConfig.player);
  1267. }
  1268. }
  1269. // 设置按钮
  1270. settingButton.onclick = function () {
  1271. let display = settingDiv.style.display;
  1272. if (display == "flex") {
  1273. settingDiv.style.display = "none";
  1274. } else {
  1275. infoDiv.style.display = "none";
  1276. settingDiv.style.display = "flex";
  1277. // 加载配置
  1278. softwarePathInput.value = currentConfig[currentConfig.player].path;
  1279. proxyInput.value = currentConfig.proxy;
  1280. $(`input:radio[name="${ID.bestQualityRadio}"][value="${currentConfig.bestQuality}"]`).prop('checked', true);
  1281. $(`input:radio[name="${ID.bilibiliCodecsRadio}"][value="${currentConfig.bilibiliCodecs}"]`).prop('checked', true);
  1282. $(`input:radio[name="${ID.playerRadio}"][value="${currentConfig.player}"]`).prop('checked', true);
  1283. playAutoInput.checked = currentConfig.playAuto == 1 ? true : false;
  1284. closeAutoInput.checked = currentConfig.closeAuto == 1 ? true : false;
  1285. syncStartTimeInput.checked = currentConfig.syncStartTime == 1 ? true : false;
  1286. $(`input:radio[name="${ID.subtitlePreferRadio}"][value="${currentConfig.subtitlePrefer}"]`).prop('checked', true);
  1287. switchPlayer($(`input:radio[name="${ID.playerRadio}"]:checked`).val());
  1288. }
  1289. }
  1290. // 播放器选择框
  1291. $(`input:radio[name="${ID.playerRadio}"]`).change(function () {
  1292. switchPlayer(this.value);
  1293. });
  1294. // 保存按钮
  1295. saveButton.onclick = function () {
  1296. let playerChecked = $(`input:radio[name="${ID.playerRadio}"]:checked`).val();
  1297. if (playerChecked == PLAYER.mpv.name || playerChecked == PLAYER.potplayer.name) {
  1298. let oldSoftwarePath = currentConfig[playerChecked].path;
  1299. let newSoftwarePath = softwarePathInput.value;
  1300. if (!newSoftwarePath) {
  1301. toast("软件路径不能为空", TOAST_TYPE.error);
  1302. return;
  1303. }
  1304. if (/.*[\u4e00-\u9fa5]+.*/g.test(newSoftwarePath)) {
  1305. toast("软件路径不能包含中文", TOAST_TYPE.error);
  1306. return;
  1307. }
  1308. newSoftwarePath = newSoftwarePath.replace(/[\\|/]+/g, "//");
  1309. if (!newSoftwarePath.endsWith(".com") && !newSoftwarePath.endsWith(".exe")) {
  1310. if (!newSoftwarePath.endsWith("//")) {
  1311. newSoftwarePath = newSoftwarePath + "//";
  1312. }
  1313. if (playerChecked == PLAYER.mpv.name) {
  1314. if (newSoftwarePath.toLowerCase().indexOf("mpvnet") != -1 || newSoftwarePath.toLowerCase().indexOf("mpv.net") != -1) {
  1315. newSoftwarePath = newSoftwarePath + "mpvnet.exe";
  1316. } else {
  1317. newSoftwarePath = newSoftwarePath + "mpv.exe";
  1318. }
  1319. } else if (playerChecked == PLAYER.potplayer.name) {
  1320. newSoftwarePath = newSoftwarePath + "PotPlayerMini64.exe";
  1321. }
  1322. }
  1323. softwarePathInput.value = newSoftwarePath;
  1324. currentConfig[playerChecked].path = newSoftwarePath;
  1325. switchStatus(downloadButton, softwarePathInput.value ? true : false);
  1326. if (oldSoftwarePath != newSoftwarePath) {
  1327. currentConfig[playerChecked].regVersion = "00000000";
  1328. }
  1329. }
  1330. currentConfig.proxy = proxyInput.value;
  1331. currentConfig.bestQuality = $(`input:radio[name="${ID.bestQualityRadio}"]:checked`).val();
  1332. currentConfig.bilibiliCodecs = $(`input:radio[name="${ID.bilibiliCodecsRadio}"]:checked`).val();
  1333. currentConfig.player = playerChecked;
  1334. currentConfig.subtitlePrefer = $(`input:radio[name="${ID.subtitlePreferRadio}"]:checked`).val();
  1335. currentConfig.playAuto = playAutoInput.checked ? 1 : 0;
  1336. currentConfig.closeAuto = closeAutoInput.checked ? 1 : 0;
  1337. currentConfig.syncStartTime = syncStartTimeInput.checked ? 1 : 0;
  1338. GM_setValue(KEY.config, currentConfig);
  1339. if (playAutoInput.checked && closeAutoInput.checked) {
  1340. toast("保存成功,如需修改配置请前往导航页");
  1341. } else {
  1342. toast("保存成功");
  1343. }
  1344. if (currentConfig.playAuto == 1) {
  1345. playLimit();
  1346. }
  1347. if (document.getElementById("iptv") && localStorage.category == "iptv") {
  1348. localStorage.player = JSON.stringify(PLAYER[currentConfig.player]);
  1349. document.getElementById("iptv").click();
  1350. }
  1351. init();
  1352. }
  1353. // 下载按钮
  1354. downloadButton.onclick = function () {
  1355. let playerChecked = $(`input:radio[name="${ID.playerRadio}"]:checked`).val();
  1356. currentConfig[playerChecked].regVersion = DEFAULT_CONFIG[playerChecked].regVersion;
  1357. GM_setValue(KEY.config, currentConfig);
  1358. let reg = REG.replace("${SOFTWARE_PATH}", currentConfig[playerChecked].path);
  1359. reg = reg.replace(/\$\{PLAYER_NAME\}/g, playerChecked);
  1360. let a = document.createElement('a');
  1361. let blob = new Blob([reg], { 'type': 'application/octet-stream' });
  1362. a.href = window.URL.createObjectURL(blob);
  1363. a.download = `${playerChecked}-install.reg`;
  1364. a.click();
  1365. }
  1366. deleteButton.onclick = function () {
  1367. let playerChecked = $(`input:radio[name="${ID.playerRadio}"]:checked`).val();
  1368. currentConfig[playerChecked].regVersion = "00000000";
  1369. GM_setValue(KEY.config, currentConfig);
  1370. let reg = REG_DELETE.replace(/\$\{PLAYER_NAME\}/g, playerChecked);
  1371. let a = document.createElement('a');
  1372. let blob = new Blob([reg], { 'type': 'application/octet-stream' });
  1373. a.href = window.URL.createObjectURL(blob);
  1374. a.download = `${playerChecked}-delete.reg`;
  1375. a.click();
  1376. }
  1377. // 关闭按钮
  1378. for (let closeButton of closeButtons) {
  1379. closeButton.onclick = function () {
  1380. settingDiv.style.display = "none";
  1381. infoDiv.style.display = "none";
  1382. }
  1383. }
  1384. // 信息按钮
  1385. infoButton.onclick = function () {
  1386. let display = infoDiv.style.display;
  1387. if (display == "flex") {
  1388. infoDiv.style.display = "none";
  1389. } else {
  1390. settingDiv.style.display = "none";
  1391. infoDiv.style.display = "flex";
  1392. let title = handler.media.title;
  1393. infoInputs[0].value = title ? title : document.title;
  1394. infoInputs[1].value = handler.media.videoUrl;
  1395. infoInputs[2].value = handler.media.audioUrl;
  1396. infoInputs[3].value = handler.media.subtitleUrl;
  1397. infoInputs[4].value = handler.media.referer;
  1398. infoInputs[5].value = handler.media.origin;
  1399. for (const infoInput of infoInputs) {
  1400. if (infoInput.value) {
  1401. infoInput.style.cursor = "pointer";
  1402. infoInput.onclick = function () {
  1403. this.select();
  1404. navigator.clipboard.writeText(this.value);
  1405. toast("已复制到剪贴板");
  1406. }
  1407. }
  1408. }
  1409. }
  1410. }
  1411. let bestQualityRadios = document.getElementsByName(ID.bestQualityRadio);
  1412. let bilibiliCodecsRadios = document.getElementsByName(ID.bilibiliCodecsRadio);
  1413. let subtitlePreferRadios = document.getElementsByName(ID.subtitlePreferRadio);
  1414. // 切换播放器
  1415. function switchPlayer(player) {
  1416. player = PLAYER[player];
  1417. // mpv 和 potplayer 专属
  1418. if (player.name == PLAYER.mpv.name || player.name == PLAYER.potplayer.name) {
  1419. switchStatus(softwarePathInput, true);
  1420. softwarePathInput.value = currentConfig[player.name].path;
  1421. if (softwarePathInput.value) {
  1422. switchStatus(downloadButton, true);
  1423. } else {
  1424. switchStatus(downloadButton, false);
  1425. }
  1426. } else {
  1427. switchStatus(softwarePathInput, false);
  1428. switchStatus(downloadButton, false);
  1429. }
  1430. // 代理
  1431. let flag = player.params.proxy ? true : false;
  1432. switchStatus(proxyInput, flag);
  1433. // 音频
  1434. flag = player.params.audioUrl ? true : false;
  1435. for (const radio of bestQualityRadios) {
  1436. switchStatus(radio, flag);
  1437. }
  1438. for (const radio of bilibiliCodecsRadios) {
  1439. switchStatus(radio, flag);
  1440. }
  1441. // 字幕
  1442. flag = player.params.subtitleUrl ? true : false;
  1443. for (const radio of subtitlePreferRadios) {
  1444. switchStatus(radio, flag);
  1445. }
  1446. // 时间
  1447. flag = player.params.startTime ? true : false;
  1448. switchStatus(syncStartTimeSpan, flag);
  1449. switchStatus(syncStartTimeInput, flag);
  1450. }
  1451. // 全屏
  1452. document.addEventListener("fullscreenchange", () => {
  1453. if (document.fullscreenElement) {
  1454. page.isFullScreen = true;
  1455. buttonDiv.style.display = "none";
  1456. } else {
  1457. page.isFullScreen = false;
  1458. if (handler.media.videoUrl) {
  1459. buttonDiv.style.display = "flex";
  1460. }
  1461. }
  1462. });
  1463. // 自定义播放器按钮
  1464. customplayerSettingButton.onclick = function () {
  1465. if (customplayerSettingTable.style.display == "flex") {
  1466. if (!videoUrlParamInput.value) {
  1467. toast("视频参数不能为空", TOAST_TYPE.error);
  1468. return;
  1469. }
  1470. currentConfig.customplayer.params.videoUrl = videoUrlParamInput.value;
  1471. currentConfig.customplayer.params.audioUrl = audioUrlParamInput.value;
  1472. currentConfig.customplayer.params.subtitleUrl = subtitleUrlParamInput.value;
  1473. currentConfig.customplayer.params.title = titleParamInput.value;
  1474. currentConfig.customplayer.params.startTime = startTimeParamInput.value;
  1475. currentConfig.customplayer.params.proxy = proxyParamInput.value;
  1476. currentConfig.customplayer.params.referer = refererParamInput.value;
  1477. currentConfig.customplayer.params.origin = originParamInput.value;
  1478. PLAYER.customplayer = currentConfig.customplayer;
  1479. GM_setValue(KEY.config, currentConfig);
  1480. switchPlayer($(`input:radio[name="${ID.playerRadio}"]:checked`).val());
  1481. settingTable.style.display = "flex";
  1482. customplayerSettingTable.style.display = "none";
  1483. customplayerSettingButton.style.backgroundImage = ICON_BASE64.customplayer;
  1484. customplayerSettingButton.dataset.tip = "设置自定义播放器";
  1485. } else {
  1486. videoUrlParamInput.value = currentConfig.customplayer.params.videoUrl;
  1487. audioUrlParamInput.value = currentConfig.customplayer.params.audioUrl;
  1488. subtitleUrlParamInput.value = currentConfig.customplayer.params.subtitleUrl;
  1489. titleParamInput.value = currentConfig.customplayer.params.title;
  1490. startTimeParamInput.value = currentConfig.customplayer.params.startTime;
  1491. proxyParamInput.value = currentConfig.customplayer.params.proxy;
  1492. refererParamInput.value = currentConfig.customplayer.params.referer;
  1493. originParamInput.value = currentConfig.customplayer.params.origin;
  1494. settingTable.style.display = "none";
  1495. customplayerSettingTable.style.display = "flex";
  1496. customplayerSettingButton.style.backgroundImage = ICON_BASE64.back;
  1497. customplayerSettingButton.dataset.tip = "保存并返回";
  1498. }
  1499. }
  1500. }
  1501. // 切换元素状态
  1502. function switchStatus(element, flag) {
  1503. if (flag) {
  1504. element.readOnly = false;
  1505. element.disabled = false;
  1506. element.classList.remove(CLASS.readOnly);
  1507. } else {
  1508. element.readOnly = true;
  1509. element.disabled = true;
  1510. element.classList.add(CLASS.readOnly);
  1511. }
  1512. }
  1513. function sleep(ms) {
  1514. return new Promise(resolve => setTimeout(resolve, ms));
  1515. }
  1516. // 加载配置
  1517. function loadConfig() {
  1518. let oldConifg = GM_getValue(KEY.config);
  1519. currentConfig = JSON.parse(JSON.stringify(DEFAULT_CONFIG));
  1520. currentConfig.mpv.regVersion = "";
  1521. currentConfig.potplayer.regVersion = "";
  1522. if (!oldConifg) {
  1523. GM_setValue(KEY.config, currentConfig);
  1524. } else {
  1525. if (oldConifg.version != DEFAULT_CONFIG.version) {
  1526. for (const key in oldConifg) {
  1527. currentConfig[key] = oldConifg[key];
  1528. }
  1529. if (!currentConfig.mpv.path && currentConfig.mpvPath) {
  1530. currentConfig.mpv.path = currentConfig.mpvPath;
  1531. delete currentConfig['mpvPath'];
  1532. }
  1533. currentConfig.version = DEFAULT_CONFIG.version;
  1534. GM_setValue(KEY.config, currentConfig);
  1535. } else {
  1536. currentConfig = oldConifg;
  1537. GM_setValue(KEY.config, currentConfig);
  1538. }
  1539. }
  1540. PLAYER.customplayer = currentConfig.customplayer;
  1541. }
  1542. class Media {
  1543. constructor() {
  1544. this.title = "";
  1545. this.videoUrl = "";
  1546. this.audioUrl = "";
  1547. this.subtitleUrl = "";
  1548. this.startTime = "";
  1549. this.referer = "";
  1550. this.origin = "";
  1551. this.proxy = "";
  1552. this.other = "";
  1553. }
  1554. setTitle(title) {
  1555. this.title = title;
  1556. }
  1557. setVideoUrl(videoUrl) {
  1558. if (this.check(videoUrl)) {
  1559. this.videoUrl = videoUrl;
  1560. if (!this.title) {
  1561. this.setTitle(document.title);
  1562. }
  1563. let nxParserIframe = document.getElementById(ID.nxParserIframe);
  1564. if (nxParserIframe) {
  1565. document.body.removeChild(nxParserIframe);
  1566. }
  1567. if (document.getElementById(ID.buttonDiv)) {
  1568. document.getElementById(ID.buttonDiv).style.display = "flex";
  1569. if (currentConfig.playAuto == 1 && page.url !== "https://www.lckp.top/play-with-mpv/index.html") {
  1570. playButtonClick(currentConfig.player);
  1571. }
  1572. document.getElementById(ID.infoButton).style.visibility = "visible";
  1573. document.getElementById(ID.settingButton).style.visibility = "visible";
  1574. document.getElementById(ID.mpvPlayButton).style.visibility = "visible";
  1575. document.getElementById(ID.potplayerPlayButton).style.visibility = "visible";
  1576. document.getElementById(ID.customplayerPlayButton).style.visibility = "visible";
  1577. setTimeout(() => {
  1578. document.getElementById(ID.infoButton).style.visibility = "hidden";
  1579. document.getElementById(ID.settingButton).style.visibility = "hidden";
  1580. document.getElementById(ID.mpvPlayButton).style.visibility = "hidden";
  1581. document.getElementById(ID.potplayerPlayButton).style.visibility = "hidden";
  1582. document.getElementById(ID.customplayerPlayButton).style.visibility = "hidden";
  1583. }, TIME.showButton);
  1584. }
  1585. }
  1586. }
  1587. setAudioUrl(audioUrl) {
  1588. this.audioUrl = audioUrl;
  1589. }
  1590. setSubtitleUrl(subtitleUrl) {
  1591. this.subtitleUrl = subtitleUrl;
  1592. }
  1593. setStartTime(startTime) {
  1594. this.startTime = Math.floor(startTime);
  1595. }
  1596. setReferer(referer) {
  1597. this.referer = referer;
  1598. }
  1599. setOrigin(origin) {
  1600. this.origin = origin;
  1601. }
  1602. setProxy(proxy) {
  1603. this.proxy = proxy;
  1604. }
  1605. setOther(other) {
  1606. this.other = other;
  1607. }
  1608. // 检查视频链接是否有效
  1609. check(videoUrl) {
  1610. if (videoUrl && videoUrl.startsWith("http")) {
  1611. if (videoUrl.match(/(\.m3u|\.m3u8)/g) && videoUrl != localStorage.iptvUrl) {
  1612. let m3u8 = "";
  1613. $.ajax({
  1614. type: "GET",
  1615. url: videoUrl,
  1616. async: false,
  1617. success: function (res) {
  1618. m3u8 = res;
  1619. }
  1620. });
  1621. if (m3u8 && m3u8.indexOf("png") != -1) {
  1622. console.log("Play-With-MPV:m3u8 链接无法播放:" + videoUrl);
  1623. return false;
  1624. }
  1625. }
  1626. if (videoUrl.startsWith("https://www.mp4")) {
  1627. return false;
  1628. }
  1629. return true;
  1630. }
  1631. console.log(`Play-With-MPV:链接无效:${videoUrl}`);
  1632. return false;
  1633. }
  1634. }
  1635. class BaseHandler {
  1636. constructor() {
  1637. loadConfig();
  1638. this.media = new Media();
  1639. this.media.setProxy(currentConfig.proxy);
  1640. for (const key in PLAYER) {
  1641. if (PLAYER[key].name == currentConfig.player) {
  1642. this.player = PLAYER[key];
  1643. break;
  1644. }
  1645. }
  1646. if (window.self == window.top) {
  1647. if (!document.getElementById(ID.buttonDiv)) {
  1648. console.log(INFO);
  1649. appendCSS();
  1650. appendHTML();
  1651. addListener();
  1652. }
  1653. document.getElementById(ID.buttonDiv).style.display = "none";
  1654. }
  1655. }
  1656. initCheck() {
  1657. return window.location.href != page.url;
  1658. }
  1659. async parse() { }
  1660. pause() {
  1661. let videos = document.getElementsByTagName("video");
  1662. if (videos && videos.length > 0) {
  1663. let i = 0;
  1664. while (i < TRY_TIME.maxPause) {
  1665. setTimeout(function () {
  1666. for (let video of videos) {
  1667. video.pause();
  1668. }
  1669. }, TIME.pauseInterval * i);
  1670. i++;
  1671. }
  1672. } else if (this.iframe) {
  1673. this.iframe.postMessage({ method: METHOD.pause }, "*");
  1674. }
  1675. }
  1676. play() {
  1677. let link = "";
  1678. for (let key in this.player.params) {
  1679. if (key == "title") {
  1680. continue;
  1681. }
  1682. if (key == "startTime") {
  1683. if (currentConfig.syncStartTime != 1) {
  1684. continue;
  1685. } else {
  1686. let videos = document.getElementsByTagName("video");
  1687. for (let i = 0; i < videos.length; i++) {
  1688. if (videos[i].currentTime != 0) {
  1689. this.media.setStartTime(videos[i].currentTime);
  1690. break;
  1691. }
  1692. }
  1693. }
  1694. }
  1695. let value = this.media[key];
  1696. if (value) {
  1697. let param = this.player.params[key];
  1698. let matchKey = '${' + key + '}';
  1699. while (param.indexOf(matchKey) != -1) {
  1700. param = param.replace(matchKey, value);
  1701. }
  1702. matchKey = '${E' + key + '}';
  1703. while (param.indexOf(matchKey) != -1) {
  1704. param = param.replace(matchKey, encodeURIComponent(value));
  1705. }
  1706. matchKey = '${D' + key + '}';
  1707. while (param.indexOf(matchKey) != -1) {
  1708. param = param.replace(matchKey, decodeURIComponent(value));
  1709. }
  1710. link = link + param;
  1711. }
  1712. }
  1713. if (this.media.title) {
  1714. let maxLength = 1950 - link.length;
  1715. let title = encodeURIComponent(this.media.title);
  1716. if (title.length > maxLength) {
  1717. title = title.substring(0, maxLength) + '...';
  1718. }
  1719. let param = this.player.params.title;
  1720. param = param.replace('${title}', title);
  1721. link = link + param;
  1722. }
  1723. window.open(link, "_self");
  1724. }
  1725. // 监听子页面事件
  1726. addIframeListener() {
  1727. let that = this;
  1728. window.addEventListener("message", function (event) {
  1729. that.iframe = event.source;
  1730. if (event.data.method == METHOD.pause) {
  1731. that.pause();
  1732. } else if (event.data.method == METHOD.report) {
  1733. that.media.setStartTime(event.data.media.startTime);
  1734. if (!that.media.videoUrl) {
  1735. that.media.setVideoUrl(event.data.media.videoUrl);
  1736. }
  1737. }
  1738. }, false);
  1739. }
  1740. // 监听顶层页面事件
  1741. addTopListener() {
  1742. let that = this;
  1743. window.addEventListener("message", function (event) {
  1744. if (event.data.method == METHOD.pause) {
  1745. that.pause();
  1746. }
  1747. }, false);
  1748. // 定时上报当前视频信息
  1749. setInterval(() => {
  1750. let video = document.getElementsByTagName("video")[0];
  1751. if (video) {
  1752. this.media.setStartTime(video.currentTime);
  1753. }
  1754. window.top.postMessage({ method: METHOD.report, media: that.media }, "*");
  1755. }, TIME.reportInterval);
  1756. }
  1757. // yt-dlp 支持网站解析器
  1758. ytDlpParser() {
  1759. return page.url;
  1760. }
  1761. // video 元素解析器
  1762. videoParser() {
  1763. let videos = document.getElementsByTagName("video");
  1764. for (let video of videos) {
  1765. let url = video.src;
  1766. if (url && url.startsWith("http")) {
  1767. return url;
  1768. }
  1769. }
  1770. }
  1771. // iframe 元素解析器
  1772. iframeParser() {
  1773. let iframes = document.getElementsByTagName("iframe");
  1774. for (let iframe of iframes) {
  1775. let urls = iframe.src.match(VIDEO_URL_REGEX);
  1776. if (urls && urls.length > 0) {
  1777. return urls[0];
  1778. }
  1779. }
  1780. }
  1781. // html 解析器
  1782. htmlParser() {
  1783. let urls = document.body.innerHTML.match(VIDEO_URL_REGEX);
  1784. if (urls && urls.length > 0) {
  1785. return urls[0];
  1786. }
  1787. }
  1788. // script 解析器
  1789. scriptParser() {
  1790. for (let script of document.scripts) {
  1791. let urls = script.innerHTML.match(VIDEO_URL_REGEX);
  1792. if (urls && urls.length > 0) {
  1793. return urls[0];
  1794. }
  1795. }
  1796. }
  1797. // url 解析器
  1798. urlParser() {
  1799. let urls = page.url.match(VIDEO_URL_REGEX);
  1800. if (urls && urls.length > 0) {
  1801. return urls[0];
  1802. }
  1803. }
  1804. // 诺讯解析
  1805. nxParser() {
  1806. handler.addIframeListener();
  1807. let nxParserIframe = document.getElementById(ID.nxParserIframe);
  1808. if (!nxParserIframe) {
  1809. nxParserIframe = document.createElement("iframe");
  1810. nxParserIframe.id = ID.nxParserIframe;
  1811. nxParserIframe.src = `https://yun.nxflv.com/?url=${page.url}`;
  1812. document.body.appendChild(nxParserIframe);
  1813. }
  1814. }
  1815. }
  1816. // 获取B站视频播放链接
  1817. async function getBilibiliPlayUrl(avid, cid) {
  1818. if (handler.player.name == PLAYER.mpv.name) {
  1819. handler.media.setOther(`--script-opts="cid=${cid}"`);
  1820. }
  1821. if (!handler.player.params.audioUrl || await getBilibiliVideoDash(avid, cid) == -1) {
  1822. getBilibiliVideoDurl(avid, cid);
  1823. }
  1824. if (currentConfig.subtitlePrefer != "off") {
  1825. getBilibiliVideoSubtitle(avid, cid);
  1826. }
  1827. }
  1828. // 获取B站 DASH 格式视频
  1829. async function getBilibiliVideoDash(avid, cid) {
  1830. let result = 0;
  1831. await $.ajax({
  1832. type: "GET",
  1833. url: `https://api.bilibili.com/x/player/playurl?qn=120&otype=json&fourk=1&fnver=0&fnval=4048&avid=${avid}&cid=${cid}`,
  1834. xhrFields: {
  1835. withCredentials: true
  1836. },
  1837. async: false,
  1838. success: function (res) {
  1839. if (!res.data) {
  1840. toast("Play-With-MPV 获取视频失败,如未登录请先登录并刷新页面", TOAST_TYPE.error);
  1841. tryTime = TRY_TIME.maxParse;
  1842. return;
  1843. }
  1844. let videoUrl = undefined;
  1845. let audioUrl = undefined;
  1846. let dash = res.data.dash;
  1847. if (!dash) {
  1848. result = -1;
  1849. return;
  1850. }
  1851. let hiRes = dash.flac;
  1852. let dolby = dash.dolby;
  1853. if (hiRes && hiRes.audio) {
  1854. audioUrl = hiRes.audio.baseUrl;
  1855. } else if (dolby && dolby.audio) {
  1856. audioUrl = dolby.audio[0].base_url;
  1857. } else if (dash.audio) {
  1858. audioUrl = dash.audio[0].baseUrl;
  1859. }
  1860. let i = 0;
  1861. while (i < dash.video.length && dash.video[i].id > BEST_QUALITY.bilibili[currentConfig.bestQuality]) {
  1862. i++;
  1863. }
  1864. videoUrl = dash.video[i].baseUrl;
  1865. let id = dash.video[i].id;
  1866. while (i < dash.video.length) {
  1867. if (dash.video[i].id != id) {
  1868. break;
  1869. }
  1870. if (dash.video[i].codecid == currentConfig.bilibiliCodecs) {
  1871. videoUrl = dash.video[i].baseUrl;
  1872. break;
  1873. }
  1874. i++;
  1875. }
  1876. handler.media.setAudioUrl(audioUrl);
  1877. handler.media.setVideoUrl(videoUrl);
  1878. result = 1;
  1879. }
  1880. });
  1881. return result;
  1882. }
  1883. // 获取B站 FLV / MP4 格式视频
  1884. function getBilibiliVideoDurl(avid, cid) {
  1885. $.ajax({
  1886. type: "GET",
  1887. url: `https://api.bilibili.com/x/player/playurl?qn=120&otype=json&fourk=1&fnver=0&fnval=128&avid=${avid}&cid=${cid}`,
  1888. xhrFields: {
  1889. withCredentials: true
  1890. },
  1891. async: false,
  1892. success: function (res) {
  1893. if (!res.data) {
  1894. toast("Play-With-MPV 获取视频失败,如未登录请先登录并刷新页面", TOAST_TYPE.error);
  1895. tryTime = TRY_TIME.maxParse;
  1896. return;
  1897. }
  1898. handler.media.setVideoUrl(res.data.durl[0].url);
  1899. }
  1900. });
  1901. }
  1902. // 获取B站视频字幕
  1903. function getBilibiliVideoSubtitle(avid, cid) {
  1904. $.ajax({
  1905. type: "GET",
  1906. url: `https://api.bilibili.com/x/player/v2?aid=${avid}&cid=${cid}`,
  1907. xhrFields: {
  1908. withCredentials: true
  1909. },
  1910. async: false,
  1911. success: function (res) {
  1912. if (res.code == 0 && res.data.subtitle && res.data.subtitle.subtitles.length > 0) {
  1913. let subtitles = res.data.subtitle.subtitles;
  1914. let url = "https:" + subtitles[0].subtitle_url;
  1915. let lan = subtitles[0].lan;
  1916. for (const subtitle of subtitles) {
  1917. if (currentConfig.subtitlePrefer.startsWith("zh") && subtitle.lan.startsWith("zh")) {
  1918. url = "https:" + subtitle.subtitle_url;
  1919. lan = subtitle.lan;
  1920. }
  1921. if (subtitle.lan == currentConfig.subtitlePrefer) {
  1922. url = "https:" + subtitle.subtitle_url;
  1923. lan = subtitle.lan;
  1924. break;
  1925. }
  1926. }
  1927. handler.media.setSubtitleUrl(`https://www.lckp.top/common/bilibili/jsonToSrt/?url=${url}&lan=${lan}`);
  1928. }
  1929. }
  1930. });
  1931. }
  1932. // 获取B站视频 aid 和 cid
  1933. function getBilibiliVideoId() {
  1934. let hasInitialState = false;
  1935. try {
  1936. if (__INITIAL_STATE__) {
  1937. hasInitialState = true;
  1938. }
  1939. } catch (error) {
  1940. hasInitialState = false;
  1941. }
  1942. if (!hasInitialState) {
  1943. return undefined;
  1944. }
  1945. let video = undefined;
  1946. if (__INITIAL_STATE__.epInfo) {
  1947. video = __INITIAL_STATE__.epInfo;
  1948. } else if (__INITIAL_STATE__.videoData) {
  1949. video = __INITIAL_STATE__.videoData;
  1950. } else if (__INITIAL_STATE__.videoInfo) {
  1951. video = __INITIAL_STATE__.videoInfo;
  1952. }
  1953. let aid = video.aid;
  1954. let cid = video.cid;
  1955. let p = __INITIAL_STATE__.p;
  1956. if (p && p > 1) {
  1957. cid = __INITIAL_STATE__.cidMap[aid].cids[p];
  1958. }
  1959. let videoId = {
  1960. aid: aid,
  1961. cid: cid
  1962. };
  1963. return videoId;
  1964. }
  1965. const BEST_QUALITY = {
  1966. bilibili: {
  1967. "unlimited": 127,
  1968. "2160p": 126,
  1969. "1440p": 116,
  1970. "1080p": 116,
  1971. "720p": 74,
  1972. "480p": 32
  1973. },
  1974. bilibiliLive: {
  1975. "unlimited": 4,
  1976. "2160p": 4,
  1977. "1440p": 4,
  1978. "1080p": 4,
  1979. "720p": 3,
  1980. "480p": 2
  1981. },
  1982. ytdlp: {
  1983. "unlimited": "",
  1984. "2160p": "--ytdl-format=bestvideo[height<=?2160]%2Bbestaudio/best",
  1985. "1440p": "--ytdl-format=bestvideo[height<=?1440]%2Bbestaudio/best",
  1986. "1080p": "--ytdl-format=bestvideo[height<=?1080]%2Bbestaudio/best",
  1987. "720p": "--ytdl-format=bestvideo[height<=?720]%2Bbestaudio/best",
  1988. "480p": "--ytdl-format=bestvideo[height<=?480]%2Bbestaudio/best"
  1989. }
  1990. }
  1991. var websiteList = [
  1992. {
  1993. // ✅ https://www.bilibili.com/bangumi/play/ep508404
  1994. // ✅ https://www.bilibili.com/bangumi/play/ep319063
  1995. name: "B站影视",
  1996. home: [
  1997. "https://www.bilibili.com"
  1998. ],
  1999. regex: /^https:\/\/www\.bilibili\.com\/bangumi\/play\/.*/g,
  2000. handler: class Handler extends BaseHandler {
  2001. constructor() {
  2002. super();
  2003. this.media.setReferer("https://www.bilibili.com");
  2004. }
  2005. async parse() {
  2006. // 直接从数据中获取 aid 和 cid
  2007. // let videoId = getBilibiliVideoId();
  2008. // if (videoId && videoId.aid && videoId.cid) {
  2009. // getBilibiliPlayUrl(videoId.aid, videoId.cid);
  2010. // return;
  2011. // }
  2012.  
  2013. // 从元素提取 epid 请求接口获取 aid 和 cid
  2014. let epid = page.url.match(/ep(\d+)/);
  2015. if (epid && epid[1]) {
  2016. epid = epid[1];
  2017. } else {
  2018. let epidElement = document.getElementsByClassName("ep-item cursor visited")[0];
  2019. if (!epidElement) {
  2020. epidElement = document.getElementsByClassName('ep-item cursor')[0];
  2021. }
  2022. if (epidElement) {
  2023. epid = epidElement.getElementsByTagName('a')[0].href.match(/ep(\d+)/)[1];
  2024. } else {
  2025. epidElement = document.getElementsByClassName("squirtle-pagelist-select-item active squirtle-blink")[0];
  2026. if (epidElement) {
  2027. epid = epidElement.dataset.value;
  2028. }
  2029. }
  2030. }
  2031. if (!epid) {
  2032. return;
  2033. }
  2034. $.ajax({
  2035. type: "GET",
  2036. url: `https://api.bilibili.com/pgc/view/web/season?ep_id=${epid}`,
  2037. xhrFields: {
  2038. withCredentials: true
  2039. },
  2040. async: false,
  2041. success: function (res) {
  2042. let currentEpisode;
  2043. let section = res.result.section;
  2044. if (!section) {
  2045. section = new Array();
  2046. }
  2047. section.push({ episodes: res.result.episodes });
  2048. for (let i = section.length - 1; i >= 0; i--) {
  2049. let episodes = section[i].episodes;
  2050. for (const episode of episodes) {
  2051. if (episode.id == epid) {
  2052. currentEpisode = episode;
  2053. break;
  2054. }
  2055. }
  2056. if (currentEpisode) {
  2057. break;
  2058. }
  2059. }
  2060. getBilibiliPlayUrl(currentEpisode.aid, currentEpisode.cid);
  2061. }
  2062. })
  2063. }
  2064. },
  2065. },
  2066. {
  2067. // ✅ https://www.bilibili.com/video/BV1Hd4y1k7Vb
  2068. // ✅ https://www.bilibili.com/video/av2
  2069. // ✅ https://www.bilibili.com/video/BV17Z4y117Qm
  2070. // ✅ https://www.bilibili.com/list/ml1806211634?oid=822115390&bvid=BV1Fg4y1p7Qe
  2071. name: "B站投稿",
  2072. regex: /^https:\/\/www\.bilibili\.com\/(video\/|list.*)(BV|av).*/g,
  2073. handler: class Handler extends BaseHandler {
  2074. constructor() {
  2075. super();
  2076. this.media.setReferer("https://www.bilibili.com");
  2077. }
  2078. initCheck() {
  2079. if (super.initCheck()) {
  2080. let newPageUrl = window.location.href;
  2081. let oldPageUrl = page.url;
  2082. let regex = /(&|\?)vd_source=\w+/;
  2083. if (regex.test(newPageUrl.replace(oldPageUrl, ""))) {
  2084. page.url = newPageUrl;
  2085. return false;
  2086. }
  2087. return true;
  2088. }
  2089. return false;
  2090. }
  2091. async parse() {
  2092. // 直接从数据中获取 aid 和 cid
  2093. // let videoId = getBilibiliVideoId();
  2094. // if (videoId && videoId.aid && videoId.cid) {
  2095. // getBilibiliPlayUrl(videoId.aid, videoId.cid);
  2096. // return;
  2097. // }
  2098.  
  2099. // 通过 bvid/avid 请求接口获取 aid 和 cid
  2100. let param = undefined;
  2101. let bvid = page.url.match(/BV([0-9a-zA-Z]+)/);
  2102. if (bvid && bvid[1]) {
  2103. param = `bvid=${bvid[1]}`;
  2104. } else {
  2105. let avid = page.url.match(/av([0-9]+)/);
  2106. if (avid && avid[1]) {
  2107. param = `aid=${avid[1]}`;
  2108. }
  2109. }
  2110. if (!param) {
  2111. return;
  2112. }
  2113. $.ajax({
  2114. type: "GET",
  2115. url: `https://api.bilibili.com/x/web-interface/view?${param}`,
  2116. xhrFields: {
  2117. withCredentials: true
  2118. },
  2119. async: false,
  2120. success: function (res) {
  2121. let aid = res.data.aid;
  2122. let cid = res.data.cid;
  2123. let index = page.url.indexOf("?p=");
  2124. if (index != -1 && res.data.pages.length > 1) {
  2125. let p = page.url.substring(index + 3);
  2126. let endIndex = p.indexOf("&");
  2127. if (endIndex != -1) {
  2128. p = p.substring(0, endIndex);
  2129. }
  2130. cid = res.data.pages[p - 1].cid;
  2131. }
  2132. getBilibiliPlayUrl(aid, cid);
  2133. }
  2134. })
  2135. }
  2136. },
  2137. },
  2138. {
  2139. // ✅ https://www.bilibili.com/festival/2023bnj?bvid=BV17G4y1X7vQ
  2140. name: "B站节日",
  2141. regex: /^https:\/\/www\.bilibili\.com\/festival\/.*/g,
  2142. handler: class Handler extends BaseHandler {
  2143. constructor() {
  2144. super();
  2145. this.media.setReferer("https://www.bilibili.com");
  2146. }
  2147. initCheck() {
  2148. if (super.initCheck()) {
  2149. return true;
  2150. }
  2151. let oldvideoId = this.videoId;
  2152. let newvideoId = getBilibiliVideoId();
  2153. if (oldvideoId && oldvideoId.cid != newvideoId.cid) {
  2154. return true;
  2155. }
  2156. return false;
  2157. }
  2158. async parse() {
  2159. let videoId = getBilibiliVideoId();
  2160. if (videoId && videoId.aid && videoId.cid) {
  2161. this.videoId = videoId;
  2162. getBilibiliPlayUrl(videoId.aid, videoId.cid);
  2163. return;
  2164. } else {
  2165. toast("Play-With-MPV 读取视频数据失败,请尝试清理B站缓存后刷新重试", TOAST_TYPE.error, 5000);
  2166. tryTime = TRY_TIME.maxParse;
  2167. }
  2168. }
  2169. },
  2170. },
  2171. {
  2172. // ✅ https://live.bilibili.com/7777
  2173. name: "B站直播",
  2174. home: [
  2175. "https://live.bilibili.com"
  2176. ],
  2177. regex: /^https:\/\/live\.bilibili\.com\/\d+.*/g,
  2178. handler: class Handler extends BaseHandler {
  2179. async parse() {
  2180. let iframes = document.getElementsByTagName("iframe");
  2181. let roomid = undefined;
  2182. for (let iframe of iframes) {
  2183. let roomids = iframe.src.match(/^https:\/\/live\.bilibili\.com.*(roomid=\d+|blanc\/\d+).*/);
  2184. if (roomids && roomids[1]) {
  2185. roomid = roomids[1].match(/\d+/)[0];
  2186. break;
  2187. }
  2188. }
  2189. if (!roomid) {
  2190. console.log("Play-With-MPV:找不到 roomid:" + roomid);
  2191. return;
  2192. }
  2193.  
  2194. let that = this;
  2195. $.ajax({
  2196. type: "GET",
  2197. url: `https://api.live.bilibili.com/room/v1/Room/playUrl?quality=${BEST_QUALITY.bilibiliLive[currentConfig.bestQuality]}&cid=${roomid}`,
  2198. async: false,
  2199. xhrFields: {
  2200. withCredentials: true
  2201. },
  2202. success: function (res) {
  2203. that.media.setVideoUrl(res.data.durl[0].url);
  2204. }
  2205. });
  2206. }
  2207. },
  2208. },
  2209. {
  2210. // ✅ https://www.ixigua.com/
  2211. name: "西瓜视频",
  2212. home: [
  2213. "https://www.ixigua.com"
  2214. ],
  2215. regex: /^https:\/\/www\.ixigua\.com\/\d.*/g,
  2216. handler: class Handler extends BaseHandler {
  2217. constructor() {
  2218. super();
  2219. this.media.setReferer("https://www.ixigua.com/");
  2220. }
  2221. async parse() {
  2222. let that = this;
  2223. $.ajax({
  2224. type: "GET",
  2225. url: page.url,
  2226. async: false,
  2227. success: function (res) {
  2228. try {
  2229. let _SSR_HYDRATED_DATA = (new Function("return " + res.match(/<script id="SSR_HYDRATED_DATA"[^<]*window._SSR_HYDRATED_DATA=({[^<]*})[^<]*<\/script>/)[1]))();
  2230. let packerData = _SSR_HYDRATED_DATA.anyVideo.gidInformation.packerData;
  2231. let main_url = undefined;
  2232. if (packerData.video) {
  2233. let videoList = packerData.video.videoResource.normal.video_list;
  2234. if (videoList) {
  2235. let video = undefined;
  2236. for (const key in videoList) {
  2237. if (!video || videoList[key].vheight > video.vheight) {
  2238. video = videoList[key];
  2239. }
  2240. }
  2241. main_url = video.main_url;
  2242. }
  2243. that.media.setVideoUrl(window.atob(main_url).replace("http://", "https://"));
  2244. return;
  2245. }
  2246. } catch (error) {
  2247. console.error("解析出错:" + error);
  2248. }
  2249. that.nxParser();
  2250. }
  2251. });
  2252. }
  2253. },
  2254. },
  2255. {
  2256. // ✅ https://yun.nxflv.com/?url=https://www.ixigua.com/7186534626612118071
  2257. name: "诺讯解析",
  2258. regex: /^https:\/\/yun\.nxflv\.com\/\?url=.+/g,
  2259. handler: class Handler extends BaseHandler {
  2260. constructor() {
  2261. super();
  2262. this.addTopListener();
  2263. }
  2264. async parse() {
  2265. let url = this.videoParser();
  2266. if (url.startsWith("blob")) {
  2267. for (let index = 0; index < sessionStorage.key.length; index++) {
  2268. url = sessionStorage.key(index);
  2269. url = url.match(/http[^#]*/g);
  2270. if (url && url.length > 0) {
  2271. url = url[0];
  2272. }
  2273. }
  2274. }
  2275. this.media.setVideoUrl(url);
  2276. }
  2277. },
  2278. },
  2279. {
  2280. // ✅ https://ddys.art/bleach-thousand-year-blood-war
  2281. name: "低端影视",
  2282. home: [
  2283. "https://ddys.art",
  2284. "https://ddys.pro"
  2285. ],
  2286. regex: /^https:\/\/(ddys\.art|ddys\.pro)\/.*/g,
  2287. handler: class Handler extends BaseHandler {
  2288. async parse() {
  2289. let video = document.getElementsByTagName("video")[0];
  2290. if (video.paused) {
  2291. document.getElementsByClassName('vjs-big-play-button')[0].click();
  2292. }
  2293. let url = this.videoParser();
  2294. if (url) {
  2295. let index = url.indexOf("?");
  2296. if (index != -1) {
  2297. url = url.substring(0, index + 1) + encodeURIComponent(url.substring(index + 1));
  2298. }
  2299. this.media.setVideoUrl(url);
  2300. let playing = document.getElementsByClassName("wp-playlist-playing")[0];
  2301. if (playing) {
  2302. let episode = playing.textContent.replace(/(\n|\t|\d\.)/g, "");
  2303. if (episode != " 全") {
  2304. this.media.title = document.getElementsByClassName("post-title")[0].textContent + episode + " - 低端影视";
  2305. }
  2306. }
  2307. }
  2308. }
  2309. },
  2310. },
  2311. {
  2312. // ✅ https://www.libvio.cc/play/714634-1-11.html
  2313. name: "LIBVIO",
  2314. home: [
  2315. "https://www.libvio.cc",
  2316. "https://libvio.fun",
  2317. "https://libvio.me",
  2318. "https://www.libvio.me"
  2319. ],
  2320. regex: /^https?:\/\/.*\.libvio\..*\/play.*/g,
  2321. handler: class Handler extends BaseHandler {
  2322. constructor() {
  2323. super();
  2324. this.addIframeListener();
  2325. }
  2326. }
  2327. },
  2328. {
  2329. name: "LIBVIO播放器",
  2330. regex: /^https:\/\/(.*\.chinaeast2\.cloudapp\.chinacloudapi\.cn|.*\.cfnode1\.xyz)\/.*\?url=.*/g,
  2331. handler: class Handler extends BaseHandler {
  2332. constructor() {
  2333. super();
  2334. this.addTopListener();
  2335. }
  2336. async parse() {
  2337. let url = urls;
  2338. let index = url.indexOf("?");
  2339. if (index != -1) {
  2340. url = url.substring(0, index + 1) + encodeURIComponent(url.substring(index + 1));
  2341. }
  2342. this.media.setVideoUrl(url);
  2343. }
  2344. }
  2345. },
  2346. {
  2347. // ✅ https://www.nivod.tv/UXEwMmLqnUjHG5e4MwmlvmVnWiAJ9rIQ-RofV7wPhhed3uoi50mYsftLPq4mYyIhB-720-0-0-play.html?x=1
  2348. name: "泥视频",
  2349. home: [
  2350. "https://www.nivod.tv",
  2351. ],
  2352. regex: /^https:\/\/www\.nivod\.tv\/.*play\.html?.*/g,
  2353. handler: class Handler extends BaseHandler {
  2354. async parse() {
  2355. this.media.setVideoUrl(__dp.options.video.url);
  2356. this.media.setTitle(document.title);
  2357. }
  2358. }
  2359. },
  2360. {
  2361. // ✅ https://www.pkmkv.com/py/268677-2-11.html
  2362. name: "片库",
  2363. home: [
  2364. "https://www.pkmkv.com",
  2365. ],
  2366. regex: /^https:\/\/www\.pkmkv\.com\/py\/.*/g,
  2367. handler: class Handler extends BaseHandler {
  2368. constructor() {
  2369. super();
  2370. this.addIframeListener();
  2371. }
  2372. async parse() {
  2373. this.media.setVideoUrl(player_aaaa.url);
  2374. }
  2375. }
  2376. },
  2377. {
  2378. name: "片库播放器",
  2379. regex: /^https:\/\/www\.pkmkv\.com\/addons\/dplayer\/\?url=.*/g,
  2380. handler: class Handler extends BaseHandler {
  2381. constructor() {
  2382. super();
  2383. this.addTopListener();
  2384. }
  2385. }
  2386. },
  2387. {
  2388. // ✅ https://www.btnull.org/py/BBnLd_9.html?167094
  2389. name: "无名小站",
  2390. home: [
  2391. "https://www.btnull.org",
  2392. "https://www.btnull.to",
  2393. "https://www.btnull.nu",
  2394. "https://www.btnull.in",
  2395. ],
  2396. regex: /^https:\/\/(www.btnull.org|www.btnull.to|www.btnull.nu|www.btnull.in)\/py\/.*/g,
  2397. handler: class Handler extends BaseHandler {
  2398. async parse() {
  2399. this.media.setVideoUrl(this.htmlParser());
  2400. }
  2401. }
  2402. },
  2403. {
  2404. // ✅ https://www.916dm.com/play/6792-1-91.html
  2405. // ✅ http://www.916dm.com/play/7800-1-10.html
  2406. // ✅ http://www.dmlaa.com/play/7696-1-10.html
  2407. // ✅ http://www.qdmsh.com/play/7663-1-10.html
  2408. // ✅ http://www.ntdm8.com/play/4973-1-1.html
  2409. name: "樱花动漫网",
  2410. home: [
  2411. "https://www.916dm.com",
  2412. "http://www.916dm.com",
  2413. "http://www.dmlaa.com",
  2414. "http://www.qdmsh.com",
  2415. "http://www.ntdm8.com"
  2416. ],
  2417. regex: /^https?:\/\/www\.(\d+dm|dmlaa|qdmsh|ntdm8)\.com\/play\/.*/g,
  2418. handler: class Handler extends BaseHandler {
  2419. constructor() {
  2420. super();
  2421. this.addIframeListener();
  2422. }
  2423. }
  2424. },
  2425. {
  2426. name: "樱花动漫网播放器",
  2427. regex: /^https:\/\/danmu\.yhdmjx\.com\/.*php\?url=.*/g,
  2428. handler: class Handler extends BaseHandler {
  2429. constructor() {
  2430. super();
  2431. this.addTopListener();
  2432. }
  2433. async parse() {
  2434. this.media.setVideoUrl(this.videoParser());
  2435. }
  2436. }
  2437. },
  2438. {
  2439. // ✅ https://dick.xfani.com/watch/582/1/11.html
  2440. name: "稀饭动漫",
  2441. home: [
  2442. "https://dick.xfani.com"
  2443. ],
  2444. regex: /^https:\/\/dick\.xfani\.com\/watch\/.*/g,
  2445. handler: class Handler extends BaseHandler {
  2446. constructor() {
  2447. super();
  2448. this.addIframeListener();
  2449. }
  2450. }
  2451. },
  2452. {
  2453. name: "稀饭动漫播放器",
  2454. regex: /(^https:\/\/dick\.xfani\.com\/addons\/dp\/player\/.*|^https:\/\/m3\.moedot\.net\/muiplayer\/\?url=.*)/g,
  2455. handler: class Handler extends BaseHandler {
  2456. constructor() {
  2457. super();
  2458. this.addTopListener();
  2459. }
  2460. async parse() {
  2461. if (config.url.indexOf(".m3u8") > 0 || config.url.indexOf(".mp4") > 0 || config.url.indexOf(".flv") > 0) {
  2462. this.media.setVideoUrl(config.url);
  2463. } else {
  2464. let that = this;
  2465. $.ajax({
  2466. type: "POST",
  2467. url: "api_currentConfig.php",
  2468. data: { "url": config.url, "time": config.time, "key": config.key, "title": config.title },
  2469. async: false,
  2470. success: function (res) {
  2471. if (res.code == "200") {
  2472. that.media.setVideoUrl(res.url);
  2473. }
  2474. }
  2475. });
  2476. }
  2477. }
  2478. }
  2479. },
  2480. {
  2481. name: "稀饭动漫播放器",
  2482. regex: /^https:\/\/player\.moedot\.net\/player\/.*/g,
  2483. handler: class Handler extends BaseHandler {
  2484. constructor() {
  2485. super();
  2486. this.addTopListener();
  2487. }
  2488. async parse() {
  2489. this.media.setVideoUrl(this.urlParser());
  2490. }
  2491. }
  2492. },
  2493. {
  2494. // ✅ https://www.mgnacg.com/bangumi/426-6-12
  2495. name: "橘子动漫",
  2496. home: [
  2497. "https://www.mgnacg.com"
  2498. ],
  2499. regex: /^https:\/\/www\.mgnacg\.com\/bangumi\/.*/g,
  2500. handler: class Handler extends BaseHandler {
  2501. constructor() {
  2502. super();
  2503. this.addIframeListener();
  2504. this.media.setReferer("https://play.mknacg.top:8585/");
  2505. }
  2506. }
  2507. },
  2508. {
  2509. name: "橘子动漫播放器",
  2510. regex: /^https:\/\/play\.mknacg\.top:8585\/.*/g,
  2511. handler: class Handler extends BaseHandler {
  2512. constructor() {
  2513. super();
  2514. this.addTopListener();
  2515. }
  2516. async parse() {
  2517. this.media.setVideoUrl(art.option.url);
  2518. }
  2519. }
  2520. },
  2521. {
  2522. // ✅ https://www.omofun.top/index.php/vod/play/id/17295/sid/2/nid/7.html
  2523. name: "OmoFun",
  2524. home: [
  2525. "https://www.omofun.top"
  2526. ],
  2527. regex: /^https:\/\/www\.omofun\.top\/index.php\/vod\/play\/id\/.*/g,
  2528. handler: class Handler extends BaseHandler {
  2529. constructor() {
  2530. super();
  2531. this.addIframeListener();
  2532. }
  2533. }
  2534. },
  2535. {
  2536. name: "OmoFun播放器",
  2537. regex: /^https:\/\/.*\.omofun\.top\/(player\/|)\?url=.*/g,
  2538. handler: class Handler extends BaseHandler {
  2539. constructor() {
  2540. super();
  2541. this.addTopListener();
  2542. }
  2543. async parse() {
  2544. this.media.setVideoUrl(this.urlParser());
  2545. }
  2546. }
  2547. },
  2548. {
  2549. // ✅ https://spdcat.net/vodplay/135443-1-23
  2550. name: "迅猫动漫",
  2551. home: [
  2552. "https://spdcat.net"
  2553. ],
  2554. regex: /^https:\/\/spdcat\.net\/vodplay\/.*/g,
  2555. handler: class Handler extends BaseHandler {
  2556. constructor() {
  2557. super();
  2558. this.addIframeListener();
  2559. }
  2560. }
  2561. },
  2562. {
  2563. name: "迅猫动漫播放器",
  2564. regex: /^https:\/\/spdcat\.net\/addons\/dp\/player\/.*/g,
  2565. handler: class Handler extends BaseHandler {
  2566. constructor() {
  2567. super();
  2568. this.addTopListener();
  2569. }
  2570. async parse() {
  2571. this.media.setVideoUrl(config.url);
  2572. setTimeout(() => {
  2573. let conplay = document.getElementsByClassName("conplay-jump")[0];
  2574. if (conplay) {
  2575. conplay.click();
  2576. }
  2577. }, 1000);
  2578. }
  2579. }
  2580. },
  2581. {
  2582. // ✅ http://www.dm88.me/player/8480-0-11.html
  2583. name: "樱花动漫",
  2584. home: [
  2585. "http://www.dm88.me"
  2586. ],
  2587. regex: /^http:\/\/www\.dm88\.me\/player\/.*/g,
  2588. handler: class Handler extends BaseHandler {
  2589. constructor() {
  2590. super();
  2591. this.addIframeListener();
  2592. }
  2593. }
  2594. },
  2595. {
  2596. name: "樱花动漫播放器",
  2597. regex: /^https:\/\/jianghu\.live2008\.com\/.*\?url=.*/g,
  2598. handler: class Handler extends BaseHandler {
  2599. constructor() {
  2600. super();
  2601. this.addTopListener();
  2602. }
  2603. async parse() {
  2604. this.media.setVideoUrl(url);
  2605. }
  2606. }
  2607. },
  2608. {
  2609. // ✅ https://www.kk151.com/play/15257-2-11.html
  2610. name: "动漫之家",
  2611. home: [
  2612. "https://www.kk151.com"
  2613. ],
  2614. regex: /^https:\/\/www\.kk151\.com\/play\/.*/g,
  2615. handler: class Handler extends BaseHandler {
  2616. constructor() {
  2617. super();
  2618. this.addIframeListener();
  2619. }
  2620. }
  2621. },
  2622. {
  2623. name: "动漫之家播放器",
  2624. regex: /^https:\/\/(www\.ikdmjx\.com\/|jx\.wolongzywcdn\.com:65\/m3u8\.php|hls\.kuaibofang\.com\/|jx\.jxbdzyw\.com\/m3u8\/|www\.m3u8\.tv\.cdn\.8old\.cn\/m3u8tv1127\/api\.php|jx\.wujinkk\.com\/dplayer\/|jx\.m3u8\.tv\/jiexi\/)\?url=.*/g,
  2625. handler: class Handler extends BaseHandler {
  2626. constructor() {
  2627. super();
  2628. this.addTopListener();
  2629. }
  2630. async parse() {
  2631. this.media.setVideoUrl(this.urlParser());
  2632. }
  2633. }
  2634. },
  2635. {
  2636. // ✅ https://hdzyk.com/?m=vod-play-id-27537-src-2-num-11.html
  2637. // ✅ https://1080zyk2.com/?m=vod-play-id-27537-src-2-num-11.html
  2638. name: "优质资源库",
  2639. home: [
  2640. "https://hdzyk.com",
  2641. "https://1080zyk1.com",
  2642. "https://1080zyk2.com",
  2643. "https://1080zyk3.com",
  2644. "https://1080zyk4.com",
  2645. "https://1080zyk5.com"
  2646. ],
  2647. regex: /^https:\/\/(hdzyk\.com|1080zyk[1-5]\.com)\/\?m=.*/g,
  2648. handler: class Handler extends BaseHandler {
  2649. constructor() {
  2650. super();
  2651. this.addIframeListener();
  2652. }
  2653. }
  2654. },
  2655. {
  2656. name: "优质资源库播放器",
  2657. regex: /^https:\/\/vip\.zykbf\.com\/\?url=.*/g,
  2658. handler: class Handler extends BaseHandler {
  2659. constructor() {
  2660. super();
  2661. this.addTopListener();
  2662. }
  2663. async parse() {
  2664. this.media.setVideoUrl(this.urlParser());
  2665. }
  2666. }
  2667. },
  2668. {
  2669. name: "优质资源库播放器",
  2670. regex: /^https:\/\/.*\.(yzzy-tv1|yzzy-tv-cdn)\.com\/.*/g,
  2671. handler: class Handler extends BaseHandler {
  2672. constructor() {
  2673. super();
  2674. this.addTopListener();
  2675. }
  2676. async parse() {
  2677. this.media.setVideoUrl("https://" + location.host + main);
  2678. }
  2679. }
  2680. },
  2681. {
  2682. // ✅ https://www.bdys10.com/play/22729-8.htm
  2683. name: "哔嘀影视",
  2684. home: [
  2685. "https://www.bdys10.com"
  2686. ],
  2687. regex: /^https:\/\/www\.bdys10\.com\/.*play\/.*/g,
  2688. handler: class Handler extends BaseHandler {
  2689. async parse() {
  2690. this.media.setVideoUrl(this.videoParser());
  2691. if (!this.media.videoUrl) {
  2692. this.media.setVideoUrl(this.htmlParser());
  2693. if (!this.media.videoUrl) {
  2694. this.media.setVideoUrl(this.scriptParser());
  2695. }
  2696. }
  2697. }
  2698. }
  2699. },
  2700. {
  2701. // ✅ https://www.dora-family.com/Resource:TV
  2702. name: "哆啦A梦新番",
  2703. home: [
  2704. "https://www.dora-family.com/Resource:TV"
  2705. ],
  2706. regex: /^https:\/\/www\.dora-family\.com\/Resource:TV/g,
  2707. handler: class Handler extends BaseHandler {
  2708. initCheck() {
  2709. if (super.initCheck()) {
  2710. return true;
  2711. }
  2712. let oldVideoUrl = this.media.videoUrl;
  2713. let newVideoUrl = this.videoParser();
  2714. if (oldVideoUrl && oldVideoUrl != newVideoUrl) {
  2715. return true;
  2716. }
  2717. return false;
  2718. }
  2719. async parse() {
  2720. this.media.setVideoUrl(this.videoParser());
  2721. }
  2722. }
  2723. },
  2724. {
  2725. // ✅ https://mypikpak.com/drive/all
  2726. name: "PikPak",
  2727. home: [
  2728. "https://mypikpak.com/drive/all"
  2729. ],
  2730. regex: /^https:\/\/mypikpak\.com\/drive\/.*/g,
  2731. handler: class Handler extends BaseHandler {
  2732. initCheck() {
  2733. if (super.initCheck()) {
  2734. return true;
  2735. }
  2736. let oldVideoUrl = this.media.videoUrl;
  2737. let newVideoUrl = this.videoParser();
  2738. if (oldVideoUrl && oldVideoUrl != newVideoUrl) {
  2739. return true;
  2740. }
  2741. return false;
  2742. }
  2743. async parse() {
  2744. while (document.getElementsByTagName("video").length == 0) {
  2745. await sleep(1000);
  2746. }
  2747. this.media.setVideoUrl(this.videoParser());
  2748. this.media.setTitle(document.getElementsByClassName("player-title")[0].textContent);
  2749. }
  2750. }
  2751. },
  2752. {
  2753. // ✅ https://www.olehdtv.com/player/vod/1/43671/1
  2754. name: "欧乐影院",
  2755. home: [
  2756. "https://www.olehdtv.com/"
  2757. ],
  2758. regex: /^https:\/\/www\.olehdtv\.com\/player\/vod\/\d+\/\d+\/\d+/g,
  2759. handler: class Handler extends BaseHandler {
  2760. async parse() {
  2761. let ids = page.url.match(/^https:\/\/www\.olehdtv\.com\/player\/vod\/(\d+)\/(\d+)\/(\d+)/);
  2762. let id = ids[2];
  2763. let index = ids[3];
  2764. let t = Date.parse(new Date) / 1e3, r = t % 20;
  2765. let _vv = MD5(t - r + "new.olelive.com");
  2766. let that = this;
  2767. $.ajax({
  2768. type: "GET",
  2769. url: `https://api.olelive.com/v1/pub/vod/detail?id=${id}&play=true&_vv=${_vv}`,
  2770. async: false,
  2771. success: function (res) {
  2772. if (res.code == 0) {
  2773. that.media.setVideoUrl(res.data.urls[index - 1].url);
  2774. }
  2775. }
  2776. })
  2777. }
  2778. play() {
  2779. this.media.setTitle(document.title);
  2780. super.play();
  2781. }
  2782. }
  2783. },
  2784. {
  2785. // ✅ https://www.olehdtv.com/player/live/tv/CCTV5HD/49
  2786. name: "欧乐影院直播",
  2787. home: [
  2788. "https://www.olehdtv.com/"
  2789. ],
  2790. regex: /^https:\/\/www\.olehdtv\.com\/player\/live\/tv\/[^/]+\/\d+/g,
  2791. handler: class Handler extends BaseHandler {
  2792. async parse() {
  2793. let ids = page.url.match(/^https:\/\/www\.olehdtv\.com\/player\/live\/tv\/([^/]+)\/(\d+)/);
  2794. let id = ids[2];
  2795. let streamId = ids[1];
  2796. let today = new Date();
  2797. let year = today.getFullYear();
  2798. let month = today.getMonth() + 1;
  2799. if (month < 10) {
  2800. month = "0" + month;
  2801. }
  2802. let day = today.getDate();
  2803. if (day < 10) {
  2804. day = "0" + day;
  2805. }
  2806. let date = year + "-" + month + "-" + day;
  2807. let t = Date.parse(today) / 1e3, r = t % 20;
  2808. let _vv = MD5(t - r + "new.olelive.com");
  2809. let that = this;
  2810. $.ajax({
  2811. type: "GET",
  2812. url: `https://api.olelive.com/v1/pub/live/info?date=${date}&streamId=${streamId}&type=tv&id=${id}&_vv=${_vv}`,
  2813. async: false,
  2814. success: function (res) {
  2815. if (res.code == 0) {
  2816. that.media.setVideoUrl(res.data.detail.hls.replace("_360", ""));
  2817. }
  2818. }
  2819. })
  2820. }
  2821. play() {
  2822. this.media.setTitle(document.title);
  2823. super.play();
  2824. }
  2825. }
  2826. },
  2827. {
  2828. // ✅ https://tkznp.com/vodplay/337990-1-2.html
  2829. name: "天空影视",
  2830. home: [
  2831. "https://tkznp.com/",
  2832. "https://www.tkznp1.com/",
  2833. "https://www.tkznp2.com/",
  2834. "https://www.tkznp3.com/",
  2835. "https://www.tkznp4.com/",
  2836. "https://www.tkznp5.com/",
  2837. "https://www.tkznp6.com/",
  2838. ],
  2839. regex: /^https?:\/\/(|www\.)tkznp(|1|2|3|4|5|6)\.com\/vodplay\/.*/g,
  2840. handler: class Handler extends BaseHandler {
  2841. constructor() {
  2842. super();
  2843. this.addIframeListener();
  2844. }
  2845. }
  2846. },
  2847. {
  2848. name: "天空影视播放器",
  2849. regex: /^https?:\/\/vip\.ckllk\.com\/\?url=.*/g,
  2850. handler: class Handler extends BaseHandler {
  2851. constructor() {
  2852. super();
  2853. this.addTopListener();
  2854. }
  2855. async parse() {
  2856. this.media.setVideoUrl(config.url);
  2857. this.media.setTitle(config.title);
  2858. }
  2859. }
  2860. },
  2861. {
  2862. // ✅ https://www.hdmoli.com/play/1630-1-0.html
  2863. // ✅ https://www.hdmoli.com/play/1630-0-0.html
  2864. name: "HDmoli",
  2865. home: [
  2866. "https://www.hdmoli.com"
  2867. ],
  2868. regex: /^https:\/\/www\.hdmoli\.com\/play\/.*/g,
  2869. handler: class Handler extends BaseHandler {
  2870. constructor() {
  2871. super();
  2872. this.addIframeListener();
  2873. }
  2874. }
  2875. },
  2876. {
  2877. name: "HDmoli 线路1",
  2878. regex: /^https:\/\/www\.hdmoli\.com\/js\/player\/dplayer\/dplayer\.html\?.*/g,
  2879. handler: class Handler extends BaseHandler {
  2880. constructor() {
  2881. super();
  2882. this.addTopListener();
  2883. setTimeout(() => {
  2884. let closeclick = document.getElementsByClassName("closeclick")[0];
  2885. if (closeclick) {
  2886. closeclick.click();
  2887. }
  2888. }, TIME.refresh);
  2889. }
  2890. async parse() {
  2891. this.media.setVideoUrl(this.videoParser());
  2892. }
  2893. }
  2894. },
  2895. {
  2896. name: "HDmoli 线路2",
  2897. regex: /^https:\/\/www\.hdmoli\.com\/js\/player\/duoduozy.html\?v=[\d\.]+/g,
  2898. handler: class Handler extends BaseHandler {
  2899. constructor() {
  2900. super();
  2901. setTimeout(() => {
  2902. let closeclick = document.getElementsByClassName("closeclick")[0];
  2903. if (closeclick) {
  2904. closeclick.click();
  2905. }
  2906. }, TIME.refresh);
  2907. }
  2908. }
  2909. },
  2910. {
  2911. name: "HDmoli 线路2",
  2912. regex: /^https:\/\/play\.qwertwe\.top\/xplay\/\?url=.*/g,
  2913. handler: class Handler extends BaseHandler {
  2914. constructor() {
  2915. super();
  2916. this.addTopListener();
  2917. }
  2918. async parse() {
  2919. let that = this;
  2920. $.ajax({
  2921. url: '555tZ4pvzHE3BpiO838.php',
  2922. type: 'GET',
  2923. dataType: 'JSON',
  2924. timeout: 3000,
  2925. data: {
  2926. tm: (new Date().getTime()),
  2927. url: config.url,
  2928. vkey: config.vkey,
  2929. token: config.token,
  2930. sign: 'F4penExTGogdt6U8'
  2931. },
  2932. success: function (data) {
  2933. if (data.code === 200) {
  2934. that.media.setVideoUrl(getVideoInfo(data.url));
  2935. }
  2936. }
  2937. });
  2938. }
  2939. }
  2940. },
  2941. {
  2942. // ✅ https://www.anfuns.cc/play/1572-1-1.html
  2943. name: "AnFuns",
  2944. home: [
  2945. "https://www.anfuns.cc"
  2946. ],
  2947. regex: /^https:\/\/www\.anfuns\.cc\/play\/.*/g,
  2948. handler: class Handler extends BaseHandler {
  2949. constructor() {
  2950. super();
  2951. this.addIframeListener();
  2952. }
  2953. }
  2954. },
  2955. {
  2956. name: "AnFuns播放器",
  2957. regex: /^https:\/\/www\.anfuns\.cc\/vapi\/(A0EPlayer|eden)\/\?url=.*/g,
  2958. handler: class Handler extends BaseHandler {
  2959. constructor() {
  2960. super();
  2961. this.addTopListener();
  2962. }
  2963. async parse() {
  2964. let url = config.url;
  2965. if (url) {
  2966. if (!url.startsWith("https://fata.peizq.online/cache/") && !url.startsWith("https://media-oss.anfuns.cn/m3u8/")) {
  2967. this.media.setVideoUrl(config.url);
  2968. } else {
  2969. tryTime = TRY_TIME.maxParse;
  2970. }
  2971. }
  2972. }
  2973. }
  2974. },
  2975. {
  2976. // ✅ https://www.youtube.com/watch?v=IkGuTYaTsLo
  2977. name: "YouTube",
  2978. home: [
  2979. "https://www.youtube.com"
  2980. ],
  2981. regex: /^https:\/\/www\.youtube\.com\/(watch|playlist)\?.*/g,
  2982. handler: class Handler extends BaseHandler {
  2983. async parse() {
  2984. this.media.setOther(BEST_QUALITY.ytdlp[currentConfig.bestQuality]);
  2985. this.media.setVideoUrl(this.ytDlpParser());
  2986. this.media.setTitle("");
  2987. }
  2988. },
  2989. },
  2990. {
  2991. // ✅ https://odysee.com/@jjlin:8/%E6%9E%97%E4%BF%8A%E5%82%91-jj-lin%E3%80%8Ajj20%E4%B8%96%E7%95%8C%E5%B7%A1%E8%BF%B4%E6%BC%94%E5%94%B1%E6%9C%83%E3%80%8B-2:8
  2992. name: "Odysee",
  2993. home: [
  2994. "https://odysee.com"
  2995. ],
  2996. regex: /^https:\/\/odysee\.com\/[^$].+/g,
  2997. handler: class Handler extends BaseHandler {
  2998. async parse() {
  2999. this.media.setOther(BEST_QUALITY.ytdlp[currentConfig.bestQuality]);
  3000. this.media.setVideoUrl(this.ytDlpParser());
  3001. this.media.setTitle("");
  3002. }
  3003. },
  3004. },
  3005. {
  3006. // ✅ https://rumble.com/v2mfr78-valheim-viking-survival-w-chaos-tricks-you-can-affect-my-game-chat-opinions.html
  3007. name: "Rumble",
  3008. home: [
  3009. "https://rumble.com"
  3010. ],
  3011. regex: /^https:\/\/rumble\.com\/v.+\.html/g,
  3012. handler: class Handler extends BaseHandler {
  3013. async parse() {
  3014. this.media.setOther(BEST_QUALITY.ytdlp[currentConfig.bestQuality]);
  3015. this.media.setVideoUrl(this.ytDlpParser());
  3016. this.media.setTitle("");
  3017. }
  3018. },
  3019. },
  3020. {
  3021. // ✅ https://www.bitchute.com/video/NoodZjmfKHXS/
  3022. name: "BitChute",
  3023. home: [
  3024. "https://www.bitchute.com"
  3025. ],
  3026. regex: /^https:\/\/www\.bitchute\.com\/video\/.+/g,
  3027. handler: class Handler extends BaseHandler {
  3028. async parse() {
  3029. this.media.setOther(BEST_QUALITY.ytdlp[currentConfig.bestQuality]);
  3030. this.media.setVideoUrl(this.ytDlpParser());
  3031. this.media.setTitle("");
  3032. }
  3033. },
  3034. },
  3035. {
  3036. // ✅ https://ani.gamer.com.tw/animeVideo.php?sn=32227
  3037. name: "巴哈姆特",
  3038. home: [
  3039. "https://ani.gamer.com.tw"
  3040. ],
  3041. regex: /^https:\/\/ani\.gamer\.com\.tw\/animeVideo.php\?sn=.*/g,
  3042. handler: class Handler extends BaseHandler {
  3043. async parse() {
  3044. this.media.setOrigin("https://ani.gamer.com.tw");
  3045. let index = page.url.indexOf("sn=") + 3;
  3046. if (index == -1) {
  3047. return;
  3048. }
  3049. let sn = page.url.substring(index);
  3050. index = sn.indexOf("&");
  3051. if (index != -1) {
  3052. sn = sn.substring(0, index);
  3053. }
  3054. let device = localStorage.ANIME_deviceid;
  3055. let that = this;
  3056. let res;
  3057. $.ajax({
  3058. type: "GET",
  3059. url: `https://ani.gamer.com.tw/ajax/m3u8.php?sn=${sn}&device=${device}`,
  3060. async: false,
  3061. xhrFields: {
  3062. withCredentials: true
  3063. },
  3064. success: function (json) {
  3065. res = JSON.parse(json);
  3066. }
  3067. })
  3068. if (res.error) {
  3069. if (res.error.code == 1015) {
  3070. let oldDuration = document.getElementsByClassName("vjs-duration-display")[0].innerHTML;
  3071. let newDuration = oldDuration;
  3072. while (oldDuration == newDuration) {
  3073. await sleep(1000);
  3074. newDuration = document.getElementsByClassName("vjs-duration-display")[0].innerHTML;
  3075. }
  3076. tryTime = 0;
  3077. }
  3078. } else {
  3079. that.media.setProxy(currentConfig.proxy);
  3080. that.media.setVideoUrl(res.src);
  3081. }
  3082. }
  3083. },
  3084. },
  3085. {
  3086. // ✅ https://hanime1.me/watch?v=26262
  3087. name: "Hanime1.me",
  3088. home: [
  3089. "https://hanime1.me"
  3090. ],
  3091. regex: /^https:\/\/hanime1\.me\/watch\?v=.*/g,
  3092. handler: class Handler extends BaseHandler {
  3093. async parse() {
  3094. let url = this.videoParser();
  3095. if (!url) {
  3096. url = this.iframeParser();
  3097. }
  3098. if (!url) {
  3099. url = page.url;
  3100. }
  3101. this.media.setVideoUrl(url);
  3102. }
  3103. },
  3104. },
  3105. {
  3106. // ✅ https://jable.tv/videos/fsdss-484/
  3107. name: "Jable.TV",
  3108. home: [
  3109. "https://jable.tv"
  3110. ],
  3111. regex: /^https:\/\/jable\.tv\/videos\/.*/g,
  3112. handler: class Handler extends BaseHandler {
  3113. async parse() {
  3114. let url = hls.url;
  3115. url = url ? url : hlsUrl;
  3116. url = url ? url : page.url;
  3117. this.media.setVideoUrl(url);
  3118. }
  3119. },
  3120. },
  3121. {
  3122. // ✅ https://ok.ru/video/2035990725937
  3123. name: "OK",
  3124. home: [
  3125. "https://ok.ru/video"
  3126. ],
  3127. regex: /^https:\/\/ok\.ru\/video\/\d+/g,
  3128. handler: class Handler extends BaseHandler {
  3129. async parse() {
  3130. this.media.setOther(BEST_QUALITY.ytdlp[currentConfig.bestQuality]);
  3131. this.media.setVideoUrl(this.ytDlpParser());
  3132. this.media.setTitle("");
  3133. }
  3134. },
  3135. },
  3136. {
  3137. // ✅ https://tver.jp/episodes/epsta1xs0z
  3138. name: "TVer",
  3139. home: [
  3140. "https://tver.jp"
  3141. ],
  3142. regex: /^https:\/\/tver\.jp\/episodes\/\w+/g,
  3143. handler: class Handler extends BaseHandler {
  3144. async parse() {
  3145. this.media.setOther(BEST_QUALITY.ytdlp[currentConfig.bestQuality]);
  3146. this.media.setVideoUrl(this.ytDlpParser());
  3147. this.media.setTitle("");
  3148. }
  3149. },
  3150. },
  3151. {
  3152. // ✅ https://www.lckp.top/play-with-mpv/index.html
  3153. name: "电视直播",
  3154. home: [
  3155. "https://www.lckp.top/play-with-mpv/index.html"
  3156. ],
  3157. regex: /^https?:\/\/(www.lckp.top\/play-with-mpv|127.0.0.1:5502\/web\/tampermonkey\/Play-With-MPV)\/index.html/g,
  3158. handler: class Handler extends BaseHandler {
  3159. async parse() {
  3160. this.media.setVideoUrl(localStorage.iptvUrl);
  3161. localStorage.player = JSON.stringify(this.player);
  3162. this.media.setTitle("");
  3163. }
  3164. },
  3165. },
  3166. {
  3167. // ✅ https://www.douyin.com/
  3168. name: "抖音",
  3169. home: [
  3170. "https://www.douyin.com/"
  3171. ],
  3172. regex: /^https?:\/\/www\.douyin\.com\/.*/g,
  3173. handler: class Handler extends BaseHandler {
  3174. constructor() {
  3175. super();
  3176. this.index = 0;
  3177. }
  3178. initCheck() {
  3179. if (super.initCheck()) {
  3180. return true;
  3181. }
  3182. let oldVideoUrl = this.media.videoUrl;
  3183. let newVideoUrl = this.videoParser();
  3184. if (oldVideoUrl && oldVideoUrl != newVideoUrl) {
  3185. return true;
  3186. }
  3187. return false;
  3188. }
  3189. async parse() {
  3190. this.media.setVideoUrl(this.videoParser());
  3191. }
  3192. videoParser() {
  3193. let videos = document.getElementsByTagName("video");
  3194. if (videos && videos.length > 0) {
  3195. this.index = videos.length > 2 ? 1 : 0;
  3196. }
  3197. let url = document.getElementsByTagName("video")[this.index].getElementsByTagName("source")[0].src;
  3198. if (url) {
  3199. return url;
  3200. }
  3201. }
  3202. },
  3203. },
  3204. {
  3205. // ✅ https://www.mengfan.tv/play/kx666U/1/3/
  3206. name: "萌番",
  3207. home: [
  3208. "https://www.mengfan.tv"
  3209. ],
  3210. regex: /^https:\/\/www\.mengfan\.tv\/play\/.*/g,
  3211. handler: class Handler extends BaseHandler {
  3212. constructor() {
  3213. super();
  3214. this.addIframeListener();
  3215. }
  3216. }
  3217. },
  3218. {
  3219. name: "萌番播放器",
  3220. regex: /^https:\/\/video1\.beijcloud\.com\/player\/\?url=.*/g,
  3221. handler: class Handler extends BaseHandler {
  3222. constructor() {
  3223. super();
  3224. this.addTopListener();
  3225. }
  3226. async parse() {
  3227. this.media.setVideoUrl(this.scriptParser());
  3228. }
  3229. }
  3230. },
  3231. {
  3232. // ✅ https://www.haitu.tv/vod/play/id/47100/sid/1/nid/4.html
  3233. name: "海兔影院",
  3234. home: [
  3235. "https://www.haitu.tv"
  3236. ],
  3237. regex: /^https:\/\/www\.haitu\.tv\/vod\/play\/.*/g,
  3238. handler: class Handler extends BaseHandler {
  3239. constructor() {
  3240. super();
  3241. this.addIframeListener();
  3242. }
  3243. }
  3244. },
  3245. {
  3246. name: "海兔影院播放器",
  3247. regex: /^https:\/\/www\.haitu\.tv\/static\/dmku\/player\/index\.php/g,
  3248. handler: class Handler extends BaseHandler {
  3249. constructor() {
  3250. super();
  3251. this.addTopListener();
  3252. }
  3253. async parse() {
  3254. this.media.setVideoUrl(config.url);
  3255. }
  3256. }
  3257. },
  3258. {
  3259. // ✅ https://www.icourse163.org/learn/ZJU-200001?tid=1470096517#/learn/content?type=detail&id=1254347726&cid=1285600901
  3260. name: "中国大学MOOC",
  3261. regex: /^https:\/\/www\.icourse163\.org\/learn\/.*\/learn\/content\?type=detail.*/g,
  3262. handler: class Handler extends BaseHandler {
  3263. async parse() {
  3264. let res = this.htmlParser();
  3265. if (!res) return;
  3266. this.media.setVideoUrl(res);
  3267. this.media.setTitle(document.querySelector('.current > .unit-name').innerText);
  3268. }
  3269. }
  3270. },
  3271. {
  3272. // ✅ https://www.iole.tv/vodplay/23711-1-1.html
  3273. name: "ioleTV",
  3274. regex: /^https:\/\/www\.iole\.tv\/vodplay\/.+/g,
  3275. handler: class Handler extends BaseHandler {
  3276. constructor() {
  3277. super();
  3278. this.addIframeListener();
  3279. }
  3280. }
  3281. },
  3282. {
  3283. name: "ioleTV播放器",
  3284. regex: /^https:\/\/www\.iole\.tv\/static\/player\/.*\.html/g,
  3285. handler: class Handler extends BaseHandler {
  3286. constructor() {
  3287. super();
  3288. this.addTopListener();
  3289. }
  3290. async parse() {
  3291. this.media.setVideoUrl(parent.MacPlayer.PlayUrl);
  3292. }
  3293. }
  3294. },
  3295. {
  3296. // ✅ https://www.zhihu.com/zvideo/1650574385558937600
  3297. name: "知乎视频",
  3298. regex: /^https:\/\/www\.zhihu\.com\/zvideo\/.+/g,
  3299. handler: class Handler extends BaseHandler {
  3300. constructor() {
  3301. super();
  3302. let that = this;
  3303. this.currentUrl = "";
  3304. //拦截请求以更新Url
  3305. const originOpen = XMLHttpRequest.prototype.open;
  3306. XMLHttpRequest.prototype.open = function (method, url, async, user, password) {
  3307. originOpen.apply(this, arguments);
  3308. if (url.match(VIDEO_URL_REGEX)) {
  3309. that.currentUrl = url;
  3310. }
  3311. };
  3312. }
  3313. async parse() {
  3314. this.media.setVideoUrl(this.currentUrl);
  3315. }
  3316. }
  3317. },
  3318. {
  3319. // ✅ https://www.tucao.cam/play/h4092670/#1
  3320. name: "吐槽弹幕网",
  3321. regex: /^https:\/\/www\.tucao\.cam\/play\/.*/g,
  3322. handler: class Handler extends BaseHandler {
  3323. async parse() {
  3324. this.media.setVideoUrl(this.videoParser());
  3325. }
  3326. }
  3327. },
  3328. {
  3329. // ✅ http://www.susudm8.com/acg/69815/3.html
  3330. name: "速速电影院",
  3331. regex: /^https?:\/\/(www\.susudm8\.com|susudyy\.com|buding3\.com|buding6\.com)\/.+\.html/g,
  3332. handler: class Handler extends BaseHandler {
  3333. constructor() {
  3334. super();
  3335. this.addIframeListener();
  3336. }
  3337. }
  3338. },
  3339. {
  3340. name: "速速电影院播放器",
  3341. regex: /^https?:\/\/(v2\.shenjw\.com:\d+|u88\.xigua88ok\.com:\d+)\/wap\.php\?url=.+/g,
  3342. handler: class Handler extends BaseHandler {
  3343. constructor() {
  3344. super();
  3345. this.addTopListener();
  3346. }
  3347. async parse() {
  3348. this.media.setVideoUrl(this.videoParser());
  3349. }
  3350. }
  3351. },
  3352. {
  3353. name: "速速电影院播放器",
  3354. regex: /^https?:\/\/test3\.gqyy8\.com:\d+\/f\/aliplayer\.php\?url=.+/g,
  3355. handler: class Handler extends BaseHandler {
  3356. constructor() {
  3357. super();
  3358. this.addTopListener();
  3359. }
  3360. async parse() {
  3361. this.media.setVideoUrl(this.scriptParser());
  3362. }
  3363. }
  3364. },
  3365. {
  3366. // ✅ https://v.mksec.cn/index.php/vod/play/id/165438/sid/2/nid/1.html
  3367. name: "小见子的视频站",
  3368. regex: /^https?:\/\/v\.mksec\.cn\/index\.php\/vod\/play\/.+\.html/g,
  3369. handler: class Handler extends BaseHandler {
  3370. constructor() {
  3371. super();
  3372. this.addIframeListener();
  3373. }
  3374. async parse() {
  3375. this.media.setVideoUrl(player_aaaa.url);
  3376. }
  3377. }
  3378. },
  3379. {
  3380. name: "小见子的视频站播放器",
  3381. regex: /^https?:\/\/v\.mksec\.cn\/static\/player\/dplayer\.html/g,
  3382. handler: class Handler extends BaseHandler {
  3383. constructor() {
  3384. super();
  3385. this.addTopListener();
  3386. }
  3387. async parse() {
  3388. tryTime = TRY_TIME.maxParse;
  3389. }
  3390. }
  3391. },
  3392. {
  3393. // ✅ https://jh642t.dshryadqp.com/index.php/vod/play/id/51434/sid/1/nid/1.html
  3394. name: "大师兄电影网",
  3395. regex: /^https?:\/\/.*dsh.*\.com\/index\.php\/vod\/play\/.+\.html/g,
  3396. handler: class Handler extends BaseHandler {
  3397. constructor() {
  3398. super();
  3399. this.addIframeListener();
  3400. }
  3401. async parse() {
  3402. this.media.setVideoUrl(player_aaaa.url);
  3403. }
  3404. }
  3405. },
  3406. {
  3407. name: "大师兄电影网播放器",
  3408. regex: /^https?:\/\/.*dsh.*\.com\/static\/player\/dplayer\.html/g,
  3409. handler: class Handler extends BaseHandler {
  3410. constructor() {
  3411. super();
  3412. this.addTopListener();
  3413. }
  3414. async parse() {
  3415. tryTime = TRY_TIME.maxParse;
  3416. }
  3417. }
  3418. },
  3419. {
  3420. // ✅ https://www.twitch.tv/yulihong22
  3421. name: "Twitch",
  3422. home: [
  3423. "https://www.twitch.tv"
  3424. ],
  3425. regex: /^https:\/\/www\.twitch\.tv\/\w+/g,
  3426. handler: class Handler extends BaseHandler {
  3427. async parse() {
  3428. if (page.url.indexOf("/directory") != -1) {
  3429. tryTime = TRY_TIME.maxParse;
  3430. return;
  3431. }
  3432. this.media.setOther(BEST_QUALITY.ytdlp[currentConfig.bestQuality]);
  3433. this.media.setVideoUrl(this.ytDlpParser());
  3434. this.media.setTitle("");
  3435. }
  3436. },
  3437. },
  3438. {
  3439. // ✅ https://jojo.bdys.top/watch/264
  3440. name: "JOJO",
  3441. home: [
  3442. "https://jojo.bdys.top"
  3443. ],
  3444. regex: /^https:\/\/jojo\.bdys\.top\/watch\/.*/g,
  3445. handler: class Handler extends BaseHandler {
  3446. async parse() {
  3447. this.media.setVideoUrl(this.videoParser());
  3448. }
  3449. },
  3450. },
  3451. {
  3452. // ✅ https://www.agemys.org/play/20220403/1/1
  3453. name: "AGE动漫",
  3454. regex: /^https?:\/\/www\.agemys\.org\/play\/.+/g,
  3455. handler: class Handler extends BaseHandler {
  3456. constructor() {
  3457. super();
  3458. this.addIframeListener();
  3459. }
  3460. }
  3461. },
  3462. {
  3463. name: "AGE动漫播放器",
  3464. regex: /^https?:\/\/vip\.sp-flv\.com:\d+\/\?url=.+/g,
  3465. handler: class Handler extends BaseHandler {
  3466. constructor() {
  3467. super();
  3468. this.addTopListener();
  3469. }
  3470. async parse() {
  3471. this.media.setVideoUrl(this.videoParser());
  3472. }
  3473. }
  3474. },
  3475. {
  3476. // ✅ https://anime.girigirilove.com/playGV25353-1-1/
  3477. name: "girigiri爱动漫",
  3478. regex: /^https:\/\/anime\.girigirilove\.com\/play.+/g,
  3479. handler: class Handler extends BaseHandler {
  3480. constructor() {
  3481. super();
  3482. this.addIframeListener();
  3483. }
  3484. }
  3485. },
  3486. {
  3487. name: "girigiri爱动漫播放器",
  3488. regex: /^https:\/\/anime\.girigirilove\.com\/addons\/dp\/player\/dp\.php\?.+/g,
  3489. handler: class Handler extends BaseHandler {
  3490. constructor() {
  3491. super();
  3492. this.addTopListener();
  3493. }
  3494. async parse() {
  3495. this.media.setVideoUrl(config.url);
  3496. }
  3497. }
  3498. },
  3499. {
  3500. name: "AList",
  3501. regex: /^https?:\/\/[^\/]+\/.*\.(mp4|mkv|flv)/g,
  3502. handler: class Handler extends BaseHandler {
  3503. async parse() {
  3504. let url = this.videoParser();
  3505. let data = {
  3506. path: decodeURIComponent(location.pathname),
  3507. password: ""
  3508. };
  3509. if (!url && tryTime < 3) {
  3510. $.ajax({
  3511. type: "POST",
  3512. url: `/api/fs/get`,
  3513. dataType: "json",
  3514. data: JSON.stringify(data),
  3515. xhrFields: {
  3516. withCredentials: true
  3517. },
  3518. headers: {
  3519. "Authorization": localStorage.getItem("token")
  3520. },
  3521. async: false,
  3522. contentType: "application/json",
  3523. success: function (res) {
  3524. if (res.code == 200) {
  3525. url = res.data.raw_url;
  3526. }
  3527. }
  3528. });
  3529. }
  3530. if (url) {
  3531. let index = url.indexOf("?");
  3532. if (index != -1) {
  3533. url = url.substring(0, index + 1) + encodeURIComponent(url.substring(index + 1));
  3534. }
  3535. this.media.setVideoUrl(url);
  3536. this.media.setTitle(document.title);
  3537. }
  3538. }
  3539. }
  3540. }
  3541. ];
  3542. // 初始化
  3543. async function init(flag) {
  3544. // 加载页面信息
  3545. page = {
  3546. host: window.location.host,
  3547. url: window.location.href,
  3548. isFullScreen: false
  3549. };
  3550. // 清除 handler
  3551. if (handler) {
  3552. handler = undefined;
  3553. if (document.getElementById(ID.buttonDiv)) {
  3554. document.getElementById(ID.buttonDiv).style.display = "none";
  3555. }
  3556. }
  3557. // 生成 handler
  3558. for (let i = 0; i < websiteList.length; i++) {
  3559. if (page.url.match(websiteList[i].regex)) {
  3560. handler = new websiteList[i].handler();
  3561. break;
  3562. }
  3563. }
  3564. if (flag && page.url.startsWith("https://www.bilibili.com/festival/")) {
  3565. await sleep(1500);
  3566. }
  3567. // 尝试解析页面视频
  3568. if (handler) {
  3569. tryTime = 0;
  3570. while (tryTime < TRY_TIME.maxParse) {
  3571. await sleep(tryTime * 1000 + 700);
  3572. if (!handler.media.videoUrl) {
  3573. try {
  3574. await handler.parse();
  3575. } catch (error) {
  3576. console.log('Play-With-MPV:解析失败:' + error);
  3577. }
  3578. }
  3579. tryTime++;
  3580. }
  3581. } else {
  3582. console.log("Play-With-MPV:暂无此网页解析器(" + page.url + ")");
  3583. }
  3584. }
  3585. // 开始执行
  3586. init();
  3587. setInterval(() => {
  3588. if (handler) {
  3589. if (handler.initCheck()) {
  3590. init(true);
  3591. }
  3592. } else if (window.location.href != page.url) {
  3593. init();
  3594. }
  3595. }, TIME.refresh);