优酷、爱奇艺、腾讯、B站等视频网站VIP视频解析,悬浮面板,多源解析(可选6/4/1个源),单源放大并替换
// ==UserScript== // @name 奇讯影视解析 (2025最新优化版) - 悬浮面板 - 多源同时解析 // @namespace qx-vip-video // @version 1.7.5 // @description 优酷、爱奇艺、腾讯、B站等视频网站VIP视频解析,悬浮面板,多源解析(可选6/4/1个源),单源放大并替换 // @author xnone // @icon  // @match *://*.youku.com/* // @match *://*.iqiyi.com/* // @match *://*.iq.com/* // @match *://v.qq.com/* // @match *://*.bilibili.com/* // @match *://*.mgtv.com/* // @match *://*.le.com/* // @match *://*.tudou.com/* // @match *://*.pptv.com/* // @match *://*.1905.com/* // @grant GM_getValue // @grant GM_setValue // @grant GM_registerMenuCommand // @license GPLv3 // ==/UserScript== (function () { 'use strict'; const isMobile = /Android|webOS|iPhone|iPod|BlackBerry/i.test(navigator.userAgent); // 解析线路配置 const parseUrls = [ "https://jx.dmflv.cc/?url=", "https://www.yemu.xyz/?url=", "https://jx.nnxv.cn/tv.php?url=", "https://jx.playerjy.com/?ads=0&url=", "https://jx.xmflv.com/?url=", "https://videocdn.ihelpy.net/jiexi/m1907.html?m1907jx=", // "https://im1907.top/?jx=", // "https://jx.jsonplayer.com/player/?url=", // "https://jx.yangtu.top/?url=", // "https://vip.bljiex.com/?v=", // "https://www.ckplayer.vip/jiexi/?url=", // "https://jx.m3u8.tv/jiexi/?url=" ]; // 网站与解析规则的映射 const siteRules = { 'v.qq.com': { node: ['.player__container', '#player-container'], area: 'playlist-list' }, 'iqiyi.com': { node: ['#video'], area: '' }, 'iq.com': { node: ['.intl-video-wrap'], area: 'm-sliding-list' }, 'youku.com': { node: ['#ykPlayer'], area: 'new-box-anthology-items' }, 'bilibili.com': { node: ['#bilibili-player', '.bpx-player-primary-area'], area: 'video-episode-card' }, 'mgtv.com': { node: ['#mgtv-player-wrap'], area: 'episode-items' }, 'le.com': { node: ['#le_playbox'], area: 'juji_grid' }, 'tudou.com': { node: ['#player'], area: '' }, 'pptv.com': { node: ['#pptv_playpage_box'], area: '' }, '1905.com': { node: ['#player', '#vodPlayer'], area: '' }, }; let originalVideoContainer = null; let originalVideoContainerSelector = null; let currentIframeContainer = null; let videoContainerWidth = null; let videoContainerHeight = null; let hidePanelTimeout = null; // 隐藏面板的定时器 function getSiteRule(host) { return siteRules[Object.keys(siteRules).find(key => host.includes(key))] || null; } function createParseElements() { const iconSize = isMobile ? 40 : GM_getValue('iconWidth', 40); const iconTop = isMobile ? 360 : GM_getValue('iconTop', 360); const iconPosition = isMobile ? 'left' : GM_getValue('iconPosition', 'left'); const iconStyle = ` #vipParseIcon { position: fixed; top: ${iconTop}px; ${iconPosition}: 5px; z-index: 999999; cursor: pointer; display: flex; flex-direction: ${iconPosition === 'left' ? 'row' : 'row-reverse'}; } #vipParseIcon img { width: ${iconSize}px; height: ${iconSize * 1.5}px; opacity: ${isMobile ? 1 : GM_getValue('iconOpacity', 100) / 100}; transition: transform 0.3s ease; } #vipParseIcon:hover img { transform: scale(1.2); } #parsePanel { position: absolute; /* 绝对定位 */ top: ${iconSize * 1.5}px; /* 图标高度+5px的间距*/ ${iconPosition === 'left' ? 'left: 0;' : 'right: 0;'} /* 根据图标位置调整 */ z-index: 999998; background-color: #fff; border: 1px solid #ccc; padding: 15px; box-shadow: 0 2px 5px rgba(0, 0, 0, 0.2); border-radius: 5px; width: 280px; /* 调整面板宽度 */ display: none; /* 初始隐藏 */ } #parsePanel button { margin: 8px 0; padding: 10px 18px; background-color: #2871a6; color: #fff; border: none; border-radius: 4px; cursor: pointer; transition: background-color 0.3s; width: 100%; box-sizing: border-box; } #parsePanel button:hover { background-color: #1e5a88; } #parsePanel .warning-tips { background: #ffeeee; /* 红色背景 */ color: #ff0000; /* 红色字体 */ padding: 15px; /* 内边距 */ border-radius: 5px; /* 圆角 */ box-shadow: 0 0 10px rgba(0,0,0,0.1); /* 投影效果 */ z-index: 1000; /* 确保提示显示在最前面 */ font-size: 12px; /* 字体大小 */ line-height: 1.5; /* 行间距 */ } #parsePanel .warning-tips p { margin: 5px 0; /* 段落间距 */ } #configPanel { margin-top: 15px; padding-top: 10px; border-top: 1px solid #eee; } #configPanel label { display: block; margin-bottom: 8px; color: #333; } #configPanel input[type="radio"] { margin-right: 6px; } #saveConfigBtn { background-color: #4CAF50 !important; } #saveConfigBtn:hover { background-color: #45a049 !important; } #aboutPanel { margin-top: 15px; padding: 15px; background-color: #f8f9fa; border-radius: 4px; } #aboutPanel h3 { margin-top: 0; color: #2c3e50; } #aboutPanel p { color: #34495e; line-height: 1.6; } #telegramLink { color: #007bff; text-decoration: underline; cursor: pointer; } /* ... 其他样式保持不变 ... */ .iframe-container { display: grid; grid-template-columns: repeat(3, 1fr); grid-template-rows: repeat(2, auto); grid-auto-rows: minmax(200px, auto); grid-gap: 1px; width: 100%; height: 100%; } .iframe-container iframe { width: 100%; height: 100%; border: 1px solid #ddd; } /* 可选:添加响应式设计 */ @media (max-width: 768px) { .iframe-container { grid-template-columns: repeat(2, 1fr); /* 在小屏幕上显示两列 */ } } @media (max-width: 480px) { .iframe-container { grid-template-columns: 1fr; /* 在非常小的屏幕上显示一列 */ } } .iframe-wrapper { position: relative; } .expand-button { position: absolute; bottom: 0; left: 0; width: 100%; background-color: rgba(0, 0, 0, 0.5); color: white; text-align: center; padding: 5px 0; cursor: pointer; opacity: 0; transition: opacity 0.3s; } .iframe-wrapper:hover .expand-button { opacity: 1; } `; const styleEl = document.createElement('style'); styleEl.textContent = iconStyle; document.head.appendChild(styleEl); const iconHtml = ` <img src="" /> <div id="parsePanel"> <div> <button id="parseBtn">👉解析</button> <button id="gotoSiteBtn" style="background:#5a6268;">去奇讯边聊边看</button> <button id="restoreBtn" style="background:#5a6268;">还原</button> </div> <div id="configPanel"> <label><input type="radio" name="iframeCount" value="6"> 6个格子解析</label> <label><input type="radio" name="iframeCount" value="4"> 4个格子解析</label> <label><input type="radio" name="iframeCount" value="1"> 1个格子解析</label> <button id="saveConfigBtn">保存配置</button> <div id="configTips" style="margin-top: 10px; padding: 5px 10px; color: red; display: none;font-size:12px;">配置已保存并生效!</div> </div> <div class="warning-tips"> <p>⚠️ 注意:</p> <p>如果全部解析失败请过段时间再试</p> <p>请勿相信视频中的任何广告,都是假的</p> </div> <div id="aboutPanel"> <h3>🎥 奇讯视频解析工具</h3> <p>一键解析多平台视频,支持多源选择,提供便捷的观看体验。</p> <a id="telegramLink" href="https://t.me/qixunyingshi" target="_blank">点击加入 Telegram 群组</a> </div> </div> `; const container = document.createElement('div'); container.id = 'vipParseIcon'; container.innerHTML = iconHtml; document.body.appendChild(container); const parsePanel = document.getElementById('parsePanel'); const vipParseIcon = document.getElementById('vipParseIcon'); const parseBtn = document.getElementById('parseBtn'); const configPanel = document.getElementById('configPanel'); const saveConfigBtn = document.getElementById('saveConfigBtn'); const restoreBtn = document.getElementById('restoreBtn'); const gotoSiteBtn = document.getElementById('gotoSiteBtn'); const icon = container.querySelector('img'); const telegramLink = document.getElementById('telegramLink'); // 初始化配置 const iframeCount = GM_getValue('iframeCount', '6'); configPanel.querySelector(`input[value="${iframeCount}"]`).checked = true; // 鼠标移入图标:显示面板,清除隐藏定时器 icon.addEventListener('mouseover', () => { clearTimeout(hidePanelTimeout); parsePanel.style.display = 'block'; }); // 鼠标移出图标:启动隐藏面板定时器 icon.addEventListener('mouseleave', () => { hidePanelTimeout = setTimeout(() => { parsePanel.style.display = 'none'; }, 300); }); // 鼠标移入面板:清除隐藏定时器 parsePanel.addEventListener('mouseover', () => { clearTimeout(hidePanelTimeout); }); // 鼠标移出面板:启动隐藏面板定时器 parsePanel.addEventListener('mouseleave', () => { hidePanelTimeout = setTimeout(() => { parsePanel.style.display = 'none'; }, 300); }); // 保存配置 saveConfigBtn.addEventListener('click', () => { const newIframeCount = configPanel.querySelector('input[name="iframeCount"]:checked').value; GM_setValue('iframeCount', newIframeCount); if (originalVideoContainer) { parseVideoMulti(); } // 获取提示元素 const tips = document.getElementById('configTips'); tips.style.display = 'block'; // 3秒后隐藏 setTimeout(() => { tips.style.display = 'none'; }, 3000); }); parsePanel.addEventListener('click', (e) => { e.stopPropagation() }); parseBtn.addEventListener('click', parseVideoMulti); vipParseIcon.addEventListener('click', parseVideoMulti); restoreBtn.addEventListener('click', restoreVideo); gotoSiteBtn.addEventListener('click', () => window.open(`https://qx.bluu.pl/#/?url=${encodeURIComponent(location.href)}`, '_blank')); telegramLink.addEventListener('click', (e) => { e.stopPropagation(); window.open('https://t.me/qixunyingshi', '_blank'); }); makeDraggable(container, icon); } function getVideoContainer() { const siteRule = getSiteRule(location.hostname); if (!siteRule) { console.log('未找到匹配的网站规则'); return null; } let videoContainer = null; for (const node of siteRule.node) { videoContainer = document.querySelector(node); if (videoContainer) { originalVideoContainerSelector = node; videoContainerWidth = videoContainer.offsetWidth; videoContainerHeight = videoContainer.offsetHeight; break; } } return videoContainer; } function expandAndReplaceIframe(iframe) { const videoContainer = getVideoContainer(); if (!videoContainer) return; const newIframe = document.createElement('iframe'); newIframe.src = iframe.src; newIframe.allowFullscreen = iframe.allowFullscreen; newIframe.allowTransparency = iframe.allowTransparency; newIframe.style.width = videoContainerWidth + 'px'; newIframe.style.height = videoContainerHeight + 'px'; newIframe.style.border = 'none'; videoContainer.innerHTML = ''; videoContainer.appendChild(newIframe); currentIframeContainer = null; } function parseVideoMulti() { const videoContainer = getVideoContainer(); if (!videoContainer) return; if (!originalVideoContainer) { originalVideoContainer = videoContainer.innerHTML; } const iframeCount = parseInt(GM_getValue('iframeCount', '6')); const urls = parseUrls.slice(0, iframeCount); let gridColumns = 1; if (iframeCount === 6) { gridColumns = 3; } else if (iframeCount === 4) { gridColumns = 2; } let iframeHTML = `<div class="iframe-container" style="grid-template-columns: repeat(${gridColumns}, 1fr);">`; urls.forEach(url => { iframeHTML += ` <div class="iframe-wrapper"> <iframe src="${url}${encodeURIComponent(location.href)}" allowfullscreen allowtransparency></iframe> <div class="expand-button">⬆️用这个视频继续播放</div> </div> `; }); iframeHTML += '</div>'; videoContainer.innerHTML = iframeHTML; currentIframeContainer = videoContainer.querySelector('.iframe-container'); const expandButtons = videoContainer.querySelectorAll('.expand-button'); expandButtons.forEach((button, index) => { button.addEventListener('click', () => { expandAndReplaceIframe(videoContainer.querySelectorAll('iframe')[index]); }); }); const siteRule = getSiteRule(location.hostname); if (siteRule && siteRule.area) { const areaSelector = `.${siteRule.area}`; if (!videoContainer.dataset.eventBound) { const bindAreaEvent = () => { const areaElement = document.querySelector(areaSelector); if (areaElement) { areaElement.addEventListener('click', () => { setTimeout(parseVideoMulti, 1000); // 延时并重新解析 }); videoContainer.dataset.eventBound = 'true'; } }; bindAreaEvent(); const observer = new MutationObserver(bindAreaEvent); observer.observe(document.body, { childList: true, subtree: true }); } } } function restoreVideo() { if (!originalVideoContainer) return; const videoContainer = document.querySelector(originalVideoContainerSelector); if (videoContainer) { videoContainer.innerHTML = originalVideoContainer; currentIframeContainer = null; } else { console.error("找不到原始视频容器:", originalVideoContainerSelector); } } function makeDraggable(element, handle) { let isDragging = false; let startX, startY, startTop; handle.addEventListener('mousedown', (e) => { e.preventDefault(); if (e.button !== 0) return; isDragging = true; startX = e.clientX; startY = e.clientY; startTop = element.offsetTop; document.addEventListener('mousemove', onMouseMove); document.addEventListener('mouseup', onMouseUp); }); function onMouseMove(e) { if (!isDragging) return; const deltaY = e.clientY - startY; let newTop = startTop + deltaY; const maxHeight = window.innerHeight - element.offsetHeight - 10; newTop = Math.max(0, Math.min(newTop, maxHeight)); element.style.top = `${newTop}px`; } function onMouseUp() { isDragging = false; document.removeEventListener('mousemove', onMouseMove); document.removeEventListener('mouseup', onMouseUp); GM_setValue('iconTop', element.offsetTop); } } window.addEventListener('load', () => { if (getSiteRule(location.hostname)) { createParseElements(); const siteRule = getSiteRule(location.hostname); if (siteRule && siteRule.area) { const areaSelector = `.${siteRule.area}`; const videoContainer = getVideoContainer(); if (videoContainer && !videoContainer.dataset.eventBound) { const bindAreaEvent = () => { const areaElement = document.querySelector(areaSelector); if (areaElement) { areaElement.addEventListener('click', () => { setTimeout(parseVideoMulti, 1000); }); videoContainer.dataset.eventBound = 'true'; } }; bindAreaEvent(); const observer = new MutationObserver(bindAreaEvent); observer.observe(document.body, { childList: true, subtree: true }); } } } }); GM_registerMenuCommand("设置解析线路", () => { const selectedLine = prompt("请选择或输入解析线路的URL:", localStorage.getItem('preferredParseLine') || parseUrls[0]); if (selectedLine) { localStorage.setItem('preferredParseLine', selectedLine); location.reload(); } }); })();