// ==UserScript==
// @name 跳转到Emby播放
// @namespace https://github.com/cgkings
// @version 0.0.8
// @description 👆👆👆👆👆👆👆在 ✅JavBus✅Javdb✅Sehuatang ✅supjav 高亮emby存在的视频,并提供标注一键跳转功能
// @author cgkings
// @match *://www.javbus.com/*
// @match *://javdb*.com/v/*
// @match *://javdb*.com/search?q=*
// @match *://www.javdb.com/*
// @match *://javdb.com/*
// @match *://supjav.com/*
// @match *://*.sehuatang.*/*
// @match *://*.sehuatang.net/*
// @match https://.*/thread-*
// @match https://.*/forum.php?mod=viewthread&tid=*
// @grant GM_xmlhttpRequest
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_registerMenuCommand
// @grant GM_addStyle
// @run-at document-start
// @license MIT
// @supportURL https://github.com/cgkings/cg_tampermonkey_script/issues
// @homepageURL https://github.com/cgkings/cg_tampermonkey_script
// @icon 
// @license MIT
// ==/UserScript==
(function () {
'use strict';
// 全局配置对象
const Config = {
get embyAPI() { return GM_getValue('embyAPI', '') },
get embyBaseUrl() { return GM_getValue('embyBaseUrl', '') },
get highlightColor() { return GM_getValue('highlightColor', '#52b54b') },
get maxConcurrentRequests() { return GM_getValue('maxConcurrentRequests', 50) },
// 添加徽章相关配置
get badgeColor() { return GM_getValue('badgeColor', '#000') },
get badgeTextColor() { return GM_getValue('badgeTextColor', '#fff') },
get badgeSize() { return GM_getValue('badgeSize', 'medium') }, // small, medium, large
set embyAPI(val) { GM_setValue('embyAPI', val) },
set embyBaseUrl(val) { GM_setValue('embyBaseUrl', val) },
set highlightColor(val) { GM_setValue('highlightColor', val) },
set maxConcurrentRequests(val) { GM_setValue('maxConcurrentRequests', val) },
// 添加徽章相关配置的setter
set badgeColor(val) { GM_setValue('badgeColor', val) },
set badgeTextColor(val) { GM_setValue('badgeTextColor', val) },
set badgeSize(val) { GM_setValue('badgeSize', val) },
isValid() { return !!this.embyAPI && !!this.embyBaseUrl }
};
// 获取徽章尺寸样式
function getBadgeSizeStyle() {
switch (Config.badgeSize) {
case 'small':
return { fontSize: '10px', padding: '1px 4px' };
case 'large':
return { fontSize: '14px', padding: '3px 7px' };
case 'medium':
default:
return { fontSize: '12px', padding: '2px 5px' };
}
}
// 初始化DOM样式
const badgeSize = getBadgeSizeStyle();
GM_addStyle(`
.emby-jump-settings-panel {position:fixed; top:50%; left:50%; transform:translate(-50%,-50%); background:#fff; border-radius:8px; box-shadow:0 0 20px rgba(0,0,0,0.3); padding:20px; z-index:10000; width:400px; max-width:90%; display:none}
.emby-jump-settings-header {display:flex; justify-content:space-between; align-items:center; margin-bottom:15px; padding-bottom:10px; border-bottom:1px solid #eee}
.emby-jump-settings-close {cursor:pointer; font-size:18px; color:#999}
.emby-jump-settings-field {margin-bottom:15px}
.emby-jump-settings-field label {display:block; margin-bottom:5px; font-weight:bold}
.emby-jump-settings-field input, .emby-jump-settings-field select {width:100%; padding:8px; border:1px solid #ddd; border-radius:4px; box-sizing:border-box}
.emby-jump-settings-buttons {display:flex; justify-content:flex-end; gap:10px; margin-top:15px}
.emby-jump-settings-buttons button {padding:8px 15px; border:none; border-radius:4px; cursor:pointer}
.emby-jump-settings-save {background-color:#52b54b; color:white}
.emby-jump-settings-cancel {background-color:#f0f0f0; color:#333}
.emby-jump-status-indicator {position:fixed; bottom:20px; right:20px; background:rgba(0,0,0,0.7); color:white; padding:8px 12px; border-radius:4px; font-size:14px; z-index:9999; transition:opacity 0.3s; box-shadow:0 2px 8px rgba(0,0,0,0.2); max-width:300px; display:flex; align-items:center; opacity:0}
.emby-jump-status-indicator .progress {display:inline-block; margin-left:10px; width:100px; height:6px; background:rgba(255,255,255,0.3); border-radius:3px}
.emby-jump-status-indicator .progress-bar {height:100%; background:#52b54b; border-radius:3px; transition:width 0.3s}
.emby-jump-status-indicator.success {background-color:rgba(82,181,75,0.9)}
.emby-jump-status-indicator.error {background-color:rgba(220,53,69,0.9)}
.emby-jump-status-indicator .close-btn {margin-left:10px; cursor:pointer; font-size:16px; font-weight:bold}
/* 徽章样式 - 保留彩虹边框,使用自定义颜色和大小 */
.emby-badge {position:absolute; top:5px; right:5px; color:${Config.badgeTextColor}; padding:${badgeSize.padding}; font-size:${badgeSize.fontSize}; font-weight:bold; z-index:10; border:2px solid transparent; border-radius:4px; background-origin:border-box; background-clip:padding-box,border-box; background-image:linear-gradient(${Config.badgeColor} 0 0), linear-gradient(50deg,#ff0000,#ff7f00,#ffff00,#00ff00,#0000ff,#4b0082,#8b00ff);}
.emby-badge:hover {color:#000; background-clip:padding-box,border-box; background-image:linear-gradient(#fff 0 0),linear-gradient(50deg,#ff0000,#ff7f00,#ffff00,#00ff00,#0000ff,#4b0082,#8b00ff);}
.emby-highlight {outline:4px solid ${Config.highlightColor} !important; position:relative;}
/* 传统链接样式 */
.emby-jump-link {background:${Config.highlightColor}; border-radius:3px; padding:3px 6px; margin-top:5px; margin-bottom:3px}
.emby-jump-link a {color:white; text-decoration:none; display:block;}
`);
// 单例状态指示器
const Status = (() => {
let el, bar, timeout;
const debounce = (fn, ms) => {
let timer;
return (...args) => {
clearTimeout(timer);
timer = setTimeout(() => fn(...args), ms);
};
};
// 创建UI
const createUI = () => {
if (el) return;
el = document.createElement('div');
el.className = 'emby-jump-status-indicator';
el.innerHTML = `<span class="status-text">准备中...</span><div class="progress"><div class="progress-bar"></div></div><span class="close-btn">×</span>`;
document.body.appendChild(el);
bar = el.querySelector('.progress-bar');
el.querySelector('.close-btn').addEventListener('click', hide);
};
// 显示消息
const show = (msg, type = '') => {
createUI();
if (timeout) clearTimeout(timeout);
el.className = 'emby-jump-status-indicator ' + type;
el.querySelector('.status-text').textContent = msg;
el.style.opacity = '1';
};
// 隐藏
const hide = () => {
if (!el) return;
el.style.opacity = '0';
timeout = setTimeout(() => {
if (el && el.parentNode) el.parentNode.removeChild(el);
el = bar = null;
}, 300);
};
// 更新进度
const updateProgress = (current, total) => {
const percent = Math.min(Math.round((current / total) * 100), 100);
if (bar) bar.style.width = `${percent}%`;
show(`查询中: ${current}/${total} (${percent}%)`);
};
return {
show,
success: (msg, autoHide) => { show(msg, 'success'); if (autoHide) setTimeout(hide, 3000); },
error: (msg, autoHide) => { show(msg, 'error'); if (autoHide) setTimeout(hide, 5000); },
updateProgress,
updateProgressDebounced: debounce(updateProgress, 100),
hide
};
})();
// 设置面板
const SettingsUI = {
show() {
let panel = document.getElementById('emby-jump-settings-panel');
if (panel) { panel.style.display = 'block'; return; }
panel = document.createElement('div');
panel.id = 'emby-jump-settings-panel';
panel.className = 'emby-jump-settings-panel';
panel.innerHTML = `
<div class="emby-jump-settings-header">
<h3 style="margin:0">Emby 设置</h3>
<span class="emby-jump-settings-close">×</span>
</div>
<div class="emby-jump-settings-field">
<label for="emby-url">Emby 服务器地址</label>
<input type="text" id="emby-url" placeholder="例如: http://192.168.1.100:8096/" value="${Config.embyBaseUrl}">
<small style="color:#666">请确保包含http://或https://前缀和最后的斜杠 /</small>
</div>
<div class="emby-jump-settings-field">
<label for="emby-api">Emby API密钥</label>
<input type="text" id="emby-api" placeholder="在Emby设置中获取API密钥" value="${Config.embyAPI}">
</div>
<div class="emby-jump-settings-field">
<label for="highlight-color">高亮颜色</label>
<input type="color" id="highlight-color" value="${Config.highlightColor}">
</div>
<div class="emby-jump-settings-field">
<label for="max-requests">最大并发请求数</label>
<input type="number" id="max-requests" min="1" max="100" value="${Config.maxConcurrentRequests}">
<small style="color:#666">因为是本地请求,可以设置较大值</small>
</div>
<!-- 添加徽章设置项 -->
<div class="emby-jump-settings-field">
<label for="badge-size">徽章大小</label>
<select id="badge-size">
<option value="small" ${Config.badgeSize === 'small' ? 'selected' : ''}>小</option>
<option value="medium" ${Config.badgeSize === 'medium' ? 'selected' : ''}>中</option>
<option value="large" ${Config.badgeSize === 'large' ? 'selected' : ''}>大</option>
</select>
</div>
<div class="emby-jump-settings-field">
<label for="badge-color">徽章背景颜色</label>
<input type="color" id="badge-color" value="${Config.badgeColor}">
<small style="color:#666">背景颜色将与彩虹边框一起显示</small>
</div>
<div class="emby-jump-settings-field">
<label for="badge-text-color">徽章文字颜色</label>
<input type="color" id="badge-text-color" value="${Config.badgeTextColor}">
</div>
<div class="emby-jump-settings-buttons">
<button class="emby-jump-settings-cancel">取消</button>
<button class="emby-jump-settings-save">保存</button>
</div>
`;
document.body.appendChild(panel);
// 绑定事件
const closePanel = () => panel.style.display = 'none';
panel.querySelector('.emby-jump-settings-close').addEventListener('click', closePanel);
panel.querySelector('.emby-jump-settings-cancel').addEventListener('click', closePanel);
panel.querySelector('.emby-jump-settings-save').addEventListener('click', () => {
const url = document.getElementById('emby-url').value;
if (!url.match(/^https?:\/\/.+\/$/)) {
alert('请输入有效的Emby服务器地址,包含http://或https://前缀和最后的斜杠 /');
return;
}
Config.embyBaseUrl = url;
Config.embyAPI = document.getElementById('emby-api').value;
Config.highlightColor = document.getElementById('highlight-color').value;
Config.maxConcurrentRequests = parseInt(document.getElementById('max-requests').value, 10);
Config.badgeSize = document.getElementById('badge-size').value;
Config.badgeColor = document.getElementById('badge-color').value;
Config.badgeTextColor = document.getElementById('badge-text-color').value;
closePanel();
alert('设置已保存!请刷新页面以应用更改。');
});
panel.style.display = 'block';
}
};
// Emby API和请求控制
class EmbyAPI {
constructor() {
this.active = 0;
this.waiting = [];
this.total = 0;
this.completed = 0;
}
// 查询单个番号
async fetchData(code) {
if (!code) return { Items: [] };
try {
const url = `${Config.embyBaseUrl}emby/Users/${Config.embyAPI}/Items?api_key=${Config.embyAPI}&Recursive=true&IncludeItemTypes=Movie&SearchTerm=${encodeURIComponent(code.trim())}&Fields=Name,Id,ServerId`;
const response = await this.request(url);
const data = JSON.parse(response.responseText);
return data;
} catch (error) {
console.error(`查询数据出错 ${code}:`, error);
return { Items: [] };
}
}
// 批量查询
async batchQuery(codes) {
if (!codes || codes.length === 0) return [];
this.total = codes.length;
this.completed = 0;
const results = new Array(this.total);
return new Promise(resolve => {
const checkComplete = () => {
if (this.completed >= this.total && this.active === 0) {
const found = results.filter(r => r?.Items?.length > 0).length;
Status.success(`查询完成: 找到${found}个匹配项`, false);
resolve(results);
}
};
const processRequest = (index) => {
const code = codes[index];
this.active++;
Status.updateProgressDebounced(this.completed, this.total);
this.fetchData(code).then(result => {
results[index] = result;
this.active--;
this.completed++;
if (this.waiting.length > 0) {
processRequest(this.waiting.shift());
}
checkComplete();
}).catch(() => {
results[index] = null;
this.active--;
this.completed++;
if (this.waiting.length > 0) {
processRequest(this.waiting.shift());
}
checkComplete();
});
};
for (let i = 0; i < this.total; i++) {
if (this.active < Config.maxConcurrentRequests) {
processRequest(i);
} else {
this.waiting.push(i);
}
}
});
}
// 通用请求方法
request(url) {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: "GET",
url,
headers: { accept: "application/json" },
timeout: 10000,
onload: res => res.status >= 200 && res.status < 300 ? resolve(res) : reject(new Error(`HTTP错误: ${res.status}`)),
onerror: () => reject(new Error("请求错误")),
ontimeout: () => reject(new Error("请求超时"))
});
});
}
// 创建Emby链接元素
createLink(data) {
if (!data?.Items?.length) return null;
const item = data.Items[0];
const embyUrl = `${Config.embyBaseUrl}web/index.html#!/item?id=${item.Id}&serverId=${item.ServerId}`;
const el = document.createElement('div');
el.className = 'emby-jump-link';
el.innerHTML = `<a href="${embyUrl}" target="_blank"><b>跳转到emby👉</b></a>`;
return el;
}
// 创建Emby徽章元素
createBadge(data) {
if (!data?.Items?.length) return null;
const item = data.Items[0];
const embyUrl = `${Config.embyBaseUrl}web/index.html#!/item?id=${item.Id}&serverId=${item.ServerId}`;
const el = document.createElement('a');
el.className = 'emby-badge';
el.href = embyUrl;
el.target = '_blank';
el.textContent = 'Emby';
return el;
}
}
// 站点处理器基类
const BaseProcessor = {
init(api) {
this.api = api;
this.processed = new WeakSet();
return this;
},
// 处理列表项 - 徽章样式
async processItemsWithBadge(items) {
if (!items?.length) return;
Status.show(`正在收集番号: 共${items.length}个项目`);
// 收集番号
const toProcess = [];
const codes = [];
for (const item of items) {
if (this.processed.has(item)) continue;
this.processed.add(item);
const code = this.extractCode(item);
if (!code) continue;
// 查找第一个包含图片的元素
const imgContainer = this.findImgContainer(item);
if (!imgContainer) continue;
toProcess.push({ item, code, imgContainer });
codes.push(code);
}
if (codes.length > 0) {
const results = await this.api.batchQuery(codes);
// 处理结果 - 批量操作
const operations = [];
for (let i = 0; i < results.length; i++) {
if (i < toProcess.length && results[i]?.Items?.length > 0) {
const { item, imgContainer } = toProcess[i];
const badge = this.api.createBadge(results[i]);
if (badge) {
// 准备添加徽章
operations.push(() => {
// 确保图片容器是相对定位
if (window.getComputedStyle(imgContainer).position === 'static') {
imgContainer.style.position = 'relative';
}
// 添加高亮样式 - 使用outline避免影响布局
item.classList.add('emby-highlight');
// 添加徽章
imgContainer.appendChild(badge);
});
}
}
}
// 批量DOM操作 - 减少重排
if (operations.length > 0) {
requestAnimationFrame(() => {
operations.forEach(op => op());
});
}
}
},
// 处理列表项 - 传统链接样式
async processItemsWithLink(items) {
if (!items?.length) return;
Status.show(`正在收集番号: 共${items.length}个项目`);
// 收集番号
const toProcess = [];
const codes = [];
for (const item of items) {
if (this.processed.has(item)) continue;
this.processed.add(item);
const code = this.extractCode(item);
const element = this.getElement(item);
if (code && element) {
toProcess.push({ element, code });
codes.push(code);
}
}
if (codes.length > 0) {
const results = await this.api.batchQuery(codes);
const processedElements = [];
// 处理结果
for (let i = 0; i < results.length; i++) {
if (i < toProcess.length && results[i]?.Items?.length > 0) {
const { element } = toProcess[i];
const link = this.api.createLink(results[i]);
if (link) {
const target = element.parentNode || element;
// 高亮条目样式
let current = element;
const containerClasses = ['item', 'masonry-brick', 'grid-item', 'movie-list', 'post'];
while (current && current !== document.body) {
for (const className of containerClasses) {
if (current.classList?.contains(className)) {
current.style.cssText += `border:3px solid ${Config.highlightColor};background-color:${Config.highlightColor}22`;
break;
}
}
current = current.parentElement;
}
processedElements.push({
target,
link,
position: element.nextSibling
});
}
}
}
// 批量插入DOM
requestAnimationFrame(() => {
processedElements.forEach(({ target, link, position }) => {
target.insertBefore(link, position);
});
});
}
},
// 主处理函数
async process() {
const items = document.querySelectorAll(this.listSelector);
if (items.length > 0) {
// 始终使用徽章样式
await this.processItemsWithBadge(items);
}
await this.processDetailPage();
this.setupObserver();
},
// 查找图片容器
findImgContainer(item) {
// 不同网站的图片容器选择器
const imgSelectors = [
'.img', // supjav
'a.movie-box', // javbus
'.cover', // javdb
'img'
];
for (const selector of imgSelectors) {
const imgContainer = item.querySelector(selector);
if (imgContainer) return imgContainer;
}
// 找不到合适的容器,尝试找第一个包含图片的元素
return item.querySelector('a') || item;
},
// 观察器设置
setupObserver() {
let pending = [];
let timer = null;
const processMutations = () => {
const newElements = [];
for (const mutation of pending) {
if (mutation.type === 'childList') {
for (const node of mutation.addedNodes) {
if (node.nodeType !== 1) continue;
if (node.matches?.(this.listSelector)) {
newElements.push(node);
}
if (node.querySelectorAll) {
node.querySelectorAll(this.listSelector).forEach(el => {
newElements.push(el);
});
}
}
}
}
if (newElements.length > 0) {
// 始终使用徽章样式
this.processItemsWithBadge(newElements);
}
pending = [];
timer = null;
};
new MutationObserver(mutations => {
pending.push(...mutations);
if (!timer) timer = setTimeout(processMutations, 300);
}).observe(document.body, { childList: true, subtree: true });
}
};
// 站点处理器
const Processors = {
javbus: Object.assign(Object.create(BaseProcessor), {
listSelector: '.item.masonry-brick, #waterfall .item',
extractCode: item => item.querySelector('.item date')?.textContent?.trim(),
getElement: item => item.querySelector('.item date'),
async processDetailPage() {
// 防止重复处理
if (document.querySelector('.emby-jump-link, .emby-badge')) return;
const infoElement = document.querySelector('.col-md-3.info p');
if (!infoElement) return;
const spans = infoElement.querySelectorAll('span');
if (spans.length > 1) {
const code = spans[1].textContent?.trim();
if (code) {
Status.show('查询中...');
const data = await this.api.fetchData(code);
if (data.Items?.length > 0) {
const link = this.api.createLink(data);
if (link) {
spans[1].parentNode.insertBefore(link, spans[1].nextSibling);
Status.success('找到匹配项', false);
}
} else {
Status.error('未找到匹配项', false);
}
}
}
}
}),
javdb: Object.assign(Object.create(BaseProcessor), {
listSelector: '.movie-list .item, .grid-item',
extractCode: item => item.querySelector('.video-title strong')?.textContent?.trim(),
getElement: item => item.querySelector('.video-title strong'),
async processDetailPage() {
// 防止重复处理
if (document.querySelector('.emby-jump-link, .emby-badge')) return;
const detailElement = document.querySelector('body > section > div > div.video-detail > h2 > strong') ||
document.querySelector('.video-detail h2 strong');
if (!detailElement) return;
const code = detailElement.textContent.trim().split(' ')[0];
if (code) {
Status.show('查询中...');
const data = await this.api.fetchData(code);
if (data.Items?.length > 0) {
const link = this.api.createLink(data);
if (link) {
detailElement.parentNode.insertBefore(link, detailElement.nextSibling);
Status.success('找到匹配项', false);
}
} else {
Status.error('未找到匹配项', false);
}
}
}
}),
supjav: Object.assign(Object.create(BaseProcessor), {
listSelector: '.post',
extractCode: function (item) {
// 从标题中提取番号
const title = item.querySelector('h3 a')?.textContent?.trim();
if (!title) return null;
// 匹配番号格式
const match = title.match(/([a-zA-Z0-9]+-\d+)/i);
return match ? match[1] : null;
},
getElement: function (item) {
return item.querySelector('h3 a');
},
async processDetailPage() {
// 防止重复处理 - 限制在主内容区域
if (document.querySelector('.video-wrap .emby-jump-link, .video-wrap .emby-badge')) return;
// 在详情页面查找标题 - 只处理主标题,不处理相关推荐
const titleElement = document.querySelector('.video-wrap .archive-title h1');
if (!titleElement) return;
const title = titleElement.textContent.trim();
const match = title.match(/([a-zA-Z0-9]+-\d+)/i);
if (!match) return;
const code = match[1];
if (code) {
Status.show('查询中...');
const data = await this.api.fetchData(code);
if (data.Items?.length > 0) {
const link = this.api.createLink(data);
if (link) {
titleElement.parentNode.insertBefore(link, titleElement.nextSibling);
Status.success('找到匹配项', false);
}
} else {
Status.error('未找到匹配项', false);
}
}
}
}),
sehuatang: Object.assign(Object.create(BaseProcessor), {
listSelector: '',
async process() {
// 防止重复处理
if (document.querySelector('.emby-jump-link, .emby-badge')) return;
const title = document.title.trim();
const codes = this.extractCodes(title);
if (codes.length > 0) {
Status.show(`找到${codes.length}个可能的番号,开始查询...`);
const results = await this.api.batchQuery(codes);
let foundAny = false;
for (const data of results) {
if (data?.Items?.length > 0) {
const container = document.querySelector('#thread_subject') ||
document.querySelector('h1.ts') ||
document.querySelector('h1');
if (container) {
const link = this.api.createLink(data);
if (link) {
container.parentNode.insertBefore(link, container.nextSibling);
foundAny = true;
}
}
}
}
if (foundAny) Status.success('找到匹配项', false);
else Status.error('未找到匹配项', false);
}
},
extractCodes(title) {
if (!title) return [];
const patterns = [
/([a-zA-Z]{2,15})[-\s]?(\d{2,15})/i,
/FC2[-\s]?PPV[-\s]?(\d{6,7})/i
];
const results = [];
for (const pattern of patterns) {
const match = title.match(pattern);
if (match) {
if (match[2]) results.push(`${match[1]}-${match[2]}`);
else if (match[1]) results.push(match[0]);
}
}
return results;
}
})
};
// 站点检测
function detectSite() {
if (location.hostname.includes('javbus') || document.querySelector('footer')?.textContent?.includes('JavBus')) {
return 'javbus';
}
if (location.hostname.includes('javdb') || document.querySelector('#footer')?.textContent?.includes('javdb')) {
return 'javdb';
}
if (location.hostname.includes('supjav') || document.title.includes('SupJav')) {
return 'supjav';
}
if (location.hostname.includes('sehuatang') || document.querySelector('#flk')?.textContent?.includes('色花堂')) {
return 'sehuatang';
}
return null;
}
// 主函数
async function main() {
console.log('Emby跳转脚本启动 (极简版)');
// 注册菜单命令
GM_registerMenuCommand("Emby 设置", () => SettingsUI.show());
// 检查API配置
if (!Config.isValid()) {
Status.error('配置无效', true);
setTimeout(() => {
alert('请先设置您的Emby服务器地址和API密钥');
SettingsUI.show();
}, 500);
return;
}
Status.show('正在初始化...');
// 检测当前站点
const site = detectSite();
if (!site) {
Status.error('未识别到支持的站点', false);
return;
}
Status.show(`检测到站点: ${site},开始处理...`);
// 创建并执行站点处理器
const processor = Processors[site].init(new EmbyAPI());
if (processor) await processor.process();
}
// 页面加载完成后启动
if (document.readyState === 'loading') {
document.addEventListener('DOMContentLoaded', main);
} else {
main();
}
})();