// ==UserScript== // @name ChromeXt脚本菜单提取器 // @namespace ChromeXt.GodMode // @version 1.2 // @description 通过劫持Array原型链,提取ChromeXt脚本菜单,会在页面右侧生成一个"CX"的悬浮图标开启。 // @author Aloazny // @match *://*/* // @icon  // @homepageURL https://scriptcat.org/zh-CN/users/157252 // @grant none // @license GPL-3.0 // @run-at document-start // ==/UserScript== (function() { 'use strict'; let captured = new Set(); const methods = ['push', 'splice', 'fill', 'unshift']; methods.forEach(method => { const original = Array.prototype[method]; Array.prototype[method] = function(...args) { const result = original.apply(this, args); args.forEach(arg => { if (arg && typeof arg === 'object' && 'title' in arg && 'listener' in arg) { captured.add(arg); } }); return result; }; }); function deepScan() { Object.getOwnPropertySymbols(window).forEach(sym => { try { const val = window[sym]; if (val && val.commands && Array.isArray(val.commands)) { val.commands.forEach(cmd => captured.add(cmd)); } } catch (e) {} }); return Array.from(captured); } const initUI = () => { if (document.getElementById('cx-v5-container')) return; const host = document.createElement('div'); host.id = 'cx-v5-container'; host.style = 'position:fixed;top:50%;right:0;z-index:2147483647;user-select:none;'; const shadow = host.attachShadow({mode: 'open'}); document.body.appendChild(host); const style = document.createElement('style'); style.textContent = ` :host { --cx-bg: rgba(255, 255, 255, 0.85); --cx-item-bg: rgba(249, 249, 249, 0.7); --cx-text: #1a1a1a; --cx-accent: #0078d4; --cx-border: rgba(0, 0, 0, 0.1); } @media (prefers-color-scheme: dark) { :host { --cx-bg: rgba(32, 32, 32, 0.85); --cx-item-bg: rgba(45, 45, 45, 0.7); --cx-text: #ffffff; --cx-accent: #60cdff; --cx-border: rgba(255, 255, 255, 0.1); } } #btn { position: absolute; right: 0; transform: translateY(-50%); width: 32px; height: 50px; background: var(--cx-accent); color: #fff; border-radius: 12px 0 0 12px; display: flex; align-items: center; justify-content: center; cursor: pointer; font: bold 12px "Segoe UI", system-ui; box-shadow: 0 4px 15px rgba(0,0,0,0.2); transition: width 0.3s cubic-bezier(0.1, 0.9, 0.2, 1); backdrop-filter: blur(8px); opacity: 0.9; } #btn:hover { width: 45px; opacity: 1; } #btn:active { transform: translateY(-50%) scale(0.9); } .panel-overlay { position: fixed; top: 0; left: 0; width: 100vw; height: 100vh; background: rgba(0,0,0,0.4); display: flex; align-items: center; justify-content: center; backdrop-filter: blur(5px); animation: fadeIn 0.2s ease; } .content { width: 85%; max-width: 420px; max-height: 70vh; background: var(--cx-bg); border: 1px solid var(--cx-border); border-radius: 12px; padding: 16px; box-shadow: 0 20px 40px rgba(0,0,0,0.3); backdrop-filter: blur(25px) saturate(180%); display: grid; grid-template-columns: 1fr; gap: 8px; overflow-y: auto; scrollbar-width: none; -ms-overflow-style: none; animation: slideUp 0.3s cubic-bezier(0.1, 0.9, 0.2, 1); } .content::-webkit-scrollbar { display: none; } @media (min-width: 768px) { .content { max-width: 600px; grid-template-columns: 1fr 1fr; } } .item { padding: 14px; background: var(--cx-item-bg); color: var(--cx-text); border-radius: 8px; font: 500 14px "Segoe UI", system-ui; text-align: center; cursor: pointer; border: 1px solid var(--cx-border); transition: all 0.15s; display: flex; align-items: center; justify-content: center; } .item:hover { background: var(--cx-accent); color: #fff; border-color: transparent; transform: translateY(-2px); } .item:active { transform: scale(0.96); } .close-btn { grid-column: 1 / -1; position: sticky; bottom: -16px; margin: 8px -16px -16px -16px; padding: 12px; background: var(--cx-bg); border-top: 1px solid var(--cx-border); text-align: center; color: var(--cx-accent); font-weight: bold; font-size: 13px; cursor: pointer; border-radius: 0 0 12px 12px; } @keyframes fadeIn { from { opacity: 0; } } @keyframes slideUp { from { transform: translateY(20px); opacity: 0; } } `; shadow.appendChild(style); const btn = document.createElement('div'); btn.id = 'btn'; btn.textContent = 'CX'; shadow.appendChild(btn); let isDragging = false, startY, startTop; btn.ontouchstart = (e) => { isDragging = true; startY = e.touches[0].clientY; startTop = host.offsetTop; btn.style.transition = 'none'; }; window.ontouchmove = (e) => { if (!isDragging) return; let moveY = e.touches[0].clientY - startY; host.style.top = (startTop + moveY) + 'px'; }; window.ontouchend = () => { if (!isDragging) return; isDragging = false; btn.style.transition = 'width 0.3s cubic-bezier(0.1, 0.9, 0.2, 1)'; let finalTop = Math.max(50, Math.min(window.innerHeight - 50, host.offsetTop)); host.style.top = finalTop + 'px'; }; btn.onclick = () => { if (isDragging) return; const menus = deepScan(); if (menus.length === 0) return alert("未发现脚本菜单。"); renderPanel(menus, shadow); }; }; function renderPanel(menus, shadow) { if (shadow.querySelector('.panel-overlay')) return; const overlay = document.createElement('div'); overlay.className = 'panel-overlay'; const content = document.createElement('div'); content.className = 'content'; menus.forEach((cmd) => { const item = document.createElement('div'); item.className = 'item'; item.innerText = typeof cmd.title === 'function' ? cmd.title() : cmd.title; item.onclick = (e) => { e.stopPropagation(); cmd.listener(); overlay.remove(); }; content.appendChild(item); }); const close = document.createElement('div'); close.className = 'close-btn'; close.innerText = "关闭菜单"; close.onclick = () => overlay.remove(); content.appendChild(close); overlay.appendChild(content); overlay.onclick = (e) => { if(e.target === overlay) overlay.remove(); }; shadow.appendChild(overlay); } if (document.readyState !== 'loading') initUI(); else window.addEventListener('DOMContentLoaded', initUI); })();