您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
基于FSU/Enhancer 的永动机滚卡助手
// ==UserScript== // @name pandaSBC // @namespace http://tampermonkey.net/ // @version 2.1.5 // @description 基于FSU/Enhancer 的永动机滚卡助手 // @license MIT // @match https://www.ea.com/ea-sports-fc/ultimate-team/web-app/* // @match https://www.easports.com/*/ea-sports-fc/ultimate-team/web-app/* // @match https://www.ea.com/*/ea-sports-fc/ultimate-team/web-app/* // @match https://signin.ea.com/* // @grant GM_getValue // @grant GM_setValue // @grant GM_addStyle // @grant GM_deleteValue // @run-at document-end // ==/UserScript== /* * 脚本使用免责声明(Script Usage Disclaimer) * * 本脚本仅供个人学习和研究使用,不得用于任何商业或非法用途。 * 作者对因使用本脚本造成的任何直接或间接损失、损害或法律责任不承担任何责任。 * 使用者须自行评估风险并对其行为负责。请务必遵守目标网站的用户协议和相关法律法规。 * * This script is provided “as is,” without warranty of any kind, express or implied. * The author shall not be liable for any damages arising out of the use of this script. * Use at your own risk and in compliance with the target site’s terms of service and applicable laws. */ (function () { 'use strict'; const GUIDE_SHOWN_KEY = 'guide_shown_v1'; const PandaSBC = (() => { const config = { version: '2.1.5', DEFAULT_TIMEOUT: 15000, LOGIN_BUTTON_CHECK_TIMEOUT: 60, UI: { SP_STABLE_FOR: 300, SP_FILL_SUCCESS_TIME: 1000, }, RANGES: [ { range: [82, 86], type: 'all' }, { range: [87, 88], type: 'all' }, { range: [89, 96], type: 'all' }, { type: 'storage' }, ], STORAGE_MAX: 100, MIN_RATING_KEY: 'minRating', DEFAULT_MIN_RATING: 98, HIGH_RATED_POPUP_THRESHOLD: 98, MAX_RATING_TOTW: 90, MAX_RATING_NORMAL: 98, targetKeywords: [ '89 阵容变异', 'TOTW 升级', '10 名 85+ 升级', '10 名 84+ 升级', ], blacklistKeywords: [ '可交易', '青铜升级', '白银升级', '黄金升级', '混合联赛升级' ], PRO: { LOWBIN_RANGE: [75, 80], LOWBIN_SAFE_MIN: 20, TOTW_RANGE: [83, 84], TOTW_NEED: 11, TOTW_SAFE_MIN: 3, LOOP_FAIL_BACKOFF_MS: 3 * 60 * 1000, TRY_LOOP_FAILS_BEFORE_FALLBACK: 2, AUX_MAX_CONSEC: 2, LOGIN_EMAIL: '', LOGIN_PASSWORD: '', }, }; const CFG_KEYS = { SP_STABLE_FOR: 'cfg_SP_STABLE_FOR', SP_FILL_SUCCESS_TIME: 'cfg_SP_FILL_SUCCESS_TIME', HIGH_RATED_POPUP_THRESHOLD: 'cfg_HIGH_RATED_POPUP_THRESHOLD', MAX_RATING_TOTW: 'cfg_MAX_RATING_TOTW', MAX_RATING_NORMAL: 'cfg_MAX_RATING_NORMAL', TOTW_SAFE_MIN: 'cfg_TOTW_SAFE_MIN', AUTO_RESTART_ON_STOP: 'cfg_AUTO_RESTART_ON_STOP', }; const DONATE = { ALIPAY_QR: '', WECHAT_QR: '', }; const state = { page: unsafeWindow, running: false, runningTask: '', isStopping: false, abortCtrl: null, currentTaskDone: Promise.resolve(), FILTERED_SETS: [], selectedLoopSetId: null, selectedDoSbcSetId: null, minRating: Number(GM_getValue(config.MIN_RATING_KEY, config.DEFAULT_MIN_RATING)) || config.DEFAULT_MIN_RATING, enableHandleDuplicate: !!GM_getValue('enableHandleDuplicate', false), _hiRatedPlayers: [], _loopFailStrike: 0, _lastLoopFailAt: 0, _xhrPromiseList: [], _xhrHooked: false, _tryLoopFailStrike2: 0, _tryLoopFbUsedForStrike: false, _auxActGuard: { lastAct: null, consec: 0 }, btn: { loop: null, open: null, do: null, auto: null }, autoRestartOnStop: !!GM_getValue(CFG_KEYS.AUTO_RESTART_ON_STOP, false), }; const log = { d: (...a) => console.debug('[pandaSBC]', ...a), i: (...a) => console.info('[pandaSBC]', ...a), w: (...a) => console.warn('[pandaSBC]', ...a), e: (...a) => console.error('[pandaSBC]', ...a), }; class AbortedError extends Error { constructor(msg = 'Aborted') { super(msg); this.name = 'AbortedError'; } } class TimeoutError extends Error { constructor(msg = 'Timeout') { super(msg); this.name = 'TimeoutError'; } } class ExpectError extends Error { constructor(msg = 'Unexpected') { super(msg); this.name = 'ExpectError'; } } const isAbort = (e) => e && (e.name === 'AbortedError' || /Aborted/.test(String(e.message))); const isHighRatedError = (e) => e && e.name === 'ExpectError' && /HighRatedInSquad/.test(String(e.message || '')); const util = { sleep: (ms) => { const s = state.abortCtrl?.signal; return new Promise((resolve, reject) => { if (s?.aborted) return reject(new AbortedError()); const onAbort = () => { clearTimeout(t); reject(new AbortedError()); }; const t = setTimeout(() => { s?.removeEventListener?.('abort', onAbort); resolve(); }, ms); s?.addEventListener?.('abort', onAbort, { once: true }); }); }, nextPaint: () => new Promise((r) => requestAnimationFrame(() => requestAnimationFrame(r))), abortPoint() { const s = state.abortCtrl?.signal; if (s?.aborted) throw new AbortedError(); if (state.isStopping || !state.running) throw new AbortedError(); return s; }, withAbort(fn) { return function (...args) { const s = state.abortCtrl?.signal; if (!s) return fn.apply(this, args); if (s.aborted) return Promise.reject(new AbortedError()); const p = fn.apply(this, args); const ap = new Promise((_, rej) => { if (s.aborted) return rej(new AbortedError()); s.addEventListener('abort', () => rej(new AbortedError()), { once: true }); }); return Promise.race([p, ap]); }; }, retry: async (fn, { tries = 2, delay = 400, factor = 1.6 } = {}) => { let err; for (let i = 0; i < tries; i++) { try { return await fn(); } catch (e) { if (isAbort(e)) throw e; err = e; await util.sleep(Math.floor(delay)); delay *= factor; } } throw err; }, clamp(n, min, max) { return Math.max(min, Math.min(max, n)); }, once(fn) { let done = false; let val; return (...a) => { if (done) return val; val = fn(...a); done = true; return val; }; }, }; const dom = { simulateClick(el) { if (!el) throw new ExpectError('simulateClick: no element'); const r = el.getBoundingClientRect(); ['mousedown', 'mouseup', 'click'].forEach((t) => el.dispatchEvent( new MouseEvent(t, { bubbles: true, cancelable: true, clientX: r.left + r.width / 2, clientY: r.top + r.height / 2, button: 0, }), ), ); }, isVisible(el) { if (!el || !el.isConnected) return false; const r = el.getBoundingClientRect(); if (r.width <= 0 || r.height <= 0) return false; const s = getComputedStyle(el); return s.visibility !== 'hidden' && s.display !== 'none'; }, isInteractable(el) { if (!dom.isVisible(el)) return false; let n = el; while (n && n !== document) { const s = getComputedStyle(n); if (s.visibility === 'hidden' || s.display === 'none' || s.pointerEvents === 'none') { return false; } n = n.parentElement || n.ownerDocument?.host; } const r = el.getBoundingClientRect(); const pts = [ [r.left + r.width / 2, r.top + r.height / 2], [r.left + r.width * 0.8, r.top + r.height / 2], [r.left + r.width * 0.2, r.top + r.height / 2], [r.left + r.width / 2, r.top + r.height * 0.3], [r.left + r.width / 2, r.top + r.height * 0.7], ]; for (const [x, y] of pts) { const top = document.elementFromPoint(x, y); if (top === el || el.contains(top)) return true; } return false; }, waitForElement: util.withAbort( async (fnOrSelector, timeout = config.DEFAULT_TIMEOUT, opts = {}) => { let { root = document, subtree = true, returnAll = false, strict = false, stableFor = 16, preferLast = false, signal = state.abortCtrl?.signal, } = opts; root = typeof root === 'string' ? document.querySelector(root) || document : root; const pass = strict ? dom.isInteractable : dom.isVisible; const getCandidates = () => { if (typeof fnOrSelector === 'string') { const list = root.querySelectorAll(fnOrSelector); const arr = list ? Array.from(list) : []; return preferLast ? arr.reverse() : arr; } else if (typeof fnOrSelector === 'function') { const res = fnOrSelector(); if (!res) return []; if (res instanceof Element) return [res]; if (NodeList.prototype.isPrototypeOf(res) || Array.isArray(res)) { const arr = Array.from(res); return preferLast ? arr.reverse() : arr; } return []; } return []; }; const watchStability = (el, onStable, onAbort) => { let stableTimer = null; const clearStableTimer = () => { if (stableTimer) { clearTimeout(stableTimer); stableTimer = null; } }; const startStableTimerIfPass = () => { clearStableTimer(); if (!el || !el.isConnected) return; if (!pass(el)) return; if (stableFor <= 32) { requestAnimationFrame(() => requestAnimationFrame(() => { if (el && el.isConnected && pass(el)) onStable(el); }), ); return; } stableTimer = setTimeout(() => onStable(el), stableFor); }; const mos = []; let node = el; while (node && node !== document && node.nodeType === 1) { const mo = new MutationObserver(startStableTimerIfPass); mo.observe(node, { attributes: true, attributeFilter: ['class', 'style', 'hidden', 'aria-hidden'], childList: true, subtree: false, }); mos.push(mo); node = node.parentElement || node.ownerDocument?.host; } const ro = new ResizeObserver(startStableTimerIfPass); try { ro.observe(el); } catch { } let io = null; try { io = new IntersectionObserver(startStableTimerIfPass, { threshold: [0, 0.01, 0.5, 1], }); io.observe(el); } catch { } const onTransEnd = startStableTimerIfPass; el.addEventListener('transitionend', onTransEnd, { passive: true }); el.addEventListener('animationend', onTransEnd, { passive: true }); startStableTimerIfPass(); const unwatch = () => { clearStableTimer(); mos.forEach((m) => m.disconnect()); try { ro.disconnect(); } catch { } try { io && io.disconnect(); } catch { } el.removeEventListener('transitionend', onTransEnd); el.removeEventListener('animationend', onTransEnd); }; if (signal) { const abortFn = () => { unwatch(); onAbort && onAbort(); }; if (signal.aborted) abortFn(); else signal.addEventListener('abort', abortFn, { once: true }); } return unwatch; }; return await new Promise((resolve, reject) => { let settled = false; let timeoutId = null; let rootObserver = null; let unwatchEl = null; const resolveOnce = (val) => { if (settled) return; settled = true; try { rootObserver && rootObserver.disconnect(); } catch { } try { unwatchEl && unwatchEl(); } catch { } if (timeoutId) clearTimeout(timeoutId); resolve(val); }; const tryPick = () => { if (settled) return; if (unwatchEl) { unwatchEl(); unwatchEl = null; } const list = getCandidates(); if (returnAll) { const okList = list.filter(pass); if (okList.length) return resolveOnce(okList); } else { const el = list.find(pass) || list[0]; if (el) { unwatchEl = watchStability( el, (stableEl) => resolveOnce(stableEl), () => resolveOnce(false), ); } } }; if (timeout > 0) timeoutId = setTimeout(() => resolveOnce(false), timeout); if (signal) { if (signal.aborted) return reject(new AbortedError()); signal.addEventListener('abort', () => reject(new AbortedError()), { once: true }); } tryPick(); rootObserver = new MutationObserver(() => requestAnimationFrame(tryPick)); rootObserver.observe(root, { childList: true, subtree, attributes: true, attributeFilter: ['class', 'style', 'hidden', 'aria-hidden'], }); }); }, ), waitGone: util.withAbort( (selector, timeout = config.DEFAULT_TIMEOUT, { interval = 200 } = {}) => new Promise((resolve, reject) => { const start = Date.now(); let timer = null; const step = () => { if (state.abortCtrl?.signal?.aborted) { if (timer) clearTimeout(timer); return reject(new AbortedError()); } if (!document.querySelector(selector)) { if (timer) clearTimeout(timer); return resolve(true); } if (Date.now() - start > timeout) { if (timer) clearTimeout(timer); return resolve(false); } timer = setTimeout(step, interval); }; step(); }), ), clickIfExists: util.withAbort( async (selectorOrFn, timeout = 2000, clickDelay = 200, opt = {}, ignoreError = false) => { const s = state.abortCtrl?.signal; if (s?.aborted) throw new AbortedError(); const el = await dom.waitForElement(selectorOrFn, timeout, opt); if (!el) { if (s?.aborted) throw new AbortedError(); if (ignoreError) return false; throw new TimeoutError(`clickIfExists: "${selectorOrFn}" not found in ${timeout}ms`); } try { if (clickDelay > 0) await util.nextPaint(); dom.simulateClick(el); return el; } catch (e) { if (ignoreError) return false; throw e; } }, ), }; const ea = { get ctrl() { try { return getAppMain() .getRootViewController() .getPresentedViewController() .getCurrentViewController() .getCurrentController(); } catch { return null; } }, get squad() { const c = ea.ctrl; return c && c._squad ? c._squad : null; }, get slots() { const sq = ea.squad; return sq ? sq.getPlayers?.() || sq._players || [] : []; }, getFilledCount() { return ea.slots.filter((s) => s && s._item && Number(s._item.definitionId) > 0).length; }, getUnassignedController() { const ctl = ea.ctrl; if (!ctl || !ctl.childViewControllers) return null; return Array.from(ctl.childViewControllers).find( (c) => c.className && c.className.includes('UTUnassigned') && c.className.includes('Controller'), ); }, waitController: util.withAbort( (name, timeout = config.DEFAULT_TIMEOUT, { pollInterval = 800 } = {}) => new Promise((resolve, reject) => { const start = Date.now(); let timer = null; const tick = () => { if (state.abortCtrl?.signal?.aborted) { clearTimeout(timer); return reject(new AbortedError()); } try { const ctrl = ea.ctrl; if (ctrl?.constructor?.name === name) { clearTimeout(timer); return resolve(ctrl); } } catch { } if (Date.now() - start > timeout) { clearTimeout(timer); return reject(new TimeoutError(`等待 ${name} 超时`)); } timer = setTimeout(tick, pollInterval); }; tick(); }), ), waitLoadingEndOnce: util.withAbort( () => new Promise((res) => { const shield = typeof gClickShield === 'object' ? gClickShield : null; if (shield && !shield.isShowing()) return res(); EAClickShieldView._onLoadingEndQueue = EAClickShieldView._onLoadingEndQueue || []; EAClickShieldView._onLoadingEndQueue.push(res); }), ), waitAllLoadingEnd: util.withAbort(async (stableDelay = 600, timeout = 10000) => { const shield = typeof gClickShield === 'object' ? gClickShield : null; const start = Date.now(); while (true) { if (shield && shield.isShowing()) { await util.sleep(300); } else { let stable = true; const t0 = Date.now(); while (Date.now() - t0 < stableDelay) { if (shield && shield.isShowing()) { stable = false; break; } await util.sleep(100); } if (stable) return true; } if (Date.now() - start > timeout) { log.w('[waitAllLoadingEnd] timeout'); return false; } } }), hookXHR: util.once(() => { if (state._xhrHooked) return true; state._xhrHooked = true; state.page._xhrPromiseList = state._xhrPromiseList; const originOpen = XMLHttpRequest.prototype.open; XMLHttpRequest.prototype.open = function (m, url, ...args) { this._xhrFlag = { method: String(m || '').toUpperCase(), url: String(url || '') }; return originOpen.apply(this, [m, url, ...args]); }; const originSend = XMLHttpRequest.prototype.send; XMLHttpRequest.prototype.send = function (body) { this.addEventListener('load', function () { const list = state.page._xhrPromiseList || []; for (const item of list) { if ( this._xhrFlag && this._xhrFlag.url.includes(item.apiPath) && (!item.method || this._xhrFlag.method === item.method.toUpperCase()) ) { if (this.status === 401) { item.status401Count = (item.status401Count || 0) + 1; if (item.status401Count >= 2 && !item.resolved) { item.resolved = true; clearTimeout(item._timer); item.resolve(null); } continue; } if (!item.resolved) { item.resolved = true; clearTimeout(item._timer); try { item.resolve(JSON.parse(this.responseText)); } catch { item.resolve(null); } } } } state.page._xhrPromiseList = (state.page._xhrPromiseList || []).filter((i) => !i.resolved); }); return originSend.apply(this, arguments); }; log.i('[hookXHR] Hooked'); return true; }), waitRequest: util.withAbort((apiPath, method, timeout = 15000) => { return new Promise((resolve, reject) => { ea.hookXHR(); const item = { apiPath, method, resolve: (v) => { cleanup(); resolve(v); }, status401Count: 0, resolved: false, _timer: null, }; const cleanup = () => { if (item.resolved) return; item.resolved = true; if (item._timer) clearTimeout(item._timer); state.page._xhrPromiseList = (state.page._xhrPromiseList || []).filter((x) => x !== item); }; item._timer = setTimeout(() => { if (!item.resolved) item.resolve(null); }, timeout); if (!Array.isArray(state.page._xhrPromiseList)) state.page._xhrPromiseList = []; state.page._xhrPromiseList.push(item); const s = state.abortCtrl?.signal; if (s) { if (s.aborted) { cleanup(); return reject(new AbortedError()); } s.addEventListener( 'abort', () => { cleanup(); reject(new AbortedError()); }, { once: true }, ); } }); }), hookRepositories: util.once(() => { try { const domain = repositories?.Item; if (!state.page.repositories || !domain) return false; if (domain._statsHooked) return true; const safeUpdate = typeof ui.updateStatsUI === 'function' ? ui.updateStatsUI : () => { }; const debouncedUpdate = ((fn) => { let t = null; return () => { clearTimeout(t); t = setTimeout(fn, 100); }; })(safeUpdate); const hook = (obj, methods) => { methods.forEach((name) => { if (!obj || typeof obj[name] !== 'function' || obj[name]._pandaHooked) return; const orig = obj[name]; obj[name] = function (...args) { const ret = orig.apply(this, args); try { debouncedUpdate(); } catch { } return ret; }; obj[name]._pandaHooked = true; }); }; hook(domain, ['add', 'remove', 'update', 'reset', 'set']); hook(domain.storage || {}, ['set', 'remove', 'reset', 'add', 'update']); hook(domain.club || {}, ['add', 'remove', 'update', 'reset', 'set']); domain._statsHooked = true; safeUpdate(); return true; } catch (e) { log.w('[hookRepositories] failed', e); return false; } }), hookEventsPopup: util.once(() => { const events = state.page.events; if (!events || typeof events.popup !== 'function') return false; if (events.popup._isPatched) return true; const interceptMap = { 珍贵球员: 44408, 快速任务: 2 }; const _orig = events.popup; events.popup = function ( title, message, callback, buttonOptions, inputPlaceholder, inputValue, inputEnabled, extraNode, ) { if (typeof title === 'string') { for (let key in interceptMap) { if (title.includes(key)) { const code = interceptMap[key]; return callback(code); } } } return _orig.call( this, title, message, callback, buttonOptions, inputPlaceholder, inputValue, inputEnabled, extraNode, ); }; events.popup._isPatched = true; log.i('[hookEventsPopup] Hooked'); return true; }), hookLoadingEnd: util.once(() => { if (EAClickShieldView._hookedForLoadingEnd) return true; const oldHideShield = EAClickShieldView.prototype.hideShield; EAClickShieldView.prototype.hideShield = function () { oldHideShield.apply(this, arguments); if (!this.isShowing()) { if (Array.isArray(EAClickShieldView._onLoadingEndQueue)) { for (const fn of EAClickShieldView._onLoadingEndQueue) { try { fn(); } catch { } } EAClickShieldView._onLoadingEndQueue = []; } } }; EAClickShieldView._onLoadingEndQueue = []; EAClickShieldView._hookedForLoadingEnd = true; log.i('[hookLoadingEnd] Hooked'); return true; }), ensureHooks() { ea.hookXHR(); ea.hookEventsPopup(); ea.hookLoadingEnd(); ea.hookRepositories(); }, }; const sbc = { _lastMainFailReason: null, getMainFailReason() { return sbc._lastMainFailReason || 'need_89'; }, sel: { rptBtn: () => Array.from(document.querySelectorAll('button.btn-standard.mini.call-to-action')).find( (b) => b.textContent.trim() === '重复球员填充阵容', ), addBtn: '.ut-image-button-control.btnAction.add', searchBtn: '.ut-image-button-control.fsu-eligibilitysearch', canvas: '.ut-squad-pitch-view--canvas', }, async addPlayer() { await dom.waitForElement(sbc.sel.rptBtn, 5000, { root: '.ut-navigation-container-view--content', strict: true, }); if (!(await dom.waitForElement('.ut-image-button-control.filter-btn.custom-player-add', 5000))) return; const maxTries = 3; const waitCountStable = async ({ baseCount, stableFor = 1000, timeout = 6000, poll = 150, }) => { const t0 = Date.now(); let last = ea.getFilledCount(); let lastAt = Date.now(); while (Date.now() - t0 <= timeout) { util.abortPoint(); await util.sleep(poll); const cur = ea.getFilledCount(); if (cur !== last) { last = cur; lastAt = Date.now(); } if (last >= baseCount + 1 && Date.now() - lastAt >= stableFor) return true; } return false; }; for (let attempt = 1; attempt <= maxTries; attempt++) { util.abortPoint(); const baseCount = ea.getFilledCount(); const ok1 = await dom.clickIfExists( sbc.sel.searchBtn, 5000, 0, { strict: false, stableFor: config.UI.SP_STABLE_FOR }, true, ); if (!ok1) { await dom.clickIfExists(sbc.sel.canvas, 800, 0, { strict: false }, true); continue; } await ea.waitAllLoadingEnd(); const ok2 = await dom.clickIfExists( sbc.sel.addBtn, 10000, 0, { strict: false, stableFor: config.UI.SP_STABLE_FOR }, true, ); if (!ok2) { await dom.clickIfExists(sbc.sel.canvas, 800, 0, { strict: false }, true); continue; } await ea.waitAllLoadingEnd(); await dom.clickIfExists(sbc.sel.canvas, 200, 0, { strict: false }, true); const ok = await waitCountStable({ baseCount, stableFor: config.UI.SP_FILL_SUCCESS_TIME, timeout: 6000, }); if (ok) return; if (attempt < maxTries) { await dom.clickIfExists(sbc.sel.canvas, 800, 0, { strict: false }, true); } else { throw new ExpectError('色卡添加失败'); } } }, collectHiRated(items, threshold = config.HIGH_RATED_POPUP_THRESHOLD) { if (!Array.isArray(items) || !items.length) return; const hi = items.filter(p => p && p.type === 'player' && p.loans === -1 && p.rating >= threshold); state._hiRatedPlayers.push(...hi); state.page.info.lock ||= []; for (const p of hi) { if (!p.isDuplicate?.() && p.rating == 99 && !state.page.info.lock.includes(p.id)) { state.page.info.lock.push(p.id); } } }, showHiRatedPopup(title = `本次高分球员(≥${config.HIGH_RATED_POPUP_THRESHOLD})`) { const list = state._hiRatedPlayers; if (!list.length) return; const popupController = new EADialogViewController({ dialogOptions: [ { labelEnum: enums.UIDialogOptions.OK } ], message: '', title, type: EADialogView.Type.MESSAGE, }); popupController.init(); popupController.onExit.observe(popupController, (e) => { e.unobserve(popupController); state._hiRatedPlayers = []; try { const cur = ea.ctrl; if ( cur && cur.constructor && cur.constructor.name === 'UTStorePackViewController' && cur.getStorePacks ) { cur.getStorePacks(true); } } catch { } }); const rootEl = popupController.getView().getRootElement(); const bodyEl = rootEl.querySelector('.ea-dialog-view--body') || rootEl; popupController.getView().getRootElement().style.width = '40rem'; const box = document.createElement('div'); box.style.cssText = 'padding:0 1rem 1.5rem 1rem;'; const players = list .slice() .sort((a, b) => Number(b.rating || 0) - Number(a.rating || 0)) .slice(0, 20); if (players.length) { const listBox = document.createElement('div'); listBox.className = 'ut-store-reveal-modal-list-view'; const ul = document.createElement('ul'); ul.className = 'itemList'; listBox.appendChild(ul); popupController.listRows = players.map((i) => { const row = new UTItemTableCellView(); row.setData( i, void 0, typeof ListItemPriority !== 'undefined' ? ListItemPriority.DEFAULT : void 0, ); row.render(); ul.appendChild(row.getRootElement()); return row; }); box.appendChild(listBox); } const maxRating = players.reduce((m, p) => Math.max(m, Number(p.rating || 0)), 0); const summary = document.createElement('div'); summary.textContent = `本次共 ${list.length} 名 ≥${config.HIGH_RATED_POPUP_THRESHOLD} 分,最高 ${maxRating}。`; summary.style.cssText = 'padding-top:.5rem;font-size:1rem;'; box.appendChild(summary); bodyEl.prepend(box); try { const rootEl = popupController.getView().getRootElement(); const footer = rootEl.querySelector('.ea-dialog-view--footer') || rootEl; const btnGroup = footer.querySelector('.ut-st-button-group') || footer.appendChild(Object.assign(document.createElement('div'), { className: 'ut-st-button-group' })); if (!btnGroup.querySelector('button[data-donate-btn="1"]')) { const donateBtn = document.createElement('button'); donateBtn.setAttribute('data-donate-btn', '1'); donateBtn.innerHTML = ` <span class="btn-text">打赏一下</span> <span class="btn-subtext"></span> `; donateBtn.addEventListener('click', (ev) => { ev.preventDefault(); ev.stopPropagation(); ui.showDonateModal(); }); btnGroup.appendChild(donateBtn); } } catch (e) { console.warn('[donate-btn] append failed:', e); } if (typeof gPopupClickShield !== 'undefined' && gPopupClickShield?.setActivePopup) { gPopupClickShield.setActivePopup(popupController); } }, async moveItems(items, pile, controller, { timeout = 15000 } = {}) { return await util.withAbort(() => new Promise((resolve, reject) => { if (!items || !items.length) return resolve({ success: true, skipped: true }); let settled = false; const done = (err, val) => { if (settled) return; settled = true; clearTimeout(tid); err ? reject(err) : resolve(val); }; try { services.Item.move(items, pile, true).observe(controller, (e, res) => { try { e.unobserve(controller); } catch { } if (!res || !res.success) return done(new ExpectError('移动失败')); done(null, res); }); } catch (e) { return done(e); } const tid = setTimeout(() => { done(new TimeoutError('moveItems超时')); }, timeout); }) )(); }, getUnassignedItemsSafe: async () => { let items = []; for (let i = 0; i < 3 && !items.length; i++) { util.abortPoint(); items = repositories.Item.getUnassignedItems(); await util.sleep(1500); } return items; }, async handleUnassigned(minRating) { const controller = ea.getUnassignedController(); let items = await sbc.getUnassignedItemsSafe(); if (!items.length) return; sbc.collectHiRated(items, config.HIGH_RATED_POPUP_THRESHOLD); const tradablePlayers = items.filter( (p) => p.type === 'player' && p.loans === -1 && !p.untradeable, ); const toStorage = items.filter( (p) => p.type === 'player' && p.loans === -1 && p.untradeable && p.isDuplicate() && ( p.rating >= minRating || (p.groups && p.groups.includes(23)) ) ); const clubPlayers = items.filter( (p) => p.type === 'player' && p.loans === -1 && p.untradeable && !p.isDuplicate(), ); if (tradablePlayers.length) await sbc.moveItems(tradablePlayers, ItemPile.TRANSFER, controller); if (clubPlayers.length) await sbc.moveItems(clubPlayers, ItemPile.CLUB, controller); const spaceLeft = config.STORAGE_MAX - repositories.Item.numItemsInCache(ItemPile.STORAGE); if (toStorage.length > spaceLeft) { alert('仓库已满'); throw new ExpectError('仓库已满'); } if (toStorage.length) await sbc.moveItems(toStorage, ItemPile.STORAGE, controller); await sbc.refreshUnassignedItems(controller); await ea.waitAllLoadingEnd(); await util.sleep(800); items = repositories.Item.getUnassignedItems(); if (!items.length) return true; const ellipsisBtn = await dom.waitForElement(() => { const root = document.querySelector('.sectioned-item-list:last-of-type'); if (!root) return null; const container = root.querySelector('.ut-section-header-view') || root; return container.querySelector('.ut-image-button-control.ellipsis-btn') || null; }, 2000); if (ellipsisBtn) { dom.simulateClick(ellipsisBtn); await util.withAbort(async () => { const modal = await dom .waitForElement('.view-modal-container.form-modal .ut-bulk-action-popup-view', 8000) .catch(() => null); if (!modal) return; const btn = [...modal.querySelectorAll('button')].find((b) => b.textContent.includes('快速出售')); if (btn) dom.simulateClick(btn); await ea.waitLoadingEndOnce(); })(); } return true; }, async handleUnassignedDuplicate(minRating) { const controller = ea.getUnassignedController(); const items = await sbc.getUnassignedItemsSafe(); if (!items.length) return; const toStorage = items.filter( (p) => p.type === 'player' && p.loans === -1 && p.untradeable && p.isDuplicate() && ( p.rating >= minRating || (p.groups && p.groups.includes(23)) ) ); const spaceLeft = config.STORAGE_MAX - repositories.Item.numItemsInCache(ItemPile.STORAGE); if (toStorage.length > spaceLeft) { alert('仓库已满'); throw new ExpectError('仓库已满'); } await sbc.moveItems(toStorage, ItemPile.STORAGE, controller); }, async refreshUnassignedItems(controller) { await util.withAbort(async () => { const req = ea.waitRequest('/purchased/items', 'GET', 10000); await services.Item.itemDao.itemRepo.unassigned.reset(); await controller.getUnassignedItems(); await req; })(); }, getStoreView() { try { const vc = ea.ctrl; if (vc?.constructor?.name !== 'UTStorePackViewController') return null; return vc.getView?.() || null; } catch { return null; } }, getSelectedPackIdFromFilter(view) { try { const id = view?._fsufilterOption?.id; return typeof id === 'number' && id > 1 ? id : null; } catch { return null; } }, getPacksNum() { const view = sbc.getStoreView(); if (!view) return 0; const packsMap = view._fsuPacks || {}; const packId = sbc.getSelectedPackIdFromFilter(view); if (!packId || !packsMap[packId]) return 0; return packsMap[packId].count || 0; }, async setPackFilterId(filterId, timeout = 2000) { const c = ea.ctrl; const view = c?.getView?.(); if (!view || !view._fsufilterOption) throw new ExpectError('FSU筛选不存在'); const curId = view._fsufilterOption.id; if (Number(curId) === Number(filterId)) return { success: true, id: filterId, skipped: true }; let finished = false; return await util.withAbort( () => new Promise((resolve, reject) => { const onChange = () => { if (finished) return; finished = true; clearTimeout(timer); view._fsufilterOption.removeTarget(view._fsufilterOption, EventType.CHANGE); resolve({ success: true, id: filterId }); }; view._fsufilterOption.addTarget(view._fsufilterOption, onChange, EventType.CHANGE); const timer = setTimeout(() => { if (finished) return; finished = true; view._fsufilterOption.removeTarget(view._fsufilterOption, EventType.CHANGE); reject(new TimeoutError('设置筛选超时')); }, timeout); view._fsufilterOption.setIndexById(Number(filterId)); }), )(); }, async goToSBCSet(opts = {}) { const set = sbc.resolveSBCSetStrict(opts); return await sbc.pushSBCSet(set, { timeout: 15000 }); }, resolveSBCSetStrict({ setId, categoryName, setName }) { const categoriesArr = Object.values(services.SBC.repository.categories._collection || {}); const setsArr = Object.values(services.SBC.repository.sets._collection || {}); if (setId != null) { const s = setsArr.find((x) => x.id === Number(setId)); if (!s) throw new ExpectError(`找不到 setId=${setId} 的 SBC`); return s; } let pool = setsArr; if (categoryName) { const cat = categoriesArr.find((c) => c.name === categoryName); if (!cat) throw new ExpectError(`找不到分类: ${categoryName}`); const idSet = new Set(cat.setIds || []); pool = pool.filter((s) => idSet.has(s.id)); } if (setName) { const found = pool.find((s) => s.name === setName) || pool.find((s) => (s.name || '').includes(setName)); if (!found) throw new ExpectError(`找不到名为/包含 ${setName} 的 SBC`); return found; } if (!pool.length) throw new ExpectError('没有可用的SBC'); return pool[0]; }, async pushSBCSet( sbcSet, { timeout = 15000 } = {} ) { const controller = ea.ctrl; const view = controller?.getView?.(); if (!controller || !view) throw new ExpectError('[pushSBCSet] 无法获得当前 controller/view'); const setInteract = (flag) => { try { view.setInteractionState && view.setInteractionState(flag); } catch { } }; setInteract(false); const challengesResp = await new Promise((resolve, reject) => { let done = false; const t = setTimeout(() => { if (!done) reject(new TimeoutError('requestChallengesForSet超时')); }, timeout); services.SBC.requestChallengesForSet(sbcSet).observe(controller, (e, resp) => { done = true; try { e.unobserve(controller); } catch { } clearTimeout(t); if (!resp || !resp.success) return reject(new ExpectError('requestChallengesForSet失败')); if (!resp.data?.challenges?.length) return reject(new ExpectError('该SBC暂无挑战')); resolve(resp); }); }); const nav = controller.getNavigationController?.(); if (!nav) { setInteract(true); throw new ExpectError('无导航控制器'); } const list = challengesResp.data.challenges || []; let target = list.find(c => { const remote = (c.status || '').toUpperCase(); if (remote && remote !== 'COMPLETED') return true; try { const ch = sbcSet.getChallenge?.(c.id); if (ch) { const local = (ch.status || '').toUpperCase(); if (local && local !== 'COMPLETED') return true; if (typeof ch.isCompleted === 'function') return !ch.isCompleted(); if ('completed' in ch) return !ch.completed; } } catch { } return false; }); if (!target) target = list[0]; const openChallenge = async (c) => { await new Promise((resolve, reject) => { let done = false; const t = setTimeout(() => { if (!done) reject(new TimeoutError('loadChallenge超时')); }, timeout); services.SBC.loadChallenge(c).observe(controller, (ee, rr) => { done = true; try { ee.unobserve(controller); } catch { } clearTimeout(t); if (!rr || !rr.success) return reject(new ExpectError('loadChallenge失败')); resolve(); }); }); try { const ch = sbcSet.getChallenge?.(c.id); if (ch && !ch.squad) ch.update?.(c); } catch { } const vc = new UTSBCSquadSplitViewController(); vc.initWithSBCSet?.(sbcSet, c.id); nav.pushViewController?.(vc, true); }; try { await openChallenge(target); } catch (e) { const vc = new UTSBCGroupChallengeSplitViewController(); vc.initWithSBCSet?.(sbcSet); nav.pushViewController?.(vc, true); try { nav.setNavigationTitle?.(sbcSet.name); } catch { } setInteract(true); throw e; } setInteract(true); return true; }, async ensureSBCHub() { const c = ea.ctrl; if (!c || c.className !== 'UTSBCHubViewController') { await dom.clickIfExists('.ut-tab-bar-item.icon-sbc', 10000, 0); await ea.waitAllLoadingEnd(); } }, async goToPacks(reentered = false) { const c = ea.ctrl; if (c?.className !== 'UTStorePackViewController') { await dom.clickIfExists('.ut-tab-bar-item.icon-store', 10000, 0); await ea.waitAllLoadingEnd(); const clickedUnassigned = await dom.clickIfExists( () => { const tiles = document.querySelectorAll( '.tile, .ut-store-tile-view, .store-tile, .tile-container', ); for (const t of tiles) { util.abortPoint(); const h = t.querySelector('h1.tileHeader, .tileHeader'); if (h && h.textContent.trim() === '未分配的物品') return t; } return null; }, 1500, 0, { strict: false }, true, ); if (clickedUnassigned) { await ea.waitAllLoadingEnd(); await sbc.handleUnassigned(state.minRating); await ea.waitAllLoadingEnd(); if (!reentered) return sbc.goToPacks(true); } await dom.clickIfExists( () => document.querySelector('.packs-tile, .ut-store-pack-tile-view, .tile.packs'), 10000, 0, { strict: false }, ); await ea.waitAllLoadingEnd(); } return true; }, getPackIdFromSbc(sbcSet) { try { return Number(sbcSet?.awards?.[0]?.value) || null; } catch { return null; } }, async setPackFilterForSetId(setId, { timeout = 2000, retry = 1 } = {}) { const set = sbc.getSbcById(setId); if (!set) return false; const packId = sbc.getPackIdFromSbc(set); if (!packId) return false; try { await ea.waitController('UTStorePackViewController', 6000); } catch { await sbc.goToPacks(); } try { await sbc.setPackFilterId(packId, timeout); return true; } catch (e) { if (retry <= 0) return false; try { const ctrl = await ea.waitController('UTStorePackViewController', 6000); ctrl?.getStorePacks?.(true); await ea.waitAllLoadingEnd(); } catch { } try { await sbc.setPackFilterId(packId, timeout); return true; } catch { return false; } } }, getSbcById(id) { id = Number(id); return ( Object.values(services.SBC.repository.sets._collection || {}).find((s) => s.id === id) || null ); }, async fetchSbcList() { await sbc.ensureSBCHub(); const categories = Object.values(services?.SBC?.repository?.categories?._collection || {}); const setsDict = services?.SBC?.repository?.sets?._collection || {}; const wanted = new Set(['升级', "我的最爱"]); const idSet = new Set(); for (const cat of categories) { if (wanted.has(cat?.name)) for (const id of (cat?.setIds || [])) idSet.add(id); } console.log('[idSet]', idSet) const filtered = Array.from(idSet) .map(id => setsDict[id]) .filter(Boolean) .filter(set => set?.name && !set.name.includes('可交易') && set.repeatabilityMode !== "REFRESH" ); state.FILTERED_SETS = filtered; return state.FILTERED_SETS; }, async openPacksOnce(expectedCount = null, depth = 0) { const MAX_DEPTH = 1; const inferExpectedFromCurrentFilter = () => { try { const view = sbc.getStoreView(); const packId = sbc.getSelectedPackIdFromFilter(view); if (!packId) return null; const sets = Object.values(services.SBC.repository.sets._collection || {}); const found = sets.find(s => Number(s?.awards?.[0]?.value) === Number(packId)); if (!found) return null; const name = String(found.name || ''); if (/10\s*名\s*8(?:4|5)\+\s*升级/.test(name)) return 10; if (name.includes('TOTW 升级')) return 1; if (/阵容变异/.test(name) && /\b89\b/.test(name)) return 50; } catch { return null; } }; if (expectedCount == null) { expectedCount = inferExpectedFromCurrentFilter(); } await ea.waitController('UTStorePackViewController', 20000); await dom.clickIfExists( () => { const btns = document.querySelectorAll('button.currency.call-to-action'); return Array.from(btns) .reverse() .find((b) => { const txt = b.querySelector('span.text')?.textContent.trim(); return txt === '打开' && b.closest('.ut-store-pack-details-view')?.style.display !== 'none'; }); }, 20000, 0, ); await ea.waitController('UTUnassignedItemsSplitViewController', 20000); const waitUnassignedCount = async (timeout = 8000) => { const t0 = Date.now(); let last = 0; while (Date.now() - t0 < timeout) { util.abortPoint(); const items = await sbc.getUnassignedItemsSafe(); const c = Array.isArray(items) ? items.length : 0; last = c; if (c > 0) return c; await util.sleep(250); } return last; }; const count = await waitUnassignedCount(8000); if (Number.isFinite(expectedCount) && expectedCount > 0 && count !== expectedCount) { await sbc.handleUnassigned(state.minRating); await ea.waitAllLoadingEnd(); if (depth < MAX_DEPTH) { await ea.waitController('UTStorePackViewController', 12000).catch(() => null); await ea.waitAllLoadingEnd(); return await sbc.openPacksOnce(expectedCount, depth + 1); } else { throw new ExpectError(`[openPacksOnce] count mismatch: got ${count}, expect ${expectedCount}`); } } await sbc.handleUnassigned(state.minRating); await ea.waitController('UTStorePackViewController'); await ea.waitAllLoadingEnd(); return true; }, async loopOnce(retry = 0) { if (state.abortCtrl?.signal?.aborted) throw new AbortedError(); log.i('[loopOnce] start'); try { await ea.waitController('UTStorePackViewController', 20000); const ok = await sbc.setPackFilterForLoop(); if (!ok) return false; await dom.clickIfExists( () => { const btns = document.querySelectorAll('button.currency.call-to-action'); return Array.from(btns) .reverse() .find((b) => { const txt = b.querySelector('span.text')?.textContent.trim(); return txt === '打开' && b.closest('.ut-store-pack-details-view')?.style.display !== 'none'; }); }, 30000, 0, ); await ea.waitController('UTUnassignedItemsSplitViewController', 20000); const getCountOnce = async () => { const items = await sbc.getUnassignedItemsSafe(); return Array.isArray(items) ? items.length : 0; }; const waitUnassignedCount = async (timeout = 8000) => { const t0 = Date.now(); let last = 0; while (Date.now() - t0 < timeout) { util.abortPoint(); const c = await getCountOnce(); last = c; if (c > 0) return c; await util.sleep(250); } return last; }; const unassignedCount = await waitUnassignedCount(8000); if (unassignedCount !== 10) { try { await sbc.handleUnassigned(state.minRating); await ea.waitAllLoadingEnd(); } catch (e) { if (!isAbort(e)) log.w('[loopOnce] handleUnassigned failed', e); } return (retry < 1) ? await sbc.loopOnce(retry + 1) : false; } if (state.enableHandleDuplicate) await sbc.handleUnassignedDuplicate(state.minRating); await sbc.goToSBCSet({ setId: Number(state.selectedLoopSetId) }); const rptBtn = await dom.waitForElement(sbc.sel.rptBtn, 5000, { strict: true }); ea.squad.removeAllItems() if (rptBtn) { dom.simulateClick(rptBtn); await ea.waitAllLoadingEnd(); } await sbc.addPlayer(); await dom.clickIfExists( () => Array.from(document.querySelectorAll('button.btn-standard.mini.call-to-action')).find( (b) => b.textContent.includes('阵容补全'), ), 5000, 0, { strict: true }, ); await dom.clickIfExists( () => Array.from(document.querySelectorAll('button')).find( (b) => b.textContent.trim() === '确定', ), 5000, 0, ); await ea.waitAllLoadingEnd(); let hasSwapPlayer = false; ea.waitRequest('/item?idList=', 'GET', 15000).then((data) => { if (data) hasSwapPlayer = true; }).catch(() => { }); const req = ea.waitRequest('?skipUserSquadValidation=', 'PUT'); const submit = await dom.clickIfExists( 'button.ut-squad-tab-button-control.actionTab.right.call-to-action:not(.disabled)', 5000, 0, { strict: true }, true, ); if (!submit) { log.w('[loopOnce] 提交按钮未出现,本轮失败'); return false; } const data = await req; if (!data?.grantedSetAwards?.length) return false; await ea.waitLoadingEndOnce(); const ctrl = await ea .waitController('UTUnassignedItemsSplitViewController', 12000) .catch(() => null); if (ctrl) { if (hasSwapPlayer) await util.sleep(3000); await sbc.handleUnassigned(state.minRating); } await ea.waitController('UTStorePackViewController'); await ea.waitAllLoadingEnd(); log.i('[loopOnce] done'); return true; } catch (e) { log.e(e); await tasks.stopAsync(); return false; } }, getSelectedLoopPackId() { const set = sbc.getSbcById(state.selectedLoopSetId); return sbc.getPackIdFromSbc(set); }, async setPackFilterForLoop(retry = 0) { if (!state.selectedLoopSetId) return false; const packId = sbc.getSelectedLoopPackId(); if (!packId) return false; try { await sbc.setPackFilterId(packId, 2000); return true; } catch (e) { log.w('[setPackFilterForLoop] failed', e); if (retry >= 1) return false; const oldDo = state.selectedDoSbcSetId; state.selectedDoSbcSetId = state.selectedLoopSetId; try { await sbc.doSBCOnce(); } catch { } finally { state.selectedDoSbcSetId = oldDo; } return await sbc.setPackFilterForLoop(retry + 1); } }, async doSBCOnce() { if (state.abortCtrl?.signal?.aborted) throw new AbortedError(); await sbc.goToSBCSet({ setId: Number(state.selectedDoSbcSetId) }); const sbcSet = services.SBC.repository.getSetById(state.selectedDoSbcSetId); const sbcTitle = sbcSet.name || ''; const isTOTW = sbcTitle.includes('TOTW'); await Promise.all([ ea.waitController('UTSBCSquadSplitViewController', 20000), ea.waitLoadingEndOnce(), ]); ea.squad.removeAllItems() const doFill = async () => { if (!isTOTW) await sbc.addPlayer(); await dom.clickIfExists( () => Array.from(document.querySelectorAll('button.btn-standard.mini.call-to-action')).find( (b) => b.textContent.includes('阵容补全'), ), 10000, 0, ); await dom.clickIfExists( () => Array.from(document.querySelectorAll('button')).find( (b) => b.textContent.trim() === '确定', ), 10000, 0, ); }; const tryFastFill = async () => { const fastBtn = await util.withAbort(async () => { const start = Date.now(); while (Date.now() - start < 2000) { util.abortPoint(); const btn = Array.from( document.querySelectorAll('button.btn-standard.mini.call-to-action'), ).find((el) => el.innerText.trim().includes('一键填充')); if (btn) return btn; await util.sleep(200); } return null; })(); if (fastBtn) { dom.simulateClick(fastBtn); return true; } return false; }; const squadPromise = ea.waitRequest('/squad', 'GET'); if (isTOTW) await doFill(); else if (!(await tryFastFill())) await doFill(); await ea.waitAllLoadingEnd(); const squad = await squadPromise; const threshold = isTOTW ? config.MAX_RATING_TOTW : config.MAX_RATING_NORMAL; const hiList = ea.slots .map((p) => p?._item || p) .filter((it) => it && it.rating >= threshold && it.pile!=10); console.log(hiList,ea.slots) if (hiList.length > 0) { const maxRating = hiList.reduce((m, p) => Math.max(m, Number(p.rating || 0)), 0); alert(`检测到高分球员(≥${threshold}),已取消提交。\n数量:${hiList.length},最高:${maxRating}`); try { state.abortCtrl?.abort?.(); } catch { } setTimeout(() => tasks.stopAsync({ suppressAutoRestartOnce: true, timeout: 2000 }), 0); throw new AbortedError('HighRatedInSquad'); } const req = ea.waitRequest('?skipUserSquadValidation=', 'PUT'); await dom.clickIfExists( 'button.ut-squad-tab-button-control.actionTab.right.call-to-action:not(.disabled)', 5000, 0, ); const data = await req; if (!data?.grantedSetAwards?.length && !data?.grantedChallengeAwards?.length) return false; await ea.waitAllLoadingEnd(); log.i('[doSBCOnce] done'); return true; }, async openPacksForSet(setId, n) { n = Number(n) || 0; if (n <= 0) return 0; const ok = await sbc.goPacksSelectFromSet(setId); if (!ok) return 0; let opened = 0; let spin = 0; while (opened < n && !state.abortCtrl?.signal?.aborted) { let count = Number(sbc.getPacksNum()) || 0; if (count <= 0) { try { const ctrl = await ea.waitController('UTStorePackViewController', 6000); ctrl?.getStorePacks?.(true); await ea.waitAllLoadingEnd(); } catch { } count = Number(sbc.getPacksNum()) || 0; if (count <= 0) { if (++spin >= 2) break; await util.sleep(600); continue; } } const okOpen = await sbc.openPacksOnce().catch(() => false); if (okOpen) { opened++; try { await ea.waitController('UTStorePackViewController', 12000); await ea.waitAllLoadingEnd(); } catch { } } else { spin++; if (spin > 2) break; try { const ctrl = await ea.waitController('UTStorePackViewController', 6000); ctrl?.getStorePacks?.(true); await ea.waitAllLoadingEnd(); } catch { } } } return opened; }, async openPacksAfterDo(setId = state.selectedDoSbcSetId, expected = null) { if (!setId) return false; if (state.abortCtrl?.signal?.aborted) throw new AbortedError(); if (expected != null) { const opened = await sbc.openPacksForSet(setId, Number(expected) || 0); return opened >= (Number(expected) || 0); } const ok = await sbc.goPacksSelectFromSet(setId); if (!ok) return false; let count = Number(sbc.getPacksNum()) || 0; if (count <= 0) { util.abortPoint(); try { const ctrl = await ea.waitController('UTStorePackViewController', 6000); ctrl?.getStorePacks?.(true); await ea.waitAllLoadingEnd(); count = Number(sbc.getPacksNum()) || 0; } catch { } } if (count <= 0) return true; for (let i = 0; i < count; i++) { util.abortPoint(); await sbc.openPacksOnce(); try { await ea.waitController('UTStorePackViewController', 12000); await ea.waitAllLoadingEnd(); } catch { } } return true; }, async goPacksSelectFromSet(setId) { await sbc.goToPacks(); const ok = await sbc.setPackFilterForSetId(setId, { timeout: 3000, retry: 1 }); if (!ok) { log.w('[goPacksSelectFromSet] 选择筛选失败', setId); return false; } return true; }, getInventorySummary() { try { const clubIter = repositories?.Item?.club?.items?.values?.(); const clubItems = clubIter ? Array.from(clubIter) : []; const storageItems = repositories?.Item?.getStorageItems?.() || []; const valid = (p) => p?.isPlayer?.() && p.loans === -1 && !p.isEnrolledInAcademy?.() && p.endTime === -1; const all = clubItems.concat(storageItems).filter(valid); const excludeSet = new Set(state?.page?.info?.lock || []); const inRange = (range) => all.reduce((n, p) => { const r = p.rating | 0; const g = Array.isArray(p?.groups) ? p.groups : Array.isArray(p?._data?.groups) ? p._data.groups : []; if ( (!range || (r >= range[0] && r <= range[1])) && Array.isArray(g) && !g.includes(23) && !excludeSet.has(p.id) ) { n++; } return n; }, 0); let totw = 0; for (const p of all) { const g = Array.isArray(p?.groups) ? p.groups : Array.isArray(p?._data?.groups) ? p._data.groups : []; if (!excludeSet.has(p.id) && g.includes(23)) totw++; } return { totw, cntTotw: inRange(config.PRO.TOTW_RANGE), cntLow: inRange(config.PRO.LOWBIN_RANGE), total: all.length, }; } catch { return { totw: 0, cntTotw: 0, cntLow: 0, total: 0 }; } }, decideAction() { const inv = sbc.getInventorySummary(); if (inv.cntLow < config.PRO.LOWBIN_SAFE_MIN) return 'do_var89'; if (inv.totw >= config.PRO.TOTW_SAFE_MIN) return 'try_loop'; if (inv.cntTotw >= config.PRO.TOTW_NEED) return 'do_totw'; return 'do_var89'; }, async ensureAutoTargets() { if (!Array.isArray(state.FILTERED_SETS) || !state.FILTERED_SETS.length) { try { await sbc.fetchSbcList(); } catch (e) { log.w('[auto] fetchSbcList failed', e); } } const list = Array.isArray(state.FILTERED_SETS) ? state.FILTERED_SETS : []; const isLoop = (s) => /10\s*名\s*8(?:4|5)\+\s*升级/.test(s?.name || ''); let loopId = null; if (state.selectedLoopSetId != null) { const sel = list.find((s) => String(s.id) === String(state.selectedLoopSetId)); if (sel && isLoop(sel)) loopId = sel.id; } const totw = list.find((s) => (s?.name || '').includes('TOTW 升级')) || null; const var89 = list.find((s) => /阵容变异/.test(s?.name || '') && /\b89\b/.test(s.name)) || null; return { loopId, totwId: totw?.id || null, var89Id: var89?.id || null }; }, async tryLoopWithBackoff(targets) { const loopId = targets?.loopId ? String(targets.loopId) : ''; if (!loopId) { sbc._lastMainFailReason = 'need_89'; return false; } state.selectedLoopSetId = loopId; try { await sbc.goToPacks(); const ok = await sbc.loopOnce(); sbc._lastMainFailReason = ok ? 'ok' : inferMainFailReason(); return !!ok; } catch (e) { if (isAbort?.(e)) throw e; log.w('[tryLoopOnce] error:', e); sbc._lastMainFailReason = inferMainFailReason(); return false; } function inferMainFailReason() { const inv = sbc.getInventorySummary?.() || {}; return (inv.totw < (config.PRO?.TOTW_SAFE_MIN ?? 0)) ? 'need_totw' : 'need_89'; } }, async runSbcNTimesPreferOpen(setId, times = 3, { openAfter = true, totw = false } = {}) { util.abortPoint(); if (!setId) return 0; state.selectedDoSbcSetId = String(setId); const targetTotal = Math.max(1, Number(times) || 3); let existingCount = 0; let packId = null; try { const set = sbc.getSbcById(state.selectedDoSbcSetId); packId = sbc.getPackIdFromSbc(set); if (!packId) { log.w('[runSbcNTimesPreferOpen] 获取 packId 失败,回退到纯做包流程'); throw new Error('NO_PACK_ID'); } const packs = services?.Store?.storeDao?.storeRepo?.myPacks?._collection || []; existingCount = packs.filter(p => String(p.id) === String(packId)).length; if (existingCount >= targetTotal) { await sbc.goToPacks(); if (openAfter) await sbc.openPacksAfterDo(state.selectedDoSbcSetId, targetTotal); return true; } } catch (e) { if (!isAbort?.(e)) log.w('[runSbcNTimesPreferOpen] 预检查包数量失败或packId缺失,进入做包流程', e); } if (!totw) await sbc.ensureSBCHub(); const needToMake = Math.max(0, targetTotal - existingCount); let made = 0; for (let i = 0; i < needToMake; i++) { util.abortPoint(); try { const ok = await sbc.doSBCOnce(); if (!ok) break; made++; } catch (e) { if (isAbort(e)) throw e; if (isHighRatedError?.(e)) throw e; break; } await util.sleep(300 + Math.random() * 400); } const totalAvailableThisRound = existingCount + made; if (openAfter && totalAvailableThisRound > 0) { const toOpen = Math.min(targetTotal, totalAvailableThisRound); try { await sbc.openPacksAfterDo(state.selectedDoSbcSetId, toOpen); } catch (e) { if (!isAbort?.(e)) log.w('[runSbcNTimesPreferOpen] 开包失败:', e); } } return made > 0 || existingCount >= targetTotal; }, async runVar89NTimes(targets, times = 3, { openAfter = true } = {}) { util.abortPoint(); if (!targets?.var89Id) return 0; return sbc.runSbcNTimesPreferOpen(targets.var89Id, times, { openAfter }); }, async runTotwNTimes(targets, times = 1, { openAfter = true, totw = true } = {}) { util.abortPoint(); if (!targets?.totwId) return 0; return sbc.runSbcNTimesPreferOpen(targets.totwId, times, { openAfter, totw }); }, async autoRound() { if (state.abortCtrl?.signal?.aborted) throw new AbortedError(); if (!Array.isArray(state.FILTERED_SETS) || !state.FILTERED_SETS.length) { try { await sbc.fetchSbcList(); } catch (e) { log.w('[auto] fetchSbcList failed', e); } } const targets = await sbc.ensureAutoTargets(); util.abortPoint(); state._auxActGuard ??= { lastAct: null, consec: 0 }; const stepAuxGuard = (act) => { if (act === 'do_totw' || act === 'do_var89') { const g = state._auxActGuard; if (g.lastAct === act) g.consec += 1; else { g.lastAct = act; g.consec = 1; } return g.consec <= (config.PRO?.AUX_MAX_CONSEC ?? 2); } else { state._auxActGuard.lastAct = null; state._auxActGuard.consec = 0; return true; } }; const getMainFailReason = async () => { try { if (typeof sbc.getMainFailReason === 'function') { const r = await sbc.getMainFailReason(targets); if (r === 'need_totw' || r === 'need_89') return r; if (r === 'ok') return 'ok'; } if (typeof sbc.getLastLoopFailReason === 'function') { const r2 = await sbc.getLastLoopFailReason(targets); if (r2 === 'need_totw' || r2 === 'need_89') return r2; if (r2 === 'ok') return 'ok'; } } catch (e) { log.w('[auto] reason-check error:', e); } const inv = sbc.getInventorySummary?.() || {}; return (inv.totw < (config.PRO?.TOTW_SAFE_MIN ?? 0)) ? 'need_totw' : 'need_89'; }; const runVar89 = async () => { util.abortPoint(); if (!targets?.var89Id) { log.i('[auto] 无 var89Id'); return false; } if (!stepAuxGuard('do_var89')) { log.w('[auto] do_var89 达到连续上限'); return false; } try { return !!(await sbc.runVar89NTimes(targets, 3, { openAfter: true })); } catch (e) { if (isAbort?.(e)) throw e; log.w('[auto] runVar89NTimes error:', e); return false; } }; const runTotw = async () => { util.abortPoint(); if (!targets?.totwId) { log.i('[auto] 无 totwId'); return false; } if (!stepAuxGuard('do_totw')) { log.w('[auto] do_totw 达到连续上限'); return false; } try { return !!(await sbc.runTotwNTimes(targets, 1, { openAfter: true, totw: true })); } catch (e) { if (isAbort?.(e)) throw e; log.w('[auto] runTotwNTimes error:', e); return false; } }; const runMain = async () => { util.abortPoint(); const loopId = targets?.loopId ? String(targets.loopId) : ''; if (!loopId) { log.i('[auto] 无可用主线 loopId'); return false; } stepAuxGuard('try_loop'); state.selectedLoopSetId = loopId; try { await sbc.goToPacks(); return !!(await sbc.tryLoopWithBackoff(targets)); } catch (e) { if (isAbort?.(e)) throw e; log.w('[auto] tryLoop error:', e); return false; } }; const primary = sbc.decideAction() || 'do_var89'; if (primary === 'try_loop') { util.abortPoint(); const okMain = await runMain(); if (okMain) return true; let reason = await getMainFailReason(); if (reason === 'ok') reason = 'need_89'; if (reason === 'need_totw') { const okTotw = await runTotw(); util.abortPoint(); if (okTotw) return true; const ok89 = await runVar89(); util.abortPoint(); if (ok89) return true; log.w('[auto] 本回合失败:主线→TOTW→89 全部失败'); return false; } else { const ok89 = await runVar89(); util.abortPoint(); if (ok89) return true; log.w('[auto] 本回合失败:主线→89 全部失败'); return false; } } if (primary === 'do_totw') { const okTotw = await runTotw(); util.abortPoint(); if (okTotw) return true; const ok89 = await runVar89(); util.abortPoint(); if (ok89) return true; log.w('[auto] 本回合失败:TOTW→89 全部失败'); return false; } const ok89 = await runVar89(); util.abortPoint(); if (ok89) return true; log.w('[auto] 本回合失败:89 失败(无其他链路可尝试)'); return false; }, getSbcByNameCandidates(filteredSets = []) { const all = Array.isArray(filteredSets) ? filteredSets : []; const KW = config?.targetKeywords || [ '89 阵容变异', 'TOTW 升级', '10 名 85+ 升级', '10 名 84+ 升级', ]; const LOOP_KEYS = ['10 名 85+ 升级', '10 名 84+ 升级']; const BLACKLIST = config?.blacklistKeywords || [ '可交易', '青铜升级', '白银升级', '黄金升级', '混合联赛升级', ]; const hitAny = (name, arr) => arr.some(k => name.includes(k)); const kwIndex = (name) => { const i = KW.findIndex(k => name.includes(k)); return i === -1 ? Number.POSITIVE_INFINITY : i; }; const loopCandidates = []; const doCandidates = []; for (const s of all) { if (!s?.name) continue; if (hitAny(s.name, BLACKLIST)) continue; if (hitAny(s.name, LOOP_KEYS)) loopCandidates.push(s); else doCandidates.push(s); } loopCandidates.sort((a, b) => kwIndex(a.name) - kwIndex(b.name)); doCandidates.sort((a, b) => kwIndex(a.name) - kwIndex(b.name)); return { doCandidates, loopCandidates }; } }; const tasks = { startAsync({ button, taskName, asyncLoop, startText, stopText, countLimit, randomPause, pauseEveryRange, bigPauseRange, }) { if (state._starting) return; state._starting = true; (async () => { try { if (state.running) { if (state.abortCtrl?.signal?.aborted) { try { await (state.currentTaskDone || Promise.resolve()); } catch { } state.running = false; state.runningTask = ''; state.abortCtrl = null; } else { return; } } tasks.resetAutoGuards({ full: true }); state.running = true; state.runningTask = taskName; if (['openPacks', 'loop', 'auto'].includes(taskName)) state._hiRatedPlayers = []; ui.updateButtonState(); button.textContent = stopText; let _doneResolve; state.abortCtrl = new AbortController(); state.currentTaskDone = new Promise((r) => (_doneResolve = r)); log.i('start', taskName); try { let count = 0; let pauseEvery = pauseEveryRange ? pauseEveryRange[0] + Math.floor(Math.random() * (pauseEveryRange[1] - pauseEveryRange[0] + 1)) : 0; let pauseTime = bigPauseRange ? bigPauseRange[0] + Math.floor(Math.random() * (bigPauseRange[1] - bigPauseRange[0] + 1)) : 0; while (state.running && !state.abortCtrl.signal.aborted) { util.abortPoint(); if (state.isStopping) throw new AbortedError(); if (countLimit != null) { const remainingRaw = typeof countLimit === 'function' ? await Promise.resolve(countLimit()) : countLimit; const remaining = Number(remainingRaw); if (!Number.isFinite(remaining) || remaining <= 0) break; } const canContinue = await asyncLoop(); util.abortPoint(); if (canContinue === false) break; count++; util.abortPoint(); if (state.isStopping) throw new AbortedError(); if (pauseEvery && count >= pauseEvery) { for (let s = Math.floor(pauseTime / 1000); s > 0; s--) { if (!state.running || state.abortCtrl.signal.aborted) break; button.textContent = `等待${s}秒`; await util.sleep(1000); } button.textContent = stopText; pauseEvery = pauseEveryRange ? pauseEveryRange[0] + Math.floor(Math.random() * (pauseEveryRange[1] - pauseEveryRange[0] + 1)) : pauseEvery; pauseTime = bigPauseRange ? bigPauseRange[0] + Math.floor(Math.random() * (bigPauseRange[1] - bigPauseRange[0] + 1)) : pauseTime; count = 0; } if (randomPause) { log.i(`等待 ${randomPause[0]} - ${randomPause[1]} 秒`); await util.sleep(randomPause[0] + Math.random() * (randomPause[1] - randomPause[0])); } } } catch (e) { if (!isAbort(e)) log.e(`[${taskName}] 中断:`, e?.message); } finally { state.running = false; state.runningTask = ''; button.textContent = startText; ui.updateButtonState(); if (['openPacks', 'loop', 'auto'].includes(taskName)) { sbc.showHiRatedPopup(`本次高分球员(≥${config.HIGH_RATED_POPUP_THRESHOLD})`); } _doneResolve(); } } finally { state._starting = false; } })(); }, resetAutoGuards({ full = true } = {}) { state._auxActGuard = { lastAct: null, consec: 0 }; state._tryLoopFailStrike2 = 0; state._tryLoopFbUsedForStrike = false; if (full) { state._loopFailStrike = 0; state._lastLoopFailAt = 0; } }, _stopPromise: null, async stopAsync({ timeout = 6000, suppressAutoRestartOnce = false, minHold = 300 } = {}) { if (tasks._stopPromise) return tasks._stopPromise; tasks._stopPromise = (async () => { const wasAuto = state.runningTask === 'auto'; try { if (!state.isStopping) { state.isStopping = true; state.running = false; ui.updateButtonState(); } try { state.abortCtrl?.abort(); } catch { } const hold = new Promise((r) => setTimeout(r, minHold)); const done = state.currentTaskDone || Promise.resolve(); const wait = Promise.race([ done, new Promise((_, rej) => setTimeout(() => rej(new Error('stop timeout')), timeout)), ]); try { await Promise.all([wait, hold]); } catch (e) { log.w('[stopAsync] timeout/err, force cleanup'); } state.running = false; state.runningTask = ''; state.abortCtrl = null; state._tryLoopFailStrike2 = 0; tasks.resetAutoGuards({ full: true }); } finally { state.isStopping = false; ui.resetButtonText(); ui.updateButtonState(); if (wasAuto && state.autoRestartOnStop && !suppressAutoRestartOnce) { try { recover.triggerAndReload('pro_stopped'); } catch { } } tasks._stopPromise = null; } })(); return tasks._stopPromise; }, startLoop() { tasks.startAsync({ button: state.btn.loop, taskName: 'loop', asyncLoop: sbc.loopOnce, startText: '永动机', stopText: '停止循环', randomPause: [500, 1000], pauseEveryRange: [35, 45], bigPauseRange: [20000, 30000], }); }, startOpen() { tasks.startAsync({ button: state.btn.open, taskName: 'openPacks', asyncLoop: sbc.openPacksOnce, startText: '开包', stopText: '停止开包', countLimit: () => sbc.getPacksNum(), randomPause: [1000, 1500], }); }, startDoSBC() { tasks.startAsync({ button: state.btn.do, taskName: 'doSBC', asyncLoop: sbc.doSBCOnce, startText: '猛猛干', stopText: '不干了', randomPause: [500, 1000], pauseEveryRange: [35, 45], bigPauseRange: [20000, 30000], }); }, startAuto() { tasks.startAsync({ button: state.btn.auto, taskName: 'auto', asyncLoop: sbc.autoRound, startText: '永动机Pro', stopText: '停止Pro', randomPause: [500, 900], }); }, }; const ui = { injectStyle: util.once(() => { GM_addStyle(` .panda-modal-mask{position:fixed;inset:0;background:rgba(0,0,0,.55);z-index:100000;display:flex;align-items:center;justify-content:center} .panda-modal{width:720px;max-width:calc(100vw - 40px);max-height:calc(100vh - 40px);background:#1f1f1f;border:1px solid #333;border-radius:12px;box-shadow:0 10px 30px rgba(0,0,0,.5);color:#eee;overflow:hidden;display:flex;flex-direction:column;font-family:inherit} .panda-modal__hd{padding:12px 16px;background:#252525;border-bottom:1px solid #333;display:flex;align-items:center;justify-content:space-between} .panda-modal__title{font-weight:700;font-size:15px} .panda-modal__close{border:none;background:transparent;color:#bbb;cursor:pointer;font-size:18px;line-height:1} .panda-modal__bd{padding:14px;display:grid;grid-template-columns:1fr 1fr;gap:12px;overflow:auto;flex:1;min-height:260px} .panda-col{background:#171717;border:1px solid #333;border-radius:10px;padding:10px;display:flex;flex-direction:column} .panda-col__title{font-size:13px;font-weight:700;color:#fff;margin-bottom:8px;display:flex;align-items:center;gap:8px} .panda-col__tip{font-size:12px;color:#aaa} .panda-col__list{display:flex;flex-direction:column;gap:6px;overflow:auto} .panda-row{display:flex;gap:8px;align-items:center;color:#ddd;font-size:13px} .panda-row input[type="radio"]{accent-color:#ffc800;cursor:pointer} .panda-modal__ft{padding:10px 14px;border-top:1px solid #333;display:flex;gap:10px;justify-content:flex-end;background:#202020} .panda-btn{min-width:80px;height:32px;border-radius:8px;border:1px solid #555;cursor:pointer;font-weight:600;background:#2b2b2b;color:#ddd} .panda-btn--ok{background:#ffd76a;color:#222;border-color:#caa84b} #sbc-panel{position:fixed;bottom:20px;right:20px;z-index:99999;display:flex;flex-direction:column;align-items:center;gap:12px;min-width:110px;padding:16px 8px 10px 8px;background:rgba(30,30,30,0.96);border-radius:16px;box-shadow:0 4px 24px #0005;font-family:inherit} .sbc-input{width:70px;height:30px;font-size:18px;text-align:center;border-radius:8px;border:1px solid #ccc;margin-bottom:2px;background:#252525;color:#ffc800} .sbc-btn{width:90px;height:38px;border:none;outline:none;cursor:pointer;border-radius:10px;box-shadow:0 2px 8px #0002;font-weight:bold;transition:all .15s;user-select:none} .sbc-btn--open{background:#ffa600;color:#333} .sbc-btn--open:hover{background:#ffd700} .sbc-btn--loop{background:#ffe066;color:#333} .sbc-btn--loop:hover{background:#fffeb2} .sbc-btn--do{background:#ff6347;color:#fff} .sbc-btn--do:hover{background:#fd8578} .sbc-btn--assign{width:90px;height:38px;border-radius:8px;background:#5bc0de;color:#111;font-weight:bold} .sbc-btn--assign:hover{background:#74d5f1} .sbc-btn--settings{background-color:#4caf50 !important;color:#fff !important;border:1px solid #3e8e41 !important} .sbc-btn--settings:hover{background-color:#45a049 !important} .sbc-btn--donate{background:#ff9f0a !important;color:#111 !important;border:1px solid #d67f00 !important} .sbc-btn--donate:hover{background:#ffb23c !important} .sbc-chk{width:14px;height:14px;margin:2px 0 0 0;border-radius:3px;cursor:pointer;accent-color:#ffc800} .sbc-chklabel{color:#fff;font-size:13px;display:flex;align-items:flex-start;gap:6px;cursor:pointer;max-width:80px;line-height:1.3} #panda-dock{position:fixed;top:140px;right:0;z-index:99998;display:flex;align-items:stretch;transform:translateX(calc(100% - 28px));transition:transform .18s ease,opacity .12s ease} #panda-dock.left{left:0;right:auto;transform:translateX(calc(-100% + 28px))} #panda-dock.expanded.right,#panda-dock.expanded.left{transform:translateX(0)} #panda-dock.dragging{transition:none;opacity:.96} #panda-dock .dock-handle{width:38px;min-height:132px;background:#1e1e1e;border:1px solid #333;border-right:none;border-radius:12px 0 0 12px;box-shadow:0 4px 24px #0005;display:flex;align-items:center;justify-content:center;cursor:pointer;user-select:none;color:#ffc800;font-weight:700;writing-mode:vertical-rl;text-orientation:mixed;letter-spacing:2px} #panda-dock.left .dock-handle{border-right:1px solid #333;border-left:none;border-radius:0 12px 12px 0} #panda-dock .dock-panel{background:rgba(30,30,30,.96);border:1px solid #333;border-radius:12px;box-shadow:0 4px 24px #0005;padding:12px;display:flex;flex-direction:column;gap:10px;min-width:120px} #panda-dock #sbc-panel{all:unset;display:flex;flex-direction:column;align-items:center;gap:12px;min-width:110px} #panda-dock .dock-foot{display:flex;flex-direction:column;gap:8px;justify-content:center;align-items:center;font-size:12px;color:#bbb;margin-top:6px} #panda-dock .dock-toggle{cursor:pointer;user-select:none;padding:4px 6px;border:1px solid #555;border-radius:8px;background:#2b2b2b} .sbc-stats{display:flex;flex-direction:column;align-items:center;gap:6px} .sbc-stat-card{width:90px;height:38px;background:#141414;border:1px solid #2a2a2a;border-radius:8px;display:flex;flex-direction:column;justify-content:center;align-items:center;gap:2px} .sbc-stat-label{font-size:11px;color:#9aa;line-height:1} .sbc-stat-value{font-weight:800;font-size:14px;color:#ffd76a;line-height:1} .panda-donate-mask{position:fixed;inset:0;background:rgba(0,0,0,.55);z-index:100001;display:flex;align-items:center;justify-content:center} .panda-donate{width:560px;max-width:calc(100vw - 40px);background:#1f1f1f;border:1px solid #333;border-radius:12px;box-shadow:0 10px 30px rgba(0,0,0,.5);color:#eee;overflow:hidden} .panda-donate__hd{padding:12px 16px;background:#252525;border-bottom:1px solid #333;display:flex;align-items:center;justify-content:space-between} .panda-donate__title{font-weight:700;font-size:15px} .panda-donate__close{border:none;background:transparent;color:#bbb;cursor:pointer;font-size:18px;line-height:1} .panda-donate__bd{padding:16px;display:grid;grid-template-columns:1fr 1fr;gap:14px} .panda-donate__col{background:#171717;border:1px solid #333;border-radius:10px;padding:10px;display:flex;flex-direction:column;gap:8px;align-items:center;justify-content:center} .panda-donate__col h4{margin:0;font-size:14px;color:#ffd76a} .panda-donate__qr{width:220px;max-width:100%;height:auto;border-radius:8px;border:1px solid #2a2a2a;background:#111;object-fit:contain} .panda-donate__ft{padding:10px 14px;border-top:1px solid #333;display:flex;gap:10px;justify-content:flex-end;background:#202020} .dock-toggle.dock-guide{border-color:#3f58d1;background:#2b2b2b;color:#cfd7ff} .dock-toggle.dock-guide:hover{filter:brightness(1.08)} .SBCSquadPanel .ut-sbc-challenge-details-view{overflow-y:initial !important;margin-bottom:20px !important} `); }), updateStatsUI() { try { const clubItemsIter = repositories?.Item?.club?.items?.values?.(); const clubItems = clubItemsIter ? Array.from(clubItemsIter) : []; const storageItems = repositories?.Item?.getStorageItems?.() || []; const isValid = (p) => p?.isPlayer?.() && p.loans === -1 && !p.isEnrolledInAcademy?.() && p.endTime === -1; const club = clubItems.filter(isValid); const storage = storageItems.filter(isValid); const all = club.concat(storage); const excludeIds = state.page.info?.lock || []; const excludeSet = new Set(excludeIds); const countPlayersInRange = (players, range, excludeSet) => { const [min, max] = range; return players.reduce((n, p) => { if ( (!range || (p.rating >= min && p.rating <= max)) && Array.isArray(p.groups) && !p.groups.includes(23) && !excludeSet.has(p.id) ) { return n + 1; } return n; }, 0); }; for (const cfg of config.RANGES) { let val = 0; let id; if (cfg.type === 'all') { val = countPlayersInRange(all, cfg.range, excludeSet); id = `stat-${cfg.range[0]}-${cfg.range[1]}`; } else { val = storage.length; id = 'stat-storage'; } const node = document.getElementById(id); if (node) node.textContent = String(val); } } catch (err) { log.w('[updateStatsUI] fail', err); } }, showDonateModal() { const alipay = DONATE.ALIPAY_QR; const wechat = DONATE.WECHAT_QR; if (!alipay && !wechat) { alert('尚未设置收款码'); return; } const mask = document.createElement('div'); mask.className = 'panda-donate-mask'; const box = document.createElement('div'); box.className = 'panda-donate'; box.innerHTML = ` <div class="panda-donate__hd"> <div class="panda-donate__title">感谢支持 🧡</div> <button class="panda-donate__close">×</button> </div> <div class="panda-donate__bd"> <div class="panda-donate__col"> <h4>支付宝</h4> ${alipay ? `<img class="panda-donate__qr" src="${alipay}" alt="Alipay QR">` : '<div style="color:#888;font-size:12px">未配置</div>'} </div> <div class="panda-donate__col"> <h4>微信</h4> ${wechat ? `<img class="panda-donate__qr" src="${wechat}" alt="WeChat QR">` : '<div style="color:#888;font-size:12px">未配置</div>'} </div> </div> <div class="panda-donate__ft"> <button class="panda-btn panda-btn--ok">关闭</button> </div> `; const close = () => { try { document.body.removeChild(mask); } catch { } }; box.querySelector('.panda-donate__close').onclick = close; box.querySelector('.panda-btn--ok').onclick = close; mask.addEventListener('click', (e) => { if (e.target === mask) close(); }); mask.appendChild(box); document.body.appendChild(mask); }, updateButtonState() { const set = (btn, text, disabled) => { if (!btn) return; if (text != null) btn.textContent = text; if (typeof disabled !== 'undefined') btn.disabled = !!disabled; }; if (state.isStopping) { set(state.btn.loop, '停止中…', true); set(state.btn.open, '停止中…', true); set(state.btn.do, '停止中…', true); set(state.btn.auto, '停止中…', true); return; } if (!state.running) { set(state.btn.loop, '永动机', false); set(state.btn.open, '开包', false); set(state.btn.do, '猛猛干', false); set(state.btn.auto, '永动机Pro', false); return; } if (state.runningTask === 'loop') { set(state.btn.loop, '停止循环', false); set(state.btn.open, null, true); set(state.btn.do, null, true); set(state.btn.auto, null, true); } else if (state.runningTask === 'openPacks') { set(state.btn.open, '停止开包', false); set(state.btn.loop, null, true); set(state.btn.do, null, true); set(state.btn.auto, null, true); } else if (state.runningTask === 'doSBC') { set(state.btn.do, '不干了', false); set(state.btn.loop, null, true); set(state.btn.open, null, true); set(state.btn.auto, null, true); } else if (state.runningTask === 'auto') { set(state.btn.auto, '停止Pro', false); set(state.btn.loop, null, true); set(state.btn.open, null, true); set(state.btn.do, null, true); } }, resetButtonText() { if (state.btn.loop) state.btn.loop.textContent = '永动机'; if (state.btn.open) state.btn.open.textContent = '开包'; if (state.btn.do) state.btn.do.textContent = '猛猛干'; if (state.btn.auto) state.btn.auto.textContent = '永动机Pro'; }, async ensureConfigThenAssign(kind, { autostart = true } = {}) { if (!state.FILTERED_SETS.length) { alert('未获取配置,点TMD获取配置。网页需要设置成简体中文,是不是中国人?'); return; } await sbc.fetchSbcList(); const pick = await ui.SBCListPop( state.FILTERED_SETS, state.selectedDoSbcSetId, state.selectedLoopSetId, kind || null, ); if (!pick) return; state.selectedDoSbcSetId = pick.doId; state.selectedLoopSetId = pick.loopId; if (!autostart) return { doId: state.selectedDoSbcSetId, loopId: state.selectedLoopSetId }; if (kind === 'do' && pick.doId) { if (state.running && state.runningTask !== 'doSBC') await tasks.stopAsync({ suppressAutoRestartOnce: true }); if (!state.running) { await sbc.ensureSBCHub(); tasks.startDoSBC(); } } else if (kind === 'loop' && pick.loopId) { if (state.running && state.runningTask !== 'loop') await tasks.stopAsync({ suppressAutoRestartOnce: true }); if (!state.running) { await sbc.goToPacks(); tasks.startLoop(); } } return { doId: state.selectedDoSbcSetId, loopId: state.selectedLoopSetId }; }, SBCListPop(filteredSets, currentDoId, currentLoopId, preferColumn = null) { const el = (tag, className, props = {}) => { const node = document.createElement(tag); if (className) node.className = className; Object.assign(node, props); return node; }; const buildRadioList = (items, name, currentId) => { const listBox = el('div', 'panda-col__list'); items.forEach((s) => { const row = el('label', 'panda-row'); const r = el('input'); r.type = 'radio'; r.name = name; r.value = String(s.id); r.checked = String(currentId || '') === String(s.id); const span = el('span'); span.textContent = s.name; row.append(r, span); listBox.appendChild(row); }); return listBox; }; const buildColumn = ({ title, tipHTML = '', name, items, currentId, highlight = false, clearText = '清除绑定', }) => { const col = el('div', 'panda-col' + (highlight ? ' panda-col--highlight' : '')); const titleEl = el('div', 'panda-col__title'); titleEl.innerHTML = tipHTML ? `${title} ${tipHTML}` : title; const listBox = buildRadioList(items, name, currentId); const spacer = document.createElement('div'); spacer.style.height = '8px'; const clearBtn = el('button', 'panda-btn', { textContent: clearText }); clearBtn.onclick = () => { [...listBox.querySelectorAll('input[type="radio"]')].forEach((x) => (x.checked = false)); }; col.append(titleEl, listBox, spacer, clearBtn); return col; }; const { doCandidates, loopCandidates } = sbc.getSbcByNameCandidates(filteredSets); const mask = el('div', 'panda-modal-mask'); const modal = el('div', 'panda-modal'); mask.appendChild(modal); const hd = el('div', 'panda-modal__hd'); const title = el('div', 'panda-modal__title', { textContent: '分配SBC' }); const btnX = el('button', 'panda-modal__close'); btnX.innerHTML = '×'; hd.append(title, btnX); modal.appendChild(hd); const bd = el('div', 'panda-modal__bd'); const colDo = buildColumn({ title: '猛猛干(单选)', name: 'assign-do', items: doCandidates, currentId: currentDoId, highlight: preferColumn === 'do', }); const colLoop = buildColumn({ title: '永动机(仅 10x85 / 10x84)', tipHTML: '<span class="panda-col__tip"></span>', name: 'assign-loop', items: loopCandidates, currentId: currentLoopId, highlight: preferColumn === 'loop', }); bd.append(colDo, colLoop); modal.appendChild(bd); const ft = el('div', 'panda-modal__ft'); const btnCancel = el('button', 'panda-btn', { textContent: '取消' }); const btnOK = el('button', 'panda-btn panda-btn--ok', { textContent: '确定' }); ft.append(btnCancel, btnOK); modal.appendChild(ft); const autoStartName = preferColumn === 'do' ? 'assign-do' : preferColumn === 'loop' ? 'assign-loop' : null; if (autoStartName) bd.querySelectorAll(`input[name="${autoStartName}"]`).forEach((r) => r.addEventListener('change', () => btnOK.click(), { once: true }), ); return new Promise((resolve) => { const close = (res) => { try { document.body.removeChild(mask); } catch { } resolve(res); }; const onKey = (e) => { if (e.key === 'Escape') { e.preventDefault(); close(null); } else if (e.key === 'Enter') { e.preventDefault(); btnOK.click(); } }; btnX.onclick = () => close(null); btnCancel.onclick = () => close(null); mask.addEventListener('click', (e) => { if (e.target === mask) close(null); }); document.addEventListener('keydown', onKey); btnOK.onclick = () => { document.removeEventListener('keydown', onKey); const pick = (name) => { const r = modal.querySelector(`input[name="${name}"]:checked`); return r ? r.value : null; }; close({ doId: pick('assign-do'), loopId: pick('assign-loop') }); }; document.body.appendChild(mask); setTimeout(() => btnOK.focus(), 0); }); }, openSbcSettings() { const cur = { SP_STABLE_FOR: Number(config.UI.SP_STABLE_FOR) || 300, SP_FILL_SUCCESS_TIME: Number(config.UI.SP_FILL_SUCCESS_TIME) || 1000, HIGH_RATED_POPUP_THRESHOLD: Number(config.HIGH_RATED_POPUP_THRESHOLD) || 98, MAX_RATING_TOTW: Number(config.MAX_RATING_TOTW) || 87, MAX_RATING_NORMAL: Number(config.MAX_RATING_NORMAL) || 98, TOTW_SAFE_MIN: Number(config.PRO.TOTW_SAFE_MIN) || 3, }; const mask = document.createElement('div'); mask.className = 'panda-modal-mask'; const modal = document.createElement('div'); modal.className = 'panda-modal'; mask.appendChild(modal); const hd = document.createElement('div'); hd.className = 'panda-modal__hd'; hd.innerHTML = ` <div class="panda-modal__title">参数设置</div> <button class="panda-modal__close">×</button> `; const bd = document.createElement('div'); bd.className = 'panda-modal__bd'; bd.style.gridTemplateColumns = '1fr'; const row = (label, id, val, hint = '', min = 0, max = 99999) => ` <div class="panda-col"> <div class="panda-col__title">${label}</div> <div style="display:flex;gap:8px;align-items:center;"> <input id="${id}" type="number" value="${val}" min="${min}" max="${max}" style="width:140px;height:30px;border-radius:8px;border:1px solid #555;background:#252525;color:#ffd76a;text-align:center;"> ${hint ? `<div class="panda-col__tip">${hint}</div>` : ''} </div> </div> `; bd.innerHTML = [ row( '色卡按钮稳定检测 (ms)', 'in-SP_STABLE_FOR', cur.SP_STABLE_FOR, '如果周黑添加失败可适当调大该数值', 0, 5000, ), row( '高分弹窗阈值', 'in-HIGH_RATED_POPUP_THRESHOLD', cur.HIGH_RATED_POPUP_THRESHOLD, '≥高分球员展示阈值,自动滚卡/开包结束以后展示高分球员弹窗', 85, 99, ), row('周黑保护阈值', 'in-MAX_RATING_TOTW', cur.MAX_RATING_TOTW, '做周黑升级时检测到 ≥该分数则取消提交', 80, 99), row('普通SBC保护阈', 'in-MAX_RATING_NORMAL', cur.MAX_RATING_NORMAL, '普通SBC检测到 ≥该分数则取消提交', 80, 99), // row( // '永动机Pro周黑/TOTS判定数量', // 'in-TOTW_SAFE_MIN', // cur.TOTW_SAFE_MIN, // '周黑/TOTS低于该数量自动做周黑', // 0, // 50, // ), ].join(''); const ft = document.createElement('div'); ft.className = 'panda-modal__ft'; ft.innerHTML = ` <button class="panda-btn" id="btn-cancel">取消</button> <button class="panda-btn panda-btn--ok" id="btn-save">保存</button> `; modal.append(hd, bd, ft); document.body.appendChild(mask); const close = () => { try { document.body.removeChild(mask); } catch { } }; hd.querySelector('.panda-modal__close').onclick = close; ft.querySelector('#btn-cancel').onclick = close; ft.querySelector('#btn-save').onclick = () => { const valNum = (id, def, min, max) => { const el = document.getElementById(id); if (!el) return def; const raw = (el.value ?? '').toString().trim(); const n = Number(raw); if (!Number.isFinite(n)) return def; return Math.max(min, Math.min(max, Math.floor(n))); }; const next = { SP_STABLE_FOR: valNum('in-SP_STABLE_FOR', cur.SP_STABLE_FOR, 0, 5000), SP_FILL_SUCCESS_TIME: valNum('in-SP_FILL_SUCCESS_TIME', cur.SP_FILL_SUCCESS_TIME, 0, 10000), HIGH_RATED_POPUP_THRESHOLD: valNum('in-HIGH_RATED_POPUP_THRESHOLD', cur.HIGH_RATED_POPUP_THRESHOLD, 80, 99), MAX_RATING_TOTW: valNum('in-MAX_RATING_TOTW', cur.MAX_RATING_TOTW, 80, 99), MAX_RATING_NORMAL: valNum('in-MAX_RATING_NORMAL', cur.MAX_RATING_NORMAL, 80, 99), // TOTW_SAFE_MIN: valNum('in-TOTW_SAFE_MIN', cur.TOTW_SAFE_MIN, 0, 50), }; config.UI.SP_STABLE_FOR = next.SP_STABLE_FOR; config.UI.SP_FILL_SUCCESS_TIME = next.SP_FILL_SUCCESS_TIME; config.HIGH_RATED_POPUP_THRESHOLD = next.HIGH_RATED_POPUP_THRESHOLD; config.MAX_RATING_TOTW = next.MAX_RATING_TOTW; config.MAX_RATING_NORMAL = next.MAX_RATING_NORMAL; // config.PRO.TOTW_SAFE_MIN = next.TOTW_SAFE_MIN; GM_setValue(CFG_KEYS.SP_STABLE_FOR, next.SP_STABLE_FOR); GM_setValue(CFG_KEYS.SP_FILL_SUCCESS_TIME, next.SP_FILL_SUCCESS_TIME); GM_setValue(CFG_KEYS.HIGH_RATED_POPUP_THRESHOLD, next.HIGH_RATED_POPUP_THRESHOLD); GM_setValue(CFG_KEYS.MAX_RATING_TOTW, next.MAX_RATING_TOTW); GM_setValue(CFG_KEYS.MAX_RATING_NORMAL, next.MAX_RATING_NORMAL); // GM_setValue(CFG_KEYS.TOTW_SAFE_MIN, next.TOTW_SAFE_MIN); try { const tip = document.createElement('div'); tip.textContent = '已保存'; tip.style.cssText = 'position:fixed;bottom:22px;right:26px;background:#2b2b2b;color:#ffd76a;padding:8px 10px;border:1px solid #555;border-radius:8px;z-index:100000;'; document.body.appendChild(tip); setTimeout(() => { try { document.body.removeChild(tip); } catch { } }, 1200); } catch { } close(); }; }, initDock() { if (document.getElementById('panda-dock')) return; ui.injectStyle(); const el = (tag, className, props = {}) => { const node = document.createElement(tag); if (className) node.className = className; Object.assign(node, props); return node; }; const panel = el('div'); panel.id = 'sbc-panel'; const inputBox = el('input', 'sbc-input', { type: 'number', value: state.minRating, min: 80, max: 99, title: '最低评分阈值', }); inputBox.onchange = () => { const v = Math.floor(Number(inputBox.value)); if (Number.isFinite(v) && v >= 45 && v <= 99) { state.minRating = v; GM_setValue(config.MIN_RATING_KEY, v); } else { inputBox.value = state.minRating; } }; const mkBtn = (text, cls, id) => el('button', `sbc-btn ${cls}`, { textContent: text, id }); const statsBox = el('div', 'sbc-stats'); const mkCard = (cfg) => { const card = document.createElement('div'); card.className = 'sbc-stat-card'; let label; let id; if (cfg.type === 'storage') { label = '仓库'; id = 'stat-storage'; } else { label = `${cfg.range[0]}–${cfg.range[1]}`; id = `stat-${cfg.range[0]}-${cfg.range[1]}`; } card.innerHTML = ` <div class="sbc-stat-label">${label}</div> <div class="sbc-stat-value" id="${id}">0</div> `; return card; }; for (const cfg of config.RANGES) statsBox.appendChild(mkCard(cfg)); state.btn.do = mkBtn('猛猛干', 'sbc-btn--do', 'btn-do-sbc'); state.btn.do.onclick = async () => { if (state.isStopping) return; if (state.running && state.runningTask === 'doSBC') return await tasks.stopAsync(); if (state.running && state.runningTask !== 'doSBC') await tasks.stopAsync({ suppressAutoRestartOnce: true });; await ui.ensureConfigThenAssign('do'); }; state.btn.open = mkBtn('开包', 'sbc-btn--open', 'btn-open-packs'); state.btn.open.onclick = async () => { if (state.isStopping) return; if (state.running && state.runningTask !== 'openPacks') await tasks.stopAsync({ suppressAutoRestartOnce: true });; if (!state.running) tasks.startOpen(); else if (state.runningTask === 'openPacks') await tasks.stopAsync(); }; state.btn.loop = mkBtn('永动机', 'sbc-btn--loop', 'btn-loop'); state.btn.loop.onclick = async () => { if (state.isStopping) return; if (!state.selectedLoopSetId) { await ui.ensureConfigThenAssign('loop'); return; } if (state.running && state.runningTask !== 'loop') await tasks.stopAsync({ suppressAutoRestartOnce: true });; if (!state.running) { await sbc.goToPacks(); tasks.startLoop(); } else if (state.runningTask === 'loop') { await tasks.stopAsync(); } }; state.btn.auto = mkBtn('永动机Pro', 'sbc-btn--loop', 'btn-auto'); state.btn.auto.onclick = async () => { if (state.isStopping) return; if (!state.selectedLoopSetId) { await ui.ensureConfigThenAssign('loop', { autostart: false }); if (!state.selectedLoopSetId) return; } if (state.running && state.runningTask !== 'auto') await tasks.stopAsync({ suppressAutoRestartOnce: true });; if (!state.running) tasks.startAuto(); else if (state.runningTask === 'auto') await tasks.stopAsync({ suppressAutoRestartOnce: true });; }; const chkHandleDup = el('input', 'sbc-chk', { type: 'checkbox', checked: state.enableHandleDuplicate, }); chkHandleDup.onchange = () => { state.enableHandleDuplicate = chkHandleDup.checked; GM_setValue('enableHandleDuplicate', state.enableHandleDuplicate); }; const chkLabel = el('label', 'sbc-chklabel'); chkLabel.append(chkHandleDup, document.createTextNode('提前分配重复球员')); const chkAutoRestart = el('input', 'sbc-chk', { type: 'checkbox', checked: state.autoRestartOnStop, }); chkAutoRestart.onchange = () => { state.autoRestartOnStop = chkAutoRestart.checked; GM_setValue(CFG_KEYS.AUTO_RESTART_ON_STOP, state.autoRestartOnStop); }; const chkARLabel = el('label', 'sbc-chklabel'); chkARLabel.append(chkAutoRestart, document.createTextNode('Pro停止后\n自动重启')); const btnAssign = el('button', 'sbc-btn sbc-btn--assign', { textContent: '获取配置' }); btnAssign.onclick = async () => { const txt0 = btnAssign.textContent; try { if (!state.FILTERED_SETS.length) { btnAssign.disabled = true; btnAssign.textContent = '获取中…'; const sets = await sbc.fetchSbcList(); const list = Array.isArray(sets) ? sets : state.FILTERED_SETS; btnAssign.textContent = Array.isArray(list) && list.length > 0 ? '分配SBC' : txt0; return; } await ui.ensureConfigThenAssign(); } catch (e) { btnAssign.textContent = txt0; alert('获取配置失败,请稍后重试'); } finally { btnAssign.disabled = false; } }; const btnCfg = mkBtn('参数设置', 'sbc-btn--settings', 'btn-sbc-settings'); btnCfg.onclick = () => ui.openSbcSettings(); const btnDonate = mkBtn('打赏一下', 'sbc-btn--donate', 'btn-donate'); btnDonate.onclick = () => ui.showDonateModal(); panel.append( statsBox, inputBox, state.btn.do, state.btn.open, state.btn.loop, state.btn.auto, btnAssign, btnCfg, btnDonate, chkLabel, chkARLabel ); let side = GM_getValue('pandaDockSide', 'right'); let top = Number(GM_getValue('pandaDockTop', 140)) || 140; let autohide = !!GM_getValue('pandaDockAutohide', false); const dock = document.createElement('div'); dock.id = 'panda-dock'; dock.className = side; dock.style.top = `${top}px`; const handle = document.createElement('div'); handle.className = 'dock-handle'; handle.title = '点击展开/收起;拖动上下移动;双击切换左右'; handle.textContent = `PANDA SBC v${config.version}`; const panelWrap = document.createElement('div'); panelWrap.className = 'dock-panel'; panelWrap.appendChild(panel); const foot = document.createElement('div'); foot.className = 'dock-foot'; const toggle = document.createElement('span'); toggle.className = 'dock-toggle'; const setAutoText = () => (toggle.textContent = autohide ? '自动隐藏:开' : '自动隐藏:关'); setAutoText(); toggle.onclick = () => { autohide = !autohide; GM_setValue('pandaDockAutohide', autohide); setAutoText(); if (!autohide) expand(); else collapse(); }; const guideBtn = document.createElement('span'); guideBtn.className = 'dock-toggle dock-guide'; guideBtn.textContent = '功能引导'; guideBtn.onclick = () => { try { Guide.init({ force: true, showUI: true }); } catch (_) { alert('引导模块未就绪'); } }; foot.appendChild(toggle); foot.appendChild(guideBtn); panelWrap.appendChild(foot); if (side === 'right') { dock.append(panelWrap, handle); } else { dock.append(handle, panelWrap); } document.body.appendChild(dock); let expanded = !autohide; const expand = () => { dock.classList.add('expanded'); expanded = true; }; const collapse = () => { if (autohide) { dock.classList.remove('expanded'); expanded = false; } }; if (expanded) dock.classList.add('expanded'); let hovering = false; let leaveTimer = null; let clickTimer = null; let ignoreLeaveUntil = 0; dock.addEventListener('pointerenter', () => { hovering = true; if (autohide) expand(); if (leaveTimer) clearTimeout(leaveTimer); }); dock.addEventListener('pointerleave', () => { hovering = false; if (!autohide) return; if (Date.now() < ignoreLeaveUntil) return; if (leaveTimer) clearTimeout(leaveTimer); leaveTimer = setTimeout(() => { if (!hovering) collapse(); }, 220); }); handle.addEventListener('click', (e) => { if (e.detail > 1) return; if (clickTimer) clearTimeout(clickTimer); clickTimer = setTimeout(() => { expanded ? collapse() : expand(); clickTimer = null; }, 180); }); handle.addEventListener('dblclick', () => { if (clickTimer) { clearTimeout(clickTimer); clickTimer = null; } side = side === 'right' ? 'left' : 'right'; GM_setValue('pandaDockSide', side); dock.classList.remove('left', 'right'); dock.classList.add(side); panelWrap.remove(); handle.remove(); if (side === 'right') { dock.append(panelWrap, handle); } else { dock.append(handle, panelWrap); } expand(); ignoreLeaveUntil = Date.now() + 300; setTimeout(() => { if (autohide && !hovering) collapse(); }, 350); }); let dragging = false; let startY = 0; let startTop = 0; const onMove = (e) => { if (!dragging) return; const dy = e.clientY - startY; const newTop = util.clamp(startTop + dy, 20, window.innerHeight - 160); dock.style.top = `${newTop}px`; }; const onUp = () => { if (!dragging) return; dragging = false; dock.classList.remove('dragging'); document.removeEventListener('mousemove', onMove); document.removeEventListener('mouseup', onUp); GM_setValue('pandaDockTop', parseInt(dock.style.top, 10) || 140); }; handle.addEventListener('mousedown', (e) => { if (e.button !== 0) return; dragging = true; dock.classList.add('dragging'); startY = e.clientY; startTop = parseInt(dock.style.top || '140', 10) || 140; document.addEventListener('mousemove', onMove); document.addEventListener('mouseup', onUp); }); document.addEventListener('mouseout', (e) => { if (autohide && e.relatedTarget == null) { hovering = false; collapse(); } }); document.addEventListener('pointermove', (e) => { if (!autohide || !expanded) return; const margin = 8; if (side === 'right' && e.clientX < window.innerWidth - 240 - margin && !hovering) collapse(); if (side === 'left' && e.clientX > 240 + margin && !hovering) collapse(); }); window.addEventListener('blur', () => { if (autohide) collapse(); }); window.addEventListener('resize', () => { const curTop = parseInt(dock.style.top || '140', 10) || 140; const maxTop = Math.max(20, window.innerHeight - 160); if (curTop > maxTop) { dock.style.top = `${maxTop}px`; GM_setValue('pandaDockTop', maxTop); } }); ui.updateButtonState(); }, }; function init() { const ensure = () => { try { ea.ensureHooks(); } catch (e) { } }; ensure(); state.page._eaHookTimer = setInterval(() => { ensure(); if ( ea.hookXHR() && ea.hookEventsPopup() && ea.hookLoadingEnd() && ea.hookRepositories() && state.page._eaHookTimer ) { clearInterval(state.page._eaHookTimer); state.page._eaHookTimer = null; log.i('[init] hooks ready, timer stopped'); } }, 1500); ui.initDock(); } function loadSbcSettingsFromStorage() { const num = (k, def) => { const v = Number(GM_getValue(k, def)); return Number.isFinite(v) ? v : def; }; config.UI.SP_STABLE_FOR = num(CFG_KEYS.SP_STABLE_FOR, config.UI.SP_STABLE_FOR); config.UI.SP_FILL_SUCCESS_TIME = num( CFG_KEYS.SP_FILL_SUCCESS_TIME, config.UI.SP_FILL_SUCCESS_TIME, ); config.HIGH_RATED_POPUP_THRESHOLD = num( CFG_KEYS.HIGH_RATED_POPUP_THRESHOLD, config.HIGH_RATED_POPUP_THRESHOLD, ); config.MAX_RATING_TOTW = num(CFG_KEYS.MAX_RATING_TOTW, config.MAX_RATING_TOTW); config.MAX_RATING_NORMAL = num(CFG_KEYS.MAX_RATING_NORMAL, config.MAX_RATING_NORMAL); // config.PRO.TOTW_SAFE_MIN = num(CFG_KEYS.TOTW_SAFE_MIN, config.PRO.TOTW_SAFE_MIN); } const SHIELD_SEL = '.ut-click-shield.showing, .ut-click-shield.showing.fsu-loading'; function _waitForLoadingStart(timeout = 2000, opts = {}) { const { interval = 120, signal = (state?.abortCtrl?.signal) } = opts; return new Promise((resolve, reject) => { const t0 = Date.now(); let timer = null; const onAbort = () => { clearTimeout(timer); reject(new Error('Aborted')); }; if (signal) { if (signal.aborted) return onAbort(); signal.addEventListener('abort', onAbort, { once: true }); } const step = () => { if (signal?.aborted) return; if (document.querySelector(SHIELD_SEL)) { if (signal) signal.removeEventListener('abort', onAbort); return resolve(true); } if (Date.now() - t0 > timeout) { if (signal) signal.removeEventListener('abort', onAbort); return resolve(false); } timer = setTimeout(step, interval); }; step(); }); } function _waitForLoadingEnd(timeout = 8000, opts = {}) { const { interval = 200, stableGoneMs = 600, signal = (state?.abortCtrl?.signal) } = opts; return new Promise((resolve, reject) => { const start = Date.now(); let lastGoneAt = 0; let timer = null; const onAbort = () => { clearTimeout(timer); reject(new Error('Aborted')); }; if (signal) { if (signal.aborted) return onAbort(); signal.addEventListener('abort', onAbort, { once: true }); } const step = () => { if (signal?.aborted) return; const present = !!document.querySelector(SHIELD_SEL); if (!present) { if (!lastGoneAt) lastGoneAt = Date.now(); if (Date.now() - lastGoneAt >= stableGoneMs) { if (signal) signal.removeEventListener('abort', onAbort); return resolve(true); } } else { lastGoneAt = 0; } if (Date.now() - start > timeout) { if (signal) signal.removeEventListener('abort', onAbort); return resolve(false); } timer = setTimeout(step, interval); }; step(); }); } async function _waitFSULoading(timeout = 8000, opts = {}) { const appeared = await _waitForLoadingStart(10000, opts); if (!appeared) return false; const vanished = await _waitForLoadingEnd(timeout, opts); return vanished; } const recover = (() => { const KEY = 'panda_recover_ticket_v3'; const NAME_PREFIX = '__PANDA_RECOVER__::'; const CFG = { MAX_ATTEMPTS: 2, STEP_RETRY_MAX: 2, WAIT_QUERY: 10000, WAIT_BTN_TRANSITION: 15000, POLL_INTERVAL: 150, CLICK_DEBOUNCE_MS: 1500, WAIT_HOME_MS: 180000, WAIT_LOADING_MS: 120000, AUTOFILL_TIMEOUT: 15000, }; const GMX = { async set(key, val) { const s = typeof val === 'string' ? val : JSON.stringify(val); try { if (typeof GM?.setValue === 'function') return await GM.setValue(key, s); } catch { } try { if (typeof GM_setValue === 'function') { GM_setValue(key, s); return; } } catch { } try { localStorage.setItem(key, s); } catch { } }, async get(key) { let s = null; try { if (typeof GM?.getValue === 'function') s = await GM.getValue(key, null); } catch { } if (s == null) { try { if (typeof GM_getValue === 'function') s = GM_getValue(key); } catch { } } if (s == null) { try { s = localStorage.getItem(key); } catch { } } return s; }, async del(key) { try { if (typeof GM?.deleteValue === 'function') return await GM.deleteValue(key); } catch { } try { if (typeof GM_deleteValue === 'function') { GM_deleteValue(key); return; } } catch { } try { localStorage.removeItem(key); } catch { } }, }; const store = { async set(v) { const s = JSON.stringify(v); await GMX.set(KEY, s); try { window.name = NAME_PREFIX + s; } catch { } }, async get() { let s = await GMX.get(KEY); if (!s && typeof window.name === 'string' && window.name.startsWith(NAME_PREFIX)) { s = window.name.slice(NAME_PREFIX.length); } try { return s ? JSON.parse(s) : null; } catch { return null; } }, async del() { await GMX.del(KEY); try { if (typeof window.name === 'string' && window.name.startsWith(NAME_PREFIX)) window.name = ''; } catch { } }, async patch(extra = {}) { const old = await store.get() || {}; await store.set({ ...old, ...extra, lastAt: Date.now() }); }, }; function hasTaskId(payload) { const id = String( state?.selectedLoopSetId || payload?.selectedLoopSetId || payload?.targetId || '' ).trim(); return !!id; } const isLoginPage = () => /signin|login|auth|juno\/login/i.test(location.hostname + location.pathname + location.search + location.hash); function findFutLoginBtn(rootSel = '.ut-login-content') { const root = document.querySelector(rootSel); if (!root) return null; const exact = root.querySelector('button.btn-standard.call-to-action'); if (exact && (exact.textContent || '').trim() === '登录') return exact; const nodes = root.querySelectorAll('button,[role="button"],.btn,.call-to-action'); return Array.from(nodes).find(n => (n.textContent || '').trim() === '登录') || null; } const getLogInBtn = () => document.querySelector('#logInBtn'); const getLogInBtnText = () => { const el = getLogInBtn(); return el ? (el.textContent || el.value || '').trim() : ''; }; const waitUntil = util.withAbort((pred, timeout = CFG.WAIT_BTN_TRANSITION, interval = CFG.POLL_INTERVAL) => { return new Promise((resolve, reject) => { const signal = state?.abortCtrl?.signal; const start = Date.now(); let timer = null; const step = () => { if (signal?.aborted) { if (timer) clearTimeout(timer); return reject(new AbortedError()); } let ok = false; try { ok = !!pred(); } catch { ok = false; } if (ok) { if (timer) clearTimeout(timer); return resolve(true); } if (Date.now() - start >= timeout) { if (timer) clearTimeout(timer); return resolve(false); } timer = setTimeout(step, interval); }; step(); }); }); const waitBtnTransition = util.withAbort(async ({ expectTextChangeTo = null, timeout = CFG.WAIT_BTN_TRANSITION } = {}) => { const before = getLogInBtnText(); const goneP = dom.waitGone('#logInBtn', timeout, { interval: CFG.POLL_INTERVAL }) .then(() => true).catch(() => false); const textChangedP = waitUntil(() => { const now = getLogInBtnText(); if (!now) return false; return expectTextChangeTo ? (now === expectTextChangeTo) : (now !== before); }, timeout, CFG.POLL_INTERVAL).then(() => true).catch(() => false); const leftPageP = waitUntil(() => !isLoginPage(), timeout, CFG.POLL_INTERVAL) .then(() => true).catch(() => false); return await Promise.race([goneP, textChangedP, leftPageP]); }); function __jq() { return (typeof unsafeWindow !== 'undefined' && unsafeWindow.jQuery) || (typeof window !== 'undefined' && (window.jQuery || window.$)) || null; } function __jqCommit($el, val) { if (!$el || !$el.length) return false; const el = $el.get(0); try { const desc = Object.getOwnPropertyDescriptor(HTMLInputElement.prototype, 'value'); if (desc && desc.set) desc.set.call(el, val); } catch { } $el.val(val); try { el.dispatchEvent(new InputEvent('input', { bubbles: true })); } catch { el.dispatchEvent(new Event('input', { bubbles: true })); } try { el.dispatchEvent(new KeyboardEvent('keyup', { bubbles: true, cancelable: true, key: '', code: '', keyCode: 0, which: 0 })); } catch { } el.dispatchEvent(new Event('change', { bubbles: true })); el.blur?.(); return true; } async function __jqClick(selectorOrPick) { const $ = __jq(); if (!$) return false; let el = null; if (typeof selectorOrPick === 'string') el = document.querySelector(selectorOrPick); else if (typeof selectorOrPick === 'function') el = selectorOrPick(); if (el) { try { $(el).trigger('click'); return true; } catch { } } let $btn = $('#logInBtn'); if (!$btn.length) { $btn = $('button, a, .btn, [role="button"]').filter(function () { const t = ($(this).val() || $(this).text() || '').trim(); return t === '下一步' || t === '登录'; }); $btn = $btn.first(); } if ($btn && $btn.length) { $btn.trigger('click'); return true; } return false; } function __ensureAutofillCSS() { if (document.getElementById('panda-autofill-style')) return; const style = document.createElement('style'); style.id = 'panda-autofill-style'; style.textContent = ` @keyframes pandaAutofillStart { from { opacity: 1; } to { opacity: 1; } } input:-webkit-autofill { animation-name: pandaAutofillStart; } `; document.head.appendChild(style); } const __waitAutofill = util.withAbort(async (elOrSelector, timeout = CFG.AUTOFILL_TIMEOUT) => { const el = typeof elOrSelector === 'string' ? document.querySelector(elOrSelector) : elOrSelector; if (!el) return ''; if (!document.getElementById('panda-autofill-style')) { const style = document.createElement('style'); style.id = 'panda-autofill-style'; style.textContent = ` @keyframes pandaAutofillStart { from { opacity: 1; } to { opacity: 1; } } input:-webkit-autofill { animation-name: pandaAutofillStart; } `; document.head.appendChild(style); } const signal = state?.abortCtrl?.signal; const start = Date.now(); let autoFilled = false; const onAnim = (e) => { if (e.animationName === 'pandaAutofillStart') autoFilled = true; }; try { el.addEventListener('animationstart', onAnim, { passive: true }); try { el.focus(); } catch { } while (Date.now() - start < timeout) { util.abortPoint(); if ((el.value || '').trim()) break; if (autoFilled && el.value) break; await util.sleep(CFG.POLL_INTERVAL); } return (el.value || '').trim(); } finally { try { el.removeEventListener('animationstart', onAnim); } catch { } } }); function _isVisible(el) { if (!el) return false; const cs = getComputedStyle(el); if (cs.display === 'none' || cs.visibility === 'hidden' || el.hidden) return false; const r = el.getBoundingClientRect(); return r.width > 0 && r.height > 0; } function detectPagePhase() { if (!isLoginPage()) { const futBtn = findFutLoginBtn(); return { page: 'FUT', futLoginBtn: futBtn }; } const btn = getLogInBtn(); const btnTxt = btn ? (btn.textContent || btn.value || '').trim() : ''; const emailEl = document.querySelector('#email'); const passEl = document.querySelector('#password') || document.querySelector('input[type="password"]'); const emailVis = _isVisible(emailEl); const passVis = _isVisible(passEl); if (passVis || btnTxt === '登录') return { page: 'LOGIN_SUBMIT', btnTxt }; if (emailVis || btnTxt === '下一步') return { page: 'LOGIN_NEXT', btnTxt }; return { page: 'LOGIN_UNKNOWN', btnTxt }; } async function buildTicketForStep2(email, password) { const old = await store.get() || {}; await store.set({ phase: 'STEP2_NEXT', attempts: (old.attempts || 0) + 1, step2Tries: 0, step3Tries: 0, lastClickAt_step2: 0, lastClickAt_step3: 0, email: (email || '').trim(), password: String(password || ''), at: Date.now(), }); return true; } async function reconcileStateWithDOM() { const payload = await store.get(); const phaseDOM = detectPagePhase(); if (phaseDOM.page === 'FUT') { return { payload, phase: phaseDOM }; } if (!payload) { const base = { attempts: 1, step2Tries: 0, step3Tries: 0, lastClickAt: 0, email: (config?.PRO?.LOGIN_EMAIL || '').trim(), password: String(config?.PRO?.LOGIN_PASSWORD ?? ''), selectedLoopSetId: String(state?.selectedLoopSetId || ''), at: Date.now() }; const phase = (phaseDOM.page === 'LOGIN_SUBMIT') ? 'STEP3_LOGIN' : (phaseDOM.page === 'LOGIN_NEXT') ? 'STEP2_NEXT' : 'STEP2_NEXT'; await store.set({ ...base, phase }); return { payload: await store.get(), phase: phaseDOM }; } if (phaseDOM.page === 'LOGIN_SUBMIT' && payload.phase !== 'STEP3_LOGIN') { await store.patch({ phase: 'STEP3_LOGIN' }); } else if (phaseDOM.page === 'LOGIN_NEXT' && payload.phase !== 'STEP2_NEXT') { await store.patch({ phase: 'STEP2_NEXT' }); } return { payload: await store.get(), phase: detectPagePhase() }; } async function step1_FUT_GotoLogin() { const futLogin = findFutLoginBtn(); if (!futLogin) return false; const email = (config?.PRO?.LOGIN_EMAIL || '').trim(); const password = (config?.PRO?.LOGIN_PASSWORD || ''); await buildTicketForStep2(email, password); log.i('[recover] Step1: FUT 发现“登录”,已记录 phase=STEP2_NEXT,并跳转登录页'); await dom.clickIfExists( () => findFutLoginBtn(), 5000, 60, { strict: false, stableFor: 64, preferLast: true, signal: state?.abortCtrl?.signal }, true ); return true; } const step2_Login_Next = util.withAbort(async () => { await dom.waitForElement(() => document.querySelector('#email') || getLogInBtn(), CFG.WAIT_QUERY, { strict: false, stableFor: 64, preferLast: true, signal: state?.abortCtrl?.signal }); const payload = await store.get(); const attempts = payload?.attempts || 1; if (attempts > CFG.MAX_ATTEMPTS) { await store.del(); return false; } const tries = payload?.step2Tries || 0; if (tries >= CFG.STEP_RETRY_MAX) { log.w('[recover] Step2: 重试超限'); await store.del(); return false; } await store.patch({ step2Tries: tries + 1 }); { const now = Date.now(); const last = payload?.lastClickAt_step2 || 0; const gap = now - last; if (gap < CFG.CLICK_DEBOUNCE_MS) { await util.sleep(CFG.CLICK_DEBOUNCE_MS - gap + 50); } await store.patch({ lastClickAt_step2: Date.now() }); } const $ = __jq(); const email = (payload?.email || (config?.PRO?.LOGIN_EMAIL || '')).trim(); const $mail = $ ? $('#email') : null; const mailEl = document.querySelector('#email'); if (!email || !mailEl) { log.w('[recover] Step2: 缺少邮箱或 #email,不执行“下一步”;清空进度'); await store.del(); return false; } if ($) { __jqCommit($mail, email); $('#loginMethod').val('emailPassword'); $('#online-input-error-email,#online-general-error,#offline-auth-error').removeClass('otkform-group-haserror'); $('.otkinput-grouped').removeClass('otkinput-iserror'); } else { try { const desc = Object.getOwnPropertyDescriptor(HTMLInputElement.prototype, 'value'); desc && desc.set && desc.set.call(mailEl, email); } catch { } mailEl.value = email; mailEl.dispatchEvent(new Event('input', { bubbles: true })); mailEl.dispatchEvent(new Event('change', { bubbles: true })); mailEl.blur?.(); const lm = document.querySelector('#loginMethod'); if (lm) lm.value = 'emailPassword'; } await store.patch({ phase: 'STEP3_LOGIN' }); await __jqClick('#logInBtn'); await dom.clickIfExists( () => getLogInBtn(), 5000, 60, { strict: false, stableFor: 64, preferLast: true, signal: state?.abortCtrl?.signal }, true ); const ok = await waitBtnTransition({ expectTextChangeTo: '登录', timeout: CFG.WAIT_BTN_TRANSITION }); const stillLogin = isLoginPage(); const btnTxt = getLogInBtnText(); if (!ok && stillLogin && btnTxt === '下一步') { log.w('[recover] Step2: 仍是“下一步”,判失败,清空进度'); await store.del(); return false; } log.i('[recover] Step2: 下一步完成,进入 Step3 或已离开登录页'); return true; }); const step3_Login_Submit = util.withAbort(async () => { await dom.waitForElement(() => document.querySelector('#password') || getLogInBtn(), CFG.WAIT_QUERY, { strict: false, stableFor: 64, preferLast: true, signal: state?.abortCtrl?.signal }); { const payload = await store.get(); const now = Date.now(); const last = payload?.lastClickAt_step3 || 0; const gap = now - last; if (gap < CFG.CLICK_DEBOUNCE_MS) { const wait = CFG.CLICK_DEBOUNCE_MS - gap + 50; log.i(`debounce ${gap}ms < ${CFG.CLICK_DEBOUNCE_MS}ms → sleep ${wait}ms`); await util.sleep(wait); } await store.patch({ lastClickAt_step3: Date.now() }); } const $ = __jq(); const passEl = document.querySelector('#password') || document.querySelector('input[type="password"]'); let payload = await store.get(); let pwd = payload?.password ?? config?.PRO?.LOGIN_PASSWORD ?? ''; const btnBefore = getLogInBtnText(); log.i(`start. domPhase=%o btnText="%s"`, detectPagePhase(), btnBefore); if (passEl && !String(pwd)) { const got = await __waitAutofill(passEl, CFG.AUTOFILL_TIMEOUT); if (got) pwd = got; } if (!passEl || !String(pwd)) { log.i('密码为空. clear ticket.'); await store.del(); return false; } if ($) { __jqCommit($(passEl), String(pwd)); } else { try { const desc = Object.getOwnPropertyDescriptor(HTMLInputElement.prototype, 'value'); desc && desc.set && desc.set.call(passEl, String(pwd)); } catch { } passEl.value = String(pwd); try { passEl.dispatchEvent(new InputEvent('input', { bubbles: true })); } catch { passEl.dispatchEvent(new Event('input', { bubbles: true })); } passEl.dispatchEvent(new Event('change', { bubbles: true })); passEl.blur?.(); } try { const ev = new KeyboardEvent('keydown', { bubbles: true, cancelable: true, key: 'Enter', code: 'Enter', keyCode: 13, which: 13 }); passEl.dispatchEvent(ev); } catch { } if ($) { try { $('#logInBtn').trigger('click'); } catch { } } await dom.clickIfExists( () => getLogInBtn(), 5000, 60, { strict: false, stableFor: 64, preferLast: true, signal: state?.abortCtrl?.signal }, true ); const ok = await Promise.race([ dom.waitGone('#logInBtn', CFG.WAIT_BTN_TRANSITION, { interval: CFG.POLL_INTERVAL }).then(() => true).catch(() => false), waitUntil(() => !isLoginPage(), CFG.WAIT_BTN_TRANSITION, CFG.POLL_INTERVAL).then(() => true).catch(() => false), waitUntil(() => getLogInBtnText() !== btnBefore, CFG.WAIT_BTN_TRANSITION, CFG.POLL_INTERVAL).then(() => true).catch(() => false), ]); if (isLoginPage()) { const errBox = document.querySelector('#online-general-error .otkc'); const errCode = (document.querySelector('#errorCode') || {}).value || ''; const errCodeDesc = (document.querySelector('#errorCodeWithDescription') || {}).value || ''; const errTxt = errBox ? (errBox.textContent || '').trim() : ''; return false; } await store.del(); return true; }); async function resumeProAfterReload() { try { await Guide.onceHomeReady(CFG.WAIT_HOME_MS); } catch { } await util.sleep(5000); await _waitFSULoading(12 * 60 * 1000); await dom.clickIfExists( () => document.querySelector('.sbc-btn--assign'), 6000, 60, { strict: false, stableFor: 64, preferLast: true, signal: state?.abortCtrl?.signal }, true ); try { await ea.waitAllLoadingEnd?.(800, 60000); } catch { } try { await sbc.fetchSbcList?.(); } catch { } const payload = await store.get(); const targets = await sbc.ensureAutoTargets?.() || {}; state.selectedLoopSetId = String(payload?.selectedLoopSetId || targets?.loopId || ''); await util.sleep(500); if (typeof tasks.startAuto === 'function') { state.isStopping = false; try { await tasks.startAuto.call(tasks); log.i('[recover] resume: 已触发 Pro 启动'); } catch (e) { log.w('[recover] resume: 启动异常', e); } finally { await store.del(); } return true; } else { log.w('[recover] resume: 未找到启动函数'); await store.del(); return false; } } async function triggerAndReload(reason = 'chain_exhausted') { if (state.__recovering) { return false; } const loopId = String(state?.selectedLoopSetId || '').trim(); if (!loopId) { await store.del(); return false; } state.__recovering = true; const old = await store.get(); const attempts = (old?.attempts || 0) + 1; await store.set({ phase: 'RESUME_PRO', attempts, reason, selectedLoopSetId: loopId, email: (config?.PRO?.LOGIN_EMAIL || '').trim(), password: String(config?.PRO?.LOGIN_PASSWORD ?? ''), lastOrigin: location.origin, seq: (old?.seq || 0) + 1, at: Date.now(), }); log.w(`[recover] 第 ${attempts} 次尝试,刷新页面…`, { reason, loopId: state?.selectedLoopSetId }); location.reload(); return false; } const tryOnBoot = util.withAbort(async () => { state.__recovering = false; const { payload, phase } = await reconcileStateWithDOM(); console.log(payload, phase) if (!hasTaskId(payload)) { log.i('[recover] 无凭证 -> 跳过恢复 & 清理凭证'); try { store.del(); } catch { } return false; } if (phase.page === 'FUT') { if (payload?.phase === 'RESUME_PRO') { const appear = await dom.waitForElement( () => findFutLoginBtn(), 20000, { strict: false, stableFor: 64, preferLast: true, signal: state?.abortCtrl?.signal } ); if (!appear) return false; const vanished = await dom.waitGone('.ut-login-content .btn-standard.call-to-action', 30000, { interval: 200 }); const stillThere = !!findFutLoginBtn(); if (vanished && !stillThere) { log.i('[recover] FUT 登录按钮在等待内消失,判定已登录或不需登录。'); return await resumeProAfterReload(); } else { return await step1_FUT_GotoLogin(); } } } const fresh = await store.get(); const attempts = fresh?.attempts || 1; if (attempts > CFG.MAX_ATTEMPTS) { log.w('[recover] 超过最大尝试次数,清理进度'); await store.del(); return false; } if (phase.page === 'LOGIN_NEXT') { return await step2_Login_Next(); } if (phase.page === 'LOGIN_SUBMIT') { console.log('step3_Login_Submit') return await step3_Login_Submit(); } log.i(`[recover] 登录页但状态不明(${phase.page}),等待下次机会。`); return false; }); return { tryOnBoot, triggerAndReload, _store: store }; })(); return { config, state, log, util, dom, ea, sbc, tasks, ui, init, loadSbcSettingsFromStorage, recover }; })(); const Guide = (() => { const KEY_SHOWN = 'panda_guide_shown_v1'; const HOME_H1_TEXT = '主页'; const WAIT_TIMEOUT_MS = 600000; const OBS_ROOT = document.body || document.documentElement; const hasShown = () => !!GM_getValue(KEY_SHOWN, false); const setShown = () => GM_setValue(KEY_SHOWN, true); function findHomeTitleNode() { const nodes = document.querySelectorAll('h1.title'); for (const n of nodes) { const txt = (n.textContent || '').trim(); if (txt === HOME_H1_TEXT) return n; } return null; } function detectDeps() { const fsuInstalled = !!document.querySelector('.fsu-loading-close'); const enhancerInstalled = !!( document.querySelector('.icon-enhancer') || document.querySelector('[class*="icon-enhancer"]') ); return { fsuInstalled, enhancerInstalled }; } function showOverlay({ fsuInstalled, enhancerInstalled }) { if (document.querySelector('.panda-guide-mask')) return; const RESOURCES = [ { label: '作者:伯纳乌书童甲', href: 'https://space.bilibili.com/23274961', note: 'B站链接' }, { label: 'FC25 PandaSBC 1群', href: 'https://qm.qq.com/q/zSDFaDZ1UA', note: '点击入群求助' }, { label: '安装教程', href: 'https://b23.tv/rg1dQVR', note: 'B站链接' }, { label: '使用教程', href: 'https://b23.tv/qo1svsY', note: 'B站链接' }, ]; const mask = document.createElement('div'); mask.style.cssText = ` position:fixed;inset:0;z-index:999999; background:rgba(0,0,0,0.55);display:flex;align-items:center;justify-content:center; font-family:system-ui,-apple-system,Segoe UI,Roboto,Helvetica,Arial; `; const box = document.createElement('div'); box.style.cssText = ` width:520px;max-width:calc(100vw - 40px);background:#1f1f1f;color:#eee;border:1px solid #333; border-radius:12px;box-shadow:0 10px 30px rgba(0,0,0,.5);padding:18px; `; const ok = (flag) => (flag ? '✅ 已检测到' : '⚠️ 未检测到'); const linksHTML = RESOURCES.map( (r) => ` <li style=" margin:6px 0; display:flex; align-items:center; justify-content:space-between; gap:10px; "> <div style="display:flex;align-items:center;gap:8px;"> ${r.note ? `<span style="font-size:12px;color:#bbb;">${r.note}</span>` : ''} <a href="${r.href}" target="_blank" rel="noopener noreferrer" style="color:#4ec9ff;text-decoration:none;">${r.label}</a> </div> <button data-copy="${r.href}" style=" min-width:64px;height:26px;border-radius:6px;border:1px solid #555; background:#2b2b2b;color:#ddd;cursor:pointer;font-size:12px; ">复制</button> </li> `, ).join(''); const html = ` <div style="font-size:16px;font-weight:700;margin-bottom:10px;">pandaSBC 新手引导(免费插件,谨防受骗)</div> <div style="font-size:14px;line-height:1.6;"> <div>• FSU:${ok(fsuInstalled)}</div> <div>• Enhancer:${ok(enhancerInstalled)}</div> </div> <div style="font-size:13px;margin:12px 0 6px 0;font-weight:700;">资源与帮助</div> <ul style="list-style:none;padding:0;margin:0;">${linksHTML}</ul> <div style="font-size:12px;color:#bbb;margin-top:10px;"> ${!fsuInstalled || !enhancerInstalled ? '提示:请先安装/启用以上依赖后再使用脚本。' : '环境检测通过,可开始使用。' } </div> <div style="display:flex;gap:8px;justify-content:flex-end;margin-top:14px;"> <button id="panda-guide-close" style=" min-width:88px;height:32px;border-radius:8px;border:1px solid #555; background:#2b2b2b;color:#ddd;cursor:pointer; ">我知道了</button> </div> `; box.innerHTML = html; mask.appendChild(box); document.body.appendChild(mask); box.addEventListener('click', async (e) => { const btn = e.target.closest('button[data-copy]'); if (!btn) return; const text = btn.getAttribute('data-copy') || ''; try { if (navigator.clipboard && navigator.clipboard.writeText) { await navigator.clipboard.writeText(text); } else { const ta = document.createElement('textarea'); ta.value = text; ta.style.position = 'fixed'; ta.style.opacity = '0'; document.body.appendChild(ta); ta.select(); document.execCommand('copy'); document.body.removeChild(ta); } btn.textContent = '已复制'; setTimeout(() => (btn.textContent = '复制'), 1200); } catch { alert('复制失败:' + text); } }); const closeBtn = box.querySelector('#panda-guide-close'); closeBtn.addEventListener('click', () => { try { document.body.removeChild(mask); } catch { } }); } function onceHomeReady() { const SEL = '.ut-tab-bar-item.icon-home'; return new Promise((resolve) => { const node = document.querySelector(SEL); if (node) return resolve(node); const mo = new MutationObserver(() => { const n = document.querySelector(SEL); if (n) { mo.disconnect(); resolve(n); } }); mo.observe(document.body || document.documentElement, { childList: true, subtree: true }); setTimeout(() => { try { mo.disconnect(); } catch { } resolve(null); }, WAIT_TIMEOUT_MS); }); } async function init({ force = false, showUI = true } = {}) { if (!force && hasShown()) { return; } const homeBtn = await onceHomeReady(); if (!homeBtn) return; const status = detectDeps(); setShown(); if (showUI) { showOverlay(status); } else { console.info('[pandaSBC][Guide]', status); } } return { init, detectDeps, onceHomeReady }; })(); window.addEventListener('load', () => { try { PandaSBC.init(); PandaSBC.loadSbcSettingsFromStorage() PandaSBC.recover.tryOnBoot(); const hasShownGuide = GM_getValue(GUIDE_SHOWN_KEY, false); if (!hasShownGuide) { Guide.init({ force: true, showUI: true }); GM_setValue(GUIDE_SHOWN_KEY, true); } } catch (e) { console.error(e); } }); })();