External Player

Play web video via external player

当前为 2025-06-04 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name External Player
  3. // @name:zh-CN 外部播放器
  4. // @namespace https://github.com/LuckyPuppy514/external-player
  5. // @copyright 2024, Grant LuckyPuppy514 (https://github.com/LuckyPuppy514)
  6. // @version 1.1.9
  7. // @license MIT
  8. // @description Play web video via external player
  9. // @description:zh-CN 使用外部播放器播放网页中的视频
  10. // @icon https://www.lckp.top/gh/LuckyPuppy514/pic-bed/common/mpv.png
  11. // @author LuckyPuppy514
  12. // @homepage https://github.com/LuckyPuppy514/external-player
  13. // @include *://*
  14. // @connect v.anime1.me
  15. // @grant GM_setValue
  16. // @grant GM_getValue
  17. // @grant GM.xmlHttpRequest
  18. // @run-at document-end
  19. // @require https://lf26-cdn-tos.bytecdntp.com/cdn/expire-1-y/pako/2.0.4/pako.min.js
  20. // ==/UserScript==
  21.  
  22. 'use strict';
  23.  
  24. const DEBUG = false;
  25.  
  26. const PROJECT_NAME = 'external-player';
  27.  
  28. const SETTING_URL = DEBUG === true ? 'http://127.0.0.1:5500/setting.html' : undefined;
  29.  
  30. const VIDEO_URL_REGEX_GLOBAL = /https?:\/\/((?![^"^']*http)[^"^']+(\.|%2e)(mp4|mkv|flv|m3u8|m4s|m3u|mov|avi|wmv|webm)(\?[^"^']+|))|((?![^"^']*http)[^"^']+\?[^"^']+(\.|%2e|video_)(mp4|mkv|flv|mov|avi|wmv|webm|m3u8|m3u)[^"^']*)/ig;
  31.  
  32. const VIDEO_URL_REGEX_EXACT = /^https?:\/\/((?![^"^']*http)[^"^']+(\.|%2e)(mp4|mkv|flv|m3u8|m4s|m3u|mov|avi|wmv|webm)(\?[^"^']+|))|((?![^"^']*http)[^"^']+\?[^"^']+(\.|%2e|video_)(mp4|mkv|flv|mov|avi|wmv|webm|m3u8|m3u)[^"^']*)$/ig;
  33.  
  34. const defaultConfig = {
  35. global: {
  36. version: '1.1.9',
  37. language: (navigator.language || navigator.userLanguage) === 'zh-CN' ? 'zh' : 'en',
  38. buttonXCoord: '0',
  39. buttonYCoord: '0',
  40. buttonScale: '1.00',
  41. buttonVisibilityDuration: '5000',
  42. networkProxy: '',
  43. parser: {
  44. ytdlp: {
  45. regex: [
  46. "https://www.youtube.com/shorts/.+",
  47. "https://www.youtube.com/watch\\?.+",
  48. "https://www.youtube.com/playlist\\?list=.+",
  49. ],
  50. preferredQuality: 'unlimited',
  51. },
  52. video: {
  53. regex: [
  54. "https://www.moepoi.net/static/player/artplayer.html",
  55. "https://.*libvio\\..+/vid/plyr/vr2.php\\?url=.+",
  56. "https://danmu.yhdmjx.com/m3u8.php\\?url=.+",
  57. "https://player.cycanime.com/\\?url=.+",
  58. "https://www.tucao.my/play/.+",
  59. "https://ddys.pro/.+",
  60. ]
  61. },
  62. url: {
  63. regex: [
  64. "https://m3u8.girigirilove.com/.+\\.php\\?.+",
  65. "https://m3u8.girigirilove.icu/.+\\.php\\?.+",
  66. "https://cnys.tv/addons/dp/player/dp.php\\?.+",
  67. ]
  68. },
  69. html: {
  70. regex: []
  71. },
  72. script: {
  73. regex: [
  74. "https://.*libvio\\..+/vid/yd.php\\?url=.+"
  75. ]
  76. },
  77. request: {
  78. regex: []
  79. },
  80. bilibili: {
  81. regex: [
  82. "https://www.bilibili.com/bangumi/play/.+",
  83. "https://www.bilibili.com/video/.+",
  84. "https://www.bilibili.com/list/.+",
  85. "https://www.bilibili.com/festival/.+"
  86. ],
  87. preferredQuality: '127',
  88. preferredSubtitle: 'off',
  89. preferredCodec: '12',
  90. },
  91. bilibiliLive: {
  92. regex: [
  93. "https://live.bilibili.com/\\d+.*",
  94. "https://live.bilibili.com/roomid=\\d+.*",
  95. "https://live.bilibili.com/blanc/\\d+.*"
  96. ],
  97. preferredQuality: '4',
  98. preferredLine: '0',
  99. },
  100. aniGamer: {
  101. regex: [
  102. "https://ani.gamer.com.tw/animeVideo.php\\?sn=.+"
  103. ]
  104. },
  105. anime1: {
  106. regex: [
  107. "https://anime1.me/\\d+"
  108. ]
  109. }
  110. }
  111. },
  112. players: [{
  113. name: 'IINA',
  114. system: 'mac',
  115. icon: '',
  116. iconSize: 53,
  117. playEvent: "const delimiter = '&';\n\nlet args = [\n `url=${encodeURIComponent(media.video)}`,\n media.origin ? `mpv_http-header-fields=${encodeURIComponent('origin: ' + media.origin)}` : '',\n media.referer ? `mpv_http-header-fields=${encodeURIComponent('referer: ' + media.referer)}` : '',\n]\nargs = args.filter(item => item !== '');\n\nconsole.log(args);\n\nwindow.open(`iina://weblink?${args.join(delimiter)}`, '_self');",
  118. presetEvent: {
  119. playAuto: false,
  120. pauseAuto: true,
  121. closeAuto: false,
  122. syncTime: false,
  123. },
  124. enable: true,
  125. readonly: true,
  126. },
  127. {
  128. name: 'PotPlayer',
  129. system: 'windows',
  130. icon: '',
  131. iconSize: 50,
  132. playEvent: "let args = [\n `\"${media.video}\"`,\n media.subtitle ? `/sub=\"${media.subtitle}\"` : '',\n media.origin ? `/headers=\"origin: ${media.origin}\"` : '',\n media.referer ? `/referer=\"${media.referer}\"` : '',\n config.networkProxy ? `/user_agent=\"${config.networkProxy}\"` : '',\n media.title ? `/title=\"${media.title}\"` : '',\n media.time ? `/seek=\"${media.time}\"` : '',\n]\nargs = args.filter(item => item !== '');\n\nconsole.log(args);\n\nwindow.open(`ush://${player.name}?${compress(args.join(' '))}`, '_self');",
  133. presetEvent: {
  134. playAuto: false,
  135. pauseAuto: true,
  136. closeAuto: false,
  137. syncTime: false,
  138. },
  139. enable: true,
  140. readonly: true,
  141. },
  142. {
  143. name: 'MPV',
  144. system: 'windows',
  145. icon: '',
  146. iconSize: 52,
  147. playEvent: "let args = [\n `\"${media.video}\"`,\n media.audio ? `--audio-file=\"${media.audio}\"` : '',\n media.subtitle ? `--sub-file=\"${media.subtitle}\"` : '',\n media.origin ? `--http-header-fields=\"origin: ${media.origin}\"` : '',\n media.referer ? `--http-header-fields=\"referer: ${media.referer}\"` : '',\n media.cookie ? `--http-header-fields=\"cookie: ${media.cookie}\"` : '',\n config.networkProxy ? `--http-proxy=\"${config.networkProxy}\"` : '',\n media.ytdlp.networkProxy ? `--ytdl-raw-options=\"proxy=[${media.ytdlp.networkProxy}]\"` : '',\n media.ytdlp.quality ? `--ytdl-format=\"bestvideo[height<=?${media.ytdlp.quality}]+bestaudio/best\"` : '',\n media.bilibili.cid ? `--script-opts-append=\"cid=${media.bilibili.cid}\"` : '',\n media.title ? `--force-media-title=\"${media.title}\"` : '',\n media.time ? `--start=\"${media.time}\"` : '',\n]\nargs = args.filter(item => item !== '');\n\nconsole.log(args);\n\nwindow.open(`ush://${player.name}?${compress(args.join(' '))}`, '_self');",
  148. presetEvent: {
  149. playAuto: false,
  150. pauseAuto: true,
  151. closeAuto: false,
  152. syncTime: false,
  153. },
  154. enable: true,
  155. readonly: true,
  156. },
  157. {
  158. name: 'MPVNET',
  159. system: 'windows',
  160. icon: '',
  161. iconSize: 50,
  162. playEvent: "let args = [\n `\"${media.video}\"`,\n media.audio ? `--audio-file=\"${media.audio}\"` : '',\n media.subtitle ? `--sub-file=\"${media.subtitle}\"` : '',\n media.origin ? `--http-header-fields=\"origin: ${media.origin}\"` : '',\n media.referer ? `--http-header-fields=\"referer: ${media.referer}\"` : '',\n media.cookie ? `--http-header-fields=\"cookie: ${media.cookie}\"` : '',\n config.networkProxy ? `--http-proxy=\"${config.networkProxy}\"` : '',\n media.ytdlp.networkProxy ? `--ytdl-raw-options=\"proxy=[${media.ytdlp.networkProxy}]\"` : '',\n media.ytdlp.quality ? `--ytdl-format=\"bestvideo[height<=?${media.ytdlp.quality}]+bestaudio/best\"` : '',\n media.bilibili.cid ? `--script-opts=\"cid=${media.bilibili.cid}\"` : '',\n media.title ? `--force-media-title=\"${media.title}\"` : '',\n media.time ? `--start=\"${media.time}\"` : '',\n]\nargs = args.filter(item => item !== '');\n\nconsole.log(args);\n\nwindow.open(`ush://${player.name}?${compress(args.join(' '))}`, '_self');",
  163. presetEvent: {
  164. playAuto: false,
  165. pauseAuto: true,
  166. closeAuto: false,
  167. syncTime: false,
  168. },
  169. enable: true,
  170. readonly: true,
  171. }
  172. ]
  173. }
  174.  
  175. if (DEBUG === true) {
  176. defaultConfig.global.parser.ytdlp.regex.push(SETTING_URL);
  177. }
  178.  
  179. const translations = {
  180. en: {
  181. loadSuccessfully: 'Load successfully',
  182. loadTimeout: 'Load timeout ......',
  183. saveSuccessfully: 'Save successfully',
  184. loadFail: 'Load fail',
  185. requireLoginOrVip: 'Require login or vip',
  186. noMatchingParserFound: 'No matching parser found',
  187. onlyNewTabsCanCloseAutomatically: 'Only new tabs can close automatically'
  188. },
  189. zh: {
  190. loadSuccessfully: '加载成功',
  191. loadTimeout: '加载超时 ......',
  192. saveSuccessfully: '保存成功',
  193. loadFail: '加载失败',
  194. requireLoginOrVip: '需要登录(不可用)或会员',
  195. noMatchingParserFound: '没有匹配的解析器',
  196. onlyNewTabsCanCloseAutomatically: '只有新标签页才能自动关闭'
  197. }
  198. };
  199.  
  200. const REFRESH_INTERVAL = 500;
  201. const MAX_TRY_COUNT = 5;
  202.  
  203. var currentTryCount;
  204. var currentConfig;
  205. var currentUrl;
  206. var currentParser;
  207. var currentMedia;
  208. var currentPlayer;
  209. var translation;
  210. var iframe;
  211.  
  212. class BaseParser {
  213. constructor() {
  214. currentMedia = {
  215. video: undefined,
  216. audio: undefined,
  217. subtitle: undefined,
  218. title: undefined,
  219. origin: undefined,
  220. referer: undefined,
  221. time: undefined,
  222. bilibili: {
  223. cid: undefined
  224. },
  225. ytdlp: {
  226. quality: undefined,
  227. networkProxy: undefined
  228. }
  229. }
  230. }
  231. async execute() {}
  232. async parseVideo() {
  233. currentMedia.video = location.href;
  234. }
  235. async parseAudio() {}
  236. async parseSubtitle() {}
  237. async parseTitle() {
  238. currentMedia.title = document.title;
  239. }
  240. async parseOrigin() {
  241. currentMedia.origin = location.origin || location.href;
  242. }
  243. async parseReferer() {
  244. let index = currentUrl.indexOf('?');
  245. currentMedia.referer = index > 0 ? currentUrl.substring(0, index) : currentUrl;
  246. }
  247. async parseTime() {
  248. try {
  249. for (const video of document.getElementsByTagName('video')) {
  250. currentMedia.time = video.currentTime;
  251. return;
  252. }
  253. } catch (error) {
  254. console.error("获取开始时间失败", error);
  255. }
  256. }
  257. async check(video) {
  258. if (!video) {
  259. video = currentMedia.video;
  260. }
  261. if (!video || !video.startsWith('http') || video.startsWith('https://www.mp4')) {
  262. return false;
  263. }
  264.  
  265. if (video.indexOf('.m3u8') > -1 || video.indexOf('.m3u') > -1) {
  266. try {
  267. const response = await (await fetch(video, {
  268. method: 'GET',
  269. credentials: 'include'
  270. })).body();
  271. return response && response.indexOf('png') === -1;
  272. } catch (error) {}
  273. }
  274.  
  275. if (video.match("https?:\/\/[a-zA-Z0-9-/]+")) {
  276. return true;
  277. }
  278.  
  279. return new RegExp(VIDEO_URL_REGEX_EXACT).test(video);
  280. }
  281. async pause() {
  282. for (let index = 0; index < MAX_TRY_COUNT; index++) {
  283. try {
  284. for (const video of document.getElementsByTagName('video')) {
  285. video.pause();
  286. }
  287. } catch (error) {
  288. console.error('暂停失败', error);
  289. } finally {
  290. await sleep(REFRESH_INTERVAL * 3);
  291. }
  292. }
  293. }
  294. async close() {
  295. try {
  296. await sleep(REFRESH_INTERVAL * 2);
  297. if (window.top.history.length === 1) {
  298. window.top.location.href = "about:blank";
  299. window.top.close();
  300. } else {
  301. showToast(translation.onlyNewTabsCanCloseAutomatically);
  302. }
  303. } catch (error) {
  304. console.error('关闭失败', error);
  305. }
  306. }
  307. async play(player) {
  308. try {
  309. showLoading(6000);
  310.  
  311. // 别名,方便播放事件使用
  312. currentPlayer = player;
  313. let media = currentMedia;
  314. let parser = currentParser;
  315. let config = currentConfig.global;
  316.  
  317. currentTryCount = 0;
  318. let latestError = undefined;
  319. do {
  320. currentTryCount++;
  321. try {
  322. // 低端影视
  323. if (currentUrl.startsWith("https://ddys")) {
  324. document.getElementsByClassName("vjs-big-play-button")[0].click();
  325. await this.parseReferer();
  326. }
  327.  
  328. await parser.execute();
  329. if (await parser.check()) {
  330. latestError = undefined;
  331. break;
  332. }
  333. await sleep(REFRESH_INTERVAL * 2);
  334. } catch (error) {
  335. latestError = error;
  336. console.error(`第${currentTryCount}次尝试解析失败:`, error);
  337. }
  338. }
  339. while (currentTryCount < MAX_TRY_COUNT);
  340. if (latestError) {
  341. showToast(translation.loadFail + ': ' + latestError.message);
  342. return;
  343. }
  344. if (!await parser.check()) {
  345. showToast(translation.loadFail);
  346. return;
  347. }
  348. media = currentMedia;
  349.  
  350. if (player.playEvent) {
  351. eval(policy.createScript(player.playEvent));
  352. }
  353.  
  354. if (player.presetEvent.closeAuto) {
  355. parser.close();
  356. }
  357. if (player.presetEvent.pauseAuto) {
  358. parser.pause();
  359. }
  360. } catch (error) {
  361. showToast(translation.loadFail + ': ' + error.message);
  362. } finally {
  363. hideLoading();
  364. }
  365. }
  366. }
  367.  
  368. const PARSER = {
  369. YTDLP: class Parser extends BaseParser {
  370. async execute() {
  371. currentMedia.ytdlp.quality = currentConfig.global.parser.ytdlp.preferredQuality === 'unlimited' ?
  372. undefined :
  373. currentConfig.global.parser.ytdlp.preferredQuality;
  374. currentMedia.ytdlp.networkProxy = currentConfig.global.networkProxy ?
  375. currentConfig.global.networkProxy :
  376. undefined;
  377. await this.parseVideo();
  378. await this.parseTime();
  379. }
  380. async check() {
  381. return currentMedia.video ? true : false;
  382. }
  383. },
  384. VIDEO: class Parser extends BaseParser {
  385. async execute() {
  386. await this.parseVideo();
  387. await this.parseTitle();
  388. await this.parseTime();
  389. }
  390. async parseVideo() {
  391. for (const video of document.getElementsByTagName('video')) {
  392. if (await this.check(video.src)) {
  393. currentMedia.video = video.src;
  394. return;
  395. }
  396. }
  397. }
  398. },
  399. URL: class Parser extends BaseParser {
  400. async execute() {
  401. await this.parseVideo();
  402. await this.parseTitle();
  403. await this.parseTime();
  404. }
  405. async parseVideo() {
  406. let urls = currentUrl.match(VIDEO_URL_REGEX_GLOBAL) || [];
  407. for (const url of urls) {
  408. if (await this.check(url)) {
  409. currentMedia.video = url;
  410. return;
  411. }
  412. }
  413.  
  414. for (const iframe of document.getElementsByTagName('iframe')) {
  415. let urls = iframe.src.match(VIDEO_URL_REGEX_GLOBAL) || [];
  416. for (const url of urls) {
  417. if (await this.check(url)) {
  418. currentMedia.video = url;
  419. return;
  420. }
  421. }
  422. }
  423. }
  424. },
  425. HTML: class Parser extends BaseParser {
  426. async execute() {
  427. await this.parseVideo();
  428. await this.parseTitle();
  429. await this.parseTime();
  430. }
  431. async parseVideo() {
  432. let urls = document.body.innerHTML.match(VIDEO_URL_REGEX_GLOBAL) || [];
  433. for (const url of urls) {
  434. if (await this.check(url)) {
  435. currentMedia.video = url;
  436. return;
  437. }
  438. }
  439. }
  440. },
  441. SCRIPT: class Parser extends BaseParser {
  442. async execute() {
  443. await this.parseVideo();
  444. await this.parseTitle();
  445. await this.parseTime();
  446. }
  447. async parseVideo() {
  448. for (const script of document.scripts) {
  449. let urls = script.innerHTML.match(VIDEO_URL_REGEX_GLOBAL) || [];
  450. for (const url of urls) {
  451. if (await this.check(url)) {
  452. currentMedia.video = url;
  453. return;
  454. }
  455. }
  456. }
  457. }
  458. },
  459. REQUEST: class Parser extends BaseParser {
  460. constructor() {
  461. super();
  462. this.video = undefined;
  463. let that = this;
  464. const open = XMLHttpRequest.prototype.open;
  465. XMLHttpRequest.prototype.open = function (method, url, async, user, password) {
  466. if (!that.video) {
  467. let urls = url.match(VIDEO_URL_REGEX_GLOBAL) || [];
  468. for (const vurl of urls) {
  469. that.check(vurl).check().then(
  470. result => {
  471. if (result === true) {
  472. that.video = vurl;
  473. }
  474. }
  475. )
  476. }
  477.  
  478. }
  479. return open.apply(this, arguments);
  480. };
  481.  
  482. const originalFetch = fetch;
  483.  
  484. window.fetch = function (url, options) {
  485. return originalFetch(url, options).then(response => {
  486. if (!that.video) {
  487. let urls = url.match(VIDEO_URL_REGEX_GLOBAL) || [];
  488. for (const vurl of urls) {
  489. that.check(vurl).check().then(
  490. result => {
  491. if (result === true) {
  492. that.video = vurl;
  493. }
  494. }
  495. )
  496. }
  497. }
  498. return response;
  499. });
  500. };
  501. }
  502. async execute() {
  503. await this.parseTitle();
  504. await this.parseVideo();
  505. await this.parseReferer();
  506. await this.parseTime();
  507. }
  508. async parseVideo() {
  509. currentMedia.video = this.video;
  510. }
  511. },
  512. BILIBILI: class Parser extends BaseParser {
  513. async execute() {
  514. await this.parseTitle();
  515. await this.parseVideo();
  516. await this.parseReferer();
  517. await this.parseTime();
  518. }
  519. async parseVideo() {
  520. let videoInfo = undefined;
  521. if (currentUrl.startsWith('https://www.bilibili.com/bangumi/')) {
  522. videoInfo = await this.getVideoInfoByEpid();
  523. } else if (currentUrl.startsWith('https://www.bilibili.com/video/')) {
  524. videoInfo = await this.getVideoInfoByBvid();
  525. } else {
  526. videoInfo = await this.getVideoInfo();
  527. }
  528.  
  529. if (!videoInfo || !videoInfo.aid || !videoInfo.cid) {
  530. throw new Error('can not find aid and cid');
  531. }
  532.  
  533. const aid = videoInfo.aid;
  534. const cid = videoInfo.cid;
  535. const title = videoInfo.title;
  536. const codecid = currentConfig.global.parser.bilibili.preferredCodec;
  537. const quality = currentConfig.global.parser.bilibili.preferredQuality;
  538.  
  539. currentMedia.bilibili.cid = cid;
  540. currentMedia.title = title ? title : currentMedia.title;
  541. if (currentConfig.global.parser.bilibili.preferredSubtitle &&
  542. currentConfig.global.parser.bilibili.preferredSubtitle !== 'off') {
  543. currentMedia.subtitle = await this.getSubtitle(aid, cid);
  544. }
  545. // 支持传入音频优先获取 dash 格式视频,以支持更高分辨率
  546. if (currentPlayer.playEvent && currentPlayer.playEvent.indexOf('audio') > -1) {
  547. const dash = await this.getDash(aid, cid, codecid, quality);
  548. if (dash) {
  549. currentMedia.audio = dash.audio;
  550. currentMedia.video = dash.video;
  551. return;
  552. }
  553. }
  554. currentMedia.video = await this.getFlvOrMP4(aid, cid);
  555. }
  556. async getVideoInfo() {
  557. try {
  558. const initialState = __INITIAL_STATE__;
  559. if (!initialState) {
  560. return;
  561. }
  562. const videoInfo = initialState.epInfo || initialState.videoData || initialState.videoInfo;
  563. const aid = videoInfo.aid;
  564. const page = initialState.p;
  565. let cid = videoInfo.cid;
  566. let title = videoInfo.title;
  567. if (page && page > 1) {
  568. cid = initialState.cidMap[aid].cids[page];
  569. }
  570.  
  571. return {
  572. aid: aid,
  573. cid: cid,
  574. title: title
  575. };
  576. } catch (error) {
  577. console.error(error.message);
  578. }
  579. }
  580. async getVideoInfoByBvid() {
  581. let param = undefined;
  582. const bvids = currentUrl.match(/BV([0-9a-zA-Z]+)/);
  583. if (bvids && bvids[1]) {
  584. param = `bvid=${bvids[1]}`;
  585. } else {
  586. const avids = currentUrl.match(/av([0-9]+)/);
  587. param = `aid=${avids[1]}`;
  588. }
  589.  
  590. if (!param) {
  591. throw new Error('can not find bvid or avid');
  592. }
  593.  
  594. const response = await (await fetch(`https://api.bilibili.com/x/web-interface/view?${param}`, {
  595. method: 'GET',
  596. credentials: 'include'
  597. })).json();
  598.  
  599. let aid = response.data.aid;
  600. let cid = response.data.cid;
  601. let title = response.data.title;
  602.  
  603. // 分 p 视频
  604. const ps = currentUrl.match(/[?&]p=([^&]+)/);
  605. if (ps && response.data.pages.length > 1) {
  606. const p = ps[1];
  607. const currentPage = response.data.pages[p - 1];
  608. cid = currentPage.cid;
  609. title = currentPage.part;
  610. }
  611.  
  612. return {
  613. aid: aid,
  614. cid: cid,
  615. title: title
  616. };
  617. }
  618. async getVideoInfoByEpid() {
  619. let epid = undefined;
  620. let epids = currentUrl.match(/ep(\d+)/);
  621. if (epids && epids[1]) {
  622. epid = epids[1];
  623. } else {
  624. let epidElement = undefined;
  625. let epidElementClassNames = [
  626. "ep-item cursor visited",
  627. "ep-item cursor",
  628. "numberListItem_select__WgCVr",
  629. "imageListItem_wrap__o28QW",
  630. ];
  631. for (const className of epidElementClassNames) {
  632. epidElement = document.getElementsByClassName(className)[0];
  633. if (epidElement) {
  634. epid = epidElement.getElementsByTagName("a")[0].href.match(/ep(\d+)/)[1];
  635. break;
  636. }
  637. }
  638.  
  639. if (!epid) {
  640. epidElement = document.getElementsByClassName("squirtle-pagelist-select-item active squirtle-blink")[0];
  641. if (epidElement) {
  642. epid = epidElement.dataset.value;
  643. }
  644. }
  645. }
  646.  
  647. if (!epid) {
  648. throw new Error('can not find epid');
  649. }
  650.  
  651. const response = await (await fetch(`https://api.bilibili.com/pgc/view/web/season?ep_id=${epid}`, {
  652. method: 'GET',
  653. credentials: 'include'
  654. })).json();
  655. let section = response.result.section;
  656. if (!section) {
  657. section = new Array();
  658. }
  659. section.push({
  660. episodes: response.result.episodes
  661. });
  662. let currentEpisode;
  663. for (let i = section.length - 1; i >= 0; i--) {
  664. let episodes = section[i].episodes;
  665. for (const episode of episodes) {
  666. if (episode.id == epid) {
  667. currentEpisode = episode;
  668. break;
  669. }
  670. }
  671. if (currentEpisode) {
  672. return {
  673. aid: currentEpisode.aid,
  674. cid: currentEpisode.cid,
  675. title: currentEpisode.share_copy
  676. }
  677. }
  678. }
  679. }
  680. async getDash(aid, cid, codecid, quality) {
  681. const url = `https://api.bilibili.com/x/player/playurl?qn=120&otype=json&fourk=1&fnver=0&fnval=4048&avid=${aid}&cid=${cid}`;
  682. const response = await (await fetch(url, {
  683. method: 'GET',
  684. credentials: 'include'
  685. })).json();
  686. if (!response.data) {
  687. currentTryCount = MAX_TRY_COUNT;
  688. throw new Error(translation.requireLoginOrVip);
  689. }
  690. let video = undefined;
  691. let audio = undefined;
  692. let dash = response.data.dash;
  693. if (!dash) {
  694. return undefined;
  695. }
  696. let hiRes = dash.flac;
  697. let dolby = dash.dolby;
  698. if (hiRes && hiRes.audio) {
  699. audio = hiRes.audio.baseUrl;
  700. } else if (dolby && dolby.audio) {
  701. audio = dolby.audio[0].base_url;
  702. } else if (dash.audio) {
  703. audio = dash.audio[0].baseUrl;
  704. }
  705. let i = 0;
  706. while (i < dash.video.length &&
  707. dash.video[i].id > quality) {
  708. i++;
  709. }
  710. video = dash.video[i].baseUrl;
  711. let id = dash.video[i].id;
  712. while (i < dash.video.length) {
  713. if (dash.video[i].id != id) {
  714. break;
  715. }
  716. if (dash.video[i].codecid == codecid) {
  717. video = dash.video[i].baseUrl;
  718. break;
  719. }
  720. i++;
  721. }
  722. return {
  723. video: video,
  724. audio: audio
  725. };
  726. }
  727. async getFlvOrMP4(aid, cid) {
  728. const url = `https://api.bilibili.com/x/player/playurl?qn=120&otype=json&fourk=1&fnver=0&fnval=128&avid=${aid}&cid=${cid}`;
  729. const response = await (await fetch(url, {
  730. method: 'GET',
  731. credentials: 'include'
  732. })).json();
  733. if (!response.data) {
  734. currentTryCount = MAX_TRY_COUNT;
  735. throw new Error(translation.requireLoginOrVip);
  736. }
  737. return response.data.durl[0].url;
  738. }
  739. async getSubtitle(avid, cid) {
  740. const url = `https://api.bilibili.com/x/player/wbi/v2?aid=${avid}&cid=${cid}`;
  741. const response = await (await fetch(url, {
  742. method: 'GET',
  743. credentials: 'include'
  744. })).json();
  745.  
  746. if (response.code === 0 && response.data.subtitle && response.data.subtitle.subtitles.length > 0) {
  747. let subtitles = response.data.subtitle.subtitles;
  748. let url = subtitles[0].subtitle_url;
  749. let lan = subtitles[0].lan;
  750. for (const subtitle of subtitles) {
  751. if (currentConfig.global.parser.bilibili.preferredSubtitle.startsWith("zh") &&
  752. subtitle.lan.startsWith("zh")) {
  753. url = subtitle.subtitle_url;
  754. lan = subtitle.lan;
  755. }
  756. if (subtitle.lan == currentConfig.subtitlePrefer) {
  757. url = subtitle.subtitle_url;
  758. lan = subtitle.lan;
  759. break;
  760. }
  761. }
  762. if (url) {
  763. return `https://www.lckp.top/common/bilibili/jsonToSrt/?url=https:${url}&lan=${lan}`;
  764. }
  765. }
  766. }
  767. },
  768. BILIBILI_LIVE: class Parser extends BaseParser {
  769. async execute() {
  770. await this.parseVideo();
  771. await this.parseTitle();
  772. await this.parseReferer();
  773. }
  774. async parseVideo() {
  775. const roomids = currentUrl.match(
  776. /.*(roomid=|blanc\/|live.bilibili.com\/)(\d+).*/
  777. );
  778. const roomid = roomids ? roomids[2] : undefined;
  779.  
  780. if (!roomid) {
  781. throw new Error('can not find roomid');
  782. }
  783.  
  784. const quality = currentConfig.global.parser.bilibiliLive.preferredQuality;
  785. const url = `https://api.live.bilibili.com/room/v1/Room/playUrl?quality=${quality}&cid=${roomid}`;
  786. const response = await (await fetch(url, {
  787. method: 'GET',
  788. credentials: 'include'
  789. })).json();
  790.  
  791. const durls = response.data.durl;
  792. const line = currentConfig.global.parser.bilibiliLive.preferredLine;
  793. let durl = durls[durls.length - 1];
  794. for (let index = 0; index < durls.length; index++) {
  795. if (line == index) {
  796. durl = durls[index];
  797. break;
  798. }
  799. }
  800. currentMedia.video = durl.url;
  801. }
  802. },
  803. ANI_GAMER: class Parser extends BaseParser {
  804. async execute() {
  805. await this.parseVideo();
  806. await this.parseOrigin();
  807. await this.parseTitle();
  808. await this.parseTime();
  809. }
  810. async parseVideo() {
  811. let match = currentUrl.match(/[?&]sn=([^&]+)/);
  812. const sn = match ? match[1] : undefined;
  813. if (!sn) {
  814. return;
  815. }
  816. const device = localStorage.ANIME_deviceid;
  817. const url = `https://ani.gamer.com.tw/ajax/m3u8.php?sn=${sn}&device=${device}`;
  818. const response = await (await fetch(url, {
  819. method: 'GET',
  820. credentials: 'include'
  821. })).json();
  822. if (response.error && response.error.code === 1015) {
  823. throw new Error("請先跳過廣告后再嘗試");
  824. }
  825. currentMedia.video = response ? response.src : undefined;
  826. }
  827. },
  828. ANIME1: class Parser extends BaseParser {
  829. async execute() {
  830. await this.parseVideo();
  831. await this.parseTitle();
  832. await this.parseTime();
  833. }
  834. async parseVideo() {
  835. const anime1_api_url = 'https://v.anime1.me/api';
  836. const body = `d=${document.querySelector("video").getAttribute("data-apireq")}`;
  837. const response = await new Promise(res => {
  838. GM.xmlHttpRequest({
  839. headers: {
  840. "content-type": "application/x-www-form-urlencoded",
  841. },
  842. method: "POST",
  843. url: anime1_api_url,
  844. data: body,
  845. onload: function (response) {
  846. res(response);
  847. }
  848. });
  849. });
  850.  
  851. let cookies = [];
  852. let cookieLines = response.responseHeaders.match(/set-cookie:\s*([^;]*)/gi);
  853. if (cookieLines) {
  854. cookieLines.forEach(cookieStr => {
  855. let [key, value] = cookieStr.replace(/set-cookie:\s*/i, "").split("=");
  856. cookies.push(`${key}=${value}`);
  857. });
  858. }
  859. currentMedia.cookie = cookies.join("; ");
  860.  
  861. const video = response?.responseText ? JSON.parse(response.responseText).s?. [0]?.src : undefined;
  862. currentMedia.video = video ? "https:" + video : undefined;
  863. }
  864. },
  865. IFRAME: class Parser extends BaseParser {
  866. async execute() {
  867. iframe.postMessage({
  868. name: PROJECT_NAME,
  869. method: 'execute'
  870. }, '*');
  871. await sleep(REFRESH_INTERVAL);
  872. await this.parseTitle();
  873. }
  874. async pause() {
  875. iframe.postMessage({
  876. name: PROJECT_NAME,
  877. method: 'pause'
  878. }, '*');
  879. }
  880. }
  881. };
  882.  
  883. function compress(str) {
  884. return btoa(String.fromCharCode(...pako.gzip(str)));
  885. };
  886.  
  887. function sleep(ms) {
  888. return new Promise((resolve) => setTimeout(resolve, ms));
  889. }
  890.  
  891. function loadConfig() {
  892. let config = GM_getValue('config');
  893. if (config) {
  894. if (config.global.version === defaultConfig.global.version) {
  895. return config;
  896. }
  897. console.log('更新配置 ......');
  898. config = updateConfig(defaultConfig, config);
  899. config.global.version = defaultConfig.global.version;
  900. } else {
  901. console.log('初始化配置 ......');
  902. config = JSON.parse(JSON.stringify(defaultConfig));
  903. for (const key in config.global.parser) {
  904. config.global.parser[key].regex = [];
  905. }
  906. }
  907. GM_setValue('config', config);
  908. return config;
  909. }
  910.  
  911. function updateConfig(defaultConfig, config) {
  912. function mergeDefaults(defaultObj, currentObj) {
  913. if (typeof defaultObj !== 'object' || defaultObj === null) {
  914. return currentObj !== undefined ? currentObj : defaultObj;
  915. }
  916.  
  917. if (Array.isArray(defaultObj)) {
  918. return Array.isArray(currentObj) ? currentObj : defaultObj;
  919. }
  920.  
  921. const merged = {};
  922. for (const key in defaultObj) {
  923. if (key === 'regex') {
  924. merged[key] = currentObj?. [key] || [];
  925. continue;
  926. }
  927. merged[key] = mergeDefaults(defaultObj[key], currentObj?. [key]);
  928. }
  929. return merged;
  930. }
  931.  
  932. const newConfig = mergeDefaults(defaultConfig, config);
  933. for (let index = 0; index < defaultConfig.players.length; index++) {
  934. const dp = defaultConfig.players[index];
  935. const np = newConfig.players[index];
  936. if (np && dp.name === np.name) {
  937. np.icon = dp.icon;
  938. np.readonly = dp.readonly;
  939. np.playEvent = dp.playEvent;
  940. if (!np.presetEvent.syncTime) {
  941. np.presetEvent.syncTime = dp.presetEvent.syncTime;
  942. }
  943. } else {
  944. newConfig.players.unshift(dp);
  945. }
  946. }
  947.  
  948. return newConfig;
  949. }
  950.  
  951. function matchParser(parser, url) {
  952. for (const key in parser) {
  953. for (const regex of parser[key].regex) {
  954. if (!regex || regex.startsWith('#') || regex.startsWith('//')) {
  955. continue;
  956. }
  957. if (new RegExp(regex).test(url)) {
  958. console.log(`match parser regex: ${new RegExp(regex)}\n${url}`);
  959. return new PARSER[key.replace(/[A-Z]/g, letter => `_${letter}`).toUpperCase()]();
  960. }
  961. }
  962. }
  963. }
  964.  
  965. // =================================== 按钮区域和设置页面 ===================================
  966.  
  967. var policy;
  968. try {
  969. policy = window.trustedTypes.createPolicy('externalPlayer', {
  970. createHTML: (string, sink) => string,
  971. createScript: (input) => input
  972. })
  973. } catch (error) {
  974. policy = {
  975. createHTML: (string, sink) => string,
  976. createScript: (input) => input
  977. }
  978. }
  979.  
  980. const FIRST_Z_INDEX = 999999999;
  981. const SECOND_Z_INDEX = FIRST_Z_INDEX - 1;
  982. const THIRD_Z_INDEX = SECOND_Z_INDEX - 1;
  983.  
  984. const COLORS = [{
  985. // 配色方案1
  986. PRIMARY: 'rgba(245, 166, 35, 1)',
  987. TEXT: 'rgba(90, 90, 90, 1)',
  988. TEXT_ACTIVE: 'rgba(255, 255, 255, 1)',
  989. WARNING: 'rgba(233, 78, 119, 1)',
  990. BORDER: 'rgba(243, 229, 213, 1)',
  991. }, {
  992. // 配色方案2
  993. PRIMARY: 'rgba(60, 179, 113, 1)',
  994. TEXT: 'rgba(47, 79, 79, 1)',
  995. TEXT_ACTIVE: 'rgba(255, 255, 255, 1)',
  996. WARNING: 'rgba(255, 111, 97, 1)',
  997. BORDER: 'rgba(204, 231, 208, 1)',
  998. }, {
  999. // 配色方案3
  1000. PRIMARY: 'rgba(74, 144, 226, 1)',
  1001. TEXT: 'rgba(51, 51, 51, 1)',
  1002. TEXT_ACTIVE: 'rgba(255, 255, 255, 1)',
  1003. WARNING: 'rgba(242, 95, 92, 1)',
  1004. BORDER: 'rgba(217, 227, 240, 1)',
  1005. }]
  1006. const COLOR = COLORS[2];
  1007.  
  1008. var style;
  1009. var buttonDiv;
  1010. var toastDiv;
  1011. var loadingDiv;
  1012. var settingButton;
  1013. var settingIframe;
  1014. var loadingId;
  1015. var isReloading = false;
  1016.  
  1017. function appendCss() {
  1018. if (style) {
  1019. return;
  1020. }
  1021. style = document.createElement('style');
  1022. style.innerHTML = policy.createHTML(`
  1023. #${PROJECT_NAME}-toast-div {
  1024. z-index: ${FIRST_Z_INDEX};
  1025. position: fixed;
  1026. top: 20px;
  1027. left: 50%;
  1028. transform: translate(-50%, 0);
  1029. background-color: rgba(0, 0, 0, 0.8);
  1030. color: white;
  1031. font-size: 14px;
  1032. padding: 10px 20px;
  1033. border-radius: 5px;
  1034. opacity: 0;
  1035. transition: opacity 0.5s ease;
  1036. display: none;
  1037. letter-spacing: 1px;
  1038. }
  1039. #${PROJECT_NAME}-loading-div {
  1040. z-index: ${FIRST_Z_INDEX};
  1041. display: none;
  1042. position: fixed;
  1043. bottom: 50%;
  1044. left: 50%;
  1045. transform: translate(-50%, -50%);
  1046. background-color: rgba(0, 0, 0, 0);
  1047. }
  1048. #${PROJECT_NAME}-loading-div div {
  1049. width: 50px;
  1050. height: 50px;
  1051. background-color: ${COLOR.PRIMARY};
  1052. border-radius: 0;
  1053. -webkit-animation: sk-rotateplane 1.2s infinite ease-in-out;
  1054. animation: sk-rotateplane 1.2s infinite ease-in-out;
  1055. }
  1056. @-webkit-keyframes sk-rotateplane {
  1057. 0% {
  1058. -webkit-transform: perspective(120px)
  1059. }
  1060. 50% {
  1061. -webkit-transform: perspective(120px) rotateY(180deg)
  1062. }
  1063. 100% {
  1064. -webkit-transform: perspective(120px) rotateY(180deg) rotateX(180deg)
  1065. }
  1066. }
  1067. @keyframes sk-rotateplane {
  1068. 0% {
  1069. transform: perspective(120px) rotateX(0deg) rotateY(0deg);
  1070. -webkit-transform: perspective(120px) rotateX(0deg) rotateY(0deg)
  1071. }
  1072. 50% {
  1073. transform: perspective(120px) rotateX(-180deg) rotateY(0deg);
  1074. -webkit-transform: perspective(120px) rotateX(-180deg) rotateY(0deg)
  1075. }
  1076. 100% {
  1077. transform: perspective(120px) rotateX(-180deg) rotateY(-180deg);
  1078. -webkit-transform: perspective(120px) rotateX(-180deg) rotateY(-180deg);
  1079. }
  1080. }
  1081. #${PROJECT_NAME}-button-div {
  1082. z-index: ${THIRD_Z_INDEX};
  1083. position: fixed;
  1084. display: none;
  1085. align-items: center;
  1086. width: auto;
  1087. height: auto;
  1088. left: ${currentConfig.global.buttonXCoord}px;
  1089. bottom: ${currentConfig.global.buttonYCoord}px;
  1090. padding: 5px;
  1091. border: 3px solid rgba(0, 0, 0, 0);
  1092. border-radius: 5px;
  1093. cursor: move;
  1094. gap: 10px;
  1095. background-color: rgba(0, 0, 0, 0);
  1096. min-width: ${50 * currentConfig.global.buttonScale}px;
  1097. min-height: ${50 * currentConfig.global.buttonScale}px;
  1098. }
  1099. #${PROJECT_NAME}-button-div button {
  1100. color: white;
  1101. font-size: 20px;
  1102. font-weight: bold;
  1103. width: 50px;
  1104. height: 50px;
  1105. outline: none;
  1106. border: none;
  1107. border-radius: 50%;
  1108. cursor: pointer;
  1109. background-size: cover;
  1110. background-color: rgba(0, 0, 0, 0);
  1111. transition: opacity 0.5s ease, visibility 0s linear 0.5s;
  1112. }
  1113. #${PROJECT_NAME}-button-div:hover {
  1114. background-color: rgb(255, 255, 255, 0.3) !important;
  1115. }
  1116. #${PROJECT_NAME}-button-div:hover button {
  1117. visibility: visible !important;
  1118. transition: opacity 0.5s ease, visibility 0s;
  1119. }
  1120. #${PROJECT_NAME}-button-div button:hover {
  1121. transform: scale(1.06);
  1122. box-shadow: 0px 0px 16px #e6e6e6;
  1123. }
  1124. #${PROJECT_NAME}-setting-button {
  1125. visibility: hidden;
  1126. position: absolute;
  1127. right: ${-12 * currentConfig.global.buttonScale}px !important;
  1128. top: ${-12 * currentConfig.global.buttonScale}px !important;
  1129. width: ${25 * currentConfig.global.buttonScale}px !important;
  1130. height: ${25 * currentConfig.global.buttonScale}px !important;
  1131. background-image: url('data:image/svg+xml,<svg t="1731846507027" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="4281" width="16" height="16"><path d="M616.533333 512.128c0-25.6-9.941333-49.536-28.16-67.669333a95.744 95.744 0 0 0-67.84-28.074667c-25.685333 0-49.706667 9.984-67.925333 28.074667a95.146667 95.146667 0 0 0-28.16 67.669333c0 25.6 10.069333 49.578667 28.16 67.712 18.218667 18.048 42.24 28.074667 67.925333 28.074667 25.642667 0 49.664-10.026667 67.84-28.074667 18.218667-18.133333 28.16-42.112 28.16-67.712z m-202.112 352.896l48-55.978667a309.290667 309.290667 0 0 0 99.029334 0l48 55.978667a27.52 27.52 0 0 0 30.208 7.978667l2.218666-0.768a380.074667 380.074667 0 0 0 118.186667-68.138667l1.834667-1.536a27.434667 27.434667 0 0 0 8.106666-30.037333l-24.746666-69.546667a298.666667 298.666667 0 0 0 49.322666-85.205333l72.874667-13.44a27.477333 27.477333 0 0 0 22.058667-22.101334l0.426666-2.304a384.64 384.64 0 0 0 0-135.936l-0.426666-2.304a27.477333 27.477333 0 0 0-22.058667-22.058666l-73.216-13.525334a302.293333 302.293333 0 0 0-49.194667-84.650666l25.002667-70.016a27.306667 27.306667 0 0 0-8.149333-30.037334l-1.834667-1.536a383.018667 383.018667 0 0 0-118.186667-68.138666l-2.218666-0.768a27.605333 27.605333 0 0 0-30.208 7.936l-48.512 56.661333a302.592 302.592 0 0 0-97.834667 0L414.592 159.146667a27.52 27.52 0 0 0-30.208-7.978667l-2.218667 0.768a381.056 381.056 0 0 0-118.186666 68.138667l-1.834667 1.536a27.434667 27.434667 0 0 0-8.106667 30.037333l24.96 69.973333a296.192 296.192 0 0 0-49.194666 84.693334l-73.216 13.525333a27.477333 27.477333 0 0 0-22.058667 22.058667l-0.426667 2.304a382.592 382.592 0 0 0 0 135.936l0.426667 2.304c2.048 11.221333 10.794667 20.053333 22.058667 22.101333l72.874666 13.44a300.672 300.672 0 0 0 49.365334 85.248l-24.832 69.504a27.306667 27.306667 0 0 0 8.149333 30.037333l1.834667 1.536a383.018667 383.018667 0 0 0 118.186666 68.138667l2.218667 0.768a27.733333 27.733333 0 0 0 30.037333-8.149333z m-44.8-352.853333A150.656 150.656 0 0 1 520.533333 361.642667a150.656 150.656 0 0 1 150.869334 150.442666A150.656 150.656 0 0 1 520.533333 662.613333a150.656 150.656 0 0 1-150.912-150.485333z" fill="${COLOR.PRIMARY}" p-id="4282"></path></svg>');
  1132. }
  1133. #${PROJECT_NAME}-setting-iframe {
  1134. z-index: ${SECOND_Z_INDEX};
  1135. position: fixed;
  1136. width: 1000px;
  1137. max-width: 100%;
  1138. height: 500px;
  1139. max-height: 90%;
  1140. top: 50%;
  1141. left: 50%;
  1142. transform: translate(-50%, -50%);
  1143. border: none;
  1144. border-radius: 5px;
  1145. box-shadow: 0 0 16px rgba(0, 0, 0, 0.6);
  1146. background-color: #fff;
  1147. display: none;
  1148. }
  1149. `);
  1150. document.head.appendChild(style);
  1151. }
  1152.  
  1153. function appendToastDiv() {
  1154. const TOAST_DIV_ID = `${PROJECT_NAME}-toast-div`;
  1155. if (document.getElementById(TOAST_DIV_ID)) {
  1156. return;
  1157. }
  1158. toastDiv = document.createElement('div');
  1159. toastDiv.id = TOAST_DIV_ID;
  1160. document.body.appendChild(toastDiv);
  1161. }
  1162.  
  1163. function showToast(message) {
  1164. toastDiv.textContent = message;
  1165. toastDiv.style.opacity = '0.9';
  1166. toastDiv.style.display = 'block';
  1167. setTimeout(() => {
  1168. toastDiv.style.opacity = '0';
  1169. toastDiv.style.display = 'none';
  1170. }, 5000);
  1171. }
  1172.  
  1173. function appendLoadingDiv() {
  1174. const LOADING_DIV_ID = `${PROJECT_NAME}-loading-div`;
  1175. if (document.getElementById(LOADING_DIV_ID)) {
  1176. return;
  1177. }
  1178. loadingDiv = document.createElement('div');
  1179. loadingDiv.id = LOADING_DIV_ID;
  1180. loadingDiv.appendChild(document.createElement('div'));
  1181. document.body.appendChild(loadingDiv);
  1182. }
  1183.  
  1184. function showLoading(timeout) {
  1185. if (loadingId) {
  1186. clearTimeout(loadingId);
  1187. loadingId = undefined;
  1188. }
  1189. if (!timeout) {
  1190. timeout = 10000;
  1191. }
  1192. loadingDiv.style.display = 'block';
  1193. loadingId = setTimeout(() => {
  1194. if (loadingDiv.style.display === 'block') {
  1195. hideLoading();
  1196. showToast(translation.loadTimeout);
  1197. }
  1198. }, timeout);
  1199. }
  1200.  
  1201. function hideLoading() {
  1202. loadingDiv.style.display = 'none';
  1203. }
  1204.  
  1205. function appendButtonDiv() {
  1206. const BUTTON_DIV_ID = `${PROJECT_NAME}-button-div`;
  1207. if (document.getElementById(BUTTON_DIV_ID)) {
  1208. buttonDiv.style.display = "none";
  1209. return;
  1210. }
  1211. buttonDiv = document.createElement('div');
  1212. buttonDiv.id = BUTTON_DIV_ID;
  1213. buttonDiv.addEventListener('mousedown', (e) => {
  1214. if (e.target.tagName === 'BUTTON') {
  1215. return;
  1216. }
  1217. let offsetX = e.clientX - buttonDiv.getBoundingClientRect().left;
  1218. let offsetY = e.clientY - buttonDiv.getBoundingClientRect().top;
  1219.  
  1220. document.addEventListener('mouseup', mouseUpHandler);
  1221. document.addEventListener('mousemove', mouseMoveHandler);
  1222.  
  1223. function mouseUpHandler() {
  1224. buttonDiv.style.border = '3px solid rgba(0, 0, 0, 0)';
  1225. document.removeEventListener('mousemove', mouseMoveHandler);
  1226. document.removeEventListener('mouseup', mouseUpHandler);
  1227. }
  1228.  
  1229. function mouseMoveHandler(e) {
  1230. buttonDiv.style.border = `3px solid ${COLOR.PRIMARY}`;
  1231. let newX = e.clientX - offsetX;
  1232. let newY = e.clientY - offsetY;
  1233.  
  1234. const windowWidth = window.innerWidth;
  1235. const windowHeight = window.innerHeight;
  1236. const divWidth = buttonDiv.offsetWidth;
  1237. const divHeight = buttonDiv.offsetHeight;
  1238.  
  1239. if (newX < 0) newX = 0;
  1240. if (newX + divWidth > windowWidth) newX = windowWidth - divWidth;
  1241. if (newY < 0) newY = 0;
  1242. if (newY + divHeight > windowHeight) newY = windowHeight - divHeight;
  1243.  
  1244. newY = windowHeight - newY - divHeight;
  1245. buttonDiv.style.left = `${newX}px`;
  1246. buttonDiv.style.bottom = `${newY}px`;
  1247. currentConfig.global.buttonXCoord = newX;
  1248. currentConfig.global.buttonYCoord = newY;
  1249. GM_setValue('config', currentConfig);
  1250. }
  1251. });
  1252. document.body.appendChild(buttonDiv);
  1253.  
  1254. appendPlayButton();
  1255. appendSettingButton();
  1256.  
  1257. // 全屏隐藏
  1258. document.addEventListener("fullscreenchange", () => {
  1259. if (document.fullscreenElement) {
  1260. buttonDiv.style.display = "none";
  1261. } else {
  1262. if (currentParser) {
  1263. buttonDiv.style.display = "flex";
  1264. }
  1265. }
  1266. });
  1267. }
  1268.  
  1269. function appendPlayButton() {
  1270. if (!currentConfig.players) {
  1271. return;
  1272. }
  1273. var playButtonNeedAutoClick;
  1274. currentConfig.players.forEach(player => {
  1275. if (player.enable !== true) {
  1276. return;
  1277. }
  1278. const playButton = document.createElement('button');
  1279. if (player.icon) {
  1280. const image = new Image();
  1281. image.src = player.icon;
  1282. image.onload = () => playButton.style.backgroundImage = `url(${image.src})`;
  1283. image.onerror = () => {
  1284. playButton.style.backgroundColor = COLOR.PRIMARY;
  1285. playButton.textContent = player.name ? player.name.substring(0, 1) : 'P';
  1286. };
  1287. } else {
  1288. playButton.style.backgroundColor = COLOR.PRIMARY;
  1289. playButton.textContent = player.name ? player.name.substring(0, 1) : 'P';
  1290. }
  1291. playButton.style.width = `${player.iconSize * currentConfig.global.buttonScale}px`;
  1292. playButton.style.height = `${player.iconSize * currentConfig.global.buttonScale}px`;
  1293.  
  1294. // 自动隐藏
  1295. if (currentConfig.global.buttonVisibilityDuration == 0) {
  1296. playButton.style.visibility = 'hidden';
  1297. } else if (currentConfig.global.buttonVisibilityDuration > 0) {
  1298. setTimeout(() => {
  1299. playButton.style.visibility = 'hidden';
  1300. }, currentConfig.global.buttonVisibilityDuration);
  1301. }
  1302.  
  1303. playButton.addEventListener('click', async function () {
  1304. playButton.disabled = true;
  1305. if (currentParser) {
  1306. currentParser.play(player);
  1307. } else {
  1308. showToast(translation.noMatchingParserFound);
  1309. }
  1310. setTimeout(() => {
  1311. playButton.disabled = false;
  1312. }, REFRESH_INTERVAL * 3);
  1313. });
  1314.  
  1315. buttonDiv.appendChild(playButton);
  1316. });
  1317. }
  1318.  
  1319. function appendSettingButton() {
  1320. settingButton = document.createElement('button');
  1321. settingButton.id = `${PROJECT_NAME}-setting-button`;
  1322. settingButton.title = 'Ctrl + Alt + E';
  1323.  
  1324. settingButton.addEventListener('click', async () => {
  1325. await appendSettingIframe();
  1326. if (settingIframe.style.display === "block") {
  1327. settingIframe.style.display = "none";
  1328. } else {
  1329. settingIframe.contentWindow.postMessage({
  1330. name: PROJECT_NAME,
  1331. method: 'loadConfig',
  1332. defaultConfig: defaultConfig,
  1333. config: currentConfig
  1334. }, '*');
  1335. settingIframe.style.display = "block";
  1336. }
  1337. });
  1338. buttonDiv.appendChild(settingButton);
  1339.  
  1340. // 失去焦点隐藏设置页面
  1341. document.addEventListener('click', (event) => {
  1342. if (settingIframe && settingIframe.style.display === 'block' &&
  1343. !settingButton.contains(event.target) &&
  1344. !settingIframe.contains(event.target)) {
  1345. settingIframe.style.display = 'none';
  1346. }
  1347. });
  1348. }
  1349.  
  1350. async function appendSettingIframe() {
  1351. const SETTING_IFRAME_ID = `${PROJECT_NAME}-setting-iframe`;
  1352. if (document.getElementById(SETTING_IFRAME_ID)) {
  1353. return;
  1354. }
  1355. settingIframe = document.createElement('iframe');
  1356. settingIframe.id = SETTING_IFRAME_ID;
  1357. let settingIframeHtml = `
  1358. <!DOCTYPE html>
  1359. <html lang="en">
  1360.  
  1361. <head>
  1362. <meta charset="UTF-8">
  1363. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  1364. <title>External Player</title>
  1365. <style>
  1366. :root {
  1367. --primary-color: ${COLOR.PRIMARY};
  1368. --text-color: ${COLOR.TEXT};
  1369. --text-active-color: ${COLOR.TEXT_ACTIVE};
  1370. --warning-color: ${COLOR.WARNING};
  1371. --border-color: ${COLOR.BORDER};
  1372. }
  1373.  
  1374. body {
  1375. display: flex;
  1376. flex-direction: row;
  1377. height: 100vh;
  1378. margin: 0;
  1379. }
  1380.  
  1381. body,
  1382. button,
  1383. input,
  1384. textarea,
  1385. select {
  1386. font-family: auto;
  1387. color: var(--text-color);
  1388. }
  1389.  
  1390. ::placeholder {
  1391. font-family: auto;
  1392. color: var(--text-color);
  1393. opacity: 0.2;
  1394. }
  1395.  
  1396. #sidebar-container {
  1397. display: none;
  1398. flex: 0 0 200px;
  1399. flex-direction: column;
  1400. background-color: #f4f4f4;
  1401. box-shadow: 2px 0 5px rgba(0, 0, 0, 0.1);
  1402. padding: 25px 20px 35px 20px;
  1403. }
  1404.  
  1405. #sidebar {
  1406. flex: 1;
  1407. overflow-y: auto;
  1408. position: relative;
  1409. border: none;
  1410. border-radius: 5px;
  1411. margin-bottom: 10px;
  1412. }
  1413.  
  1414. #sidebar::-webkit-scrollbar {
  1415. display: none !important;
  1416. }
  1417.  
  1418. .reset-button,
  1419. #add-tab-button,
  1420. #save-button,
  1421. #sidebar button {
  1422. width: 200px;
  1423. padding: 10px;
  1424. margin: 0 0 10px 0;
  1425. border: none;
  1426. border-radius: 5px;
  1427. background-color: #e0e0e0;
  1428. cursor: pointer;
  1429. font-size: 15px;
  1430. white-space: nowrap;
  1431. display: inline-flex;
  1432. position: relative;
  1433. align-items: center;
  1434. justify-content: center;
  1435. }
  1436.  
  1437. #add-tab-button,
  1438. #save-button {
  1439. background-color: var(--primary-color);
  1440. color: var(--text-active-color);
  1441. margin: 0;
  1442. }
  1443.  
  1444. #add-tab-button {
  1445. font-size: 25px;
  1446. line-height: 21.45px;
  1447. }
  1448.  
  1449. #add-tab-button:hover,
  1450. #save-button:hover {
  1451. opacity: 0.9;
  1452. }
  1453.  
  1454. #reset-button-coord-button {
  1455. padding: 7px 10px;
  1456. }
  1457.  
  1458. .reset-button {
  1459. margin: 0;
  1460. width: 80px;
  1461. background-color: var(--warning-color);
  1462. color: var(--text-active-color);
  1463. opacity: 0.6;
  1464. }
  1465.  
  1466. .reset-button:hover {
  1467. opacity: 0.8;
  1468. }
  1469.  
  1470. #sidebar button svg {
  1471. width: 20px !important;
  1472. height: 20px !important;
  1473. position: absolute;
  1474. left: 10px;
  1475. fill: var(--text-color);
  1476. }
  1477.  
  1478. #content .radio-button svg {
  1479. width: 20px !important;
  1480. height: 20px !important;
  1481. fill: var(--text-color);
  1482. }
  1483.  
  1484. #sidebar button.active svg,
  1485. #sidebar button:hover svg,
  1486. #content .radio-button.active svg,
  1487. #content .radio-button:hover svg {
  1488. fill: var(--text-active-color)
  1489. }
  1490.  
  1491. #sidebar button.active {
  1492. background-color: var(--primary-color);
  1493. color: var(--text-active-color);
  1494. }
  1495.  
  1496. #sidebar button:hover {
  1497. background-color: var(--primary-color);
  1498. color: var(--text-active-color);
  1499. }
  1500.  
  1501. #content-container {
  1502. display: none;
  1503. flex-direction: column;
  1504. flex: 1;
  1505. padding: 25px 20px 0 20px;
  1506. }
  1507.  
  1508. #content {
  1509. flex: 1;
  1510. padding: 20px;
  1511. overflow-y: auto;
  1512. position: relative;
  1513. border: 1px solid var(--border-color);
  1514. border-radius: 5px;
  1515. margin-bottom: 15px;
  1516. }
  1517.  
  1518. .tab {
  1519. display: none;
  1520. position: relative;
  1521. }
  1522.  
  1523. .tab.active {
  1524. display: block;
  1525. }
  1526.  
  1527. .input-group {
  1528. margin-bottom: 15px;
  1529. }
  1530.  
  1531. label {
  1532. display: flex;
  1533. margin-bottom: 5px;
  1534. font-weight: bold;
  1535. align-items: center;
  1536. }
  1537.  
  1538. input[type="number"] {
  1539. width: calc(100% - 16px);
  1540. font-size: 14px;
  1541. border-radius: 5px;
  1542. border: 1px solid var(--border-color);
  1543. margin-right: 15px;
  1544. padding: 8px;
  1545. }
  1546.  
  1547. input[type="text"],
  1548. input[type="search"],
  1549. textarea {
  1550. width: 100%;
  1551. min-width: 400px;
  1552. padding: 8px;
  1553. border: 1px solid var(--border-color);
  1554. border-radius: 5px;
  1555. font-size: 14px;
  1556. box-sizing: border-box;
  1557. }
  1558.  
  1559. textarea {
  1560. resize: vertical;
  1561. height: 160px;
  1562. }
  1563.  
  1564. .switch {
  1565. position: relative;
  1566. display: inline-block;
  1567. width: 54px;
  1568. height: 24px;
  1569. }
  1570.  
  1571. .switch input {
  1572. opacity: 0;
  1573. width: 0;
  1574. height: 0;
  1575. }
  1576.  
  1577. .switch-slider {
  1578. position: absolute;
  1579. cursor: pointer;
  1580. top: 0;
  1581. left: 0;
  1582. right: 0;
  1583. bottom: 0;
  1584. background-color: #ccc;
  1585. transition: 0.4s;
  1586. border-radius: 34px;
  1587. }
  1588.  
  1589. .switch-slider:before {
  1590. position: absolute;
  1591. content: "";
  1592. height: 16px;
  1593. width: 16px;
  1594. border-radius: 50%;
  1595. left: 4px;
  1596. bottom: 4px;
  1597. background-color: var(--text-active-color);
  1598. transition: 0.4s;
  1599. }
  1600.  
  1601. input:checked+.switch-slider {
  1602. background-color: var(--primary-color);
  1603. }
  1604.  
  1605. input:checked+.switch-slider:before {
  1606. transform: translateX(30px);
  1607. }
  1608.  
  1609. .remove-button {
  1610. position: absolute;
  1611. opacity: 0.9;
  1612. top: -10px;
  1613. right: 0;
  1614. background: var(--warning-color);
  1615. color: var(--text-active-color);
  1616. border: none;
  1617. padding: 5px 10px;
  1618. cursor: pointer;
  1619. border-radius: 5px;
  1620. font-size: 14px;
  1621. }
  1622.  
  1623. .remove-button:hover {
  1624. opacity: 1;
  1625. }
  1626.  
  1627. .radio-button-group,
  1628. .checkbox-group {
  1629. display: flex;
  1630. flex-wrap: wrap;
  1631. gap: 10px;
  1632. margin-bottom: 15px;
  1633. }
  1634.  
  1635. .radio-button,
  1636. .checkbox-group .chekbox-label {
  1637. padding: 8px 12px;
  1638. background-color: #e0e0e0;
  1639. cursor: pointer;
  1640. border-radius: 5px;
  1641. font-size: 14px;
  1642. font-weight: normal;
  1643. min-width: 132px;
  1644. display: inline-flex;
  1645. justify-content: center;
  1646. align-items: center;
  1647. gap: 12px;
  1648. height: 20px;
  1649. margin: 0;
  1650. }
  1651.  
  1652. .parser .radio-button,
  1653. .parser .checkbox-group .chekbox-label {
  1654. min-width: 122.5px;
  1655. }
  1656.  
  1657. .radio-button.active,
  1658. .checkbox-group input:checked+.chekbox-label {
  1659. background-color: var(--primary-color);
  1660. color: var(--text-active-color);
  1661. }
  1662.  
  1663. .radio-button:hover {
  1664. background-color: var(--primary-color);
  1665. color: var(--text-active-color);
  1666. }
  1667.  
  1668. .checkbox-group input[type="checkbox"] {
  1669. display: none;
  1670. }
  1671.  
  1672. #language {
  1673. padding: 8px;
  1674. border-radius: 5px;
  1675. cursor: pointer;
  1676. width: 100%;
  1677. border: 1px solid var(--border-color);
  1678. }
  1679.  
  1680. .parser {
  1681. border: 1px solid var(--border-color);
  1682. border-radius: 5px;
  1683. padding: 10px 20px;
  1684. }
  1685.  
  1686. .parser textarea {
  1687. margin-bottom: 10px;
  1688. resize: none;
  1689. }
  1690.  
  1691. a {
  1692. color: var(--text-color);
  1693. text-decoration: none;
  1694. font-weight: bold;
  1695. transition: color 0.3s ease, border-bottom 0.3s ease;
  1696. border-bottom: 2px solid transparent;
  1697. }
  1698.  
  1699. a:hover {
  1700. color: var(--primary-color);
  1701. border-bottom-color: var(--primary-color);
  1702. }
  1703.  
  1704. #tab-container {
  1705. flex: 1;
  1706. padding: 20px;
  1707. overflow-y: auto;
  1708. position: relative;
  1709. border: 1px solid var(--border-color);
  1710. border-radius: 5px;
  1711. margin-bottom: 10px;
  1712. }
  1713.  
  1714. :disabled {
  1715. opacity: 0.6;
  1716. }
  1717.  
  1718. div.disabled,
  1719. button:disabled {
  1720. pointer-events: none !important;
  1721. cursor: not-allowed !important;
  1722. }
  1723.  
  1724. .parser textarea:disabled {
  1725. height: 30px;
  1726. overflow-y: hidden;
  1727. line-height: 20px;
  1728. }
  1729.  
  1730. textarea:disabled::-webkit-scrollbar {
  1731. display: none;
  1732. }
  1733.  
  1734. ::-webkit-scrollbar {
  1735. width: 20px !important;
  1736. height: 20px !important;
  1737. }
  1738.  
  1739. ::-webkit-scrollbar-thumb {
  1740. background: var(--border-color) !important;
  1741. border-radius: 5px !important;
  1742. }
  1743.  
  1744. ::-webkit-scrollbar-thumb:hover {
  1745. background: var(--primary-color) !important;
  1746. }
  1747.  
  1748. ::-webkit-scrollbar-track {
  1749. background: rgb(245, 245, 245) !important;
  1750. border-radius: 5px !important;
  1751. }
  1752.  
  1753. select:focus,
  1754. input:focus,
  1755. textarea:focus {
  1756. border-color: var(--primary-color);
  1757. outline: none;
  1758. }
  1759.  
  1760. #footer {
  1761. font-size: 14px;
  1762. height: 35px;
  1763. display: flex;
  1764. justify-content: center;
  1765. align-items: center;
  1766. }
  1767.  
  1768. #footer svg {
  1769. width: 20px;
  1770. height: 20px;
  1771. margin-bottom: -3px;
  1772. }
  1773.  
  1774. #footer a,
  1775. #footer a:hover {
  1776. margin-left: 3px;
  1777. margin-right: 3px;
  1778. font-weight: normal;
  1779. border-bottom: none !important;
  1780. text-decoration: none !important;
  1781. }
  1782. </style>
  1783. </head>
  1784.  
  1785. <body>
  1786. <div id="sidebar-container">
  1787. <div id="sidebar">
  1788. <button id="global-button" class="tab-button active" data-tab="global">
  1789. <svg t="1732015880724" class="icon" viewBox="0 0 1024 1024" version="1.1"
  1790. xmlns="http://www.w3.org/2000/svg" p-id="4317" width="32" height="32">
  1791. <path
  1792. d="M386.35 112.05h-228.7c-25.2 0-45.7 20.5-45.7 45.7v228.5c0 25.2 20.4 45.7 45.6 45.8h228.6c25.2 0 45.7-20.4 45.8-45.6V157.65c0.1-25.2-20.4-45.6-45.6-45.6z"
  1793. p-id="4318"></path>
  1794. <path
  1795. d="M157.55 80.05h229c42.8 0 77.5 34.7 77.5 77.5v229c0 42.8-34.7 77.5-77.5 77.5h-229c-42.8 0-77.5-34.7-77.5-77.5v-229c0-42.8 34.7-77.5 77.5-77.5z m228.9 320.5c7.8 0 14.1-6.3 14.1-14.1v-229c0-7.8-6.3-14.1-14.1-14.1h-229c-7.8 0-14.1 6.3-14.1 14.1v229c0 7.8 6.3 14.1 14.1 14.1h229z"
  1796. p-id="4319"></path>
  1797. <path
  1798. d="M387.55 590.25h-231.1c-25.5 0-46.2 20.7-46.2 46.2v231.1c0 25.5 20.7 46.2 46.2 46.2h231.1c25.5 0 46.2-20.7 46.2-46.2v-231.1c0-25.5-20.7-46.2-46.2-46.2z"
  1799. p-id="4320"></path>
  1800. <path
  1801. d="M157.55 560.05h229c42.8 0 77.5 34.7 77.5 77.5v229c0 42.8-34.7 77.5-77.5 77.5h-229c-42.8 0-77.5-34.7-77.5-77.5v-229c0-42.8 34.7-77.5 77.5-77.5z m228.9 320.5c7.8 0 14.1-6.3 14.1-14.1v-229c0-7.8-6.3-14.1-14.1-14.1h-229c-7.8 0-14.1 6.3-14.1 14.1v229c0 7.8 6.3 14.1 14.1 14.1h229zM637.55 80.05h229c42.8 0 77.5 34.7 77.5 77.5v229c0 42.8-34.7 77.5-77.5 77.5h-229c-42.8 0-77.5-34.7-77.5-77.5v-229c0-42.8 34.7-77.5 77.5-77.5z m228.9 320.5c7.8 0 14.1-6.3 14.1-14.1v-229c0-7.8-6.3-14.1-14.1-14.1h-229c-7.8 0-14.1 6.3-14.1 14.1v229c0 7.8 6.3 14.1 14.1 14.1h229z"
  1802. p-id="4321"></path>
  1803. <path
  1804. d="M866.306 592.006h-228.6c-25.2 0-45.7 20.5-45.7 45.7v228.5c0 25.2 20.5 45.7 45.7 45.7h228.5c25.2 0 45.7-20.4 45.8-45.6v-228.6c0-25.2-20.5-45.7-45.7-45.7z"
  1805. p-id="4322"></path>
  1806. <path
  1807. d="M637.506 560.006h229c42.8 0 77.5 34.7 77.5 77.5v229c0 42.8-34.7 77.5-77.5 77.5h-229c-42.8 0-77.5-34.7-77.5-77.5v-229c0-42.8 34.7-77.5 77.5-77.5z m229 320.6c7.8 0 14.1-6.3 14.1-14.1v-229c0-7.8-6.3-14.1-14.1-14.1h-229c-7.8 0-14.1 6.3-14.1 14.1v229c0 7.8 6.3 14.1 14.1 14.1h229z"
  1808. p-id="4323"></path>
  1809. </svg>
  1810. <span data-translate="global">全局配置</span>
  1811. </button>
  1812. </div>
  1813. <button id="add-tab-button">+</button>
  1814. </div>
  1815.  
  1816. <div id="content-container">
  1817. <div id="content">
  1818. <div id="global" class="tab active">
  1819. <div class="input-group">
  1820. <label data-translate="version">版本</label>
  1821. <input type="text" id="version" readonly></input>
  1822. </div>
  1823. <div class="input-group">
  1824. <label data-translate="language">语言</label>
  1825. <select id="language">
  1826. <option value="zh" selected>中文</option>
  1827. <option value="en">English</option>
  1828. </select>
  1829. </div>
  1830. <div class="input-group">
  1831. <label data-translate="buttonCoord">按钮坐标</label>
  1832. <label>
  1833. <input type="number" id="buttonXCoord" min="0" placeholder="0">
  1834. <input type="number" id="buttonYCoord" min="0" placeholder="0">
  1835. <button id="reset-button-coord-button" class="reset-button" data-translate="reset">重置</button>
  1836. </label>
  1837. </div>
  1838. <div class="input-group">
  1839. <label data-translate="buttonScale">按钮比例</label>
  1840. <input type="number" id="buttonScale" min="0.01" max="10" step="0.01" placeholder="1.00">
  1841. </div>
  1842. <div class="input-group">
  1843. <label data-translate="buttonVisibilityDuration">按钮可见时长(毫秒,-1:一直可见)</label>
  1844. <input type="number" id="buttonVisibilityDuration" min="-1" placeholder="3000">
  1845. </div>
  1846. <div class="input-group">
  1847. <label data-translate="networkProxy">网络代理</label>
  1848. <input type="text" id="networkProxy" placeholder="http://127.0.0.1:7890"></input>
  1849. </div>
  1850. <label data-translate="parser">解析器</label>
  1851. <div class="input-group parser" id="ytdlp">
  1852. <label><a href="https://github.com/yt-dlp/yt-dlp" target="_blank">YTDLP</a></label>
  1853. <textarea name="regex" disabled></textarea>
  1854. <textarea name="regex"></textarea>
  1855. <label data-translate="preferredQuality">首选画质</label>
  1856. <div class="radio-button-group" name="preferredQuality">
  1857. <div class="radio-button active" value="unlimited" data-translate="unlimited">无限制</div>
  1858. <div class="radio-button" value="2160">2160P</div>
  1859. <div class="radio-button" value="1440">1440P</div>
  1860. <div class="radio-button" value="1080">1080P</div>
  1861. <div class="radio-button" value="720">720P</div>
  1862. </div>
  1863. </div>
  1864. <div class="input-group parser" id="video">
  1865. <label><a href="https://github.com/LuckyPuppy514/external-player" target="_blank">VIDEO</a></label>
  1866. <textarea name="regex" disabled></textarea>
  1867. <textarea name="regex"></textarea>
  1868. </div>
  1869. <div class="input-group parser" id="url">
  1870. <label><a href="https://github.com/LuckyPuppy514/external-player" target="_blank">URL</a></label>
  1871. <textarea name="regex" disabled></textarea>
  1872. <textarea name="regex"></textarea>
  1873. </div>
  1874. <div class="input-group parser" id="html">
  1875. <label><a href="https://github.com/LuckyPuppy514/external-player" target="_blank">HTML</a></label>
  1876. <textarea name="regex" disabled></textarea>
  1877. <textarea name="regex"></textarea>
  1878. </div>
  1879. <div class="input-group parser" id="script">
  1880. <label><a href="https://github.com/LuckyPuppy514/external-player" target="_blank">SCRIPT</a></label>
  1881. <textarea name="regex" disabled></textarea>
  1882. <textarea name="regex"></textarea>
  1883. </div>
  1884. <div class="input-group parser" id="request">
  1885. <label><a href="https://github.com/LuckyPuppy514/external-player"
  1886. target="_blank">REQUEST</a></label>
  1887. <textarea name="regex" disabled></textarea>
  1888. <textarea name="regex"></textarea>
  1889. </div>
  1890. <div class="input-group parser" id="bilibili">
  1891. <label><a href="https://github.com/SocialSisterYi/bilibili-API-collect"
  1892. target="_blank">BILIBILI</a></label>
  1893. <textarea name="regex" disabled></textarea>
  1894. <textarea name="regex" style="display: none;"></textarea>
  1895. <label data-translate="preferredQuality">首选画质</label>
  1896. <div class="radio-button-group" name="preferredQuality">
  1897. <div class="radio-button active" value="127" data-translate="unlimited">无限制</div>
  1898. <div class="radio-button" value="126">2160P</div>
  1899. <div class="radio-button" value="116">1080P</div>
  1900. <div class="radio-button" value="74">720P</div>
  1901. </div>
  1902. <label data-translate="preferredSubtitle">首选字幕</label>
  1903. <div class="radio-button-group" name="preferredSubtitle">
  1904. <div class="radio-button active" value="off" data-translate="off">关闭</div>
  1905. <div class="radio-button" value="zh-Hans">简体</div>
  1906. <div class="radio-button" value="zh-Hant">繁体</div>
  1907. <div class="radio-button" value="en-US">English</div>
  1908. </div>
  1909. <label data-translate="preferredCodec">首选编码</label>
  1910. <div class="radio-button-group" name="preferredCodec">
  1911. <div class="radio-button active" value="12">HEVC</div>
  1912. <div class="radio-button" value="13">AV1</div>
  1913. <div class="radio-button" value="7">AVC</div>
  1914. </div>
  1915. </div>
  1916. <div class="input-group parser" id="bilibiliLive">
  1917. <label><a href="https://github.com/SocialSisterYi/bilibili-API-collect" target="_blank">BILIBILI
  1918. LIVE</a></label>
  1919. <textarea name="regex" disabled></textarea>
  1920. <textarea name="regex" style="display: none;"></textarea>
  1921. <label data-translate="preferredQuality">首选画质</label>
  1922. <div class="radio-button-group" name="preferredQuality">
  1923. <div class="radio-button active" value="4" data-translate="original">原画</div>
  1924. <div class="radio-button active" value="3" data-translate="hd">高清</div>
  1925. <div class="radio-button active" value="2" data-translate="smooth">流畅</div>
  1926. </div>
  1927. <label data-translate="preferredLine">首选线路</label>
  1928. <div class="radio-button-group" name="preferredLine">
  1929. <div class="radio-button active" value="0" data-translate="mainLine">主线</div>
  1930. <div class="radio-button active" value="1" data-translate="backupLine1">备线1</div>
  1931. <div class="radio-button active" value="2" data-translate="backupLine2">备线2</div>
  1932. <div class="radio-button active" value="3" data-translate="backupLine3">备线3</div>
  1933. </div>
  1934. </div>
  1935. </div>
  1936. </div>
  1937.  
  1938. <div style="margin: 0 auto;">
  1939. <button id="save-button" data-translate="save">保存</button>
  1940. <button id="reset-button" class="reset-button" data-translate="reset">重置</button>
  1941. </div>
  1942.  
  1943. <div id="footer">
  1944. <span>
  1945. <a href="https://github.com/LuckyPuppy514" target="_blank">
  1946. &copy 2024 LuckyPuppy514
  1947. </a>
  1948. <svg t="1731923678389" class="icon" viewBox="0 0 1024 1024" version="1.1"
  1949. xmlns="http://www.w3.org/2000/svg" p-id="5894" width="32" height="32">
  1950. <path
  1951. d="M20.48 503.72608c0 214.4256 137.4208 396.73856 328.94976 463.6672 25.8048 6.5536 21.87264-11.8784 21.87264-24.33024v-85.07392c-148.93056 17.44896-154.86976-81.1008-164.94592-97.52576-20.23424-34.52928-67.91168-43.33568-53.69856-59.76064 33.91488-17.44896 68.48512 4.42368 108.46208 63.61088 28.95872 42.88512 85.44256 35.6352 114.15552 28.4672a138.8544 138.8544 0 0 1 38.0928-66.7648c-154.25536-27.60704-218.60352-121.77408-218.60352-233.79968 0-54.31296 17.94048-104.2432 53.0432-144.54784-22.36416-66.43712 2.08896-123.24864 5.3248-131.6864 63.81568-5.7344 130.00704 45.6704 135.168 49.68448 36.2496-9.78944 77.57824-14.9504 123.82208-14.9504 46.4896 0 88.064 5.3248 124.5184 15.23712 12.288-9.4208 73.80992-53.53472 133.12-48.128 3.15392 8.43776 27.0336 63.93856 6.02112 129.4336 35.59424 40.38656 53.69856 90.76736 53.69856 145.24416 0 112.18944-64.7168 206.4384-219.42272 233.71776a140.0832 140.0832 0 0 1 41.7792 99.9424v123.4944c0.86016 9.87136 0 19.6608 16.50688 19.6608 194.31424-65.49504 334.2336-249.15968 334.2336-465.5104C1002.57792 232.48896 782.66368 12.77952 511.5904 12.77952 240.18944 12.65664 20.48 232.40704 20.48 503.72608z"
  1952. fill="#000000" opacity=".65" p-id="5895"></path>
  1953. </svg>
  1954. <a href="https://github.com/LuckyPuppy514/external-player" target="_blank">
  1955. Powered by External Player
  1956. </a>
  1957. </span>
  1958. </div>
  1959. </div>
  1960. </body>
  1961. <script>
  1962. const translations = {
  1963. en: {
  1964. global: 'Global Config',
  1965. version: 'Version',
  1966. language: 'Language',
  1967. buttonCoord: 'Button Coord',
  1968. buttonScale: 'Button Scale',
  1969. buttonVisibilityDuration: 'Button Visibility Duration (ms, -1: Keep Visible)',
  1970. networkProxy: 'Network Proxy',
  1971. reset: 'Reset',
  1972. save: 'Save',
  1973. delete: 'Delete',
  1974. name: 'Name',
  1975. system: 'System',
  1976. icon: 'Icon',
  1977. iconSize: 'Icon Size',
  1978. playEvent: 'Play Event',
  1979. enable: 'Enable',
  1980. parser: 'Parser',
  1981. preferredQuality: 'Preferred Quality',
  1982. preferredSubtitle: 'Preferred Subtitle',
  1983. preferredCodec: 'Preferred Codec',
  1984. preferredLine: 'Preferred Line',
  1985. original: 'Original',
  1986. hd: 'HD',
  1987. smooth: 'Smooth',
  1988. mainLine: 'Main',
  1989. backupLine1: 'Backup 1',
  1990. backupLine2: 'Backup 2',
  1991. backupLine3: 'Backup 3',
  1992. unlimited: 'Unlimited',
  1993. off: 'OFF',
  1994. presetEvent: 'Preset Event',
  1995. playAuto: 'Play Automatically',
  1996. pauseAuto: 'Pause Automatically',
  1997. closeAuto: 'Close Automatically',
  1998. syncTime: 'Synchronize Time',
  1999. },
  2000. zh: {
  2001. global: '全局配置',
  2002. version: '版本',
  2003. language: '语言',
  2004. buttonCoord: '按钮坐标',
  2005. buttonScale: '按钮比例',
  2006. buttonVisibilityDuration: '按钮可见时长(毫秒,-1:一直可见)',
  2007. networkProxy: '网络代理',
  2008. reset: '重置',
  2009. save: '保存',
  2010. delete: '删除',
  2011. name: '名称',
  2012. system: '系统',
  2013. icon: '图标',
  2014. iconSize: '图标大小',
  2015. playEvent: '播放事件',
  2016. enable: '启用',
  2017. parser: '解析器',
  2018. preferredQuality: '首选画质',
  2019. preferredSubtitle: '首选字幕',
  2020. preferredCodec: '首选编码',
  2021. preferredLine: '首选线路',
  2022. original: '原画',
  2023. hd: '高清',
  2024. smooth: '流畅',
  2025. mainLine: '主线',
  2026. backupLine1: '备线1',
  2027. backupLine2: '备线2',
  2028. backupLine3: '备线3',
  2029. unlimited: '无限制',
  2030. off: '关闭',
  2031. presetEvent: '预设事件',
  2032. playAuto: '自动播放',
  2033. pauseAuto: '自动暂停',
  2034. closeAuto: '自动关闭',
  2035. syncTime: '同步时间',
  2036. }
  2037. };
  2038.  
  2039. const SYSTEM_SVG = {
  2040. windows: '<svg t="1732017849573" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="5376" width="32" height="32"><path d="M523.8 191.4v288.9h382V128.1zM523.8 833.6l382 62.2v-352h-382zM120.1 480.2H443V201.9l-322.9 53.5zM120.1 770.6L443 823.2V543.8H120.1z" p-id="5377"></path></svg>',
  2041. linux: '<svg t="1732017810402" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="4326" width="32" height="32"><path d="M834.198588 918.588235c-30.659765 15.661176-71.559529 50.115765-86.618353 64.572236-11.324235 10.782118-58.066824 16.203294-84.449882 2.710588-30.659765-15.661176-14.516706-40.417882-61.861647-41.923765-23.672471-0.602353-46.802824-0.602353-69.933177-0.602353-20.419765 0.602353-40.839529 1.626353-61.861647 2.108235-70.957176 1.626353-77.944471 47.405176-123.723294 45.778824-31.201882-1.084235-70.415059-25.840941-138.24-39.755294-47.344941-9.758118-93.003294-12.348235-102.761412-33.370353-9.637647-21.022118 11.866353-44.634353 13.432471-65.054118 1.626353-27.467294-20.419765-64.572235-4.276706-78.607059 13.974588-12.348235 43.550118-3.252706 62.885647-13.914352 20.419765-11.806118 29.033412-21.022118 29.033412-46.260706 7.529412 25.720471-0.542118 46.682353-17.227294 56.922353-10.24 6.445176-29.033412 9.697882-44.694588 8.131764-12.348235-1.144471-19.877647 0.481882-23.130353 5.360941-4.818824 5.903059-3.252706 16.685176 2.710588 30.659765 5.903059 13.974588 12.890353 23.130353 11.806118 40.297412-0.542118 17.227294-19.877647 37.707294-16.624942 52.224 1.084235 5.421176 6.445176 10.24 19.877647 13.974588 21.504 5.903059 60.777412 11.806118 98.966589 21.022118 42.526118 10.721882 86.618353 30.057412 114.085647 26.322823 81.739294-11.324235 34.936471-98.966588 22.046117-119.868235-69.391059-108.724706-115.109647-179.681882-151.67247-151.732706-9.155765 7.529412-9.697882-18.311529-9.155765-28.551529 1.626353-35.538824 19.395765-48.368941 30.117647-75.836236 20.419765-52.224 36.020706-111.856941 67.222588-142.516705 23.311059-30.177882 59.873882-79.088941 66.921412-104.869647-5.963294-55.958588-7.589647-115.109647-8.613647-166.671059-1.084235-55.416471 7.529412-103.905882 69.933177-137.697883C453.391059 33.310118 473.268706 30.117647 494.290824 30.117647c37.104941-0.602353 78.486588 10.24 104.869647 29.575529 41.984 31.201882 68.306824 97.340235 65.114353 144.624942-2.168471 37.104941 4.276706 75.294118 16.143058 115.109647 13.974588 46.802824 36.080941 79.570824 71.55953 117.217882 42.526118 45.176471 75.836235 133.903059 85.534117 190.343529 8.613647 52.826353-3.252706 85.594353-14.516705 87.220706-17.227294 2.590118-27.949176 56.922353-81.739295 54.814118-34.394353-1.626353-37.647059-22.046118-47.344941-39.815529-15.600941-27.407059-31.201882-18.793412-37.104941 10.24-3.252706 14.516706-1.144471 36.080941 3.734588 52.103529 9.697882 33.912471 6.445176 65.656471 0.542118 104.929882-11.324235 74.209882 52.163765 88.184471 94.689882 52.645647 41.923765-34.876235 51.079529-40.297412 103.785412-58.608941 80.112941-27.467294 53.248-51.621647 10.179765-66.138353-38.731294-12.950588-40.297412-78.064941-26.383059-90.413176 3.252706 69.933176 39.815529 80.173176 54.874353 89.810823 66.138353 41.020235-24.756706 74.932706-64.030118 94.810353z m-90.352941-259.734588c14.516706-48.489412 8.071529-67.764706-1.566118-113.543529-7.529412-34.394353-39.273412-81.257412-64.030117-95.713883 6.445176 5.360941 18.311529 20.961882 30.659764 44.574118 21.504 40.417882 43.008 100.050824 29.033412 149.564235-5.360941 19.275294-18.251294 21.985882-26.864941 22.528-37.647059 4.336941-15.600941-45.176471-31.201882-112.338823-17.769412-75.354353-36.020706-80.715294-40.297412-86.618353-22.166588-97.822118-46.320941-88.124235-53.368471-124.687059-5.903059-32.828235 28.551529-59.693176-18.251294-68.848941-14.516706-2.710588-34.936471-17.227294-43.008-18.31153-8.071529-1.024-12.408471-54.332235 17.709177-55.958588 29.575529-2.168471 34.996706 33.370353 29.575529 47.405177-8.553412 13.914353 0.542118 19.335529 15.119059 14.45647 11.806118-3.734588 4.276706-34.936471 6.987294-39.213176-7.529412-45.176471-26.383059-51.621647-45.718588-55.416471-74.270118 5.903059-40.899765 87.702588-48.429177 80.173177-10.782118-11.324235-41.923765-1.084235-41.923764-8.131765 0.542118-41.923765-13.492706-66.138353-32.828236-66.680471-21.504-0.542118-30.117647 29.575529-31.201882 46.742589-1.626353 16.143059 9.155765 50.115765 17.227294 47.405176 5.360941-1.626353 14.516706-12.408471 4.818824-11.806118-4.818824 0-12.348235-11.866353-13.432471-25.840941-0.542118-14.034824 4.879059-28.009412 23.130353-27.467294 20.961882 0.542118 20.961882 42.465882 18.793412 44.092235-6.927059 4.818824-15.600941 14.034824-16.685177 15.600942-6.927059 11.324235-20.359529 14.456471-25.780706 19.395764-9.155765 9.637647-11.264 20.419765-4.276705 24.154353 24.696471 13.974588 16.624941 30.057412 51.079529 31.262118 22.588235 1.084235 39.213176-3.252706 54.874353-8.07153 11.806118-3.734588 50.055529-11.806118 58.066823-25.840941 3.734588-5.903059 8.071529-5.903059 10.721883-4.276706 5.360941 2.650353 6.445176 12.890353-6.987294 16.143059-18.793412 5.421176-37.647059 15.661176-54.814118 22.106353-16.685176 6.927059-22.046118 9.637647-37.647059 12.288-35.478588 6.445176-61.801412-12.890353-38.189176 10.24 8.071529 7.529412 15.600941 12.348235 36.020706 11.866353 45.176471-1.626353 95.232-56.018824 100.050823-31.804235 1.024 5.360941-14.034824 11.806118-25.840941 17.769412-41.923765 20.419765-71.499294 61.319529-98.424471 47.284705-24.214588-12.890353-48.368941-72.643765-47.887058-45.658353 0.542118 41.381647-54.332235 77.944471-29.033412 125.289412-16.685176 4.216471-53.790118 83.365647-59.151059 124.205177-3.252706 23.672471 2.168471 52.705882-3.794824 68.848941-8.071529 23.672471-44.634353-22.588235-32.768-79.028706 2.108235-9.637647 0-11.866353-2.710588-6.927059-14.516706 26.322824-6.445176 63.427765 5.360941 89.208471 4.879059 11.324235 17.227294 16.143059 26.383059 25.840941 18.793412 21.443765 93.003294 76.378353 105.953883 89.810823a33.008941 33.008941 0 0 1-22.588236 55.898353c17.769412 33.370353 34.936471 36.623059 34.454588 90.895059 20.419765-10.721882 12.408471-34.394353 3.734589-49.392941-5.963294-10.842353-13.432471-15.661176-11.866353-18.311529 1.084235-1.626353 11.866353-10.842353 17.769412-3.734589 18.251294 20.419765 52.705882 24.154353 89.268705 19.33553 37.104941-4.336941 76.920471-17.227294 95.171765-46.802824 8.613647-13.974588 14.516706-18.793412 18.31153-16.143059 4.276706 2.108235 5.963294 11.806118 5.360941 27.949177-0.542118 17.227294-7.529412 34.996706-12.348236 49.513412-4.879059 16.685176-6.445176 27.949176 9.697883 28.551529 4.276706-30.177882 12.890353-59.753412 15.058823-89.871059 2.710588-34.394353-22.046118-97.822118 4.879059-129.626353 6.987294-8.613647 15.540706-9.637647 27.407059-9.637647 1.566118-43.068235 67.764706-39.755294 89.810823-22.046117 0-9.758118-20.961882-18.853647-29.575529-22.648471zM304.971294 503.988706c-3.794824 6.927059-13.432471 12.288-5.963294 13.43247 2.710588 0.542118 10.24-6.023529 13.492706-13.43247 2.650353-9.155765 5.360941-14.034824 1.084235-15.661177-4.879059-1.566118-3.794824 8.071529-8.613647 15.661177z m123.120941-291.538824c-6.445176-1.626353-5.360941 8.011294-2.108235 6.987294 2.168471 0 4.879059 3.252706 3.734588 8.07153-1.084235 6.445176-0.542118 10.842353 4.336941 10.842353 0.542118 0 1.566118 0 1.566118-1.626353 2.228706-13.552941-4.276706-23.190588-7.529412-24.274824z m14.576941 49.453177c-5.360941 0.542118-4.336941-11.866353 12.890353-10.782118-10.782118 1.084235-6.987294 10.782118-12.890353 10.782118z m44.092236-9.155765c15.600941-6.927059 20.961882 3.794824 15.600941 5.963294-5.421176 1.566118-5.963294-8.673882-15.600941-5.963294z m65.054117-43.550118c-6.987294 0.602353-4.818824 3.734588-1.566117 4.818824 4.276706 1.204706 8.613647 8.673882 9.697882 16.685176 0 1.084235 5.360941-1.084235 5.360941-2.710588 0.481882-12.830118-10.782118-19.275294-13.492706-18.793412z m31.201883-116.133647c-4.276706-4.336941-8.613647-8.131765-12.890353-8.131764-10.782118 1.084235-5.421176 12.348235-6.987294 17.769411-2.168471 5.903059-10.179765 10.782118-4.818824 15.058824 4.879059 3.734588 8.071529-5.903059 18.31153-9.637647 2.650353-1.144471 15.058824 0.481882 17.709176-5.421177 0.481882-2.710588-6.445176-5.903059-11.324235-9.637647z m59.693176 237.628236c-10.179765-6.384941-12.348235-17.167059-16.082823-13.432471-11.324235 12.348235 13.974588 38.189176 24.69647 40.417882 6.445176 1.084235 11.324235-7.589647 9.697883-15.119058-2.168471-10.179765-9.697882-6.445176-18.31153-11.866353z" p-id="4327"></path></svg>',
  2042. mac: '<svg t="1731999754869" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="7764" width="32" height="32"><path d="M849.124134 704.896288c-1.040702 3.157923-17.300015 59.872622-57.250912 118.190843-34.577516 50.305733-70.331835 101.018741-126.801964 101.909018-55.532781 0.976234-73.303516-33.134655-136.707568-33.134655-63.323211 0-83.23061 32.244378-135.712915 34.110889-54.254671 2.220574-96.003518-54.951543-130.712017-105.011682-70.934562-102.549607-125.552507-290.600541-52.30118-416.625816 36.040844-63.055105 100.821243-103.135962 171.364903-104.230899 53.160757-1.004887 103.739712 36.012192 136.028093 36.012192 33.171494 0 94.357018-44.791136 158.90615-38.089503 27.02654 1.151219 102.622262 11.298324 151.328567 81.891102-3.832282 2.607384-90.452081 53.724599-89.487104 157.76107C739.079832 663.275355 847.952448 704.467523 849.124134 704.896288M633.69669 230.749408c29.107945-35.506678 48.235584-84.314291 43.202964-132.785236-41.560558 1.630127-92.196819 27.600615-122.291231 62.896492-26.609031 30.794353-50.062186 80.362282-43.521213 128.270409C557.264926 291.935955 604.745311 264.949324 633.69669 230.749408" p-id="7765"></path></svg>'
  2043. };
  2044.  
  2045. var policy;
  2046. try {
  2047. policy = window.trustedTypes.createPolicy('default', {
  2048. createHTML: (string, sink) => string,
  2049. createScript: (input) => input
  2050. })
  2051. } catch (error) {
  2052. policy = {
  2053. createHTML: (string, sink) => string,
  2054. createScript: (input) => input
  2055. }
  2056. }
  2057.  
  2058. var defaultConfig;
  2059. var tabCount = 0;
  2060. var projectName;
  2061.  
  2062. function translatePage(language) {
  2063. const trans = translations[language];
  2064. document.querySelectorAll('[data-translate]').forEach(el => {
  2065. el.textContent = trans[el.getAttribute('data-translate')] || el.textContent;
  2066. });
  2067. }
  2068.  
  2069. function createTab(tabId, tabName = \`Player \${tabCount}\`, config = {}) {
  2070. const tabButton = document.createElement('button');
  2071. tabButton.className = 'tab-button';
  2072. tabButton.textContent = tabName;
  2073. tabButton.dataset.tab = tabId;
  2074. sidebar.insertBefore(tabButton, document.getElementById('global-button').nextSibling);
  2075.  
  2076. const tab = document.createElement('div');
  2077. tab.id = tabId;
  2078. tab.name = tabName;
  2079. tab.className = 'tab';
  2080. tab.setAttribute('readonly', config.readonly === true)
  2081. const disabled = config.readonly === true ? 'disabled' : '';
  2082. config.presetEvent = config.presetEvent || {
  2083. pauseAuto: true,
  2084. syncTime: false,
  2085. };
  2086. tab.innerHTML = policy.createHTML(\`
  2087. <div class="header">
  2088. <button class="remove-button" data-translate="delete" \${disabled}>删除</button>
  2089. </div>
  2090. <div class="input-group">
  2091. <label data-translate="name">名称</label>
  2092. <input type="text" value="\${config.name || tabName}" name="name" placeholder="\${tabName}" required \${disabled}>
  2093. </div>
  2094. <div class="input-group">
  2095. <label data-translate="system">系统</label>
  2096. <div class="radio-button-group" name="system">
  2097. <div class="radio-button active \${disabled}" value="windows">\${SYSTEM_SVG.windows} Windows</div>
  2098. <div class="radio-button \${disabled}" value="linux">\${SYSTEM_SVG.linux} Linux</div>
  2099. <div class="radio-button \${disabled}" value="mac">\${SYSTEM_SVG.mac} Mac</div>
  2100. </div>
  2101. </div>
  2102. <div class="input-group">
  2103. <label data-translate="iconSize">图标大小</label>
  2104. <input type="number" value="\${config.iconSize || 50}" name="iconSize" min="1" required>
  2105. </div>
  2106. <div class="input-group">
  2107. <label data-translate="icon">图标</label>
  2108. <input type="search" value="\${config.icon || ''}" name="icon" required>
  2109. </div>
  2110. <div class="input-group">
  2111. <label data-translate="presetEvent">预设事件</label>
  2112. <div class="checkbox-group">
  2113. <input type="checkbox" id="\${tabId}-play-auto" name="playAuto" \${config.presetEvent.playAuto ? 'checked' : ''}/>
  2114. <label for="\${tabId}-play-auto" data-translate="playAuto" class="chekbox-label">自动播放</label>
  2115. <input type="checkbox" id="\${tabId}-pause-auto" name="pauseAuto" \${config.presetEvent.pauseAuto ? 'checked' : ''}/>
  2116. <label for="\${tabId}-pause-auto" data-translate="pauseAuto" class="chekbox-label">自动暂停</label>
  2117. <input type="checkbox" id="\${tabId}-close-auto" name="closeAuto" \${config.presetEvent.closeAuto ? 'checked' : ''}/>
  2118. <label for="\${tabId}-close-auto" data-translate="closeAuto" class="chekbox-label">自动关闭</label>
  2119. <input type="checkbox" id="\${tabId}-sync-time" name="syncTime" \${config.presetEvent.syncTime ? 'checked' : ''}/>
  2120. <label for="\${tabId}-sync-time" data-translate="syncTime" class="chekbox-label">同步时间</label>
  2121. </div>
  2122. </div>
  2123. <div class="input-group">
  2124. <label data-translate="playEvent">播放事件</label>
  2125. <textarea class="tab-textarea" wrap="off" \${disabled} name="playEvent">\${config.playEvent || ''}</textarea>
  2126. </div>
  2127. <div class="input-group">
  2128. <label data-translate="enable">启用</label>
  2129. <label class="switch">
  2130. <input type="checkbox" class="tab-switch" \${config.enable || config.enable === undefined ? 'checked' : ''} name="enable"><span class="switch-slider"></span>
  2131. </label>
  2132. </div>
  2133. \`);
  2134. content.appendChild(tab);
  2135.  
  2136. tab.querySelector('.remove-button').onclick = () => {
  2137. const previousElement = tabButton.previousElementSibling;
  2138. sidebar.removeChild(tabButton);
  2139. content.removeChild(tab);
  2140. activateTab(previousElement.getAttribute('data-tab'));
  2141. };
  2142.  
  2143. const nameInput = tab.querySelector('[name="name"]');
  2144. nameInput.oninput = () => {
  2145. tabButton.innerHTML = policy.createHTML(SYSTEM_SVG[tab.querySelector('[name=system] .active').getAttribute('value')] + (
  2146. nameInput.value || tabName));
  2147. };
  2148.  
  2149. config.system = config.system || 'windows';
  2150. tab.querySelectorAll('[name=system]').forEach(radioButtonGroup => {
  2151. const radioButtons = radioButtonGroup.querySelectorAll('.radio-button');
  2152. radioButtons.forEach(radioButton => {
  2153. radioButton.onclick = () => {
  2154. radioButtons.forEach(btn => btn.classList.remove('active'));
  2155. radioButton.classList.add('active');
  2156. tabButton.innerHTML = policy.createHTML(SYSTEM_SVG[radioButton.getAttribute('value')] + (nameInput
  2157. .value || tabName));
  2158. };
  2159. if (radioButton.getAttribute('value') === config.system) {
  2160. radioButton.classList.add('active');
  2161. tabButton.innerHTML = policy.createHTML(SYSTEM_SVG[radioButton.getAttribute('value')] + (nameInput
  2162. .value || tabName));
  2163. } else {
  2164. radioButton.classList.remove('active');
  2165. }
  2166. });
  2167. })
  2168.  
  2169. tabButton.onclick = () => activateTab(tabId);
  2170.  
  2171. activateTab(tabId);
  2172. }
  2173.  
  2174. function activateTab(tabId) {
  2175. document.querySelectorAll('.tab').forEach(tab => tab.classList.remove('active'));
  2176. document.querySelectorAll('.tab-button').forEach(btn => btn.classList.remove('active'));
  2177. document.getElementById(tabId).classList.add('active');
  2178. document.querySelector(\`[data-tab="\${tabId}"]\`).classList.add('active');
  2179. document.querySelector('#content').scrollTop = 0;
  2180. }
  2181.  
  2182. function saveConfig() {
  2183. const ytdlp = document.querySelector('#ytdlp');
  2184. const ytdlpRegex = ytdlp.querySelector('[name="regex"]:not([disabled])').value;
  2185.  
  2186. const bilibili = document.querySelector('#bilibili');
  2187. const bilibiliRegex = bilibili.querySelector('[name="regex"]:not([disabled])').value;
  2188.  
  2189. const config = {
  2190. global: {
  2191. parser: {}
  2192. },
  2193. players: []
  2194. };
  2195.  
  2196. for (const id in defaultConfig.global.parser) {
  2197. const parser = document.getElementById(id);
  2198. if (!parser) {
  2199. continue;
  2200. }
  2201. const regex = parser.querySelector('[name="regex"]:not([disabled])').value;
  2202. config.global.parser[id] = {};
  2203. config.global.parser[id].regex = regex ? regex.split('\\n') : [];
  2204. for (const name in defaultConfig.global.parser[id]) {
  2205. if (name === 'regex') {
  2206. continue;
  2207. }
  2208. config.global.parser[id][name] = parser.querySelector(\`[name=\${name}] .active\`).getAttribute('value');
  2209. }
  2210. }
  2211.  
  2212. for (const key in defaultConfig.global) {
  2213. if (key === 'parser') {
  2214. continue;
  2215. }
  2216. config.global[key] = document.getElementById(key)?.value || defaultConfig.global[key];
  2217. }
  2218.  
  2219. document.querySelectorAll('.tab').forEach(tab => {
  2220. if (tab.id !== 'global') {
  2221. config.players.push({
  2222. readonly: tab.getAttribute('readonly') === "true",
  2223. name: tab.querySelector('[name="name"]').value || tab.name || 'Player',
  2224. system: tab.querySelector('[name="system"] .active').getAttribute('value') ||
  2225. 'windows',
  2226. icon: tab.querySelector('[name="icon"]').value || '',
  2227. iconSize: tab.querySelector('[name="iconSize"]').value || 50,
  2228. playEvent: tab.querySelector('[name="playEvent"]').value || '',
  2229. presetEvent: {
  2230. playAuto: tab.querySelector('[name="playAuto"]').checked,
  2231. pauseAuto: tab.querySelector('[name="pauseAuto"]').checked,
  2232. closeAuto: tab.querySelector('[name="closeAuto"]').checked,
  2233. syncTime: tab.querySelector('[name="syncTime"]').checked,
  2234. },
  2235. enable: tab.querySelector('[name="enable"]').checked,
  2236. });
  2237. }
  2238. });
  2239.  
  2240. parent.postMessage({
  2241. name: projectName,
  2242. method: 'saveConfig',
  2243. config: config
  2244. }, '*');
  2245. };
  2246.  
  2247. function resetButtonCoord() {
  2248. document.getElementById('buttonXCoord').value = defaultConfig.global.buttonXCoord;
  2249. document.getElementById('buttonYCoord').value = defaultConfig.global.buttonYCoord;
  2250. }
  2251.  
  2252. function loadConfig(config) {
  2253. // 全局配置
  2254. for (const key in config.global) {
  2255. if (key === 'parser' || !document.getElementById(key)) {
  2256. continue;
  2257. }
  2258. document.getElementById(key).value = config.global[key];
  2259. }
  2260.  
  2261. document.getElementById('language').value = config.global.language;
  2262. language.dispatchEvent(new Event("change"));
  2263.  
  2264. document.querySelectorAll('.parser').forEach(parser => {
  2265. parser.querySelectorAll('.radio-button-group').forEach(radioButtonGroup => {
  2266. const radioButtons = radioButtonGroup.querySelectorAll('.radio-button');
  2267. radioButtons.forEach(radioButton => {
  2268. if (radioButton.getAttribute('value') === config.global.parser[parser.id][
  2269. radioButtonGroup.getAttribute('name')
  2270. ]) {
  2271. radioButton.classList.add('active');
  2272. } else {
  2273. radioButton.classList.remove('active');
  2274. }
  2275. });
  2276. })
  2277. parser.querySelectorAll('textarea').forEach(textarea => {
  2278. if (textarea.disabled) {
  2279. const regex = defaultConfig.global.parser[parser.id][textarea.getAttribute(
  2280. 'name')] || [];
  2281. if (regex.length > 0) {
  2282. textarea.value = regex.join('\\n');
  2283. textarea.style.height = regex.length * 20 + 20 + 'px';
  2284. } else {
  2285. textarea.style.display = 'none';
  2286. }
  2287. } else {
  2288. const regex = config.global.parser[parser.id][textarea.getAttribute('name')] || [];
  2289. textarea.value = regex.join('\\n');
  2290. }
  2291. })
  2292. })
  2293.  
  2294. // 播放器配置
  2295. removeAllTab();
  2296. config.players.forEach(player => createTab(\`player\${tabCount++}\`, player.name, player));
  2297.  
  2298. // 默认选中全局配置
  2299. activateTab('global');
  2300. }
  2301.  
  2302. function removeAllTab() {
  2303. document.querySelectorAll('.tab-button').forEach(tabButton => {
  2304. if (tabButton.id === 'global-button') {
  2305. return;
  2306. }
  2307. sidebar.removeChild(tabButton);
  2308. })
  2309. document.querySelectorAll('.tab').forEach(tab => {
  2310. if (tab.id === 'global') {
  2311. return;
  2312. }
  2313. content.removeChild(tab);
  2314. })
  2315. }
  2316.  
  2317. function resetConfig() {
  2318. let config = JSON.parse(JSON.stringify(defaultConfig));
  2319. for (const key in config.global.parser) {
  2320. config.global.parser[key].regex = [];
  2321. }
  2322. loadConfig(config);
  2323. }
  2324.  
  2325. function init() {
  2326. if (window.self === window.top) {
  2327. return;
  2328. }
  2329.  
  2330. window.addEventListener('message', function (event) {
  2331. const data = event.data;
  2332. if (!data || !data.name || data.method !== 'loadConfig') {
  2333. return;
  2334. }
  2335. projectName = data.name;
  2336. defaultConfig = data.defaultConfig;
  2337. loadConfig(data.config);
  2338. translatePage(data.config.global.language);
  2339. document.getElementById('sidebar-container').style.display = 'flex';
  2340. document.getElementById('content-container').style.display = 'flex';
  2341. });
  2342.  
  2343. document.getElementById('language').addEventListener('change', (e) => {
  2344. translatePage(e.target.value);
  2345. });
  2346. document.getElementById('add-tab-button').onclick = () => createTab(\`tab\${tabCount++}\`);
  2347. document.getElementById('global-button').onclick = () => activateTab('global');
  2348. document.getElementById('save-button').onclick = () => saveConfig();
  2349. document.getElementById('reset-button').onclick = () => resetConfig();
  2350. document.getElementById('reset-button-coord-button').onclick = () => resetButtonCoord();
  2351.  
  2352. document.querySelectorAll('#global .radio-button-group').forEach(radioButtonGroup => {
  2353. const radioButtons = radioButtonGroup.querySelectorAll('.radio-button');
  2354. radioButtons.forEach(radioButton => {
  2355. radioButton.onclick = () => {
  2356. radioButtons.forEach(btn => btn.classList.remove('active'));
  2357. radioButton.classList.add('active');
  2358. };
  2359. });
  2360. })
  2361. }
  2362.  
  2363. init();
  2364. </script>
  2365.  
  2366. </html>
  2367. `;
  2368. if (SETTING_URL) {
  2369. const response = await fetch(SETTING_URL);
  2370. settingIframeHtml = await response.text();
  2371. }
  2372. settingIframe.onload = function () {
  2373. const doc = settingIframe.contentDocument || settingIframe.contentWindow.document;
  2374. doc.open();
  2375. doc.write(policy.createHTML(settingIframeHtml));
  2376. doc.close();
  2377. };
  2378. document.body.appendChild(settingIframe);
  2379.  
  2380. try {
  2381. showLoading();
  2382. await sleep(REFRESH_INTERVAL);
  2383. } finally {
  2384. hideLoading();
  2385. }
  2386. }
  2387.  
  2388. function saveConfig(config) {
  2389. // 保存配置
  2390. currentConfig = config;
  2391. GM_setValue('config', currentConfig);
  2392. showToast(translation.saveSuccessfully);
  2393.  
  2394. // 移除旧元素
  2395. document.head.removeChild(style);
  2396. document.body.removeChild(buttonDiv);
  2397. style = undefined;
  2398. buttonDiv = undefined;
  2399.  
  2400. // 重新初始化
  2401. isReloading = true;
  2402. init(currentUrl);
  2403. }
  2404.  
  2405. function startFlashing(element) {
  2406. let visibility = element.style.visibility;
  2407. let transition = element.style.transition;
  2408. let boxShadow = element.style.boxShadow;
  2409.  
  2410. element.style.visibility = 'visible';
  2411. element.style.transition = 'box-shadow 0.5s ease';
  2412. let isGlowing = false;
  2413. const interval = setInterval(() => {
  2414. isGlowing = !isGlowing;
  2415. element.style.boxShadow = isGlowing ? `0 0 10px 10px ${COLOR.PRIMARY}` : 'none';
  2416. }, 500);
  2417.  
  2418. setTimeout(() => {
  2419. clearInterval(interval);
  2420. element.style.visibility = visibility;
  2421. element.transition = transition;
  2422. element.boxShadow = boxShadow;
  2423. }, 5000);
  2424. }
  2425.  
  2426. function showButtonDiv() {
  2427. buttonDiv.style.display = 'flex';
  2428. if (!isReloading) {
  2429. for (const player of currentConfig.players) {
  2430. if (player.presetEvent.playAuto === true) {
  2431. setTimeout(() => {
  2432. currentParser.play(player);
  2433. }, REFRESH_INTERVAL);
  2434. }
  2435. }
  2436. }
  2437. isReloading = false;
  2438. }
  2439.  
  2440. // ======================================== 开始执行 =======================================
  2441.  
  2442. function initTop() {
  2443. appendCss();
  2444. appendToastDiv();
  2445. appendLoadingDiv();
  2446. appendButtonDiv();
  2447.  
  2448. if (currentParser) {
  2449. showButtonDiv();
  2450. }
  2451.  
  2452. // 监听子页面事件
  2453. window.addEventListener('message', function (event) {
  2454. const data = event.data;
  2455. if (!data) {
  2456. return;
  2457. }
  2458. if (!data.name || data.name !== PROJECT_NAME) {
  2459. return;
  2460. }
  2461. if (data.method === 'init') {
  2462. iframe = event.source;
  2463. // 子页面覆盖父页面解析器
  2464. currentParser = new PARSER.IFRAME();
  2465. isReloading = data.isReloading;
  2466. showButtonDiv();
  2467. return;
  2468. }
  2469. if (data.method === 'currentMedia') {
  2470. currentMedia = data.currentMedia;
  2471. return;
  2472. }
  2473. });
  2474.  
  2475. // 快捷键
  2476. document.addEventListener('keydown', (event) => {
  2477. // 打开设置:Ctrl + Alt + E
  2478. if (event.ctrlKey && event.altKey && (event.key === 'e' || event.key === 'E')) {
  2479. event.preventDefault();
  2480. startFlashing(settingButton);
  2481. settingButton.click();
  2482. }
  2483. });
  2484.  
  2485. // 保存配置
  2486. window.addEventListener('message', function (event) {
  2487. const data = event.data;
  2488. if (!data || data.name !== PROJECT_NAME || data.method !== 'saveConfig') {
  2489. return;
  2490. }
  2491. saveConfig(data.config);
  2492. if (iframe) {
  2493. iframe.postMessage({
  2494. name: PROJECT_NAME,
  2495. method: 'reload'
  2496. }, '*');
  2497. }
  2498. });
  2499. }
  2500.  
  2501. function initIframe() {
  2502. if (currentParser) {
  2503. // 通知顶层窗口初始化按钮
  2504. setTimeout(() => {
  2505. parent.postMessage({
  2506. name: PROJECT_NAME,
  2507. method: 'init',
  2508. isReloading: isReloading
  2509. }, '*');
  2510. isReloading = false;
  2511. }, REFRESH_INTERVAL);
  2512.  
  2513. // 监听父页面事件
  2514. window.addEventListener("message", async function (event) {
  2515. const data = event.data;
  2516. if (!data) {
  2517. return;
  2518. }
  2519. if (!data.name || data.name !== PROJECT_NAME) {
  2520. return;
  2521. }
  2522. if (data.method === 'execute') {
  2523. await currentParser.execute();
  2524. parent.postMessage({
  2525. name: PROJECT_NAME,
  2526. method: 'currentMedia',
  2527. currentMedia: currentMedia
  2528. }, '*');
  2529. return;
  2530. }
  2531. if (data.method === 'pause') {
  2532. currentParser.pause();
  2533. return;
  2534. }
  2535. if (data.method === 'reload') {
  2536. isReloading = true;
  2537. init(currentUrl);
  2538. }
  2539. });
  2540. }
  2541. }
  2542.  
  2543. async function init(url) {
  2544. currentConfig = loadConfig();
  2545. translation = translations[currentConfig.global.language];
  2546. currentParser = matchParser(currentConfig.global.parser, url) || matchParser(defaultConfig.global.parser, url);
  2547. if (self === top) {
  2548. initTop();
  2549. } else {
  2550. initIframe();
  2551. }
  2552. currentUrl = url;
  2553. }
  2554.  
  2555. onload = () => {
  2556. setInterval(() => {
  2557. const url = location.href;
  2558. if (currentUrl !== url || (self === top && !buttonDiv)) {
  2559. console.log(`current url update: ${currentUrl ? currentUrl + ' => ' : ''}${url}`);
  2560. if (currentUrl && currentUrl.indexOf('?') > -1 &&
  2561. url.replace(/\/\?/, '?').startsWith(currentUrl.replace(/\/\?/, '?'))) {
  2562. currentUrl = url;
  2563. return;
  2564. }
  2565. init(url);
  2566. }
  2567. }, REFRESH_INTERVAL);
  2568. };

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址