- // ==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();
- }
- });
- })();