// ==UserScript==
// @name runnansbc
// @namespace http://tampermonkey.net/
// @version 1.0.8
// @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/*
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_addStyle
// @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.2',
DEFAULT_TIMEOUT: 15000,
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,
STORAGE_THRESHOLD: 80,
MIN_RATING_KEY: 'minRating',
DEFAULT_MIN_RATING: 85,
HIGH_RATED_POPUP_THRESHOLD: 98,
MAX_RATING_TOTW: 87,
MAX_RATING_NORMAL: 98,
SLOW_RATE: 1.0,
targetKeywords: [
'阵容变异',
'TOTW 升级',
'FUTTIES 阵容',
'10 名 85+ 升级',
'10 名 84+ 升级',
],
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,
},
};
const CFG_KEYS = {
SLOW_RATE: 'cfg_SLOW_RATE',
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',
};
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,
btn: { loop: null, open: null, do: null, auto: null },
};
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) => new Promise((r) => setTimeout(r, ms * (config.SLOW_RATE || 1))),
nextPaint: () =>
new Promise((r) => requestAnimationFrame(() => requestAnimationFrame(r))),
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 * (config.SLOW_RATE || 1), 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) => {
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 resolveOnce(false);
signal.addEventListener('abort', () => resolveOnce(false), { 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 el = await dom.waitForElement(selectorOrFn, timeout, opt);
if (!el) {
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 * (config.SLOW_RATE || 1), { pollInterval = 800 * (config.SLOW_RATE || 1) } = {}) =>
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 = {
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() {
// Runnan's method: Find and click empty slot directly
log.i('[addPlayer] 使用Runnan方法: 查找并点击空槽位');
// First check if duplicate needs to be added
const hasUnassigned = repositories?.Item?.getUnassignedItems?.()?.length > 0;
if (hasUnassigned) {
log.i('[addPlayer] 检测到未分配卡片,尝试添加重复卡');
// Try to find and click duplicate button
const dupBtn = await dom.waitForElement('button[data-r="add-duplicated"]', 3000 * (config.SLOW_RATE || 1));
if (!dupBtn) {
// Fallback: search by text
const dupBtnAlt = Array.from(document.querySelectorAll('button')).find(b =>
b.textContent.includes('添加重复') ||
b.textContent.includes('Add Duplicate') ||
b.textContent.includes('添加重覆'));
if (dupBtnAlt) {
log.i('[addPlayer] 点击添加重复卡按钮 (通过文本查找)');
dom.simulateClick(dupBtnAlt);
await ea.waitAllLoadingEnd();
await util.sleep(1000);
}
} else {
log.i('[addPlayer] 点击添加重复卡按钮');
dom.simulateClick(dupBtn);
await ea.waitAllLoadingEnd();
await util.sleep(500);
}
}
// Find and click empty slot
const emptySlot = await dom.waitForElement('.ut-squad-slot-view .player.item.empty', 3000 * (config.SLOW_RATE || 1));
if (!emptySlot) {
log.w('[addPlayer] 未找到空槽位,尝试查找其他选择器');
// Try alternative selectors
const altSlot = await dom.waitForElement(() => {
// Look for empty slot with different selectors
const slots = document.querySelectorAll('.ut-squad-slot-view');
for (const slot of slots) {
const playerDiv = slot.querySelector('.player.item');
if (playerDiv && playerDiv.classList.contains('empty')) {
return slot;
}
}
return null;
}, 3000 * (config.SLOW_RATE || 1));
if (!altSlot) {
log.w('[addPlayer] 完全未找到空槽位');
return false;
}
}
log.i('[addPlayer] 找到空槽位,点击它');
const slotView = emptySlot ? emptySlot.closest('.ut-squad-slot-view') : null;
if (slotView) {
dom.simulateClick(slotView);
await util.sleep(800);
// Look for eligibility search button
log.i('[addPlayer] 查找资格搜索按钮');
const eligibilityBtn = await dom.waitForElement(() => {
// First try data attribute
const buttons = document.querySelectorAll('button[data-r="eligibilitysearch"]');
if (buttons.length > 0) return buttons[0];
// Then try by text content
return Array.from(document.querySelectorAll('button')).find(b => {
const text = b.textContent || '';
return text.includes('添加 任意') ||
text.includes('Add Any') ||
text.includes('资格搜索') ||
text.includes('Eligibility') ||
text.includes('Search');
});
}, 5000 * (config.SLOW_RATE || 1));
if (eligibilityBtn) {
log.i('[addPlayer] 执行资格搜索');
dom.simulateClick(eligibilityBtn);
await util.sleep(800);
// Wait for player list to load
await dom.waitForElement('.ut-pinned-list', 3000 * (config.SLOW_RATE || 1));
await util.sleep(300);
// Click the "范围" (Range) filter button to show all players
const rangeBtn = await dom.waitForElement(() => {
const containers = document.querySelectorAll('.pagingContainer');
for (let container of containers) {
const divs = container.querySelectorAll('div');
for (let div of divs) {
// Find the div with "范围" text
if (div.textContent === '范围') {
// Find the button that's a sibling of this div
const btn = div.parentElement.querySelector('button.btn-standard.call-to-action.listfilter-btn');
if (btn) return btn;
}
}
}
return null;
}, 3000 * (config.SLOW_RATE || 1));
if (rangeBtn) {
log.i('[addPlayer] 点击范围按钮,当前状态: ' + rangeBtn.textContent);
dom.simulateClick(rangeBtn);
await util.sleep(500);
}
// Try to add the first player
let firstAddBtn = await dom.waitForElement('.ut-image-button-control.btnAction.add', 2000 * (config.SLOW_RATE || 1));
// If no player found and range button exists, click it again (will change to "仅仓库")
if (!firstAddBtn && rangeBtn) {
log.i('[addPlayer] 没有找到可添加的球员,切换到仅仓库模式');
dom.simulateClick(rangeBtn);
await util.sleep(500);
// Try again to find the first player
firstAddBtn = await dom.waitForElement('.ut-image-button-control.btnAction.add', 2000 * (config.SLOW_RATE || 1));
}
if (firstAddBtn) {
log.i('[addPlayer] 添加第一个球员');
dom.simulateClick(firstAddBtn);
await util.sleep(500);
// Click canvas once to fill remaining slots
log.i('[addPlayer] 点击画布自动填充剩余位置');
const canvas = await dom.waitForElement('.ut-squad-pitch-view--canvas', 3000 * (config.SLOW_RATE || 1));
if (canvas) {
dom.simulateClick(canvas);
await util.sleep(500);
} else {
log.w('[addPlayer] 未找到画布元素');
}
// Check if player was added successfully
const newCount = ea.getFilledCount();
log.i(`[addPlayer] 球员添加完成,当前填充数: ${newCount}`);
return true;
} else {
log.w('[addPlayer] 未找到可添加的球员');
return false;
}
} else {
log.w('[addPlayer] 未找到资格搜索按钮');
}
}
log.w('[addPlayer] 添加球员失败');
return false;
},
collectHiRated(items, threshold = config.HIGH_RATED_POPUP_THRESHOLD) {
const hi = items.filter(
(p) => p.type === 'player' && p.loans === -1 && p.rating >= threshold,
);
state._hiRatedPlayers.push(...hi);
},
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) {
return await util.withAbort(
() =>
new Promise((resolve, reject) => {
if (!items || !items.length) return resolve({ success: true });
services.Item.move(items, pile, true).observe(controller, (e, t) => {
e.unobserve(controller);
if (!t.success) {
alert('移动失败');
return reject(new ExpectError('移动失败'));
}
resolve(t);
});
}),
)();
},
getUnassignedItemsSafe: async () => {
let items = [];
for (let i = 0; i < 3 && !items.length; i++) {
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,
);
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 currentStorage = repositories.Item.numItemsInCache(ItemPile.STORAGE);
const spaceLeft = config.STORAGE_MAX - currentStorage;
// Check if storage is approaching threshold (80+)
if (currentStorage >= config.STORAGE_THRESHOLD) {
log.i(`[handleUnassignedDuplicate] 存储已达阈值 (${currentStorage}/${config.STORAGE_MAX}),执行89 SBC清理空间`);
await sbc.executeStorageCleanup();
}
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 * (config.SLOW_RATE || 1))
.catch(() => null);
if (!modal) return;
const btn = [...modal.querySelectorAll('button')].find((b) => b.textContent.includes('快速出售'));
if (btn) {
console.log('[QuickSell] 点击快速出售按钮');
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,
);
const currentStorage = repositories.Item.numItemsInCache(ItemPile.STORAGE);
const spaceLeft = config.STORAGE_MAX - currentStorage;
// Check if storage is approaching threshold (80+)
if (currentStorage >= config.STORAGE_THRESHOLD) {
log.i(`[handleUnassigned] 存储已达阈值 (${currentStorage}/${config.STORAGE_MAX}),执行89 SBC清理空间`);
await sbc.executeStorageCleanup();
}
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');
try {
view.setInteractionState && view.setInteractionState(false);
} catch { }
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) throw new ExpectError('无导航控制器');
if (sbcSet.hidden) {
const first = challengesResp.data.challenges[0];
await new Promise((resolve, reject) => {
let done = false;
const t = setTimeout(() => {
if (!done) reject(new TimeoutError('loadChallenge超时'));
}, timeout);
services.SBC.loadChallenge(first).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?.(first.id);
if (ch && !ch.squad) ch.update?.(first);
} catch { }
const vc = new UTSBCSquadSplitViewController();
vc.initWithSBCSet?.(sbcSet, first.id);
nav.pushViewController?.(vc);
} else {
const vc = new UTSBCGroupChallengeSplitViewController();
vc.initWithSBCSet?.(sbcSet);
nav.pushViewController?.(vc, true);
try {
nav.setNavigationTitle?.(sbcSet.name);
} catch { }
}
try {
view.setInteractionState && view.setInteractionState(true);
} catch { }
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) {
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 * (config.SLOW_RATE || 1));
} 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 * (config.SLOW_RATE || 1));
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 filteredSets = Object.values(services.SBC.repository.categories._collection)
.filter((cat) => cat.name === '升级')
.flatMap((cat) => cat.setIds)
.map((id) => services.SBC.repository.sets._collection[id])
.filter(
(set) =>
config.targetKeywords.some((keyword) => set.name.includes(keyword)) &&
!set.name.includes('可交易') &&
set.challengesCount === 1 &&
set.repeatabilityMode !== 'NON_REPEATABLE',
);
state.FILTERED_SETS = filteredSets || [];
log.i('FILTERED_SETS', state.FILTERED_SETS);
return state.FILTERED_SETS;
},
async openPacksOnce() {
log.i('[openPacksOnce]');
try {
await ea.waitController('UTStorePackViewController', 20000 * (config.SLOW_RATE || 1));
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 * (config.SLOW_RATE || 1));
await sbc.handleUnassigned(state.minRating);
await ea.waitController('UTStorePackViewController');
await ea.waitAllLoadingEnd();
return true;
} catch (e) {
log.e(e);
tasks.stopAsync();
return false;
}
},
async loopOnce() {
log.i('[loopOnce] start');
try {
// Check storage level at the beginning of loop
const currentStorage = repositories.Item.numItemsInCache(ItemPile.STORAGE);
if (currentStorage >= config.STORAGE_THRESHOLD) {
log.i(`[loopOnce] 检测到存储达阈值 (${currentStorage}/${config.STORAGE_MAX}),执行89 SBC清理空间`);
await sbc.executeStorageCleanup();
}
// Check TOTW count and execute 20 TOTW SBCs if below threshold
const inv = sbc.getInventorySummary();
log.i(`[loopOnce] TOTW检查 - 当前TOTW: ${inv.totw}, 安全阈值: ${config.PRO.TOTW_SAFE_MIN}, TOTW素材: ${inv.cntTotw}, 需要: ${config.PRO.TOTW_NEED}`);
if (inv.totw <= (config.PRO.TOTW_SAFE_MIN+ 1) && inv.cntTotw >= config.PRO.TOTW_NEED) {
log.i(`[loopOnce] TOTW数量不足 (${inv.totw}/${config.PRO.TOTW_SAFE_MIN}),执行20次TOTW SBC`);
await sbc.executeTOTWBatch();
} else {
if (inv.totw >= config.PRO.TOTW_SAFE_MIN) {
log.i(`[loopOnce] TOTW充足,跳过TOTW批处理`);
} else if (inv.cntTotw < config.PRO.TOTW_NEED) {
log.i(`[loopOnce] TOTW素材不足 (${inv.cntTotw}/${config.PRO.TOTW_NEED}),跳过TOTW批处理`);
}
}
await ea.waitController('UTStorePackViewController', 20000 * (config.SLOW_RATE || 1));
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 * (config.SLOW_RATE || 1));
if (state.enableHandleDuplicate) await sbc.handleUnassignedDuplicate(state.minRating);
await sbc.goToSBCSet({ setId: Number(state.selectedLoopSetId) });
const filledCount = ea.getFilledCount();
if (filledCount > 0) ea.squad?.removeAllItems?.();
const rptBtn = await dom.waitForElement(sbc.sel.rptBtn, 5000, { strict: true });
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;
});
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 * (config.SLOW_RATE || 1))
.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);
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() {
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 * (config.SLOW_RATE || 1)),
ea.waitLoadingEndOnce(),
]);
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) {
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 = (squad?.squad?.players || [])
.map((p) => p?.itemData || p)
.filter((it) => it && it.rating >= threshold);
if (hiList.length > 0) {
const maxRating = hiList.reduce((m, p) => Math.max(m, Number(p.rating || 0)), 0);
alert(`检测到高分球员(≥${threshold}),已取消提交。\n数量:${hiList.length},最高:${maxRating}`);
throw new ExpectError('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) return false;
await ea.waitAllLoadingEnd();
log.i('[doSBCOnce] done');
return true;
},
async executeStorageCleanup() {
log.i('[executeStorageCleanup] 开始执行89 SBC清理存储');
try {
// Use existing targets logic to find var89 SBC
const targets = await sbc.ensureAutoTargets();
if (!targets?.var89Id) {
log.w('[executeStorageCleanup] 未找到89变异SBC');
return false;
}
// Use existing runVar89NTimes function (execute 1 time for storage cleanup)
const result = await sbc.runVar89NTimes(targets, 1, { openAfter: true });
if (result > 0) {
log.i(`[executeStorageCleanup] 89 SBC执行成功,完成 ${result} 次`);
return true;
} else {
log.w('[executeStorageCleanup] 89 SBC执行失败');
return false;
}
} catch (error) {
log.e(`[executeStorageCleanup] 错误: ${error.message || error}`);
return false;
}
},
async executeTOTWBatch() {
log.i('[executeTOTWBatch] 开始执行20次TOTW SBC');
try {
// Get targets to find TOTW SBC
const targets = await sbc.ensureAutoTargets();
if (!targets?.totwId) {
log.w('[executeTOTWBatch] 未找到TOTW SBC');
return false;
}
// Fixed: Execute 20 TOTW SBCs
const toExecute = 20;
const originalSetId = state.selectedDoSbcSetId;
state.selectedDoSbcSetId = String(targets.totwId);
let successCount = 0;
try {
// Execute 20 TOTW SBCs without opening packs individually
for (let i = 0; i < toExecute; i++) {
if (state.abortCtrl?.signal?.aborted) break;
const inv = sbc.getInventorySummary();
log.i(`[executeTOTWBatch] 执行第 ${i + 1}/${toExecute} 次TOTW SBC`);
if (inv.cntTotw < config.PRO.TOTW_NEED) {
log.i(`[executeTOTWBatch] 当前TOTW数量: ${inv.cntTotw}, 需要: ${config.PRO.TOTW_NEED}, 执行TOTW SBC`);
break;
}
const ok = await sbc.doSBCOnce();
if (ok) {
successCount++;
} else {
log.w(`[executeTOTWBatch] 第 ${i + 1} 次TOTW SBC失败,停止执行`);
break;
}
// Small delay between SBCs
await util.sleep(300 + Math.random() * 200);
}
// Open all packs at once after completing all SBCs
if (successCount > 0) {
log.i(`[executeTOTWBatch] 开始批量打开 ${successCount} 个包`);
try {
await sbc.openPacksAfterDo(state.selectedDoSbcSetId, successCount);
log.i(`[executeTOTWBatch] 成功打开 ${successCount} 个包`);
} catch (e) {
log.w('[executeTOTWBatch] 批量开包失败:', e.message);
}
}
} finally {
// Restore original selected set ID
state.selectedDoSbcSetId = originalSetId;
}
log.i(`[executeTOTWBatch] 完成! 成功执行 ${successCount}/${toExecute} 次TOTW SBC`);
return successCount > 0;
} catch (error) {
log.e(`[executeTOTWBatch] 错误: ${error.message || error}`);
return false;
}
},
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 * (config.SLOW_RATE || 1));
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 * (config.SLOW_RATE || 1));
await ea.waitAllLoadingEnd();
} catch { }
} else {
spin++;
if (spin > 2) break;
try {
const ctrl = await ea.waitController('UTStorePackViewController', 6000 * (config.SLOW_RATE || 1));
ctrl?.getStorePacks?.(true);
await ea.waitAllLoadingEnd();
} catch { }
}
}
return opened;
},
async openPacksAfterDo(setId = state.selectedDoSbcSetId, expected = null) {
if (!setId) return false;
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) {
try {
const ctrl = await ea.waitController('UTStorePackViewController', 6000 * (config.SLOW_RATE || 1));
ctrl?.getStorePacks?.(true);
await ea.waitAllLoadingEnd();
count = Number(sbc.getPacksNum()) || 0;
} catch { }
}
if (count <= 0) return true;
for (let i = 0; i < count; i++) {
if (state.abortCtrl?.signal?.aborted) throw new AbortedError();
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();
log.i('[auto] 库存:', inv);
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() {
const now = Date.now();
if (state._loopFailStrike >= 2 && now - state._lastLoopFailAt < config.PRO.LOOP_FAIL_BACKOFF_MS) {
log.w('[auto] loop短期失败过多,改由上层兜底89');
return false;
}
await sbc.goToPacks();
const ok = await sbc.loopOnce();
if (!ok) {
state._loopFailStrike++;
state._lastLoopFailAt = now;
} else {
state._loopFailStrike = 0;
}
return ok;
},
async runVar89NTimes(targets, times = 3, { openAfter = true } = {}) {
if (!targets?.var89Id) return 0;
await sbc.ensureSBCHub();
state.selectedDoSbcSetId = String(targets.var89Id);
let okCount = 0;
for (let i = 0; i < times; i++) {
if (state.abortCtrl?.signal?.aborted) throw new AbortedError();
try {
const ok = await sbc.doSBCOnce();
if (!ok) break;
okCount++;
} catch (e) {
if (isAbort(e)) throw e;
if (isHighRatedError(e)) throw e;
break;
}
await util.sleep(300 + Math.random() * 400);
}
if (openAfter && okCount > 0) {
try {
await sbc.openPacksAfterDo(state.selectedDoSbcSetId, okCount);
} catch (e) {
if (!isAbort(e)) log.w(e);
}
}
return okCount > 0;
},
async autoRound() {
if (state.abortCtrl?.signal?.aborted) throw new AbortedError();
const act = sbc.decideAction();
log.i('[auto] 决策:', act);
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();
log.i('[auto] action:', act, 'targets:', targets, 'presetLoopId:', state.selectedLoopSetId);
if (act === 'try_loop') {
const loopId = targets.loopId;
if (!loopId) {
log.i('[auto] 没有可用的SBC');
return false;
}
state.selectedLoopSetId = String(loopId);
try {
await sbc.goToPacks();
const ok = await sbc.tryLoopWithBackoff();
if (ok) return true;
const ok89 = await sbc.runVar89NTimes(targets, 3, { openAfter: true });
if (!ok89) {
log.w('89x3 failed stopping auto.');
return false;
}
return true;
} catch (e) {
if (isAbort(e)) throw e;
const ok89 = await sbc.runVar89NTimes(targets, 3, { openAfter: true });
if (!ok89) {
log.w('89x3 failed stopping auto.');
return false;
}
return true;
}
}
if (act === 'do_totw') {
if (targets.totwId) {
state.selectedDoSbcSetId = String(targets.totwId);
try {
const ok = await sbc.doSBCOnce();
if (ok) {
try {
await sbc.openPacksAfterDo(state.selectedDoSbcSetId, 1);
} catch { }
return true;
}
if (targets.var89Id) {
const ok89 = await sbc.runVar89NTimes(targets, 3, { openAfter: true });
if (!ok89) {
log.w('89x3 failed stopping auto.');
return false;
}
return true;
}
return false;
} catch (e) {
if (isAbort(e)) throw e;
if (targets.var89Id) {
const ok89 = await sbc.runVar89NTimes(targets, 3, { openAfter: true });
if (!ok89) {
log.w('89x3 failed stopping auto.');
return false;
}
return true;
}
return false;
}
} else if (targets.var89Id) {
const ok89 = await sbc.runVar89NTimes(targets, 3, { openAfter: true });
if (!ok89) {
log.w('89x3 failed stopping auto.');
return false;
}
return true;
}
return false;
}
if (act === 'do_var89' && targets.var89Id) {
const ok89 = await sbc.runVar89NTimes(targets, 3, { openAfter: true });
if (!ok89) {
log.w('89x3 failed stopping auto.');
return false;
}
return true;
}
log.i('[auto] 决策失败');
return false;
},
getSbcByNameCandidates(list = state.FILTERED_SETS) {
const loopAllowNames = ['10 名 85+ 升级', '10 名 84+ 升级'];
const loopCandidates = list.filter((s) => loopAllowNames.some((n) => s.name.includes(n)));
const doCandidates = list.filter((s) => !loopAllowNames.some((n) => s.name.includes(n)));
return { doCandidates: doCandidates.length ? doCandidates : list, loopCandidates };
},
};
const tasks = {
startAsync({
button,
taskName,
asyncLoop,
startText,
stopText,
countLimit,
randomPause,
pauseEveryRange,
bigPauseRange,
}) {
if (state.running) return;
state.running = true;
if (['openPacks', 'loop', 'auto'].includes(taskName)) state._hiRatedPlayers = [];
state.runningTask = taskName;
ui.updateButtonState();
button.textContent = stopText;
let _doneResolve;
state.abortCtrl = new AbortController();
state.currentTaskDone = new Promise((r) => (_doneResolve = r));
log.i('start', taskName);
(async () => {
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) {
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();
if (canContinue === false) break;
count++;
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) {
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 = '';
state.abortCtrl = null;
button.textContent = startText;
ui.updateButtonState();
if (['openPacks', 'loop', 'auto'].includes(taskName)) {
sbc.showHiRatedPopup(`本次高分球员(≥${config.HIGH_RATED_POPUP_THRESHOLD})`);
}
_doneResolve();
}
})();
},
async stopAsync() {
if (!state.isStopping) {
state.isStopping = true;
ui.updateButtonState();
}
if (state.running) {
state.running = false;
state.runningTask = '';
state.abortCtrl?.abort();
}
const minHold = new Promise((r) => setTimeout(r, 300));
try {
await Promise.all([state.currentTaskDone.catch(() => { }), minHold]);
} finally {
state.isStopping = false;
ui.resetButtonText();
ui.updateButtonState();
}
},
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('未获取配置');
return;
}
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();
if (!state.running) {
await sbc.ensureSBCHub();
tasks.startDoSBC();
}
} else if (kind === 'loop' && pick.loopId) {
if (state.running && state.runningTask !== 'loop') await tasks.stopAsync();
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 = {
SLOW_RATE: Number(config.SLOW_RATE) || 1.0,
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, step = 1) => `
<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}" step="${step}"
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(
'慢速模式倍率',
'in-SLOW_RATE',
cur.SLOW_RATE,
'所有等待时间乘以此倍率 (1.0=正常速度, 2.0=2倍慢速)',
0.5,
5.0,
0.1,
),
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 valFloat = (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, n));
};
const next = {
SLOW_RATE: valFloat('in-SLOW_RATE', cur.SLOW_RATE, 0.5, 5.0),
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.SLOW_RATE = next.SLOW_RATE;
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.SLOW_RATE, next.SLOW_RATE);
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 tasks.stopAsync();
if (state.running && state.runningTask !== 'doSBC') await tasks.stopAsync();
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();
if (!state.running) tasks.startOpen();
else if (state.runningTask === 'openPacks') 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();
if (!state.running) {
await sbc.goToPacks();
tasks.startLoop();
} else if (state.runningTask === 'loop') {
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();
if (!state.running) tasks.startAuto();
else if (state.runningTask === 'auto') tasks.stopAsync();
};
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 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,
);
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.SLOW_RATE = num(CFG_KEYS.SLOW_RATE, config.SLOW_RATE);
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);
}
return { config, state, log, util, dom, ea, sbc, tasks, ui, init, loadSbcSettingsFromStorage };
})();
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 };
})();
window.addEventListener('load', () => {
try {
PandaSBC.init();
PandaSBC.loadSbcSettingsFromStorage()
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);
}
});
})();