Greasy Fork is available in English.

奇讯影视解析 (2025最新优化版) - 悬浮面板 - 多源同时解析

优酷、爱奇艺、腾讯、B站等视频网站VIP视频解析,悬浮面板,多源解析(可选6/4/1个源),单源放大并替换

  1. // ==UserScript==
  2. // @name 奇讯影视解析 (2025最新优化版) - 悬浮面板 - 多源同时解析
  3. // @namespace qx-vip-video
  4. // @version 1.7.5
  5. // @description 优酷、爱奇艺、腾讯、B站等视频网站VIP视频解析,悬浮面板,多源解析(可选6/4/1个源),单源放大并替换
  6. // @author xnone
  7. // @icon 
  8. // @match *://*.youku.com/*
  9. // @match *://*.iqiyi.com/*
  10. // @match *://*.iq.com/*
  11. // @match *://v.qq.com/*
  12. // @match *://*.bilibili.com/*
  13. // @match *://*.mgtv.com/*
  14. // @match *://*.le.com/*
  15. // @match *://*.tudou.com/*
  16. // @match *://*.pptv.com/*
  17. // @match *://*.1905.com/*
  18. // @grant GM_getValue
  19. // @grant GM_setValue
  20. // @grant GM_registerMenuCommand
  21. // @license GPLv3
  22. // ==/UserScript==
  23.  
  24. (function () {
  25. 'use strict';
  26.  
  27. const isMobile = /Android|webOS|iPhone|iPod|BlackBerry/i.test(navigator.userAgent);
  28.  
  29. // 解析线路配置
  30. const parseUrls = [
  31. "https://jx.dmflv.cc/?url=",
  32. "https://www.yemu.xyz/?url=",
  33. "https://jx.nnxv.cn/tv.php?url=",
  34. "https://jx.playerjy.com/?ads=0&url=",
  35. "https://jx.xmflv.com/?url=",
  36. "https://videocdn.ihelpy.net/jiexi/m1907.html?m1907jx=",
  37.  
  38. // "https://im1907.top/?jx=",
  39. // "https://jx.jsonplayer.com/player/?url=",
  40. // "https://jx.yangtu.top/?url=",
  41. // "https://vip.bljiex.com/?v=",
  42. // "https://www.ckplayer.vip/jiexi/?url=",
  43. // "https://jx.m3u8.tv/jiexi/?url="
  44. ];
  45.  
  46. // 网站与解析规则的映射
  47. const siteRules = {
  48. 'v.qq.com': { node: ['.player__container', '#player-container'], area: 'playlist-list' },
  49. 'iqiyi.com': { node: ['#video'], area: '' },
  50. 'iq.com': { node: ['.intl-video-wrap'], area: 'm-sliding-list' },
  51. 'youku.com': { node: ['#ykPlayer'], area: 'new-box-anthology-items' },
  52. 'bilibili.com': { node: ['#bilibili-player', '.bpx-player-primary-area'], area: 'video-episode-card' },
  53. 'mgtv.com': { node: ['#mgtv-player-wrap'], area: 'episode-items' },
  54. 'le.com': { node: ['#le_playbox'], area: 'juji_grid' },
  55. 'tudou.com': { node: ['#player'], area: '' },
  56. 'pptv.com': { node: ['#pptv_playpage_box'], area: '' },
  57. '1905.com': { node: ['#player', '#vodPlayer'], area: '' },
  58. };
  59.  
  60. let originalVideoContainer = null;
  61. let originalVideoContainerSelector = null;
  62. let currentIframeContainer = null;
  63. let videoContainerWidth = null;
  64. let videoContainerHeight = null;
  65. let hidePanelTimeout = null; // 隐藏面板的定时器
  66.  
  67. function getSiteRule(host) {
  68. return siteRules[Object.keys(siteRules).find(key => host.includes(key))] || null;
  69. }
  70.  
  71. function createParseElements() {
  72. const iconSize = isMobile ? 40 : GM_getValue('iconWidth', 40);
  73. const iconTop = isMobile ? 360 : GM_getValue('iconTop', 360);
  74. const iconPosition = isMobile ? 'left' : GM_getValue('iconPosition', 'left');
  75.  
  76. const iconStyle = `
  77. #vipParseIcon {
  78. position: fixed;
  79. top: ${iconTop}px;
  80. ${iconPosition}: 5px;
  81. z-index: 999999;
  82. cursor: pointer;
  83. display: flex;
  84. flex-direction: ${iconPosition === 'left' ? 'row' : 'row-reverse'};
  85. }
  86. #vipParseIcon img {
  87. width: ${iconSize}px;
  88. height: ${iconSize * 1.5}px;
  89. opacity: ${isMobile ? 1 : GM_getValue('iconOpacity', 100) / 100};
  90. transition: transform 0.3s ease;
  91. }
  92. #vipParseIcon:hover img {
  93. transform: scale(1.2);
  94. }
  95.  
  96. #parsePanel {
  97. position: absolute; /* 绝对定位 */
  98. top: ${iconSize * 1.5}px; /* 图标高度+5px的间距*/
  99. ${iconPosition === 'left' ? 'left: 0;' : 'right: 0;'} /* 根据图标位置调整 */
  100. z-index: 999998;
  101. background-color: #fff;
  102. border: 1px solid #ccc;
  103. padding: 15px;
  104. box-shadow: 0 2px 5px rgba(0, 0, 0, 0.2);
  105. border-radius: 5px;
  106. width: 280px; /* 调整面板宽度 */
  107. display: none; /* 初始隐藏 */
  108. }
  109.  
  110. #parsePanel button {
  111. margin: 8px 0;
  112. padding: 10px 18px;
  113. background-color: #2871a6;
  114. color: #fff;
  115. border: none;
  116. border-radius: 4px;
  117. cursor: pointer;
  118. transition: background-color 0.3s;
  119. width: 100%;
  120. box-sizing: border-box;
  121. }
  122. #parsePanel button:hover {
  123. background-color: #1e5a88;
  124. }
  125.  
  126. #parsePanel .warning-tips {
  127. background: #ffeeee; /* 红色背景 */
  128. color: #ff0000; /* 红色字体 */
  129. padding: 15px; /* 内边距 */
  130. border-radius: 5px; /* 圆角 */
  131. box-shadow: 0 0 10px rgba(0,0,0,0.1); /* 投影效果 */
  132. z-index: 1000; /* 确保提示显示在最前面 */
  133. font-size: 12px; /* 字体大小 */
  134. line-height: 1.5; /* 行间距 */
  135. }
  136. #parsePanel .warning-tips p {
  137. margin: 5px 0; /* 段落间距 */
  138. }
  139.  
  140. #configPanel {
  141. margin-top: 15px;
  142. padding-top: 10px;
  143. border-top: 1px solid #eee;
  144. }
  145.  
  146. #configPanel label {
  147. display: block;
  148. margin-bottom: 8px;
  149. color: #333;
  150. }
  151. #configPanel input[type="radio"] {
  152. margin-right: 6px;
  153. }
  154.  
  155. #saveConfigBtn {
  156. background-color: #4CAF50 !important;
  157. }
  158. #saveConfigBtn:hover {
  159. background-color: #45a049 !important;
  160. }
  161.  
  162. #aboutPanel {
  163. margin-top: 15px;
  164. padding: 15px;
  165. background-color: #f8f9fa;
  166. border-radius: 4px;
  167. }
  168.  
  169. #aboutPanel h3 {
  170. margin-top: 0;
  171. color: #2c3e50;
  172. }
  173.  
  174. #aboutPanel p {
  175. color: #34495e;
  176. line-height: 1.6;
  177. }
  178.  
  179. #telegramLink {
  180. color: #007bff;
  181. text-decoration: underline;
  182. cursor: pointer;
  183. }
  184.  
  185. /* ... 其他样式保持不变 ... */
  186. .iframe-container {
  187. display: grid;
  188. grid-template-columns: repeat(3, 1fr);
  189. grid-template-rows: repeat(2, auto);
  190. grid-auto-rows: minmax(200px, auto);
  191. grid-gap: 1px;
  192. width: 100%;
  193. height: 100%;
  194. }
  195. .iframe-container iframe {
  196. width: 100%;
  197. height: 100%;
  198. border: 1px solid #ddd;
  199. }
  200. /* 可选:添加响应式设计 */
  201. @media (max-width: 768px) {
  202. .iframe-container {
  203. grid-template-columns: repeat(2, 1fr); /* 在小屏幕上显示两列 */
  204. }
  205. }
  206.  
  207. @media (max-width: 480px) {
  208. .iframe-container {
  209. grid-template-columns: 1fr; /* 在非常小的屏幕上显示一列 */
  210. }
  211. }
  212.  
  213. .iframe-wrapper {
  214. position: relative;
  215. }
  216.  
  217. .expand-button {
  218. position: absolute;
  219. bottom: 0;
  220. left: 0;
  221. width: 100%;
  222. background-color: rgba(0, 0, 0, 0.5);
  223. color: white;
  224. text-align: center;
  225. padding: 5px 0;
  226. cursor: pointer;
  227. opacity: 0;
  228. transition: opacity 0.3s;
  229. }
  230. .iframe-wrapper:hover .expand-button {
  231. opacity: 1;
  232. }
  233. `;
  234.  
  235. const styleEl = document.createElement('style');
  236. styleEl.textContent = iconStyle;
  237. document.head.appendChild(styleEl);
  238.  
  239. const iconHtml = `
  240. <img src="" />
  241. <div id="parsePanel">
  242. <div>
  243. <button id="parseBtn">👉解析</button>
  244. <button id="gotoSiteBtn" style="background:#5a6268;">去奇讯边聊边看</button>
  245. <button id="restoreBtn" style="background:#5a6268;">还原</button>
  246. </div>
  247. <div id="configPanel">
  248. <label><input type="radio" name="iframeCount" value="6"> 6个格子解析</label>
  249. <label><input type="radio" name="iframeCount" value="4"> 4个格子解析</label>
  250. <label><input type="radio" name="iframeCount" value="1"> 1个格子解析</label>
  251. <button id="saveConfigBtn">保存配置</button>
  252. <div id="configTips" style="margin-top: 10px; padding: 5px 10px; color: red; display: none;font-size:12px;">配置已保存并生效!</div>
  253. </div>
  254. <div class="warning-tips">
  255. <p>⚠️ 注意:</p>
  256. <p>如果全部解析失败请过段时间再试</p>
  257. <p>请勿相信视频中的任何广告,都是假的</p>
  258. </div>
  259.  
  260. <div id="aboutPanel">
  261. <h3>🎥 奇讯视频解析工具</h3>
  262. <p>一键解析多平台视频,支持多源选择,提供便捷的观看体验。</p>
  263. <a id="telegramLink" href="https://t.me/qixunyingshi" target="_blank">点击加入 Telegram 群组</a>
  264. </div>
  265. </div>
  266. `;
  267.  
  268.  
  269. const container = document.createElement('div');
  270. container.id = 'vipParseIcon';
  271. container.innerHTML = iconHtml;
  272. document.body.appendChild(container);
  273.  
  274. const parsePanel = document.getElementById('parsePanel');
  275. const vipParseIcon = document.getElementById('vipParseIcon');
  276. const parseBtn = document.getElementById('parseBtn');
  277. const configPanel = document.getElementById('configPanel');
  278. const saveConfigBtn = document.getElementById('saveConfigBtn');
  279. const restoreBtn = document.getElementById('restoreBtn');
  280. const gotoSiteBtn = document.getElementById('gotoSiteBtn');
  281. const icon = container.querySelector('img');
  282. const telegramLink = document.getElementById('telegramLink');
  283.  
  284. // 初始化配置
  285. const iframeCount = GM_getValue('iframeCount', '6');
  286. configPanel.querySelector(`input[value="${iframeCount}"]`).checked = true;
  287.  
  288. // 鼠标移入图标:显示面板,清除隐藏定时器
  289. icon.addEventListener('mouseover', () => {
  290. clearTimeout(hidePanelTimeout);
  291. parsePanel.style.display = 'block';
  292. });
  293.  
  294. // 鼠标移出图标:启动隐藏面板定时器
  295. icon.addEventListener('mouseleave', () => {
  296. hidePanelTimeout = setTimeout(() => {
  297. parsePanel.style.display = 'none';
  298. }, 300);
  299. });
  300.  
  301. // 鼠标移入面板:清除隐藏定时器
  302. parsePanel.addEventListener('mouseover', () => {
  303. clearTimeout(hidePanelTimeout);
  304. });
  305.  
  306. // 鼠标移出面板:启动隐藏面板定时器
  307. parsePanel.addEventListener('mouseleave', () => {
  308. hidePanelTimeout = setTimeout(() => {
  309. parsePanel.style.display = 'none';
  310. }, 300);
  311. });
  312.  
  313. // 保存配置
  314. saveConfigBtn.addEventListener('click', () => {
  315. const newIframeCount = configPanel.querySelector('input[name="iframeCount"]:checked').value;
  316. GM_setValue('iframeCount', newIframeCount);
  317. if (originalVideoContainer) {
  318. parseVideoMulti();
  319. }
  320. // 获取提示元素
  321. const tips = document.getElementById('configTips');
  322. tips.style.display = 'block';
  323. // 3秒后隐藏
  324. setTimeout(() => {
  325. tips.style.display = 'none';
  326. }, 3000);
  327. });
  328.  
  329. parsePanel.addEventListener('click', (e) => {
  330. e.stopPropagation()
  331. });
  332. parseBtn.addEventListener('click', parseVideoMulti);
  333. vipParseIcon.addEventListener('click', parseVideoMulti);
  334. restoreBtn.addEventListener('click', restoreVideo);
  335. gotoSiteBtn.addEventListener('click', () => window.open(`https://qx.bluu.pl/#/?url=${encodeURIComponent(location.href)}`, '_blank'));
  336. telegramLink.addEventListener('click', (e) => {
  337. e.stopPropagation();
  338. window.open('https://t.me/qixunyingshi', '_blank');
  339. });
  340.  
  341. makeDraggable(container, icon);
  342. }
  343.  
  344. function getVideoContainer() {
  345. const siteRule = getSiteRule(location.hostname);
  346. if (!siteRule) {
  347. console.log('未找到匹配的网站规则');
  348. return null;
  349. }
  350. let videoContainer = null;
  351. for (const node of siteRule.node) {
  352. videoContainer = document.querySelector(node);
  353. if (videoContainer) {
  354. originalVideoContainerSelector = node;
  355. videoContainerWidth = videoContainer.offsetWidth;
  356. videoContainerHeight = videoContainer.offsetHeight;
  357. break;
  358. }
  359. }
  360. return videoContainer;
  361. }
  362.  
  363. function expandAndReplaceIframe(iframe) {
  364. const videoContainer = getVideoContainer();
  365. if (!videoContainer) return;
  366.  
  367. const newIframe = document.createElement('iframe');
  368. newIframe.src = iframe.src;
  369. newIframe.allowFullscreen = iframe.allowFullscreen;
  370. newIframe.allowTransparency = iframe.allowTransparency;
  371.  
  372. newIframe.style.width = videoContainerWidth + 'px';
  373. newIframe.style.height = videoContainerHeight + 'px';
  374. newIframe.style.border = 'none';
  375.  
  376. videoContainer.innerHTML = '';
  377. videoContainer.appendChild(newIframe);
  378. currentIframeContainer = null;
  379. }
  380.  
  381. function parseVideoMulti() {
  382. const videoContainer = getVideoContainer();
  383. if (!videoContainer) return;
  384.  
  385. if (!originalVideoContainer) {
  386. originalVideoContainer = videoContainer.innerHTML;
  387. }
  388.  
  389. const iframeCount = parseInt(GM_getValue('iframeCount', '6'));
  390. const urls = parseUrls.slice(0, iframeCount);
  391.  
  392. let gridColumns = 1;
  393. if (iframeCount === 6) {
  394. gridColumns = 3;
  395. } else if (iframeCount === 4) {
  396. gridColumns = 2;
  397. }
  398.  
  399. let iframeHTML = `<div class="iframe-container" style="grid-template-columns: repeat(${gridColumns}, 1fr);">`;
  400. urls.forEach(url => {
  401. iframeHTML += `
  402. <div class="iframe-wrapper">
  403. <iframe src="${url}${encodeURIComponent(location.href)}" allowfullscreen allowtransparency></iframe>
  404. <div class="expand-button">⬆️用这个视频继续播放</div>
  405. </div>
  406. `;
  407. });
  408. iframeHTML += '</div>';
  409.  
  410. videoContainer.innerHTML = iframeHTML;
  411. currentIframeContainer = videoContainer.querySelector('.iframe-container');
  412.  
  413. const expandButtons = videoContainer.querySelectorAll('.expand-button');
  414. expandButtons.forEach((button, index) => {
  415. button.addEventListener('click', () => {
  416. expandAndReplaceIframe(videoContainer.querySelectorAll('iframe')[index]);
  417. });
  418. });
  419.  
  420. const siteRule = getSiteRule(location.hostname);
  421. if (siteRule && siteRule.area) {
  422. const areaSelector = `.${siteRule.area}`;
  423. if (!videoContainer.dataset.eventBound) {
  424. const bindAreaEvent = () => {
  425. const areaElement = document.querySelector(areaSelector);
  426. if (areaElement) {
  427. areaElement.addEventListener('click', () => {
  428. setTimeout(parseVideoMulti, 1000); // 延时并重新解析
  429. });
  430. videoContainer.dataset.eventBound = 'true';
  431. }
  432. };
  433. bindAreaEvent();
  434. const observer = new MutationObserver(bindAreaEvent);
  435. observer.observe(document.body, { childList: true, subtree: true });
  436. }
  437. }
  438. }
  439.  
  440. function restoreVideo() {
  441. if (!originalVideoContainer) return;
  442. const videoContainer = document.querySelector(originalVideoContainerSelector);
  443. if (videoContainer) {
  444. videoContainer.innerHTML = originalVideoContainer;
  445. currentIframeContainer = null;
  446. } else {
  447. console.error("找不到原始视频容器:", originalVideoContainerSelector);
  448. }
  449. }
  450.  
  451. function makeDraggable(element, handle) {
  452. let isDragging = false;
  453. let startX, startY, startTop;
  454.  
  455. handle.addEventListener('mousedown', (e) => {
  456. e.preventDefault();
  457. if (e.button !== 0) return;
  458.  
  459. isDragging = true;
  460. startX = e.clientX;
  461. startY = e.clientY;
  462. startTop = element.offsetTop;
  463.  
  464. document.addEventListener('mousemove', onMouseMove);
  465. document.addEventListener('mouseup', onMouseUp);
  466. });
  467.  
  468. function onMouseMove(e) {
  469. if (!isDragging) return;
  470.  
  471. const deltaY = e.clientY - startY;
  472. let newTop = startTop + deltaY;
  473. const maxHeight = window.innerHeight - element.offsetHeight - 10;
  474. newTop = Math.max(0, Math.min(newTop, maxHeight));
  475. element.style.top = `${newTop}px`;
  476. }
  477.  
  478. function onMouseUp() {
  479. isDragging = false;
  480. document.removeEventListener('mousemove', onMouseMove);
  481. document.removeEventListener('mouseup', onMouseUp);
  482. GM_setValue('iconTop', element.offsetTop);
  483. }
  484. }
  485.  
  486. window.addEventListener('load', () => {
  487. if (getSiteRule(location.hostname)) {
  488. createParseElements();
  489.  
  490. const siteRule = getSiteRule(location.hostname);
  491. if (siteRule && siteRule.area) {
  492. const areaSelector = `.${siteRule.area}`;
  493. const videoContainer = getVideoContainer();
  494. if (videoContainer && !videoContainer.dataset.eventBound) {
  495. const bindAreaEvent = () => {
  496. const areaElement = document.querySelector(areaSelector);
  497. if (areaElement) {
  498. areaElement.addEventListener('click', () => {
  499. setTimeout(parseVideoMulti, 1000);
  500. });
  501. videoContainer.dataset.eventBound = 'true';
  502. }
  503. };
  504.  
  505. bindAreaEvent();
  506. const observer = new MutationObserver(bindAreaEvent);
  507. observer.observe(document.body, { childList: true, subtree: true });
  508. }
  509. }
  510. }
  511. });
  512.  
  513. GM_registerMenuCommand("设置解析线路", () => {
  514. const selectedLine = prompt("请选择或输入解析线路的URL:", localStorage.getItem('preferredParseLine') || parseUrls[0]);
  515. if (selectedLine) {
  516. localStorage.setItem('preferredParseLine', selectedLine);
  517. location.reload();
  518. }
  519. });
  520. })();