pandaSBC

基于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);
    }
  });
})();
长期地址
遇到问题?请前往 GitHub 提 Issues。