auto-task

自动完成 Freeanywhere,Giveawaysu,GiveeClub,Givekey,Gleam,Indiedb,keyhub,OpiumPulses,Opquests,SweepWidget 等网站的任务。

2025-10-16 يوللانغان نەشرى. ئەڭ يېڭى نەشرىنى كۆرۈش.

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @name               auto-task
// @namespace          auto-task
// @version            5.0.6
// @description        自动完成 Freeanywhere,Giveawaysu,GiveeClub,Givekey,Gleam,Indiedb,keyhub,OpiumPulses,Opquests,SweepWidget 等网站的任务。
// @description:en     Automatically complete the tasks of FreeAnyWhere, GiveawaySu, GiveeClub, Givekey, Gleam, Indiedb, keyhub, OpiumPulses, Opquests, SweepWidget websites.
// @author             HCLonely
// @license            MIT
// @run-at             document-start
// @homepage           https://auto-task-doc.js.org/
// @supportURL         https://github.com/HCLonely/auto-task/issues
// @icon               https://auto-task.hclonely.com/favicon.ico
// @tag                games

// @include            *://freeanywhere.net/*
// @include            *://giveaway.su/giveaway/view/*
// @include            *://givee.club/*/event/*
// @include            *://givekey.ru/giveaway/*
// @include            *://www.indiedb.com/giveaways*
// @include            *://key-hub.eu/giveaway/*
// @include            *://keylol.com/*
// @include            *://www.opiumpulses.com/giveaways
// @include            *://prys.revadike.com/giveaway/?id=*
// @include            *://opquests.com/quests/*
// @include            *://gleam.io/*
// @include            *://sweepwidget.com/view/*
// @include            *://giveawayhopper.com/c/*

// @include            *://discord.com/*
// @include            *://www.twitch.tv/*
// @include            *://www.youtube.com/*
// @include            *://m.youtube.com/*
// @include            *://*.reddit.com/*
// @include            *://twitter.com/settings/account?k*
// @include            *://x.com/settings/account*
// @include            *://steamcommunity.com/*
// @include            *://store.steampowered.com/*

// @include            *://give.gamesforfarm.local/*
// @include            *://gamesforfarm-testing.ru/*
// @include            *://mee6.xyz/*
// @include            *://gamesforfarm.com/*

// @include            https://auto-task.hclonely.com/setting.html
// @include            https://auto-task.hclonely.com/history.html
// @include            https://auto-task-doc.js.org/setting.html
// @include            https://auto-task-doc.js.org/history.html

// @grant              GM_setValue
// @grant              GM_getValue
// @grant              GM_listValues
// @grant              GM_deleteValue
// @grant              GM_addStyle
// @grant              GM_xmlhttpRequest
// @grant              GM_registerMenuCommand
// @grant              GM_info
// @grant              GM_openInTab
// @grant              GM_setClipboard
// @grant              GM_getResourceText
// @grant              GM_cookie
// @grant              GM_addValueChangeListener
// @grant              GM_removeValueChangeListener
// @grant              unsafeWindow
// @grant              window.close
// @grant              window.localStorage
// @grant              window.sessionStorage
// @grant              window.focus

// @connect            auto-task.hclonely.com
// @connect            auto-task-doc.js.org
// @connect            cdn.jsdelivr.net
// @connect            store.steampowered.com
// @connect            steamcommunity.com
// @connect            login.steampowered.com
// @connect            twitter.com
// @connect            x.com
// @connect            abs.twimg.com
// @connect            api.twitter.com
// @connect            youtube.com
// @connect            www.youtube.com
// @connect            facebook.com
// @connect            instagram.com
// @connect            vk.com
// @connect            twitch.tv
// @connect            www.twitch.tv
// @connect            gql.twitch.tv
// @connect            github.com
// @connect            discordapp.com
// @connect            discord.gg
// @connect            discord.com
// @connect            www.reddit.com
// @connect            oauth.reddit.com
// @connect            raw.githubusercontent.com
// @connect            t.me
// @connect            bit.ly
// @connect            giveaway.su
// @connect            google.com
// @connect            www.vloot.io
// @connect            givee.club
// @connect            gleam.io
// @connect            www.indiedb.com
// @connect            key-hub.eu
// @connect            opquests.com
// @connect            itch.io
// @connect            auto-task.hclonely.com
// @connect            giveawayhopper.com
// @connect            freeanywhere.net
// @connect            *

// @require            https://cdn.jsdelivr.net/npm/[email protected]/dist/jquery.min.js
// @require            https://cdn.jsdelivr.net/npm/[email protected]/dist/js.cookie.min.js
// @require            https://cdn.jsdelivr.net/npm/[email protected]/src/sha1.min.js
// @require            https://cdn.jsdelivr.net/npm/sweetalert2@11/dist/sweetalert2.min.js
// @resource           style https://cdn.jsdelivr.net/npm/[email protected]/dist/sweetalert2.min.css
// @require            https://cdn.jsdelivr.net/npm/[email protected]/dist/keyboard.min.js
// @require            https://cdn.jsdelivr.net/npm/[email protected]/dayjs.min.js
// @require            https://cdn.jsdelivr.net/gh/tinygo-org/tinygo@3e60eeb368f25f237a512e7553fd6d70f36dc74c/targets/wasm_exec.min.js
// @require            https://cdn.jsdelivr.net/npm/[email protected]/dist/inspect.min.js
// @require            https://cdn.jsdelivr.net/npm/[email protected]/dist/browser.min.js

// @noframes
// ==/UserScript==

console.log('%c%s', 'color:blue', 'Auto-Task[Load]: 脚本开始加载');

/*
 * @Author       : HCLonely
 * @Date         : 2025-06-15 14:59:17
 * @LastEditTime : 2025-08-18 19:05:01
 * @LastEditors  : HCLonely
 * @FilePath     : /auto-task/src/scripts/checkDependence.js
 * @Description  :
 */
const neededDependencies = ['jQuery', 'Cookies', 'sha1', 'Swal', 'keyboardJS', 'dayjs', 'Go', 'util', 'browser'];

const missingDependencies = neededDependencies.filter(dependency => typeof window[dependency] === 'undefined');

if (missingDependencies.length > 0) {
  console.log('%c%s', 'color:red', `[Auto-Task] 脚本加载失败,缺少的依赖:${missingDependencies.join(', ')}`);
  if (confirm(`[Auto-Task] 脚本依赖加载失败,请刷新重试或安装全依赖版本,是否前往安装全依赖版本?\n缺少的依赖:${missingDependencies.join(', ')}`)) {
    GM_openInTab('https://github.com/HCLonely/auto-task/raw/main/dist/auto-task.all.user.js', { active: true });
  }
}


(function(Swal, Cookies, browser, util, dayjs, keyboardJS) {
  'use strict';
  const tokenKeyPattern = /token|auth|session|jwt|key|secret|api[-_]?key|bearer|authorization|access[-_]?token|refresh[-_]?token|sid/i;
  const tokenStringPatterns = [ /([A-Za-z0-9-_]{10,})\.([A-Za-z0-9-_]{10,})\.([A-Za-z0-9-_]{10,})/g, /(Bearer|Basic)\s+([A-Za-z0-9\-._~+/]+=*)/gi, /\b([a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12})\b/gi, /\b(eyJ[A-Za-z0-9\-_]+)\b/g ];
  const maskToken = str => {
    if (typeof str !== 'string' || str.length < 8) {
      return str;
    }
    return str.replace(/^([A-Za-z0-9\-_+/=]{4})[A-Za-z0-9\-_+/=]+([A-Za-z0-9\-_+/=]{4})$/, '$1***$2');
  };
  const maskObject = obj => {
    if (Array.isArray(obj)) {
      return obj.map(maskObject);
    } else if (obj && typeof obj === 'object') {
      const newObj = {};
      for (const key in obj) {
        if (tokenKeyPattern.test(key) && typeof obj[key] === 'string') {
          newObj[key] = maskToken(obj[key]);
        } else {
          newObj[key] = maskObject(obj[key]);
        }
      }
      return newObj;
    }
    if (typeof obj === 'string' && obj.length > 8) {
      return maskString(obj);
    }
    return obj;
  };
  const maskString = str => {
    let masked = str;
    for (const pattern of tokenStringPatterns) {
      masked = masked.replace(pattern, ((match, ...groups) => {
        if (groups.length >= 3 && match.includes('.')) {
          return groups.map((seg => seg.length > 8 ? `${seg.slice(0, 4)}***${seg.slice(-4)}` : seg)).join('.');
        }
        if (match.length > 8) {
          return `${match.slice(0, 4)}***${match.slice(-4)}`;
        }
        return match;
      }));
    }
    return masked;
  };
  const maskArgs = args => args.map((arg => {
    if (typeof arg === 'string') {
      return maskString(arg);
    } else if (typeof arg === 'object' && arg !== null) {
      return maskObject(arg);
    }
    return arg;
  }));
  const consoleLogHook = () => {
    const originalLog = console.log;
    window.__allLogs = window.__allLogs || [];
    console.log = function(...args) {
      const maskedArgs = maskArgs(args);
      window.__allLogs.push(maskedArgs);
      originalLog.apply(console, maskedArgs);
    };
  };
  const defaultGlobalOptions = {
    doTask: {
      discord: {
        servers: true
      },
      twitch: {
        channels: true
      },
      twitter: {
        users: true,
        retweets: true
      },
      vk: {
        names: true
      },
      youtube: {
        channels: true,
        likes: true
      },
      reddit: {
        reddits: true
      },
      steam: {
        groups: true,
        officialGroups: true,
        wishlists: true,
        follows: true,
        forums: true,
        workshops: true,
        curators: true,
        workshopVotes: true,
        announcements: true,
        licenses: true,
        playtests: true,
        playTime: true
      }
    },
    undoTask: {
      discord: {
        servers: true
      },
      twitch: {
        channels: true
      },
      twitter: {
        users: true,
        retweets: true
      },
      vk: {
        names: true
      },
      youtube: {
        channels: true,
        likes: true
      },
      reddit: {
        reddits: true
      },
      steam: {
        groups: true,
        officialGroups: true,
        wishlists: true,
        follows: true,
        forums: true,
        workshops: true,
        curators: true,
        playTime: true
      }
    },
    ASF: {
      AsfEnabled: false,
      AsfIpcUrl: '',
      AsfIpcPassword: '',
      AsfBotname: 'asf',
      steamWeb: false,
      preferASF: false,
      steamWebApiKey: ''
    },
    position: {
      buttonSideX: 'right',
      buttonSideY: 'top',
      buttonDistance: '15,30',
      showButtonSideX: 'right',
      showButtonSideY: 'top',
      showButtonDistance: '15,30',
      logSideX: 'right',
      logSideY: 'bottom',
      logDistance: '10,10'
    },
    hotKey: {
      doTaskKey: 'alt + d',
      undoTaskKey: 'alt + u',
      toggleLogKey: 'alt + l'
    },
    other: {
      twitterVerifyId: '783214',
      youtubeVerifyChannel: 'UCrXUsMBcfTVqwAS7DKg9C0Q',
      autoUpdateSource: 'jsdelivr',
      language: 'zh',
      checkLogin: true,
      checkLeftKey: true,
      defaultShowButton: true,
      defaultShowLog: true,
      debug: false,
      receivePreview: true
    }
  };
  const userDefinedGlobalOptions = GM_getValue('globalOptions') || {};
  const deepMerge = (target, source) => {
    try {
      const result = {
        ...target
      };
      for (const [key, value] of Object.entries(source)) {
        const targetValue = target[key];
        if (isObject(value) && isObject(targetValue)) {
          result[key] = deepMerge(targetValue, value);
        } else if (value !== undefined) {
          if (typeof value === typeof targetValue) {
            result[key] = value;
          } else {
            console.log('%c%s', 'color:yellow;background:black', `Auto-Task[Warning]: Type mismatch for key "${key}". Expected ${typeof targetValue}, got ${typeof value}. Using default value.`);
          }
        }
      }
      return result;
    } catch (error) {
      console.log('%c%s', 'color:white;background:red', `Auto-Task[Error]: deepMerge\n${error.stack}`);
      return target;
    }
  };
  const isObject = value => value !== null && typeof value === 'object' && !Array.isArray(value);
  const globalOptions = deepMerge(defaultGlobalOptions, userDefinedGlobalOptions);
  var style = '.colorful-button,#auto-task-buttons a.auto-task-website-btn,.show-button-div a.auto-task-website-btn,body.auto-task-options .auto-task-form table button{position:relative !important;padding:14px 28px !important;text-align:center !important;color:#fff !important;text-decoration:none !important;background:#2196f3 !important;border-radius:30px !important;text-transform:capitalize !important;font-weight:600 !important;letter-spacing:.5px !important;border:none !important;transition:all .2s ease !important;display:inline-block !important;line-height:1.5 !important;margin:8px !important;margin-bottom:12px !important;box-sizing:border-box !important;min-height:50px !important;min-width:140px !important;outline:none !important;vertical-align:middle !important;white-space:nowrap !important;font-size:18px !important}.colorful-button:hover,#auto-task-buttons a.auto-task-website-btn:hover,.show-button-div a.auto-task-website-btn:hover,body.auto-task-options .auto-task-form table button:hover{background:#1976d2 !important;box-shadow:0 4px 8px rgba(0,0,0,.1) !important;cursor:pointer !important;color:#fff !important;text-decoration:none !important}.colorful-button:active,#auto-task-buttons a.auto-task-website-btn:active,.show-button-div a.auto-task-website-btn:active,body.auto-task-options .auto-task-form table button:active{transform:translateY(1px) !important;color:#fff !important;text-decoration:none !important}.colorful-button:focus,#auto-task-buttons a.auto-task-website-btn:focus,.show-button-div a.auto-task-website-btn:focus,body.auto-task-options .auto-task-form table button:focus{color:#fff !important;text-decoration:none !important;outline:none !important}#auto-task-info{position:fixed;bottom:10px;right:10px;width:60%;max-width:500px;max-height:60%;overflow-y:auto;color:#000;background-color:#fff;padding-left:5px;z-index:999999999 !important;border:solid 2px #add8e6;border-radius:10px;font-size:14px !important}#auto-task-info li{text-align:left;display:block !important;align-items:baseline !important}#auto-task-info li .before-icon{display:inline-block !important;width:14px !important;height:14px !important;position:relative !important;top:2px !important;margin-right:5px !important;background-size:14px !important;background-repeat:no-repeat !important;flex-shrink:0 !important}#auto-task-info li font.before{color:#57bae8;margin-right:5px}#auto-task-info li a.high-light{color:#00aeff;font-weight:bold}#auto-task-info .success{color:green}#auto-task-info .error{color:red}#auto-task-info .warning{color:blue}#auto-task-info .info{color:#ff0}#auto-task-info .update-text{color:green;border:solid 2px #8dcb69;margin:5px 10px 5px 20px;border-radius:10px;padding:5px 20px}.auto-task-keylol{display:inline-block;text-transform:capitalize;margin-left:10px;text-decoration:none !important;border:solid 1px;border-radius:5px;padding:0 2px}.auto-task-keylol[selected=selected]{background-color:blue !important;color:#fff !important}.auto-task-form table{font-family:verdana,arial,sans-serif;font-size:11px;color:#333;border-width:1px;border-color:#999;border-collapse:collapse;width:100%}.auto-task-form table thead td{border-width:1px;padding:8px;border-style:solid;border-color:#a9c6c9;font-weight:bold;background-color:#fff}.auto-task-form table tbody tr{background-color:#d4e3e5}.auto-task-form table tbody tr:hover{background-color:#ff6 !important}.auto-task-form table tbody tr th{background-color:#c3dde0;border-width:1px;padding:8px;border-style:solid;border-color:#a9c6c9;text-transform:capitalize}.auto-task-form table tbody tr td{border-width:1px;padding:8px;border-style:solid;border-color:#a9c6c9}.swal2-modal{width:70% !important;max-width:1000px !important}.swal2-modal #swal2-title{text-align:center !important}body.auto-task-options{padding-top:10px;text-align:center}body.auto-task-options .auto-task-form{width:80%;max-width:1000px;margin:0 auto;padding-bottom:20px}body.auto-task-options .auto-task-form table input.editOption{width:80%}body.auto-task-options .auto-task-form table #getTwitterUserId,body.auto-task-options .auto-task-form table #getYoutubeChannelId{margin-top:5px}body.auto-task-options .auto-task-form table button{z-index:1;position:relative !important;padding:5px 20px !important;text-align:center !important;color:#fff !important;text-decoration:none !important;background:#2196f3 !important;border-radius:30px !important;text-transform:capitalize !important;font-weight:600 !important;letter-spacing:.5px !important;border:none !important;transition:all .2s ease !important;display:inline-block !important;line-height:1 !important;margin:8px !important;box-sizing:border-box !important;min-height:30px !important;min-width:140px !important;outline:none !important;vertical-align:middle !important;white-space:nowrap !important;font-size:15px !important}body.auto-task-options .auto-task-form table input[type=text]{outline-style:none;border:1px solid #ccc;border-radius:3px;padding:5px 10px;font-size:14px}body.auto-task-options .auto-task-form table input[type=text]:focus{border-color:#66afe9;outline:0;box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 8px rgba(102,175,233,.6)}body.auto-task-options .auto-task-form table label{position:relative;width:160px;height:80px;cursor:pointer;transform:scale(0.25);margin:-25% 0;top:-30px;display:inline-block}body.auto-task-options .auto-task-form table label input{position:relative;z-index:1;appearance:none}body.auto-task-options .auto-task-form table label input:checked~span{background:#05be05;box-shadow:0 15px 25px rgba(5,190,5,.4)}body.auto-task-options .auto-task-form table label input:checked~span i{left:84px}body.auto-task-options .auto-task-form table label input:checked~span i::before{background:#05be05;box-shadow:35px 0 0 #05be05}body.auto-task-options .auto-task-form table label input:checked~span i::after{bottom:12px;height:15px;border-bottom-left-radius:15px;border-bottom-right-radius:15px;background:#05be05}body.auto-task-options .auto-task-form table label span{position:absolute;top:0;left:0;width:100%;height:100%;background:#fe0000;border-radius:80px;transition:.5s;box-shadow:0 15px 25px rgba(254,0,0,.4)}body.auto-task-options .auto-task-form table label span i{position:absolute;top:4px;left:4px;width:72px;height:72px;background:#fff;border-radius:50%}body.auto-task-options .auto-task-form table label span i::before{content:"";position:absolute;top:22px;left:12px;width:12px;height:12px;border-radius:50%;background:#fe0000;box-shadow:35px 0 0 #fe0000;transition:.5s}body.auto-task-options .auto-task-form table label span i::after{content:"";position:absolute;bottom:15px;left:calc(50% - 15px);width:30px;height:6px;border-radius:6px;background:#fe0000;transition:.5s}body.auto-task-history{font-size:15px;font-weight:400;line-height:1.5}body.auto-task-history .container a{color:#007bff;text-decoration:none;background-color:rgba(0,0,0,0)}body.auto-task-history .container .card{width:80%;max-width:800px;border-radius:10px;background:rgba(118,118,118,.1019607843);border-top:1px solid hsla(0,0%,100%,.5019607843);-webkit-backdrop-filter:blur(20px);backdrop-filter:blur(20px);box-shadow:0 15px 25px rgba(0,0,0,.1019607843);margin:20px auto;position:relative;display:flex;flex-direction:column;word-wrap:break-word;-webkit-background-clip:border-box;background-clip:border-box;border:1px solid rgba(0,0,0,.125);border-radius:.25rem}body.auto-task-history .container .card .title{text-align:center;font-size:30px;font-weight:bold;margin:5px 0}body.auto-task-history .container .card .title a:hover{text-decoration:none;background:#93e1ff;border-radius:10px;padding:3px}body.auto-task-history .container .card ul{margin-bottom:25px}body.auto-task-history .container .card ul li{margin-bottom:5px;line-height:20px}body.auto-task-history .container .card ul a:hover{text-decoration:underline}body.auto-task-history .container .card .delete-task{right:10px;width:38px;height:35px;position:absolute;font-size:24px;cursor:pointer;border-radius:10px}body.auto-task-history .container .card .delete-task:hover{background:#fff}body.auto-task-history .container .card .time{right:5px;position:absolute;bottom:0;color:#e83e8c;font-family:\'SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace\';font-size:15px}#auto-task-buttons,.show-button-div{position:fixed !important;top:30px;right:15px;width:150px !important;z-index:999999999 !important;padding:8px !important;border-radius:12px !important}#auto-task-buttons p,.show-button-div p{line-height:normal !important;height:auto !important;text-align:center !important;margin:8px 0 !important;padding:0 !important;font-size:16px !important;color:#333 !important}#auto-task-buttons a.auto-task-website-btn,.show-button-div a.auto-task-website-btn{width:140px !important;min-height:30px !important;line-height:1.5 !important;font-size:16px !important;display:block !important;margin:0 auto !important;padding:8px 16px !important}.show-button-div{width:40px !important;cursor:pointer !important;padding:4px !important}.show-button-div a.auto-task-website-btn{right:-15px !important}.show-button-div a.auto-task-website-btn::after{content:"✓" !important;position:absolute !important;left:12px !important;top:50% !important;transform:translateY(-50%) !important;font-size:18px !important;font-weight:bold !important;color:#fff !important}.auto-task-capitalize{text-transform:capitalize !important}.swal2-file:focus,.swal2-input:focus,.swal2-textarea:focus{box-shadow:inset 0px 0px 4px 1px rgba(100,150,200,.5) !important}.swal2-checkbox-custom{align-items:center;justify-content:center;background:#fff;color:inherit;margin:1em auto}.swal2-checkbox-custom input{flex-shrink:0;margin:0 .4em}.giveaway-actions #getKey{display:none !important}.auto-task-giveaway-status{color:#fff;border-radius:10px;padding:0 5px;margin-left:5px}.auto-task-giveaway-status.active{background-color:#5cb85c}.auto-task-giveaway-status.not-active{background-color:#d9534f}';
  const data$1 = {
    website: '网站',
    type: '类型',
    edit: '编辑',
    whiteList: '白名单',
    skipTask: '跳过撤销任务',
    whiteListOptions: '白名单设置',
    changeWhiteListOption: '设置白名单(%0)',
    whiteListNotFound: '找不到此项白名单: %0',
    changeWhiteListSuccess: '白名单修改成功,刷新生效!',
    changeWebsiteOptions: '网站设置',
    changeGlobalOptions: '全局设置',
    ok: '是',
    save: '保存',
    close: '关闭',
    return: '返回',
    option: '选项',
    value: '值',
    websiteOptions: '当前网站设置',
    changeWebsiteOptionsSuccess: '更改当前网站设置成功,刷新生效!',
    changeGlobalOptionsSuccess: '更改全局设置成功,刷新生效!',
    needLogin: '请先登录!',
    getTasksInfo: '正在获取并处理任务信息...',
    gettingKey: '正在获取Key...',
    verifyingTask: '正在验证任务',
    notice: '自动任务脚本提醒',
    noKeysLeft: '此页面已经没有剩余key了,是否关闭?',
    giveawayEnded: '此活动已结束,是否关闭?',
    giveawayNotWork: '此活动因某些原因(已结束/暂停/未开始...)不可用(如果是脚本误判请及时反馈),是否关闭?',
    confirm: '确定',
    cancel: '取消',
    unKnown: '未知',
    unKnownTaskType: '未识别的任务',
    doing: '正在做任务',
    allTasksComplete: '所有任务已完成!',
    getTaskIdFailed: '获取任务Id失败!',
    initSuccess: '%0 初始化成功!',
    initFailed: '%0 初始化失败!',
    errorLink: '链接错误: %0',
    needInit: '请先初始化',
    verifyingAuth: '正在验证%0凭证...',
    updatingAuth: '正在更新%0凭证...',
    refreshingToken: '正在刷新%0凭证...',
    settingToken: '正在设置%0凭证...',
    steamStoreTab: 'Steam商店(弹窗)',
    steamCommunityTab: 'Steam社区(弹窗)',
    initing: '正在初始化...',
    getFailed: '获取%0失败!',
    checkLoginFailed: '检测登录状态失败!',
    checkLeftKeyFailed: '检测剩余Key失败!',
    userId: '用户Id',
    joiningGiveaway: '正在加入赠Key',
    needJoinGiveaway: '需要先加入赠Key',
    cannotUndo: '此网站不支持取消任务',
    verifyAuth: '正在验证 %0 凭证...',
    closePageNotice: '如果此页面没有自动关闭,请自行关闭本页面。',
    errorReport: '检测到脚本报错,是否前往反馈BUG?',
    visitingLink: '正在访问链接...',
    doTask: '做任务',
    undoTask: '撤销任务',
    verifyTask: '验证任务',
    getKey: '获取Key',
    selectAll: '全选',
    selectNone: '全不选',
    invertSelect: '反选',
    doFreeTask: '加入免费赠品',
    doPointTask: '加入点数赠品',
    skipTaskOption: '设置中已配置跳过任务',
    other: '其他',
    globalOptions: '全局设置',
    checkLogin: '登录检测</br>需要登录的网站自动登录,部分网站支持',
    checkLeftKey: '剩余Key检测</br>赠Key活动结束提示是否关闭,部分网站支持',
    twitterVerifyId: '通过尝试关注该账号验证Twitter凭证</br>默认为Twitter官方帐号 783214</br>不想关注官方账号可以改为自己的帐号',
    youtubeVerifyChannel: '通过尝试订阅该频道验证YouTube凭证</br>默认为YouTube官方频道 UCrXUsMBcfTVqwAS7DKg9C0Q</br>不想关注官方频道可以改为自己的频道',
    autoUpdateSource: '更新源</br>github: 需代理,实时更新</br>jsdelivr: 可不用代理,更新有延迟</br>standby: 备用</br>auto: 依次使用github, jsdelivr, standby源进行尝试更新',
    saveGlobalOptions: '保存设置',
    settingPage: '设置页面',
    name: '名称',
    version: '版本',
    scriptManager: '脚本管理器',
    script: '脚本',
    environment: '环境',
    os: '系统',
    browser: '浏览器',
    getId: '获取 %0 id',
    getTwitterUserId: '获取Twitter用户id(获取id功能仅在设置页面可用)',
    getYoutubeChannelId: '获取Youtube频道id(获取id功能仅在设置页面可用)',
    showButton: '显示按钮',
    hideButton: '隐藏按钮',
    showLog: '显示日志',
    hideLog: '隐藏日志',
    defaultShowButton: '默认显示按钮',
    defaultShowLog: '默认显示日志',
    debug: '输出调试日志,不要开启此选项!',
    receivePreview: '接收预览版更新',
    position: '组件位置',
    buttonSideX: '按钮区域水平方向定位(实时预览功能仅在设置页面可用)</br>left: 靠左 | right: 靠右',
    buttonSideY: '按钮区域垂直方向定位(实时预览功能仅在设置页面可用)</br>top: 靠上 | bottom: 靠下',
    buttonDistance: '按钮区域距边缘的距离(实时预览功能仅在设置页面可用)</br>格式: X距离,Y距离',
    showButtonSideX: '显示按钮水平方向定位(实时预览功能仅在设置页面可用)</br>left: 靠左 | right: 靠右',
    showButtonSideY: '显示按钮垂直方向定位(实时预览功能仅在设置页面可用)</br>top: 靠上 | bottom: 靠下',
    showButtonDistance: '显示按钮距边缘的距离(实时预览功能仅在设置页面可用)</br>格式: X距离,Y距离',
    logSideX: '日志区域水平方向定位(实时预览功能仅在设置页面可用)</br>left: 靠左 | right: 靠右',
    logSideY: '日志区域垂直方向定位(实时预览功能仅在设置页面可用)</br>top: 靠上 | bottom: 靠下',
    logDistance: '日志区域距边缘的距离(实时预览功能仅在设置页面可用)</br>格式: X距离,Y距离',
    hotKey: '快捷键',
    doTaskKey: '做任务快捷键</br>(实时预览功能仅在设置页面可用)',
    undoTaskKey: '撤销任务快捷键</br>(实时预览功能仅在设置页面可用)',
    toggleLogKey: '显示/隐藏日志快捷键</br>(实时预览功能仅在设置页面可用)',
    tasksHistory: '任务历史',
    clearHistory: '清空历史',
    clearHistoryFinished: '已清空任务历史!',
    deleteTask: '删除任务',
    lastChangeTime: '最后一次修改时间',
    clearTaskFinished: '删除以下任务完成!',
    clearTaskFailed: '删除任务失败,没有找到任务名!',
    syncData: '数据同步',
    settingData: '正在上传数据...',
    gettingData: '正在获取数据...',
    help: '帮助',
    fileName: '文件名',
    upload2gist: '同步到Gist',
    downloadFromGist: '从Gist同步',
    saveAndTest: '保存配置并测试',
    testSuccess: '测试成功!',
    testFailed: '测试失败!',
    saveAndTestNotice: '请先保存配置并测试!',
    processingData: '正在处理数据...',
    updatingData: '正在上传数据...',
    syncDataSuccess: '同步数据成功!',
    syncDataFailed: '同步数据失败,请在控制台查看错误信息!',
    downloadingData: '正在下载数据...',
    checkedNoData: '没有检测到远程数据,请确认配置是否正确!',
    savingData: '正在保存数据...',
    syncHistory: '同步任务历史',
    checkUpdateFailed: '检测更新失败',
    newVersionNotice: '检测到新版本V%0, <a class="high-light" href="%1" target="_blank">点此更新</a>',
    language: '语言</br>目前仅支持zh: 中文, en: 英文',
    gistOptions: 'Gist 设置',
    swalNotice: '检测到您第一次安装V4版本脚本,请前往阅读用前必读内容!',
    echoNotice: '检测到您第一次安装V4版本脚本,请<a class="high-light" href="%0" target="_blank">点此前往</a>阅读用前必读内容!',
    noticeLink: 'https://auto-task-doc.js.org/guide/#用前必读',
    toGithub: '前往Github反馈',
    toKeylol: '前往其乐论坛反馈',
    copySuccess: '错误信息已复制到剪切板,是否前往其乐论坛反馈?',
    copyFailed: '请复制下方错误信息后前往Keylol论坛反馈!',
    updateText: '%0 版本更新内容:',
    Active: '进行中',
    Ended: '已结束',
    Banned: '已封禁',
    Paused: '已暂停',
    notStart: '未开始',
    noRemoteData: '检测到远程无数据',
    errorRemoteDataFormat: '远程数据格式错误',
    updateHistory: '历史更新记录<a class="high-light" href="https://auto-task-doc.js.org/logs/" target="_blank">点此查看</a>',
    AsfEnabled: '使用ASF做Steam相关任务(需<a href="https://github.com/chr233/ASFEnhance" target="_blank">ASFEnhance</a>插件)',
    steamWeb: '同时使用Steam Web API做任务',
    preferASF: '优先使用ASF',
    AsfIpcUrl: 'ASF IPC 地址',
    AsfIpcPassword: 'ASF IPC 密码',
    versionNotMatched: '脚本管理器版本过低,需TamperMonkey >= 5.2.0或TamperMonkey Beta >= 5.2.6196',
    curatorLimitNotice: '失败:可能是鉴赏家关注数量限制,请<a class="high-light" href="https://store.steampowered.com/curators/home/" target="_blank">取关部分鉴赏家</a>后再试',
    unknownScriptHandler: '未知脚本管理器,适用性未知',
    debugModeNotice: '检测到 DEBUG 模式已开启,非必要请关闭!',
    steamWebApiKey: 'Steam Web API 密钥<br>用于检测游戏挂机状态,<a href="https://steamcommunity.com/dev/apikey" target="_blank">点此申请</a>',
    groups: '组',
    officialGroups: '官方组',
    wishlists: '愿望单',
    follows: '游戏关注',
    forums: '论坛',
    workshops: '创意工坊收藏',
    curators: '鉴赏家',
    workshopVotes: '创意工坊点赞',
    announcements: '社区通知',
    steamCommunity: 'Steam社区',
    steamStore: 'Steam商店',
    licenses: '入库免费游戏',
    playtests: '请求访问权限',
    playTime: '挂时长',
    needLoginSteamStore: '请先<a href="https://store.steampowered.com/login/" target="_blank">登录Steam商店</a>',
    needLoginSteamCommunity: '请先<a href="https://steamcommunity.com/login/home/" target="_blank">登录Steam社区</a>',
    joiningSteamGroup: '正在加入Steam组',
    leavingSteamGroup: '正在退出Steam组',
    gettingSteamGroupId: '正在获取Steam组Id',
    joiningSteamOfficialGroup: '正在加入Steam官方组',
    leavingSteamOfficialGroup: '正在退出Steam官方组',
    gettingSteamOfficialGroupId: '正在获取Steam官方组Id',
    subscribingForum: '正在订阅Steam论坛',
    unsubscribingForum: '正在取消订阅Steam论坛',
    gettingForumId: '正在获取Steam论坛Id',
    followingCurator: '正在关注Steam鉴赏家',
    unfollowingCurator: '正在取关Steam鉴赏家',
    gettingCuratorId: '正在获取Steam鉴赏家Id',
    addingToWishlist: '正在添加游戏到Steam愿望单',
    removingFromWishlist: '正在从Steam愿望单移除游戏',
    followingGame: '正在关注Steam游戏',
    unfollowingGame: '正在取关Steam游戏',
    favoritingWorkshop: '正在收藏Steam创意工坊物品',
    unfavoritingWorkshop: '正在取消收藏Steam创意工坊物品',
    gettingWorkshopAppId: '正在获取Steam创意工坊物品Id',
    votingUpWorkshop: '正在点赞Steam创意工坊物品',
    gettingAnnouncementParams: '正在获取Steam通知信息',
    likingAnnouncement: '正在点赞Steam通知',
    changingArea: '正在更换Steam地区: %0...',
    notNeededChangeArea: '当前地区不需要更换',
    noAnotherArea: '请检测是否开启正确开启代理',
    gettingAreaInfo: '正在获取Steam地区信息...',
    changeAreaNotice: '疑似锁区游戏,尝试换区执行',
    steamFinishNotice: 'Steam任务完成,尝试将购物车地区换回',
    gettingSubid: '正在获取游戏subid',
    addingFreeLicense: '正在入库',
    missParams: '缺少参数',
    gettingLicenses: '正在获取Licenses...',
    requestingPlayTestAccess: '正在请求访问权限',
    gettingDemoAppid: '正在获取Steam游戏的Demo Appid',
    tryChangeAreaNotice: '此功能无法检测游戏是否限区,因此会尝试换区后再入库,换区失败也不影响后续入库',
    gettingUserInfo: '正在获取Steam用户社区凭证...',
    retry: '重试',
    owned: '已拥有',
    redirect: '重定向',
    noSubid: '无法获取,跳过',
    noASFInstance: '未启用ASF,跳过挂时长任务',
    initingASF: '正在初始化ASF...',
    playingGames: '正在挂游戏时长[%0]...',
    stoppingPlayGames: '正在停止挂游戏时长...',
    stopPlayTimeTitle: 'Steam游戏挂机时长满足,是否结束挂机?',
    stopPlayTimeText: '挂机已超时:%0 分钟',
    ASFNotSupportted: '当前功能(%0)ASF无法实现,跳过',
    checkingPlayGamesStatus: '正在检查挂游戏时长状态...',
    gettingSteamId: '正在获取Steam ID...',
    checkingPlayStatus: '正在检查挂机状态...',
    noPlayStatus: '游戏未运行',
    servers: '服务器',
    joiningDiscordServer: '正在加入Discord服务器',
    leavingDiscordServer: '正在退出Discord服务器',
    gettingDiscordGuild: '正在获取Discord服务器Id',
    getDiscordAuthFailed: '获取Discord凭证失败,请检测Discord帐号是否已登录',
    discordImportantNotice: '重要提醒!!!',
    discordImportantNoticeText: '由于Discord网站后台更新,目前使用此脚本加组后可能会导致Discord帐号被强制退出,且需要两步验证才能正常登录,请谨慎使用!!!',
    continueDiscordTask: '本次执行Discord任务',
    skipDiscordTask: '本次跳过Discord任务',
    continueAndDontRemindAgain: '总是执行Discord任务且不再提醒',
    gettingDiscordXContextProperties: '正在获取Discord加群参数',
    captchaNeeded: '检测到人机验证,请手动完成!',
    users: '用户',
    loginIns: '请先<a href="https://www.instagram.com/accounts/login/" target="_blank">登录Instagram</a>',
    insBanned: '您的Instagram账户已被封禁',
    verifyingInsAuth: '正在验证Instagram凭证...',
    gettingInsUserId: '正在获取Instagram用户Id',
    followingIns: '正在关注Instagram用户',
    unfollowingIns: '正在取关Instagram用户',
    reddits: '社区/用户',
    loginReddit: '请先<a href="https://www.reddit.com/login/" target="_blank">登录Reddit</a>',
    changingRedditVersion: '正在切换Reddit为新版页面...',
    joiningReddit: '正在加入Reddit社区',
    leavingReddit: '正在退出Reddit社区',
    followingRedditUser: '正在关注Reddit用户',
    unfollowingRedditUser: '正在取关Reddit用户',
    channels: '频道',
    followingTwitchChannel: '正在关注Twitch频道',
    unfollowingTwitchChannel: '正在取关Twitch频道',
    gettingTwitchChannelId: '正在获取Twitch频道Id',
    checkingTwitchIntegrity: '正在检查Twitch完整性...',
    twitterUser: '推特用户',
    retweets: '转推',
    gettingTwitterUserId: '正在获取推特用户Id',
    followingTwitterUser: '正在关注推特用户',
    unfollowingTwitterUser: '正在取关推特用户',
    retweetting: '正在转推',
    unretweetting: '正在撤销转推',
    names: '组/社区/动态',
    loginVk: '请先<a href="https://vk.com/login/" target="_blank">登录Vk</a>',
    gettingVkId: '正在获取Vk任务Id',
    joiningVkGroup: '正在加入Vk组',
    leavingVkGroup: '正在退出Vk组',
    joiningVkPublic: '正在加入Vk社区',
    leavingVkPublic: '正在退出Vk社区',
    sendingVkWall: '正在转发Vk动态',
    deletingVkWall: '正在撤销转发Vk动态',
    youtubeChannel: 'YouTube频道',
    likes: '点赞',
    loginYtb: '请先<a href="https://accounts.google.com/ServiceLogin?service=youtube" target="_blank">登录YouTube</a>',
    tryUpdateYtbAuth: '请尝试<a href="https://www.youtube.com/#auth" target="_blank">更新YouTube凭证</a>',
    gettingYtbToken: '正在获取YouTube Token...',
    followingYtbChannel: '正在订阅YouTube频道',
    unfollowingYtbChannel: '正在退订YouTube频道',
    likingYtbVideo: '正在点赞YouTube视频',
    unlikingYtbVideo: '正在取消点赞YouTube视频',
    giveKeyNoticeBefore: '每次验证间隔15s',
    giveKeyNoticeAfter: '如果没有key, 请在<a href="https://givekey.ru/profile" target="_blank">https://givekey.ru/profile</a>查看',
    noPoints: '点数不够,跳过抽奖',
    getNeedPointsFailed: '获取所需点数失败,跳过抽奖',
    joiningLottery: '正在加入抽奖',
    doingGleamTask: '正在做Gleam任务...',
    gettingGleamLink: '正在获取Gleam任务链接...',
    gleamTaskNotice: '如果此页面长时间未关闭,请完成任一任务后自行关闭!',
    verifiedGleamTasks: '已尝试验证所有任务,验证失败的任务请尝试手动验证或完成!',
    campaign: '检测到人机验证,请手动完成!3秒后重新检测...',
    gsNotice: '为避免得到"0000-0000-0000"key, 已自动屏蔽"Grab Key"按钮,获取key时请关闭脚本!',
    giveeClubVerifyNotice: '正在验证任务...',
    giveeClubVerifyFinished: '请等待验证完成后自行加入赠Key',
    doingKeyhubTask: '正在做Keyhub任务...',
    SweepWidgetNotice: '正在处理并验证任务,每次验证任务有1~3s间隔防止触发验证过快警告...',
    tasksNotCompleted: '任务未完成',
    notConnect: '社交平台未连接,跳过任务: %0',
    tgTaskNotice: '检测到Telegram任务,需要手动完成',
    updatingUserData: '正在更新用户数据...',
    gettingUserGames: '正在获取用户游戏...',
    confirmingTask: '正在跳过警告页面...',
    unSupporttedTaskType: '不支持的任务类型: %0',
    taskNotFinished: '有任务未完成,不获取密钥',
    logCopied: '完整日志已复制到剪切板,请前往反馈!'
  };
  const data = {
    website: 'Website',
    type: 'Type',
    edit: 'Edit',
    whiteList: 'Whitelist',
    skipTask: 'Skip undo task',
    whiteListOptions: 'Whitelist options',
    changeWhiteListOption: 'Whitelist option(%0)',
    whiteListNotFound: 'Cannot find this whitelist: %0',
    changeWhiteListSuccess: 'The whitelist is successfully modified, and the page refresh will take effect!',
    changeWebsiteOptions: 'Website options',
    changeGlobalOptions: 'Global options',
    ok: 'OK',
    save: 'Save',
    close: 'Close',
    return: 'Return',
    option: 'Option',
    value: 'Value',
    websiteOptions: 'Current website settings',
    changeWebsiteOptionsSuccess: 'The current website setting is changed successfully, and the page refresh will take effect!',
    changeGlobalOptionsSuccess: 'The global setting is changed successfully, and the refresh will take effect!',
    needLogin: 'Please log in first!',
    getTasksInfo: 'Obtaining and processing task information...',
    gettingKey: 'Getting Key...',
    verifyingTask: 'Verifying task',
    notice: 'Automatic task script notice',
    noKeysLeft: 'There are no more keys left on this page. Do you want to close it?',
    giveawayEnded: 'This event has ended, do you want to close it?',
    giveawayNotWork: 'This activity is unavailable for some reasons (banned/ended/paused/not started...)' + ' (if it is a script misjudgment, please give us feedback in time), is it closed?',
    confirm: 'Confirm',
    cancel: 'Cancel',
    unKnown: 'Unknown',
    unKnownTaskType: 'Unrecognized task',
    doing: 'Doing a task',
    allTasksComplete: 'All tasks have been completed!',
    getTaskIdFailed: 'Failed to obtain task Id!',
    initSuccess: '%0 was initialized successfully!',
    initFailed: '%0 initialization failed!',
    errorLink: 'Link error: %0',
    needInit: 'Please initialize first',
    verifyingAuth: 'Verifying %0 token...',
    updatingAuth: 'Update %0 token...',
    refreshingToken: 'Refreshing %0 token...',
    settingToken: 'Setting %0 token...',
    steamStoreTab: 'Steam store (new tab)',
    steamCommunityTab: 'Steam community(new tab)',
    initing: 'Initializing...',
    getFailed: 'Failed to get %0!',
    checkLoginFailed: 'Failed to detect login status!',
    checkLeftKeyFailed: 'Failed to detect the remaining keys!',
    userId: 'User Id',
    joiningGiveaway: 'Joining giveaway',
    needJoinGiveaway: 'Need to join the giveaway first',
    cannotUndo: 'This website does not support canceling tasks',
    verifyAuth: 'Verifying %0 token...',
    closePageNotice: 'f this page does not close automatically, please close this page yourself.',
    errorReport: 'A script error is detected, do you want to report the BUG?',
    visitingLink: 'Visiting link ...',
    doTask: 'DoTask',
    undoTask: 'UndoTask',
    verifyTask: 'Verify',
    getKey: 'GetKey',
    selectAll: 'SelectAll',
    selectNone: 'SelectNone',
    invertSelect: 'InvertSelect',
    doFreeTask: 'FreeTask',
    doPointTask: 'PointTask',
    skipTaskOption: 'Skip task has been configured in the settings',
    other: 'Other',
    globalOptions: 'Global Options',
    checkLogin: 'Login detection</br>Need to log in to the website automatically log in, part of this website supports.',
    checkLeftKey: 'Key remaining detection</br>The end of the giveaway event prompts whether to close or not, part of this website supports.',
    twitterVerifyId: 'Verify Twitter token by trying to follow the account.</br>The default is the official Twitter account 783214.</br>' + 'If you don\'t want to follow the official account, you can change it to your own account.',
    youtubeVerifyChannel: 'Verify YouTube token by trying to subscribe to the channel.</br>' + 'The default is the official YouTube channel UCrXUsMBcfTVqwAS7DKg9C0Q.</br>' + 'If you don\'t want to follow the official channel, you can change it to your own channel.',
    autoUpdateSource: 'The source to update</br>github: Fast update.</br>jsdelivr: Update is delayed.</br>' + 'standby: Standby source.</br>auto: Try to update using github, jsdelivr, standby sources in turn.',
    saveGlobalOptions: 'SaveSettings',
    settingPage: 'Setting Page',
    name: 'Name',
    version: 'Version',
    scriptManager: 'Script Manager',
    script: 'Script',
    environment: 'Environment',
    os: 'OS',
    browser: 'Browser',
    getId: 'Get %0 id',
    getTwitterUserId: 'Get Twitter user id (Get id function is only available on the settings page).',
    getYoutubeChannelId: 'Get Youtube channel id (Get id function is only available on the settings page).',
    showButton: 'ShowButton',
    hideButton: 'HideButton',
    showLog: 'ShowLog',
    hideLog: 'HideLog',
    defaultShowButton: 'Default display button',
    defaultShowLog: 'Display log by default',
    debug: 'Output debug log, do not enable this option!',
    receivePreview: 'Receive preview updates',
    position: 'Component position',
    buttonSideX: 'Horizontal positioning of the button area (real-time preview function is only available on the setting page).' + '</br>left: left | right: right',
    buttonSideY: 'The button area is positioned in the vertical direction (real-time preview function is only available on the settings page).' + '</br>top: top | bottom: bottom',
    buttonDistance: 'The distance between the button area and the edge (the real-time preview function is only available on the setting page).' + '</br> Format: X distance, Y distance',
    showButtonSideX: 'ShowButton horizontal positioning (real-time preview function is only available on the setting page).' + '</br>left: left | right: right',
    showButtonSideY: 'ShowButton vertical positioning (real-time preview function is only available on the setting page).' + '</br>top: top | bottom: bottom',
    showButtonDistance: 'The distance between the ShowButton and the edge (real-time preview function is only available on the setting page).' + '</br> Format: X distance, Y distance',
    logSideX: 'Horizontal positioning of the log area (real-time preview function is only available on the setting page).' + '</br>left: left | right: right',
    logSideY: 'Vertical positioning of the log area (real-time preview function is only available on the setting page).' + '</br>top: top | bottom: bottom',
    logDistance: 'The distance between the log area and the edge (the real-time preview function is only available on the setting page).' + '</br> Format: X distance, Y distance',
    hotKey: 'Shortcut key',
    doTaskKey: 'DoTask shortcut keys</br> (real-time preview function is only available on the settings page).',
    undoTaskKey: 'UndoTask shortcut keys</br> (real-time preview function is only available on the settings page).',
    toggleLogKey: 'ShowLog/HideLog shortcut keys</br> (real-time preview function is only available on the settings page).',
    tasksHistory: 'TasksHistory',
    clearHistory: 'Clear history',
    clearHistoryFinished: 'The mission history has been cleared!',
    deleteTask: 'Delete task',
    lastChangeTime: 'Last Change Time',
    clearTaskFinished: 'Delete the following tasks completed!',
    clearTaskFailed: 'Failed to delete the task, the task name was not found!',
    syncData: 'DataSync',
    settingData: 'Uploading data...',
    gettingData: 'Getting data...',
    help: 'Help',
    fileName: 'Filename',
    upload2gist: 'Sync to Gist',
    downloadFromGist: 'Sync from Gist',
    saveAndTest: 'Save configuration and test',
    testSuccess: 'Test success!',
    testFailed: 'Test failed!',
    saveAndTestNotice: 'Please save the configuration and test first!',
    processingData: 'Processing data...',
    updatingData: 'Uploading data...',
    syncDataSuccess: 'Synchronized data successfully!',
    syncDataFailed: 'Failed to synchronize data, please check the error message on the console!',
    downloadingData: 'Downloading data...',
    checkedNoData: 'No remote data is detected, please confirm whether the configuration is correct!',
    savingData: 'Saving data...',
    syncHistory: 'Synchronize tasks history',
    checkUpdateFailed: 'Check update failed',
    newVersionNotice: 'Checked a new version V%0, <a class="high-light" href="%1" target="_blank">click to update</a>',
    language: 'Language</br> Currently only supports zh: Chinese, en: English',
    gistOptions: 'Gist Settings',
    swalNotice: 'It is detected that you are installing the V4 version script for the first time' + ', please go to read the READ ME FIRST content before use!',
    echoNotice: 'It is detected that you are installing the V4 version script for the first time' + ', please <a class="high-light" href="%0" target="_blank">click here</a> to read the READ ME FIRST content before use!',
    noticeLink: 'https://auto-task-doc.js.org/en/guide/#read-me-first',
    toGithub: 'Feedback(Github)',
    toKeylol: 'Feedback(Keylol)',
    copySuccess: 'The error message has been copied to the clipboard. Do you want to go to the Keylol forum to give feedback?',
    copyFailed: 'Please copy the error information below and report back to the Keylol forum!',
    updateText: 'Updates in version %0:',
    Active: 'Active',
    Ended: 'Ended',
    Banned: 'Banned',
    Paused: 'Paused',
    notStart: 'notStart',
    noRemoteData: 'No data remotely',
    errorRemoteDataFormat: 'Remote data has wrong format',
    updateHistory: '<a class="high-light" href="https://auto-task-doc.js.org/logs/" target="_blank">Click here</a>' + ' to view the historical update record.',
    AsfEnabled: 'Use ASF to do Steam related tasks (requires <a href="https://github.com/chr233/ASFEnhance" target="_blank">ASFEnhance</a> plugin)',
    steamWeb: 'Use Steam Web API to do Steam related tasks simultaneously',
    preferASF: 'Prefer ASF to do Steam related tasks',
    AsfIpcUrl: 'ASF IPC URL',
    AsfIpcPassword: 'ASF IPC Password',
    curatorLimitNotice: 'Failed: Maybe the curator follow limit is reached, please <a class="high-light" href="https://store.steampowered.com/curators/home/" target="_blank">unfollow some curators</a> and try again',
    unknownScriptHandler: 'Unknown script handler, compatibility unknown',
    debugModeNotice: 'Detected DEBUG mode enabled, please close it if it is not needed!',
    steamWebApiKey: 'Steam Web API Key<br>Used to detect game idle status, <a href="https://steamcommunity.com/dev/apikey" target="_blank">click to apply</a>',
    groups: 'Group',
    officialGroups: 'Official Group',
    wishlists: 'Wishlist',
    follows: 'Follow Game',
    forums: 'Forum',
    workshops: 'Favorite Workshop',
    curators: 'Curator',
    workshopVotes: 'Voteup Workshop',
    announcements: 'Announcement',
    steamCommunity: 'Steam Community',
    steamStore: 'Steam Store',
    licenses: 'Add License',
    playtests: 'Playtest Access',
    needLoginSteamStore: 'Please <a href="https://store.steampowered.com/login/" target="_blank">log in to the Steam Store</a>',
    needLoginSteamCommunity: 'Please <a href="https://steamcommunity.com/login/home/" target="_blank">log in to the Steam Community</a>',
    joiningSteamGroup: 'Joining Steam Group',
    leavingSteamGroup: 'Leaving Steam Group',
    gettingSteamGroupId: 'Getting Steam Group Id',
    joiningSteamOfficialGroup: 'Joining Steam Official Group',
    leavingSteamOfficialGroup: 'Leaving Steam Official Group',
    gettingSteamOfficialGroupId: 'Getting Steam Official Group Id',
    subscribingForum: 'Subscribing the Steam Forum',
    unsubscribingForum: 'Unsubscribing the Steam Forum',
    gettingForumId: 'Getting Steam Forum Id',
    followingCurator: 'Following Steam Curator',
    unfollowingCurator: 'Unfollowing Steam Curator',
    gettingCuratorId: 'Getting Steam Curator Id',
    addingToWishlist: 'Adding the game to the Steam wishlist',
    removingFromWishlist: 'Removing the game from the Steam wishlist',
    followingGame: 'Following Steam games',
    unfollowingGame: 'Unfollowing Steam games',
    favoritingWorkshop: 'Favouring Steam Workshop Items',
    unfavoritingWorkshop: 'Unfavoriting Steam Workshop Items',
    gettingWorkshopAppId: 'Getting Steam Workshop Item Id',
    votingUpWorkshop: 'Liking Steam workshop items',
    gettingAnnouncementParams: 'Getting Steam announcement information',
    likingAnnouncement: 'Liking Steam announcement',
    changingArea: 'Changing Steam area: %0...',
    notNeededChangeArea: 'The current area does not need to be changed',
    noAnotherArea: 'Please check whether the proxy is turned on correctly',
    gettingAreaInfo: 'Getting Steam area information...',
    changeAreaNotice: 'Suspected of a locked zone game, try to change the zone to execute',
    steamFinishNotice: 'Steam task completed, try to change the shopping cart area back to ',
    gettingSubid: 'Getting subid',
    addingFreeLicense: 'Adding free license',
    missParams: 'Missing parameters',
    gettingLicenses: 'Getting licenses...',
    requestingPlayTestAccess: 'Requesting play test access',
    gettingDemoAppid: 'Getting demo appid for steam game',
    tryChangeAreaNotice: 'This function cannot detect whether the game is limited, so it will try to change the area before entering the library' + '. Failure to change the area will not affect the subsequent storage.',
    versionNotMatched: 'The script manager version is too low, requiring TamperMonkey >= 5.2.0 or TamperMonkey Beta >= 5.2.6196',
    gettingUserInfo: 'Getting steam user community credentials...',
    retry: 'Retry',
    owned: 'Owned',
    redirect: 'Redirect',
    noSubid: 'skip due to unrecognized',
    noASFInstance: 'ASF is not enabled, skip idle time task',
    initingASF: 'Initing ASF...',
    playingGames: 'Playing games [%0]...',
    stoppingPlayGames: 'Stopping play games...',
    stopPlayTimeTitle: 'The Steam game idle time has finished. Do you want to end it?',
    stopPlayTimeText: 'Time out: %0 minutes',
    ASFNotSupportted: 'The current function (%0) cannot be implemented by ASF, skipping',
    checkingPlayGamesStatus: 'Checking play games status...',
    gettingSteamId: 'Getting Steam ID...',
    checkingPlayStatus: 'Checking play status...',
    noPlayStatus: 'Game not running',
    servers: 'Server',
    joiningDiscordServer: 'Joining Discord Server',
    leavingDiscordServer: 'Leaving Discord Server',
    gettingDiscordGuild: 'Getting Discord server Id',
    getDiscordAuthFailed: 'Failed to get Discord token, please check whether the Discord account is logged in',
    discordImportantNotice: 'Important Reminder! ! !',
    discordImportantNoticeText: 'Due to the background update of the Discord website, currently using this script to join a group may cause the Discord account to be forcibly logged out, and two-step verification is required to log in normally, please use it with caution! ! !',
    continueDiscordTask: 'Do Discord tasks this time.',
    skipDiscordTask: 'Skip Discord tasks this time.',
    continueAndDontRemindAgain: 'Always do Discord tasks and do not remind again.',
    gettingDiscordXContextProperties: 'Getting Discord X context properties...',
    captchaNeeded: 'Captcha detected, please complete it manually!',
    users: 'User',
    loginIns: 'Please <a href="https://www.instagram.com/accounts/login/" target="_blank">log in to Instagram</a>',
    insBanned: 'Your Instagram account has been banned',
    verifyingInsAuth: 'Verifying Instagram token...',
    gettingInsUserId: 'Getting Instagram user Id',
    followingIns: 'Following Instagram user',
    unfollowingIns: 'Unfollowing Instagram user',
    reddits: 'Reddit/User',
    loginReddit: 'Please <a href="https://www.reddit.com/login/" target="_blank">log in to Reddit</a>',
    changingRedditVersion: 'Switching Reddit to a new version page...',
    joiningReddit: 'Joining the Reddit',
    leavingReddit: 'Leaving the Reddit',
    followingRedditUser: 'Following Reddit User',
    unfollowingRedditUser: 'Unfollowing Reddit User',
    channels: 'Channel',
    followingTwitchChannel: 'Following Twitch Channel',
    unfollowingTwitchChannel: 'Unfollowing Twitch Channel',
    gettingTwitchChannelId: 'Getting Twitch Channel Id',
    checkingTwitchIntegrity: 'Checking Twitch integrity...',
    twitterUser: 'Twitter User',
    retweets: 'Retweet',
    gettingTwitterUserId: 'Getting Twitter User Id',
    followingTwitterUser: 'Following Twitter User',
    unfollowingTwitterUser: 'Unfollowing Twitter User',
    retweetting: 'Retweetting',
    unretweetting: 'Unretweetting',
    names: 'Group/Public/Wall',
    loginVk: 'Please <a href="https://vk.com/login/" target="_blank">log in to Vk</a>',
    gettingVkId: 'Getting Vk task Id',
    joiningVkGroup: 'Joining Vk Group',
    leavingVkGroup: 'Leaving Vk Group',
    joiningVkPublic: 'Joining Vk Public',
    leavingVkPublic: 'Leaving Vk Public',
    sendingVkWall: 'Sending Vk Wall',
    deletingVkWall: 'Deleting Vk Wall',
    youtubeChannel: 'YouTube Channel',
    likes: 'Like',
    loginYtb: 'Please <a href="https://accounts.google.com/ServiceLogin?service=youtube" target="_blank">log in to YouTube</a>',
    tryUpdateYtbAuth: 'Please try to <a href="https://www.youtube.com/#auth" target="_blank">update YouTube token</a>',
    gettingYtbToken: 'Getting YouTube Token...',
    followingYtbChannel: 'Subscribing to YouTube channel',
    unfollowingYtbChannel: 'Unsubscribing to YouTube channel',
    likingYtbVideo: 'Liking YouTube video',
    unlikingYtbVideo: 'Unliking YouTube video',
    giveKeyNoticeBefore: 'Each verification interval is 15s',
    giveKeyNoticeAfter: 'If there is no key, please check at <a href="https://givekey.ru/profile" target="_blank">https://givekey.ru/profile</a>',
    noPoints: 'Not enough points, skip the lottery',
    getNeedPointsFailed: 'ailed to obtain the required points, skip the lottery',
    joiningLottery: 'Joining the lottery',
    doingGleamTask: 'Doing Gleam Task...',
    gettingGleamLink: 'Getting Gleam task link...',
    gleamTaskNotice: 'If this page has not been closed for a long time, please close it yourself after completing any task!',
    verifiedGleamTasks: 'Attempted to verify all tasks. If the verification fails, please try to verify manually or complete it!',
    campaign: 'ReCAPTCHA detected, please complete it manually! 3 seconds later, re-verify...',
    gsNotice: 'In order to avoid getting the "0000-0000-0000" key, the "Grab Key" button has been hidden,' + ' please close the script when obtaining the key!',
    giveeClubVerifyNotice: 'Verifying task...',
    giveeClubVerifyFinished: 'Wait for the verification to complete and join it by yourself',
    doingKeyhubTask: 'Doing Keyhub Task...',
    SweepWidgetNotice: 'The task is being processed and verified. ' + 'There is an interval of 1~3s for each verification task to prevent the triggering of too fast verification warning...',
    tasksNotCompleted: 'Tasks Not Completed',
    notConnect: 'Social platform is not connectted, skip task: %0',
    tgTaskNotice: 'The telegram task is checked, need to do it yourself!',
    updatingUserData: 'Updating user data...',
    gettingUserGames: 'Getting user games...',
    confirmingTask: 'Confirming task...',
    unSupporttedTaskType: 'Unsupportted task type: %0',
    taskNotFinished: 'There are tasks not completed, do not get the key',
    logCopied: 'Full log has been copied to the clipboard, please go to feedback!'
  };
  const languages = {
    zh: data$1,
    en: data
  };
  const SUPPORTED_LANGUAGES = [ 'zh', 'en' ];
  const getCurrentLanguage = () => {
    const userLanguage = globalOptions.other.language;
    return SUPPORTED_LANGUAGES.includes(userLanguage) ? userLanguage : 'en';
  };
  const replacePlaceholders = (text, args) => text.replace(/%([\d]+)/g, ((_, index) => args[parseInt(index, 10)] || ''));
  const I18n = (key, ...args) => {
    const currentLanguage = getCurrentLanguage();
    const translation = languages[currentLanguage]?.[key];
    if (!translation) {
      console.warn(`Missing translation for key: ${key} in language: ${currentLanguage}`);
      return key;
    }
    return replacePlaceholders(translation, args);
  };
  var DebugLevel;
  (function(DebugLevel) {
    DebugLevel['ERROR'] = 'error';
    DebugLevel['WARN'] = 'warn';
    DebugLevel['INFO'] = 'info';
    DebugLevel['DEBUG'] = 'debug';
    DebugLevel['TRACE'] = 'trace';
  })(DebugLevel || (DebugLevel = {}));
  const defaultConfig = {
    enabled: false,
    level: DebugLevel.INFO,
    prefix: 'Auto-Task',
    styles: {
      error: 'color:#ff0000;font-weight:bold',
      warn: 'color:#ffa500',
      info: 'color:#a7a7a7',
      debug: 'color:#808080',
      trace: 'color:#87ceeb'
    },
    showTimestamp: true
  };
  class Debugger {
    config;
    levelPriority;
    constructor(config = {}) {
      this.config = {
        ...defaultConfig,
        ...config
      };
      this.levelPriority = {
        error: 0,
        warn: 1,
        info: 2,
        debug: 3,
        trace: 4
      };
    }
    getTimestamp() {
      return (new Date).toLocaleString();
    }
    shouldLog(level) {
      return this.config.enabled && this.levelPriority[level] <= this.levelPriority[this.config.level];
    }
    formatMessage(level, message) {
      const parts = [ this.config.prefix ];
      if (this.config.showTimestamp) {
        parts.push(`[${this.getTimestamp()}]`);
      }
      parts.push(`[${level.toUpperCase()}]:`);
      parts.push(message);
      return parts.join(' ');
    }
    log(level, message, ...args) {
      if (!this.shouldLog(level)) {
        return;
      }
      const formattedMessage = this.formatMessage(level, message);
      const style = this.config.styles[level];
      if (args.length > 0) {
        console.groupCollapsed('%c%s', style, formattedMessage);
        args.forEach((arg => {
          console.log(util.inspect(arg, {
            showHidden: true,
            depth: null,
            colors: false
          }));
        }));
        console.groupEnd();
      } else {
        console.log('%c%s', style, formattedMessage);
      }
    }
    error(message, ...args) {
      this.log(DebugLevel.ERROR, message, ...args);
    }
    warn(message, ...args) {
      this.log(DebugLevel.WARN, message, ...args);
    }
    info(message, ...args) {
      this.log(DebugLevel.INFO, message, ...args);
    }
    debug(message, ...args) {
      this.log(DebugLevel.DEBUG, message, ...args);
    }
    trace(message, ...args) {
      this.log(DebugLevel.TRACE, message, ...args);
      if (this.shouldLog(DebugLevel.TRACE)) {
        console.trace();
      }
    }
    updateConfig(config) {
      this.config = {
        ...this.config,
        ...config
      };
    }
    enable() {
      this.config.enabled = true;
    }
    disable() {
      this.config.enabled = false;
    }
    setLevel(level) {
      this.config.level = level;
    }
  }
  let debugInstance;
  const initDebug = () => {
    if (!debugInstance) {
      debugInstance = new Debugger({
        enabled: window.DEBUG || false
      });
      if (window.DEBUG) {
        debugInstance.setLevel(DebugLevel.DEBUG);
      }
    }
    return debugInstance;
  };
  const debug = (...args) => {
    const instance = initDebug();
    return instance.debug(...args);
  };
  const getRunLogs = () => {
    debug('开始获取运行日志');
    const logElements = $('#auto-task-info>li');
    const logs = logElements.length > 0 ? $.makeArray(logElements).map((element => element.innerText)).join('\n') : '';
    debug('运行日志获取完成', {
      logsLength: logs.length
    });
    return logs;
  };
  const getEnvironmentInfo = async () => {
    debug('开始获取环境信息');
    const envInfo = {
      website: window.location.href,
      browser: JSON.stringify(await browser.getInfo(), null, 2),
      manager: `${GM_info.scriptHandler} ${GM_info.version}`,
      userScript: GM_info.script.version,
      logs: '',
      runLogs: getRunLogs()
    };
    debug('环境信息获取完成', envInfo);
    return envInfo;
  };
  const buildGithubIssueParams = async (name, errorStack, envInfo) => {
    debug('开始构建GitHub Issue参数', {
      name: name,
      errorStackLength: errorStack.length
    });
    const params = {
      title: `[BUG] 脚本报错: ${name}`,
      labels: 'bug',
      template: 'bug_report.yml',
      website: envInfo.website,
      browser: envInfo.browser,
      manager: envInfo.manager,
      'user-script': envInfo.userScript,
      logs: errorStack || '',
      'run-logs': ''
    };
    const runLogs = window.__allLogs.join('\n');
    await GM_setClipboard(runLogs);
    debug('GitHub Issue参数构建完成', params);
    return params;
  };
  const generateGithubLink = async (name, errorStack, envInfo) => {
    debug('开始生成GitHub Issue链接');
    const params = new URLSearchParams(await buildGithubIssueParams(name, errorStack, envInfo));
    const link = `https://github.com/HCLonely/auto-task/issues/new?${params.toString()}`;
    debug('GitHub Issue链接生成完成', {
      link: link
    });
    return link;
  };
  const logError = (name, errorStack) => {
    debug('记录错误日志', {
      name: name
    });
    console.log('%c%s', 'color:white;background:red', `Auto-Task[Error]: ${name}\n${errorStack}`);
  };
  const handleErrorReport = async (platform, name, errorStack, envInfo) => {
    debug('开始处理错误报告', {
      platform: platform,
      name: name
    });
    {
      const githubLink = await generateGithubLink(name, errorStack, envInfo);
      debug('打开GitHub Issue链接', {
        githubLink: githubLink
      });
      GM_openInTab(githubLink, {
        active: true
      });
    }
  };
  async function throwError(error, name) {
    debug('开始处理错误', {
      name: name,
      error: error
    });
    if (window.TRACE) {
      debug('启用跟踪模式');
      console.trace('%cAuto-Task[Trace]:', 'color:blue');
    }
    const errorStack = error.stack || '';
    logError(name, errorStack);
    debug('获取环境信息');
    const envInfo = await getEnvironmentInfo();
    envInfo.logs = errorStack;
    debug('显示错误报告对话框');
    const {isConfirmed: isConfirmed} = await Swal.fire({
      title: I18n('errorReport'),
      icon: 'error',
      showCancelButton: true,
      confirmButtonText: I18n('toGithub'),
      cancelButtonText: I18n('close')
    });
    if (isConfirmed) {
      debug('用户确认提交错误报告');
      await handleErrorReport('github', name, errorStack, envInfo);
      Swal.fire({
        title: I18n('logCopied'),
        icon: 'success',
        showConfirmButton: false,
        showCancelButton: true,
        cancelButtonText: I18n('close')
      });
    } else {
      debug('用户取消提交错误报告');
    }
  }
  const parseHeaders = headerString => {
    debug('开始解析HTTP头', {
      headerString: headerString
    });
    const headers = {};
    if (!headerString) {
      debug('HTTP头为空,返回空对象');
      return headers;
    }
    headerString.split('\n').forEach((header => {
      const [name, ...values] = header.trim().split(':');
      const value = values.join(':').trim();
      if (!name || !value) {
        return;
      }
      if (headers[name]) {
        headers[name] = Array.isArray(headers[name]) ? [ ...headers[name], value ] : [ headers[name], value ];
      } else {
        headers[name] = value;
      }
    }));
    if (headers['set-cookie'] && !Array.isArray(headers['set-cookie'])) {
      headers['set-cookie'] = [ headers['set-cookie'] ];
    }
    debug('HTTP头解析完成', {
      headers: headers
    });
    return headers;
  };
  const processResponse = (data, options) => {
    debug('开始处理响应数据', {
      responseType: options.responseType
    });
    const headers = parseHeaders(data.responseHeaders);
    data.responseHeadersText = data.responseHeaders;
    data.responseHeaders = headers;
    data.finalUrl = headers.location || data.finalUrl;
    debug('响应头处理完成', {
      finalUrl: data.finalUrl
    });
    if (options.responseType === 'json' && data?.response && typeof data.response !== 'object') {
      debug('尝试解析JSON响应');
      try {
        data.response = JSON.parse(data.responseText);
        debug('JSON解析成功');
      } catch {
        debug('JSON解析失败,保持原始响应');
      }
    }
  };
  const httpRequest = async (options, times = 0) => {
    debug('开始HTTP请求', {
      url: options.url,
      method: options.method,
      retryTimes: times
    });
    if (window.TRACE) {
      console.trace('%cAuto-Task[Trace]:', 'color:blue');
    }
    try {
      const result = await new Promise((resolve => {
        const requestObj = {
          fetch: true,
          timeout: 3e4,
          ontimeout: data => {
            debug('请求超时', {
              url: options.url
            });
            resolve({
              result: 'Error',
              statusText: 'Timeout',
              status: 601,
              data: data,
              options: options
            });
          },
          onabort: () => {
            debug('请求被中止', {
              url: options.url
            });
            resolve({
              result: 'Error',
              statusText: 'Aborted',
              status: 602,
              data: undefined,
              options: options
            });
          },
          onerror: data => {
            debug('请求发生错误', {
              url: options.url,
              error: data
            });
            resolve({
              result: 'Error',
              statusText: 'Error',
              status: 603,
              data: data,
              options: options
            });
          },
          onload: data => {
            debug('请求加载完成', {
              url: options.url,
              status: data.status
            });
            processResponse(data, options);
            resolve({
              result: 'Success',
              statusText: 'Load',
              status: 600,
              data: data,
              options: options
            });
          },
          ...options,
          responseType: options.dataType || options.responseType
        };
        debug('发送请求', {
          requestObj: requestObj
        });
        GM_xmlhttpRequest(requestObj);
      }));
      if (window.DEBUG) {
        console.log('%cAuto-Task[httpRequest]:', 'color:blue', result);
      }
      if (result.status !== 600 && times < 2) {
        debug('请求失败,准备重试', {
          status: result.status,
          retryTimes: times + 1
        });
        return await httpRequest(options, times + 1);
      }
      debug('请求完成', {
        status: result.status,
        result: result.result
      });
      return result;
    } catch (error) {
      debug('请求发生JavaScript错误', {
        error: error
      });
      console.log('%cAuto-Task[httpRequest]:', 'color:red', JSON.stringify({
        errorMsg: error,
        options: options
      }));
      throwError(error, 'httpRequest');
      return {
        result: 'JsError',
        statusText: 'Error',
        status: 604,
        error: error,
        options: options
      };
    }
  };
  var ASF = '<?xml version="1.0" encoding="UTF-8"?>\n<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="16" height="16" viewBox="0 0 16 16" version="1.1">\n<g id="surface1">\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(11.372549%,17.254902%,26.274511%);fill-opacity:1;" d="M 7.652344 0.175781 C 8.046875 0.351562 8.210938 0.59375 8.441406 0.953125 C 9.007812 1.746094 9.644531 2.539062 10.640625 2.761719 C 12.011719 2.949219 13.429688 2.882812 14.808594 2.796875 C 14.90625 2.792969 15.003906 2.785156 15.101562 2.78125 C 15.1875 2.773438 15.273438 2.769531 15.363281 2.761719 C 15.429688 2.769531 15.496094 2.777344 15.566406 2.78125 C 15.761719 3.082031 15.757812 3.191406 15.75 3.539062 C 15.75 3.625 15.746094 3.710938 15.746094 3.800781 C 15.742188 3.867188 15.742188 3.933594 15.738281 4 C 15.824219 4.027344 15.910156 4.058594 16 4.085938 C 15.988281 4.34375 15.976562 4.597656 15.960938 4.851562 C 15.957031 4.925781 15.957031 5 15.953125 5.074219 C 15.945312 5.175781 15.945312 5.175781 15.941406 5.285156 C 15.9375 5.378906 15.9375 5.378906 15.929688 5.480469 C 15.914062 5.652344 15.914062 5.652344 15.824219 5.914062 C 15.816406 6.058594 15.8125 6.203125 15.8125 6.34375 C 15.8125 6.433594 15.8125 6.519531 15.808594 6.613281 C 15.808594 6.707031 15.808594 6.800781 15.808594 6.898438 C 15.804688 7.097656 15.804688 7.300781 15.800781 7.5 C 15.796875 7.816406 15.792969 8.132812 15.792969 8.449219 C 15.789062 8.753906 15.785156 9.058594 15.78125 9.363281 C 15.78125 9.457031 15.78125 9.550781 15.78125 9.648438 C 15.777344 9.738281 15.777344 9.824219 15.773438 9.914062 C 15.773438 10.03125 15.773438 10.03125 15.773438 10.148438 C 15.738281 10.347656 15.738281 10.347656 15.625 10.503906 C 15.386719 10.671875 15.152344 10.625 14.871094 10.609375 C 14.867188 10.675781 14.859375 10.742188 14.855469 10.808594 C 14.816406 11.367188 14.746094 11.890625 14.609375 12.433594 C 14.3125 12.535156 14.105469 12.507812 13.804688 12.441406 C 13.6875 12.414062 13.6875 12.414062 13.570312 12.390625 C 13.480469 12.367188 13.480469 12.367188 13.390625 12.347656 C 13.359375 12.515625 13.359375 12.515625 13.324219 12.683594 C 13.226562 13.144531 13.054688 13.566406 12.871094 14 C 12.75 13.988281 12.628906 13.980469 12.511719 13.96875 C 12.410156 13.957031 12.410156 13.957031 12.308594 13.949219 C 12.050781 13.90625 11.8125 13.828125 11.566406 13.738281 C 11.492188 13.867188 11.492188 13.867188 11.417969 14 C 11.355469 14.113281 11.292969 14.222656 11.226562 14.335938 C 11.195312 14.394531 11.164062 14.449219 11.132812 14.507812 C 11 14.738281 10.882812 14.941406 10.695312 15.128906 C 10.257812 15.097656 9.957031 14.863281 9.601562 14.625 C 9.402344 14.503906 9.402344 14.503906 9.222656 14.542969 C 9.015625 14.617188 8.925781 14.703125 8.777344 14.863281 C 8.730469 14.914062 8.683594 14.964844 8.632812 15.019531 C 8.535156 15.125 8.433594 15.234375 8.335938 15.339844 C 7.972656 15.726562 7.972656 15.726562 7.675781 15.792969 C 7.4375 15.726562 7.367188 15.660156 7.21875 15.460938 C 7 15.1875 6.773438 14.941406 6.523438 14.695312 C 6.375 14.550781 6.230469 14.40625 6.085938 14.261719 C 6.023438 14.195312 5.957031 14.128906 5.890625 14.0625 C 2.175781 10.347656 2.175781 10.347656 1.945312 10.117188 C 1.738281 9.910156 1.527344 9.707031 1.316406 9.503906 C 1.160156 9.351562 1 9.199219 0.84375 9.042969 C 0.722656 8.925781 0.722656 8.925781 0.597656 8.808594 C 0.519531 8.734375 0.445312 8.660156 0.363281 8.582031 C 0.296875 8.515625 0.226562 8.445312 0.152344 8.375 C -0.03125 8.132812 -0.0351562 8.039062 0 7.738281 C 0.117188 7.570312 0.117188 7.570312 0.28125 7.402344 C 0.34375 7.339844 0.40625 7.277344 0.472656 7.210938 C 0.542969 7.144531 0.613281 7.074219 0.683594 7.003906 C 0.757812 6.933594 0.828125 6.859375 0.902344 6.785156 C 1.101562 6.585938 1.304688 6.386719 1.503906 6.1875 C 1.714844 5.980469 1.921875 5.773438 2.132812 5.5625 C 2.480469 5.214844 2.832031 4.863281 3.183594 4.515625 C 3.683594 4.023438 4.175781 3.53125 4.671875 3.039062 C 5.015625 2.695312 5.359375 2.355469 5.699219 2.015625 C 5.785156 1.929688 5.867188 1.847656 5.949219 1.765625 C 6.222656 1.492188 6.496094 1.222656 6.773438 0.949219 C 6.839844 0.882812 6.910156 0.8125 6.980469 0.742188 C 7.078125 0.648438 7.078125 0.648438 7.171875 0.550781 C 7.226562 0.496094 7.285156 0.441406 7.339844 0.386719 C 7.476562 0.261719 7.476562 0.261719 7.652344 0.175781 Z M 7.652344 0.175781 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(5.490196%,7.843138%,14.901961%);fill-opacity:1;" d="M 10.433594 2.78125 C 11.101562 2.777344 11.765625 2.773438 12.433594 2.769531 C 12.742188 2.769531 13.050781 2.765625 13.359375 2.761719 C 13.714844 2.757812 14.070312 2.757812 14.425781 2.757812 C 14.539062 2.753906 14.648438 2.753906 14.761719 2.753906 C 14.867188 2.753906 14.96875 2.753906 15.074219 2.753906 C 15.210938 2.753906 15.210938 2.753906 15.347656 2.75 C 15.421875 2.761719 15.492188 2.773438 15.566406 2.78125 C 15.761719 3.082031 15.757812 3.191406 15.75 3.539062 C 15.75 3.625 15.746094 3.710938 15.746094 3.800781 C 15.742188 3.867188 15.742188 3.933594 15.738281 4 C 15.824219 4.027344 15.910156 4.058594 16 4.085938 C 15.988281 4.332031 15.980469 4.578125 15.96875 4.824219 C 15.964844 4.929688 15.964844 4.929688 15.960938 5.035156 C 15.9375 5.492188 15.863281 5.90625 15.738281 6.347656 C 15.675781 6.3125 15.613281 6.273438 15.546875 6.234375 C 15.011719 5.921875 14.472656 5.609375 13.9375 5.296875 C 13.761719 5.195312 13.582031 5.089844 13.40625 4.988281 C 12.996094 4.746094 12.585938 4.507812 12.167969 4.273438 C 10.847656 3.527344 10.847656 3.527344 10.433594 2.957031 C 10.433594 2.898438 10.433594 2.839844 10.433594 2.78125 Z M 10.433594 2.78125 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(95.686275%,96.078432%,96.470588%);fill-opacity:1;" d="M 7.566406 5.304688 C 7.878906 5.410156 7.90625 5.480469 8.058594 5.761719 C 8.117188 5.859375 8.117188 5.859375 8.175781 5.964844 C 8.269531 6.195312 8.277344 6.363281 8.261719 6.609375 C 8.355469 6.550781 8.449219 6.492188 8.550781 6.433594 C 8.777344 6.296875 8.855469 6.261719 9.128906 6.261719 C 9.136719 6.425781 9.140625 6.589844 9.148438 6.753906 C 9.152344 6.894531 9.152344 6.894531 9.15625 7.035156 C 9.125 7.34375 9.058594 7.492188 8.871094 7.738281 C 8.703125 7.871094 8.703125 7.871094 8.527344 7.972656 C 8.46875 8.007812 8.410156 8.042969 8.351562 8.078125 C 8.292969 8.109375 8.234375 8.140625 8.175781 8.175781 C 8.113281 8.207031 8.054688 8.238281 7.996094 8.273438 C 7.710938 8.398438 7.511719 8.335938 7.21875 8.261719 C 6.832031 8.070312 6.476562 7.859375 6.261719 7.476562 C 6.167969 7.203125 6.164062 7.007812 6.167969 6.722656 C 6.167969 6.636719 6.171875 6.550781 6.171875 6.460938 C 6.171875 6.394531 6.171875 6.328125 6.175781 6.261719 C 6.492188 6.332031 6.753906 6.457031 7.042969 6.609375 C 7.050781 6.480469 7.050781 6.480469 7.058594 6.347656 C 7.113281 5.9375 7.332031 5.636719 7.566406 5.304688 Z M 7.566406 5.304688 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(5.882353%,8.235294%,15.294118%);fill-opacity:1;" d="M 15.652344 6.433594 C 15.679688 6.433594 15.710938 6.433594 15.738281 6.433594 C 15.746094 6.988281 15.75 7.539062 15.753906 8.09375 C 15.753906 8.28125 15.757812 8.46875 15.757812 8.65625 C 15.761719 8.925781 15.761719 9.195312 15.765625 9.464844 C 15.765625 9.550781 15.765625 9.636719 15.769531 9.722656 C 15.769531 9.800781 15.769531 9.878906 15.769531 9.960938 C 15.769531 10.027344 15.769531 10.097656 15.769531 10.167969 C 15.730469 10.398438 15.664062 10.472656 15.476562 10.609375 C 15.15625 10.625 15.15625 10.625 14.871094 10.609375 C 14.867188 10.675781 14.859375 10.742188 14.855469 10.808594 C 14.816406 11.367188 14.746094 11.890625 14.609375 12.433594 C 14.3125 12.535156 14.105469 12.507812 13.804688 12.441406 C 13.726562 12.421875 13.648438 12.40625 13.570312 12.390625 C 13.480469 12.367188 13.480469 12.367188 13.390625 12.347656 C 13.449219 12.027344 13.535156 11.730469 13.644531 11.425781 C 13.679688 11.332031 13.710938 11.238281 13.746094 11.144531 C 13.78125 11.046875 13.816406 10.953125 13.851562 10.851562 C 13.902344 10.710938 13.902344 10.710938 13.957031 10.566406 C 14.054688 10.289062 14.160156 10.015625 14.261719 9.738281 C 14.292969 9.65625 14.320312 9.574219 14.351562 9.488281 C 14.613281 8.792969 14.613281 8.792969 14.929688 8.523438 C 15.359375 8.140625 15.347656 7.753906 15.40625 7.203125 C 15.441406 6.886719 15.472656 6.707031 15.652344 6.433594 Z M 15.652344 6.433594 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(13.725491%,18.431373%,37.254903%);fill-opacity:1;" d="M 13.566406 9.824219 C 13.609375 9.914062 13.609375 9.914062 13.652344 10 C 13.539062 10.21875 13.417969 10.425781 13.289062 10.636719 C 13.207031 10.769531 13.125 10.90625 13.046875 11.039062 C 13.003906 11.109375 12.960938 11.183594 12.914062 11.253906 C 12.691406 11.632812 12.46875 12.015625 12.25 12.398438 C 12.207031 12.472656 12.160156 12.550781 12.117188 12.628906 C 11.8125 13.160156 11.53125 13.703125 11.269531 14.253906 C 11.113281 14.582031 10.933594 14.855469 10.695312 15.128906 C 10.3125 15.078125 10.042969 14.933594 9.710938 14.734375 C 9.574219 14.648438 9.574219 14.648438 9.429688 14.566406 C 9.359375 14.523438 9.289062 14.480469 9.21875 14.433594 C 9.253906 14.046875 9.523438 13.835938 9.789062 13.582031 C 9.832031 13.539062 9.875 13.496094 9.921875 13.449219 C 10.15625 13.21875 10.402344 12.996094 10.65625 12.78125 C 11.15625 12.34375 11.621094 11.871094 12.089844 11.398438 C 12.25 11.238281 12.414062 11.078125 12.574219 10.914062 C 12.679688 10.8125 12.78125 10.707031 12.886719 10.605469 C 12.933594 10.554688 12.980469 10.507812 13.03125 10.457031 C 13.234375 10.253906 13.40625 10.066406 13.566406 9.824219 Z M 13.566406 9.824219 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(95.294118%,95.686275%,96.078432%);fill-opacity:1;" d="M 6.175781 10.347656 C 6.503906 10.464844 6.769531 10.617188 7.054688 10.820312 C 7.128906 10.871094 7.203125 10.925781 7.277344 10.976562 C 7.445312 11.105469 7.59375 11.234375 7.738281 11.390625 C 7.769531 11.316406 7.769531 11.316406 7.800781 11.238281 C 7.941406 10.992188 8.105469 10.882812 8.335938 10.722656 C 8.410156 10.671875 8.488281 10.617188 8.566406 10.5625 C 8.761719 10.449219 8.90625 10.386719 9.128906 10.347656 C 9.21875 11.453125 9.21875 11.453125 8.914062 11.832031 C 8.617188 12.113281 8.285156 12.273438 7.914062 12.433594 C 7.882812 12.921875 7.855469 13.410156 7.824219 13.914062 C 7.683594 13.914062 7.539062 13.914062 7.390625 13.914062 C 7.390625 13.398438 7.390625 12.878906 7.390625 12.347656 C 7.246094 12.320312 7.105469 12.289062 6.957031 12.261719 C 6.5625 12.078125 6.359375 11.875 6.175781 11.476562 C 6.144531 11.097656 6.152344 10.726562 6.175781 10.347656 Z M 6.175781 10.347656 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(8.235294%,11.764706%,23.137255%);fill-opacity:1;" d="M 14.433594 9.042969 C 14.523438 9.21875 14.523438 9.21875 14.484375 9.378906 C 14.460938 9.445312 14.4375 9.507812 14.414062 9.574219 C 14.386719 9.648438 14.359375 9.71875 14.335938 9.792969 C 14.308594 9.871094 14.277344 9.949219 14.25 10.027344 C 14.222656 10.105469 14.195312 10.183594 14.164062 10.265625 C 14.007812 10.703125 13.84375 11.132812 13.675781 11.566406 C 13.503906 11.996094 13.386719 12.429688 13.285156 12.878906 C 13.179688 13.269531 13.03125 13.632812 12.871094 14 C 12.378906 13.96875 12 13.910156 11.566406 13.652344 C 11.703125 13.144531 11.960938 12.71875 12.222656 12.265625 C 12.269531 12.1875 12.316406 12.105469 12.363281 12.023438 C 12.460938 11.859375 12.554688 11.695312 12.648438 11.53125 C 12.765625 11.332031 12.882812 11.128906 12.996094 10.925781 C 13.402344 10.222656 13.859375 9.621094 14.433594 9.042969 Z M 14.433594 9.042969 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(0%,0%,0%);fill-opacity:1;" d="M 10.433594 2.78125 C 11.101562 2.777344 11.765625 2.773438 12.433594 2.769531 C 12.742188 2.769531 13.050781 2.765625 13.359375 2.761719 C 13.714844 2.757812 14.070312 2.757812 14.425781 2.757812 C 14.539062 2.753906 14.648438 2.753906 14.761719 2.753906 C 14.867188 2.753906 14.96875 2.753906 15.074219 2.753906 C 15.210938 2.753906 15.210938 2.753906 15.347656 2.75 C 15.421875 2.761719 15.492188 2.773438 15.566406 2.78125 C 15.761719 3.082031 15.757812 3.191406 15.75 3.539062 C 15.75 3.625 15.746094 3.710938 15.746094 3.800781 C 15.742188 3.867188 15.742188 3.933594 15.738281 4 C 14.917969 3.957031 14.128906 3.839844 13.324219 3.6875 C 13.066406 3.640625 12.808594 3.59375 12.546875 3.550781 C 12.363281 3.515625 12.175781 3.484375 11.992188 3.449219 C 11.78125 3.414062 11.566406 3.378906 11.355469 3.34375 C 11.226562 3.324219 11.226562 3.324219 11.097656 3.304688 C 11.027344 3.292969 10.953125 3.28125 10.878906 3.273438 C 10.664062 3.207031 10.570312 3.132812 10.433594 2.957031 C 10.433594 2.898438 10.433594 2.839844 10.433594 2.78125 Z M 10.433594 2.78125 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(95.686275%,96.078432%,96.470588%);fill-opacity:1;" d="M 9.128906 8.261719 C 9.183594 9.257812 9.183594 9.257812 9.011719 9.695312 C 8.605469 10.082031 8.152344 10.3125 7.597656 10.398438 C 7.09375 10.3125 6.6875 10.128906 6.347656 9.738281 C 6.074219 9.320312 6.136719 8.828125 6.175781 8.347656 C 6.539062 8.394531 6.765625 8.519531 7.066406 8.734375 C 7.140625 8.789062 7.214844 8.839844 7.292969 8.894531 C 7.476562 9.042969 7.476562 9.042969 7.566406 9.21875 C 7.621094 9.21875 7.679688 9.21875 7.738281 9.21875 C 7.835938 9.117188 7.933594 9.015625 8.03125 8.914062 C 8.738281 8.261719 8.738281 8.261719 9.128906 8.261719 Z M 9.128906 8.261719 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(7.843138%,11.372549%,22.745098%);fill-opacity:1;" d="M 11.042969 3.476562 C 11.648438 3.675781 12.246094 3.886719 12.84375 4.101562 C 12.984375 4.152344 12.984375 4.152344 13.125 4.207031 C 14.054688 4.542969 14.984375 4.878906 15.914062 5.21875 C 15.867188 5.597656 15.816406 5.972656 15.738281 6.347656 C 15.675781 6.3125 15.613281 6.273438 15.546875 6.234375 C 15.011719 5.921875 14.472656 5.609375 13.9375 5.296875 C 13.667969 5.140625 13.402344 4.984375 13.132812 4.828125 C 12.804688 4.636719 12.476562 4.445312 12.148438 4.253906 C 12.011719 4.179688 11.878906 4.101562 11.746094 4.023438 C 11.683594 3.988281 11.621094 3.949219 11.558594 3.914062 C 11.140625 3.671875 11.140625 3.671875 11.042969 3.476562 Z M 11.042969 3.476562 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(0%,0%,0.392157%);fill-opacity:1;" d="M 15.652344 6.433594 C 15.679688 6.433594 15.710938 6.433594 15.738281 6.433594 C 15.746094 6.988281 15.75 7.539062 15.753906 8.09375 C 15.753906 8.28125 15.757812 8.46875 15.757812 8.65625 C 15.761719 8.925781 15.761719 9.195312 15.765625 9.464844 C 15.765625 9.550781 15.765625 9.636719 15.769531 9.722656 C 15.769531 9.839844 15.769531 9.839844 15.769531 9.960938 C 15.769531 10.027344 15.769531 10.097656 15.769531 10.167969 C 15.730469 10.394531 15.664062 10.472656 15.476562 10.609375 C 15.199219 10.625 15.199219 10.625 14.957031 10.609375 C 14.875 10.285156 14.871094 10.046875 14.933594 9.71875 C 14.949219 9.636719 14.964844 9.550781 14.980469 9.464844 C 15 9.378906 15.015625 9.289062 15.03125 9.199219 C 15.050781 9.113281 15.066406 9.027344 15.082031 8.933594 C 15.203125 8.289062 15.203125 8.289062 15.304688 8.085938 C 15.34375 7.792969 15.375 7.5 15.40625 7.203125 C 15.441406 6.886719 15.472656 6.707031 15.652344 6.433594 Z M 15.652344 6.433594 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(86.274511%,87.450981%,89.019608%);fill-opacity:1;" d="M 8.609375 3.390625 C 8.78125 3.390625 8.953125 3.390625 9.128906 3.390625 C 9.128906 4.109375 9.128906 4.824219 9.128906 5.566406 C 8.988281 5.59375 8.84375 5.621094 8.695312 5.652344 C 8.609375 5.566406 8.609375 5.566406 8.597656 5.355469 C 8.597656 5.265625 8.601562 5.175781 8.601562 5.082031 C 8.601562 4.984375 8.601562 4.886719 8.601562 4.785156 C 8.601562 4.683594 8.601562 4.578125 8.601562 4.472656 C 8.605469 4.371094 8.605469 4.265625 8.605469 4.160156 C 8.605469 3.902344 8.605469 3.648438 8.609375 3.390625 Z M 8.609375 3.390625 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(93.725491%,94.117647%,94.509804%);fill-opacity:1;" d="M 6.175781 3.390625 C 6.316406 3.390625 6.460938 3.390625 6.609375 3.390625 C 6.609375 4.136719 6.609375 4.882812 6.609375 5.652344 C 6.464844 5.625 6.320312 5.59375 6.175781 5.566406 C 6.175781 4.847656 6.175781 4.128906 6.175781 3.390625 Z M 6.175781 3.390625 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(93.333334%,93.725491%,94.117647%);fill-opacity:1;" d="M 7.476562 2.433594 C 7.59375 2.433594 7.707031 2.433594 7.824219 2.433594 C 7.824219 3.179688 7.824219 3.925781 7.824219 4.695312 C 7.683594 4.695312 7.539062 4.695312 7.390625 4.695312 C 7.390625 4.335938 7.386719 3.976562 7.386719 3.613281 C 7.386719 3.511719 7.382812 3.410156 7.382812 3.300781 C 7.382812 3.203125 7.382812 3.105469 7.382812 3.003906 C 7.382812 2.914062 7.382812 2.824219 7.382812 2.730469 C 7.390625 2.523438 7.390625 2.523438 7.476562 2.433594 Z M 7.476562 2.433594 "/>\n<path style=" stroke:none;fill-rule:nonzero;fill:rgb(5.490196%,7.843138%,14.901961%);fill-opacity:1;" d="M 15.390625 6.871094 C 15.511719 7.203125 15.449219 7.484375 15.390625 7.824219 C 15.304688 7.769531 15.21875 7.710938 15.128906 7.652344 C 15.171875 7.328125 15.207031 7.148438 15.390625 6.871094 Z M 15.390625 6.871094 "/>\n</g>\n</svg>\n';
  var Web = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 256 256" width="16" height="16">\n\t<title>favicon</title>\n\t<defs>\n\t\t<image  width="256" height="256" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>\n';
  var Discord$1 = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 256 256" width="16" height="16">\n\t<title>favicon</title>\n\t<defs>\n\t\t<image  width="256" height="256" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>\n';
  var Twitch$1 = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 32 32" width="16" height="16">\n\t<title>favicon-32-e29e246c157142c94346</title>\n\t<defs>\n\t\t<image  width="28" height="32" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="2" y="0"/>\n</svg>\n';
  var Instagram = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 76 76" width="16" height="16">\n\t<title>lam-fZmwmvn</title>\n\t<defs>\n\t\t<image  width="76" height="76" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>';
  var Twitter$1 = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024" width="16" height="16">\n\t<title>icon-ios</title>\n\t<defs>\n\t\t<image  width="1024" height="1024" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>\n';
  var Reddit$1 = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 192 192" width="16" height="16">\n\t<title>192x192</title>\n\t<defs>\n\t\t<image  width="192" height="192" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>';
  var Youtube$1 = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 48 48" width="16" height="16">\n\t<title>favicon_48x48</title>\n\t<defs>\n\t\t<image  width="48" height="48" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>';
  var Vk$1 = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 256 256" width="16" height="16">\n\t<title>fav_logo</title>\n\t<defs>\n\t\t<image  width="256" height="256" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>';
  var AutoTask = '<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 32 32" width="16" height="16">\n\t<title>favicon</title>\n\t<defs>\n\t\t<image  width="32" height="32" id="img1" href=""/>\n\t</defs>\n\t<style>\n\t</style>\n\t<use id="Background" href="#img1" x="0" y="0"/>\n</svg>';
  const ICONS = {
    '[ASF]': ASF,
    '[Web]': Web,
    '[Discord]': Discord$1,
    '[Twitch]': Twitch$1,
    '[Instagram]': Instagram,
    '[Twitter]': Twitter$1,
    '[Reddit]': Reddit$1,
    '[Youtube]': Youtube$1,
    '[Vk]': Vk$1,
    '[AutoTask]': AutoTask
  };
  const generateLink = (url, text) => `<a href="${url}" target="_blank">${text}</a>`;
  const createBaseElement = content => $(`<li>${content}<font class="log-status"></font></li>`).addClass('card-text');
  const createPlatformElement = (type, text, id) => {
    const urlGenerators = {
      group: text => `https://steamcommunity.com/groups/${text}`,
      officialGroup: text => `https://steamcommunity.com/games/${text}`,
      forum: text => `https://steamcommunity.com/app/${text}/discussions/`,
      curator: text => `https://store.steampowered.com/${text?.includes('/') ? text : `curator/${text}`}`,
      app: text => `https://store.steampowered.com/app/${text}`,
      sub: text => `https://steamdb.info/sub/${text}/`,
      workshop: text => `https://steamcommunity.com/sharedfiles/filedetails/?id=${text}`,
      announcement: (text, id) => `https://store.steampowered.com/news/app/${text}/view/${id}`,
      discord: {
        invite: text => `https://discord.com/invite/${text}`,
        server: text => `https://discord.com/channels/@me/${text}`
      },
      twitch: text => `https://www.twitch.tv/${text}`,
      instagram: text => `https://www.instagram.com/${text}/`,
      twitter: text => `https://x.com/${text}`,
      reddit: {
        subreddit: text => `https://www.reddit.com/r/${text}/`,
        user: text => `https://www.reddit.com/user/${text?.replace('u_', '')}`
      },
      youtube: {
        channel: text => `https://www.youtube.com/channel/${text}`,
        video: text => `https://www.youtube.com/watch?v=${text}`
      },
      vk: text => `https://vk.com/${text}/`
    };
    const typeMap = {
      joiningSteamGroup: [ 'group' ],
      leavingSteamGroup: [ 'group' ],
      gettingSteamGroupId: [ 'group' ],
      joiningSteamOfficialGroup: [ 'officialGroup' ],
      leavingSteamOfficialGroup: [ 'officialGroup' ],
      gettingSteamOfficialGroupId: [ 'officialGroup' ],
      subscribingForum: [ 'forum' ],
      unsubscribingForum: [ 'forum' ],
      gettingForumId: [ 'forum' ],
      followingCurator: [ 'curator' ],
      unfollowingCurator: [ 'curator' ],
      gettingCuratorId: [ 'curator' ],
      addingToWishlist: [ 'app' ],
      removingFromWishlist: [ 'app' ],
      followingGame: [ 'app' ],
      unfollowingGame: [ 'app' ],
      gettingSubid: [ 'app' ],
      addingFreeLicense: [ 'app', 'sub' ],
      requestingPlayTestAccess: [ 'app' ],
      gettingDemoAppid: [ 'app' ],
      favoritingWorkshop: [ 'workshop' ],
      unfavoritingWorkshop: [ 'workshop' ],
      gettingWorkshopAppId: [ 'workshop' ],
      votingUpWorkshop: [ 'workshop' ],
      gettingAnnouncementParams: [ 'announcement' ],
      likingAnnouncement: [ 'announcement' ],
      joiningDiscordServer: [ 'discord', 'invite' ],
      gettingDiscordGuild: [ 'discord', 'invite' ],
      gettingDiscordXContextProperties: [ 'discord', 'invite' ],
      leavingDiscordServer: [ 'discord', 'server' ],
      followingTwitchChannel: [ 'twitch' ],
      unfollowingTwitchChannel: [ 'twitch' ],
      gettingTwitchChannelId: [ 'twitch' ],
      gettingInsUserId: [ 'instagram' ],
      followingIns: [ 'instagram' ],
      unfollowingIns: [ 'instagram' ],
      gettingTwitterUserId: [ 'twitter' ],
      followingTwitterUser: [ 'twitter' ],
      unfollowingTwitterUser: [ 'twitter' ],
      joiningReddit: [ 'reddit', 'subreddit' ],
      leavingReddit: [ 'reddit', 'subreddit' ],
      followingRedditUser: [ 'reddit', 'user' ],
      unfollowingRedditUser: [ 'reddit', 'user' ],
      followingYtbChannel: [ 'youtube', 'channel' ],
      unfollowingYtbChannel: [ 'youtube', 'channel' ],
      likingYtbVideo: [ 'youtube', 'video' ],
      unlikingYtbVideo: [ 'youtube', 'video' ],
      gettingVkId: [ 'vk' ],
      joiningVkGroup: [ 'vk' ],
      leavingVkGroup: [ 'vk' ],
      joiningVkPublic: [ 'vk' ],
      leavingVkPublic: [ 'vk' ],
      sendingVkWall: [ 'vk' ],
      deletingVkWall: [ 'vk' ]
    };
    const urlConfig = typeMap[type];
    if (!urlConfig || !text) {
      return null;
    }
    const [platform, subType] = urlConfig;
    const urlGenerator = urlGenerators[platform];
    if (typeof urlGenerator === 'function') {
      const url = urlGenerator(text, id);
      const displayText = platform === 'announcement' ? id || '' : text;
      return createBaseElement(`${I18n(type)}[${generateLink(url, displayText)}]...`);
    }
    if (subType && typeof urlGenerator === 'object') {
      const subGenerator = urlGenerator[subType];
      if (typeof subGenerator === 'function') {
        const displayText = type.includes('RedditUser') ? text.replace('u_', '') : text;
        return createBaseElement(`${I18n(type)}[${generateLink(subGenerator(text), displayText)}]...`);
      }
    }
    return null;
  };
  const createSpecialElement = (type, text, html, id) => {
    switch (type) {
     case 'retweetting':
     case 'unretweetting':
      return createBaseElement(`${I18n(type)}${text}...`);

     case 'visitingLink':
      return createBaseElement(`${I18n('visitingLink')}[${generateLink(text || '', text || '')}]...`);

     case 'verifyingInsAuth':
     case 'text':
      return createBaseElement(I18n(text || ''));

     case 'html':
      return $(text || html || '');

     case 'whiteList':
      return $(`<li><font class="warning">${I18n('skipTask')}[${text}(${id})](${I18n('whiteList')})</font></li>`);

     case 'globalOptionsSkip':
      return $(`<li>${I18n('skipTaskOption')}<font class="warning">${text}</font></li>`);

     default:
      return createBaseElement(`${I18n('unKnown')}:${type}(${text})...`);
    }
  };
  const echoLog = ({type: type, text: text, html: html, id: id, before: before}) => {
    const emptyStatus = {
      success: () => emptyStatus,
      error: () => emptyStatus,
      warning: () => emptyStatus,
      info: () => emptyStatus,
      view: () => emptyStatus,
      remove: () => emptyStatus
    };
    try {
      let ele;
      if (!type && !text && !html) {
        ele = createBaseElement('');
      } else if (text && !type) {
        ele = createBaseElement(text);
      } else if (html && !type) {
        ele = $(html);
      } else if (type) {
        const platformElement = createPlatformElement(type, text, id);
        ele = platformElement || createSpecialElement(type, text, html, id);
      } else {
        ele = createBaseElement('');
      }
      if (before) {
        if (before in ICONS) {
          const iconKey = before;
          const svgContent = ICONS[iconKey];
          const base64Svg = btoa(svgContent);
          ele.prepend(`<font class="before-icon" style="background-image: url('data:image/svg+xml;base64,${base64Svg}')"></font>`);
        } else {
          ele.prepend(`<font class="before">${before}</font>`);
        }
      } else {
        const base64Svg = btoa(ICONS['[AutoTask]']);
        ele.prepend(`<font class="before-icon" style="background-image: url('data:image/svg+xml;base64,${base64Svg}')"></font>`);
      }
      ele.addClass('card-text');
      $('#auto-task-info').append(ele);
      ele[0]?.scrollIntoView();
      const font = ele.find('font.log-status');
      const status = {
        font: font,
        success(text = 'Success', html = false) {
          this.font?.attr('class', '').addClass('success');
          html ? this.font?.html(text) : this.font?.text(text);
          return this;
        },
        error(text = 'Error', html = false) {
          this.font?.attr('class', '').addClass('error');
          html ? this.font?.html(text) : this.font?.text(text);
          return this;
        },
        warning(text = 'Warning', html = false) {
          this.font?.attr('class', '').addClass('warning');
          html ? this.font?.html(text) : this.font?.text(text);
          return this;
        },
        info(text = 'Info', html = false) {
          this.font?.attr('class', '').addClass('info');
          html ? this.font?.html(text) : this.font?.text(text);
          return this;
        },
        view() {
          this.font?.[0].scrollIntoView();
          return this;
        },
        remove() {
          this.font?.parent().remove();
          return this;
        }
      };
      return status;
    } catch (error) {
      throwError(error, 'echoLog');
      return emptyStatus;
    }
  };
  const unique = array => {
    try {
      return Array.from(new Set(array));
    } catch (error) {
      throwError(error, 'unique');
      return [];
    }
  };
  const delay = (time = 1e3) => new Promise((resolve => setTimeout((() => resolve(true)), time)));
  const getRedirectLink = async (link, redirectOnce = false) => {
    try {
      if (!link) {
        return null;
      }
      const redirectLinksCache = GM_getValue('redirectLinks') || {};
      const cachedLink = redirectLinksCache[link];
      if (cachedLink) {
        debug('使用缓存的重定向链接', {
          original: link,
          cached: cachedLink
        });
        return cachedLink;
      }
      const {data: data} = await httpRequest({
        url: link,
        method: 'GET',
        redirect: redirectOnce ? 'manual' : 'follow'
      });
      if (data?.finalUrl) {
        redirectLinksCache[link] = data.finalUrl;
        GM_setValue('redirectLinks', redirectLinksCache);
        debug('获取新的重定向链接', {
          original: link,
          final: data.finalUrl
        });
        return data.finalUrl;
      }
      debug('未找到重定向链接', {
        link: link
      });
      return null;
    } catch (error) {
      throwError(error, 'getRedirectLink');
      return null;
    }
  };
  const visitLink = async (link, options) => {
    try {
      debug('开始访问链接', {
        link: link,
        options: options
      });
      const logStatus = echoLog({
        type: 'visitLink',
        text: link
      });
      const {result: result, statusText: statusText, status: status} = await httpRequest({
        url: link,
        method: 'GET',
        ...options
      });
      if (result === 'Success') {
        debug('链接访问成功', {
          link: link
        });
        logStatus.success();
        return true;
      }
      debug('链接访问失败', {
        link: link,
        result: result,
        statusText: statusText,
        status: status
      });
      logStatus.error(`${result}:${statusText}(${status})`);
      return false;
    } catch (error) {
      throwError(error, 'visitLink');
      return false;
    }
  };
  const getUrlQuery = url => {
    try {
      debug('开始解析URL查询参数', {
        url: url || window.location.href
      });
      const searchParams = url ? new URL(url, window.location.origin).searchParams : new URLSearchParams(window.location.search);
      const query = {};
      for (const [key, value] of searchParams.entries()) {
        query[key] = value;
      }
      debug('URL查询参数解析结果', query);
      return query;
    } catch (error) {
      throwError(error, 'getUrlQuery');
      return {};
    }
  };
  const stringToColour = str => {
    try {
      let hash = 0;
      for (let i = 0; i < str.length; i++) {
        hash = str.charCodeAt(i) + ((hash << 5) - hash);
      }
      const rgb = Array.from({
        length: 3
      }, ((_, i) => {
        const value = hash >> i * 8 & 255;
        return value.toString(16).padStart(2, '0');
      }));
      const color = `#${rgb.join('')}`;
      return color;
    } catch (error) {
      throwError(error, 'stringToColour');
      return '#ffffff';
    }
  };
  const getAllLocalStorageAsObjects = localStorage => {
    try {
      debug('开始将所有LocalStorage转换为对象');
      const result = {};
      for (let i = 0; i < localStorage.length; i++) {
        const key = localStorage.key(i);
        if (!key) {
          continue;
        }
        const value = localStorage.getItem(key);
        try {
          result[key] = JSON.parse(value);
        } catch (error) {
          result[key] = value;
          console.error(error);
        }
      }
      debug('所有LocalStorage转换为对象完成', result);
      return result;
    } catch (error) {
      debug('将所有LocalStorage转换为对象失败', {
        error: error
      });
      throwError(error, 'getAllLocalStorageAsObjects');
      return {};
    }
  };
  class Social {
    tasks;
    getRealParams(name, links, doTask, link2param) {
      try {
        debug('开始获取实际参数', {
          name: name,
          linksCount: links.length,
          doTask: doTask
        });
        let realParams = [];
        if (links.length > 0) {
          debug('处理链接参数');
          const convertedLinks = links.map((link => link2param(link))).filter((link => link !== undefined));
          debug('链接参数处理结果', {
            convertedLinksCount: convertedLinks.length
          });
          realParams = [ ...realParams, ...convertedLinks ];
        }
        if (!doTask && this.tasks[name]?.length) {
          debug('处理任务参数', {
            taskCount: this.tasks[name].length
          });
          realParams = [ ...realParams, ...this.tasks[name] ];
        }
        const uniqueParams = unique(realParams);
        debug('参数处理完成', {
          originalCount: realParams.length,
          uniqueCount: uniqueParams.length
        });
        return uniqueParams;
      } catch (error) {
        debug('获取实际参数时发生错误', {
          error: error
        });
        throwError(error, 'Social.getRealParams');
        return [];
      }
    }
  }
  class Discord extends Social {
    tasks;
    whiteList;
    #auth=GM_getValue('discordAuth') || {};
    #cache=GM_getValue('discordCache') || {};
    #xContextPropertiesCache=GM_getValue('discordXContextPropertiesCache') || {};
    #initialized=false;
    constructor() {
      super();
      const defaultTasksTemplate = {
        servers: []
      };
      this.tasks = defaultTasksTemplate;
      this.whiteList = {
        ...defaultTasksTemplate,
        ...GM_getValue('whiteList')?.discord || {}
      };
    }
    async init(action) {
      try {
        debug('开始初始化Discord模块', {
          action: action
        });
        if (!GM_getValue('dontRemindDiscordAgain')) {
          debug('显示Discord重要提示对话框');
          const result = await Swal.fire({
            title: I18n('discordImportantNotice'),
            text: I18n('discordImportantNoticeText'),
            showCancelButton: true,
            showDenyButton: true,
            confirmButtonText: I18n('continueDiscordTask'),
            cancelButtonText: I18n('skipDiscordTask'),
            denyButtonText: I18n('continueAndDontRemindAgain')
          }).then((({isConfirmed: isConfirmed, isDenied: isDenied}) => {
            if (isConfirmed) {
              return true;
            }
            if (isDenied) {
              GM_setValue('dontRemindDiscordAgain', true);
              return true;
            }
            return false;
          }));
          if (!result) {
            this.#initialized = false;
            return 'skip';
          }
        }
        if (action === 'do' && !globalOptions.doTask.discord.servers || action === 'undo' && !globalOptions.undoTask.discord.servers) {
          this.#initialized = false;
          debug('检测到用户已禁用Discord任务,跳过初始化');
          return 'skip';
        }
        if (this.#initialized) {
          debug('Discord模块已初始化');
          return true;
        }
        if (!this.#auth.auth || !this.#auth.xSuperProperties) {
          debug('未找到Discord授权信息,尝试更新授权');
          if (await this.#updateAuth()) {
            this.#initialized = true;
            return true;
          }
          return false;
        }
        const isVerified = await this.#verifyAuth();
        if (isVerified) {
          debug('Discord授权验证成功');
          echoLog({
            before: '[Discord]'
          }).success(I18n('initSuccess', 'Discord'));
          this.#initialized = true;
          return true;
        }
        GM_setValue('discordAuth', {
          auth: null
        });
        debug('Discord授权验证失败,尝试重新获取授权');
        if (await this.#updateAuth()) {
          echoLog({
            before: '[Discord]'
          }).success(I18n('initSuccess', 'Discord'));
          this.#initialized = true;
          return true;
        }
        echoLog({
          before: '[Discord]'
        }).error(I18n('initFailed', 'Discord'));
        return false;
      } catch (error) {
        throwError(error, 'Discord.init');
        return false;
      }
    }
    async #verifyAuth() {
      try {
        debug('开始验证Discord授权');
        const logStatus = echoLog({
          text: I18n('verifyingAuth', 'Discord'),
          before: '[Discord]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://discord.com/api/v6/users/@me',
          method: 'HEAD',
          headers: {
            authorization: this.#auth.auth
          }
        });
        if (result !== 'Success') {
          debug('Discord授权验证请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('Discord授权验证状态码错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('Discord授权验证成功');
        logStatus.success();
        return true;
      } catch (error) {
        throwError(error, 'Discord.verifyAuth');
        return false;
      }
    }
    async #updateAuth() {
      try {
        debug('开始更新Discord授权');
        const logStatus = echoLog({
          text: I18n('updatingAuth', 'Discord'),
          before: '[Discord]'
        });
        return await new Promise((resolve => {
          const newTab = GM_openInTab('https://discord.com/channels/@me', {
            active: true,
            insert: true,
            setParent: true
          });
          newTab.name = 'ATv4_discordAuth';
          newTab.onclose = async () => {
            const {auth: auth, xSuperProperties: xSuperProperties} = GM_getValue('discordAuth');
            if (auth && xSuperProperties) {
              debug('成功获取新的Discord授权');
              this.#auth = {
                auth: auth,
                xSuperProperties: xSuperProperties
              };
              logStatus.success();
              resolve(await this.#verifyAuth());
            } else {
              debug('获取Discord授权失败');
              logStatus.error('Error: Update discord auth failed!');
              resolve(false);
            }
          };
        }));
      } catch (error) {
        throwError(error, 'Discord.updateAuth');
        return false;
      }
    }
    async #joinServer(inviteId) {
      try {
        debug('开始加入Discord服务器', {
          inviteId: inviteId
        });
        const logStatus = echoLog({
          type: 'joiningDiscordServer',
          text: inviteId,
          before: '[Discord]'
        });
        const xContextProperties = await this.#getXContextProperties(inviteId);
        if (!xContextProperties) {
          debug('获取加群参数失败,无法加入服务器', {
            inviteId: inviteId
          });
          logStatus.error('Error: Failed to get xContextProperties');
          return false;
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://discord.com/api/v9/invites/${inviteId}`,
          method: 'POST',
          dataType: 'json',
          headers: {
            'content-type': 'application/json',
            authorization: this.#auth.auth,
            origin: 'https://discord.com',
            referer: `https://discord.com/invite/${inviteId}`,
            'x-super-properties': this.#auth.xSuperProperties,
            'x-context-properties': xContextProperties
          },
          data: '{"session_id":null}'
        });
        if (result !== 'Success' || data?.status !== 200) {
          debug('加入Discord服务器失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          if (status === 400) {
            debug('加入Discord服务器失败,状态码为400,需完成人机验证');
            logStatus.error(I18n('captchaNeeded'));
            return false;
          }
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        const guild = data.response?.guild?.id;
        if (!guild) {
          debug('获取服务器ID失败');
          logStatus.error('Error: Failed to get guild ID');
          return false;
        }
        debug('成功加入Discord服务器', {
          guild: guild
        });
        logStatus.success();
        this.#setCache(inviteId, guild);
        this.tasks.servers = unique([ ...this.tasks.servers, inviteId ]);
        return true;
      } catch (error) {
        throwError(error, 'Discord.joinServer');
        return false;
      }
    }
    async #leaveServer(inviteId) {
      try {
        debug('开始退出Discord服务器', {
          inviteId: inviteId
        });
        if (this.whiteList.servers.includes(inviteId)) {
          debug('服务器在白名单中,跳过退出操作', {
            inviteId: inviteId
          });
          echoLog({
            type: 'whiteList',
            text: 'Discord.leaveServer',
            id: inviteId,
            before: '[Discord]'
          });
          return true;
        }
        const guild = await this.#getGuild(inviteId);
        if (!guild) {
          debug('获取服务器ID失败,无法退出服务器', {
            inviteId: inviteId
          });
          return false;
        }
        const logStatus = echoLog({
          type: 'leavingDiscordServer',
          text: guild,
          before: '[Discord]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://discord.com/api/v9/users/@me/guilds/${guild}`,
          method: 'DELETE',
          headers: {
            authorization: this.#auth.auth,
            'x-super-properties': this.#auth.xSuperProperties
          }
        });
        if (result !== 'Success' || data?.status !== 204) {
          debug('退出Discord服务器失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        debug('成功退出Discord服务器', {
          guild: guild
        });
        logStatus.success();
        return true;
      } catch (error) {
        throwError(error, 'Discord.leaveServer');
        return false;
      }
    }
    async #getXContextProperties(inviteId) {
      try {
        debug('开始获取Discord加群参数', {
          inviteId: inviteId
        });
        const logStatus = echoLog({
          type: 'gettingDiscordXContextProperties',
          text: inviteId,
          before: '[Discord]'
        });
        const cachedXContextProperties = this.#xContextPropertiesCache[inviteId];
        if (cachedXContextProperties) {
          debug('从缓存中获取到加群参数', {
            inviteId: inviteId,
            cachedXContextProperties: cachedXContextProperties
          });
          logStatus.success();
          return cachedXContextProperties;
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://discord.com/api/v9/invites/${inviteId}?with_counts=true&with_expiration=true&with_permissions=true`,
          responseType: 'json',
          method: 'GET',
          headers: {
            authorization: this.#auth.auth,
            'x-super-properties': this.#auth.xSuperProperties
          }
        });
        if (result !== 'Success' || data?.status !== 200) {
          debug('获取加群参数失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        const guild = data.response?.guild?.id;
        if (!guild) {
          debug('加群参数中未找到ID', {
            inviteId: inviteId
          });
          logStatus.error('Error: Failed to get guild ID');
          return false;
        }
        const xContextProperties = {
          location: 'Accept Invite Page',
          location_guild_id: data.response?.guild?.id,
          location_channel_id: data.response?.channel?.id,
          location_channel_type: data.response?.channel?.type
        };
        debug('成功获取加群参数', xContextProperties);
        logStatus.success();
        this.#setXContextPropertiesCache(inviteId, window.btoa(JSON.stringify(xContextProperties)));
        this.#setCache(inviteId, guild);
        return window.btoa(JSON.stringify(xContextProperties));
      } catch (error) {
        throwError(error, 'Discord.getXContextProperties');
        return false;
      }
    }
    async #getGuild(inviteId) {
      try {
        debug('开始获取Discord服务器ID', {
          inviteId: inviteId
        });
        const logStatus = echoLog({
          type: 'gettingDiscordGuild',
          text: inviteId,
          before: '[Discord]'
        });
        const cachedGuild = this.#cache[inviteId];
        if (cachedGuild) {
          debug('从缓存中获取到服务器ID', {
            inviteId: inviteId,
            cachedGuild: cachedGuild
          });
          logStatus.success();
          return cachedGuild;
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://discord.com/api/v9/invites/${inviteId}`,
          responseType: 'json',
          method: 'GET',
          headers: {
            authorization: this.#auth.auth,
            'x-super-properties': this.#auth.xSuperProperties
          }
        });
        if (result !== 'Success' || data?.status !== 200) {
          debug('获取服务器信息失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        const guild = data.response?.guild?.id;
        if (!guild) {
          debug('服务器信息中未找到ID', {
            inviteId: inviteId
          });
          logStatus.error('Error: Failed to get guild ID');
          return false;
        }
        debug('成功获取服务器ID', {
          inviteId: inviteId,
          guild: guild
        });
        logStatus.success();
        this.#setCache(inviteId, guild);
        return guild;
      } catch (error) {
        throwError(error, 'Discord.getGuild');
        return false;
      }
    }
    async toggle({doTask: doTask = true, serverLinks: serverLinks = []}) {
      try {
        debug('开始处理Discord服务器任务', {
          doTask: doTask,
          serverLinksCount: serverLinks.length
        });
        if (!this.#initialized) {
          debug('Discord模块未初始化');
          echoLog({
            text: I18n('needInit'),
            before: '[Discord]'
          });
          return false;
        }
        if (doTask && !globalOptions.doTask.discord.servers || !doTask && !globalOptions.undoTask.discord.servers) {
          debug('根据全局选项跳过Discord服务器任务', {
            doTask: doTask
          });
          echoLog({
            type: 'globalOptionsSkip',
            text: 'discord.servers',
            before: '[Discord]'
          });
          return true;
        }
        const realServers = this.getRealParams('servers', serverLinks, doTask, (link => link.match(/invite\/(.+)/)?.[1]));
        debug('处理后的服务器列表', {
          count: realServers.length,
          servers: realServers
        });
        if (realServers.length === 0) {
          debug('没有需要处理的服务器');
          return true;
        }
        const results = [];
        for (const server of realServers) {
          results.push(doTask ? this.#joinServer(server) : this.#leaveServer(server));
          await delay(1e3);
        }
        return await Promise.allSettled(results).then((() => true));
      } catch (error) {
        throwError(error, 'Discord.toggleServers');
        return false;
      }
    }
    #setCache(inviteId, guild) {
      try {
        debug('设置Discord服务器缓存', {
          inviteId: inviteId,
          guild: guild
        });
        this.#cache[inviteId] = guild;
        GM_setValue('discordCache', this.#cache);
        debug('Discord服务器缓存设置成功');
      } catch (error) {
        debug('设置Discord服务器缓存失败', {
          error: error
        });
        throwError(error, 'Discord.setCache');
      }
    }
    #setXContextPropertiesCache(inviteId, xContextProperties) {
      try {
        debug('设置Discord加群参数缓存', {
          inviteId: inviteId,
          xContextProperties: xContextProperties
        });
        this.#xContextPropertiesCache[inviteId] = xContextProperties;
        GM_setValue('discordXContextPropertiesCache', this.#xContextPropertiesCache);
      } catch (error) {
        debug('设置Discord加群参数缓存失败', {
          error: error
        });
        throwError(error, 'Discord.setXContextPropertiesCache');
      }
    }
  }
  class Reddit extends Social {
    tasks;
    whiteList;
    #auth;
    #initialized=false;
    constructor() {
      super();
      const defaultTasksTemplate = {
        reddits: []
      };
      debug('初始化Reddit实例');
      this.tasks = defaultTasksTemplate;
      this.whiteList = {
        ...defaultTasksTemplate,
        ...GM_getValue('whiteList')?.reddit || {}
      };
    }
    async init() {
      try {
        debug('开始初始化Reddit模块');
        if (this.#initialized) {
          debug('Reddit模块已初始化');
          return true;
        }
        const isVerified = await this.#updateAuth();
        if (isVerified) {
          debug('Reddit授权验证成功');
          echoLog({
            before: '[Reddit]'
          }).success(I18n('initSuccess', 'Reddit'));
          this.#initialized = true;
          return true;
        }
        debug('Reddit初始化失败');
        echoLog({
          before: '[Reddit]'
        }).error(I18n('initFailed', 'Reddit'));
        return false;
      } catch (error) {
        debug('Reddit初始化发生错误', {
          error: error
        });
        throwError(error, 'Reddit.init');
        return false;
      }
    }
    async #useBeta() {
      try {
        debug('开始切换Reddit为新版');
        const logStatus = echoLog({
          text: I18n('changingRedditVersion'),
          before: '[Reddit]'
        });
        return await new Promise((resolve => {
          const newTab = GM_openInTab('https://www.reddit.com/', {
            active: true,
            insert: true,
            setParent: true
          });
          newTab.name = 'ATv4_redditAuth';
          newTab.onclose = async () => {
            debug('新版Reddit标签页已关闭');
            logStatus.success();
            resolve(await this.#updateAuth(true));
          };
        }));
      } catch (error) {
        debug('切换Reddit版本时发生错误', {
          error: error
        });
        throwError(error, 'Reddit.useBeta');
        return false;
      }
    }
    async #updateAuth(beta = false) {
      try {
        debug('开始更新Reddit授权', {
          beta: beta
        });
        const logStatus = echoLog({
          text: I18n('updatingAuth', 'Reddit'),
          before: '[Reddit]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://www.reddit.com/',
          method: 'GET',
          nochche: true,
          headers: {
            'Cache-Control': 'no-cache'
          }
        });
        if (result !== 'Success') {
          debug('获取Reddit页面失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.responseText.includes('www.reddit.com/login/')) {
          debug('需要登录Reddit');
          logStatus.error(`Error:${I18n('loginReddit')}`, true);
          return false;
        }
        if (data?.status !== 200) {
          debug('Reddit页面状态码错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        if (data.responseText.includes('redesign-beta-optin-btn') && !beta) {
          debug('检测到旧版Reddit,需要切换到新版');
          return await this.#useBeta();
        }
        const accessToken = data.responseText.match(/"accessToken":"(.*?)","expires":"(.*?)"/)?.[1];
        if (!accessToken) {
          debug('未找到Reddit访问令牌');
          logStatus.error('Error: Parameter "accessToken" not found!');
          return false;
        }
        debug('成功获取Reddit访问令牌');
        this.#auth = {
          token: accessToken
        };
        logStatus.success();
        return true;
      } catch (error) {
        debug('更新Reddit授权时发生错误', {
          error: error
        });
        throwError(error, 'Reddit.updateAuth');
        return false;
      }
    }
    async #toggleTask({name: name, doTask: doTask = true}) {
      try {
        debug('开始处理Reddit任务', {
          name: name,
          doTask: doTask
        });
        if (!doTask && this.whiteList.reddits.includes(name)) {
          debug('Reddit在白名单中,跳过取消订阅', {
            name: name
          });
          echoLog({
            type: 'whiteList',
            text: 'Reddit.undoTask',
            id: name,
            before: '[Reddit]'
          });
          return true;
        }
        let type = doTask ? 'joiningReddit' : 'leavingReddit';
        if (/^u_/.test(name)) {
          type = doTask ? 'followingRedditUser' : 'unfollowingRedditUser';
        }
        debug('任务类型', {
          type: type,
          name: name
        });
        const logStatus = echoLog({
          type: type,
          text: name,
          before: '[Reddit]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://oauth.reddit.com/api/subscribe?redditWebClient=desktop2x&app=desktop2x-client-production&raw_json=1&gilding_detail=1',
          method: 'POST',
          headers: {
            authorization: `Bearer ${this.#auth.token}`,
            'content-type': 'application/x-www-form-urlencoded'
          },
          data: $.param({
            action: doTask ? 'sub' : 'unsub',
            sr_name: name,
            api_type: 'json'
          })
        });
        if (result !== 'Success') {
          debug('Reddit任务请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('Reddit任务状态码错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('Reddit任务处理成功', {
          name: name,
          doTask: doTask
        });
        logStatus.success();
        if (doTask) {
          this.tasks.reddits = unique([ ...this.tasks.reddits, name ]);
        }
        return true;
      } catch (error) {
        debug('处理Reddit任务时发生错误', {
          error: error
        });
        throwError(error, 'Reddit.toggleTask');
        return false;
      }
    }
    async toggle({doTask: doTask = true, redditLinks: redditLinks = []}) {
      try {
        debug('开始处理Reddit链接任务', {
          doTask: doTask,
          redditLinksCount: redditLinks.length
        });
        if (!this.#initialized) {
          debug('Reddit模块未初始化');
          echoLog({
            text: I18n('needInit'),
            before: '[Reddit]'
          });
          return false;
        }
        if (doTask && !globalOptions.doTask.reddit.reddits || !doTask && !globalOptions.undoTask.reddit.reddits) {
          debug('根据全局选项跳过Reddit任务', {
            doTask: doTask
          });
          echoLog({
            type: 'globalOptionsSkip',
            text: 'reddit.reddits',
            before: '[Reddit]'
          });
          return true;
        }
        const realReddits = this.getRealParams('reddits', redditLinks, doTask, (link => {
          const name = link.match(/https?:\/\/www\.reddit\.com\/r\/([^/]*)/)?.[1];
          const userName = link.match(/https?:\/\/www\.reddit\.com\/user\/([^/]*)/)?.[1];
          if (userName) {
            return name || userName;
          }
          return name;
        }));
        debug('处理后的Reddit列表', {
          count: realReddits.length,
          reddits: realReddits
        });
        if (realReddits.length === 0) {
          debug('没有需要处理的Reddit链接');
          return true;
        }
        const prom = [];
        for (const name of realReddits) {
          prom.push(this.#toggleTask({
            name: name,
            doTask: doTask
          }));
          await delay(1e3);
        }
        return await Promise.all(prom).then((() => true));
      } catch (error) {
        debug('处理Reddit链接任务时发生错误', {
          error: error
        });
        throwError(error, 'Reddit.toggle');
        return false;
      }
    }
  }
  class Twitch extends Social {
    tasks;
    whiteList;
    #auth=GM_getValue('twitchAuth') || {};
    #cache=GM_getValue('twitchCache') || {};
    #initialized=false;
    #integrityToken;
    constructor() {
      super();
      const defaultTasksTemplate = {
        channels: []
      };
      debug('初始化Twitch实例');
      this.tasks = defaultTasksTemplate;
      this.whiteList = {
        ...defaultTasksTemplate,
        ...GM_getValue('whiteList')?.twitch || {}
      };
    }
    async init() {
      try {
        debug('开始初始化Twitch模块');
        if (this.#initialized) {
          debug('Twitch模块已初始化');
          return true;
        }
        if (!this.#auth.authToken || !this.#auth.clientId || !this.#auth.clientVersion || !this.#auth.deviceId || !this.#auth.clientSessionId) {
          if (await this.#updateAuth()) {
            this.#initialized = true;
            return true;
          }
          return false;
        }
        const isVerified = await this.#verifyAuth(true);
        if (isVerified) {
          debug('Twitch授权验证成功');
          echoLog({
            before: '[Twitch]'
          }).success(I18n('initSuccess', 'Twitch'));
          this.#initialized = true;
          return true;
        }
        GM_setValue('twitchAuth', null);
        if (await this.#updateAuth()) {
          debug('Twitch重新授权成功');
          echoLog({
            before: '[Twitch]'
          }).success(I18n('initSuccess', 'Twitch'));
          this.#initialized = true;
          return true;
        }
        debug('Twitch初始化失败');
        echoLog({
          before: '[Twitch]'
        }).error(I18n('initFailed', 'Twitch'));
        return false;
      } catch (error) {
        debug('Twitch初始化发生错误', {
          error: error
        });
        throwError(error, 'Twitch.init');
        return false;
      }
    }
    async #verifyAuth(isFirst) {
      try {
        debug('开始验证Twitch授权');
        const logStatus = echoLog({
          text: I18n('verifyingAuth', 'Twitch'),
          before: '[Twitch]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://gql.twitch.tv/gql',
          method: 'POST',
          dataType: 'json',
          headers: {
            Authorization: `OAuth ${this.#auth.authToken}`,
            'Client-Id': this.#auth.clientId
          },
          data: '[{"operationName":"FrontPageNew_User","variables":{"limit":1},"extensions":{"persistedQuery":{"version":1,' + '"sha256Hash":"64bd07a2cbaca80699d62636d966cf6395a5d14a1f0a14282067dcb28b13eb11"}}}]'
        });
        if (result !== 'Success') {
          debug('Twitch授权验证请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200 || !data.response?.[0]?.data?.currentUser) {
          debug('Twitch授权验证状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        await this.#integrity(isFirst);
        debug('Twitch授权验证成功');
        logStatus.success();
        return true;
      } catch (error) {
        debug('Twitch授权验证发生错误', {
          error: error
        });
        throwError(error, 'Twitch.verifyAuth');
        return false;
      }
    }
    async #integrity(isFirst = true, ct = '') {
      try {
        debug('开始检查Twitch完整性', {
          isFirst: isFirst,
          ct: ct
        });
        const logStatus = echoLog({
          text: I18n('checkingTwitchIntegrity'),
          before: '[Twitch]'
        });
        if (isFirst && (!this.#auth.authToken || !this.#auth.clientId || !this.#auth.clientVersion || !this.#auth.deviceId || !this.#auth.clientSessionId)) {
          return await this.#updateAuth(false);
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://gql.twitch.tv/integrity',
          method: 'POST',
          dataType: 'json',
          anonymous: true,
          headers: {
            Origin: 'https://www.twitch.tv',
            Referer: 'https://www.twitch.tv/',
            Authorization: `OAuth ${this.#auth.authToken}`,
            'Client-Id': this.#auth.clientId,
            'Client-Version': this.#auth.clientVersion,
            'X-Device-Id': this.#auth.deviceId,
            'Client-Session-Id': this.#auth.clientSessionId,
            'x-kpsdk-ct': ct
          }
        });
        if (result !== 'Success') {
          debug('Twitch完整性检查请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (!ct && data?.responseHeaders?.['x-kpsdk-ct']) {
          debug('需要重新检查Twitch完整性');
          return await this.#integrity(isFirst, data.responseHeaders['x-kpsdk-ct']);
        }
        if (data?.status !== 200 || !data.response?.token) {
          debug('Twitch完整性检查状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        this.#integrityToken = data.response.token;
        debug('Twitch完整性检查成功');
        logStatus.success();
        return true;
      } catch (error) {
        debug('Twitch完整性检查发生错误', {
          error: error
        });
        throwError(error, 'Twitch.integrity');
        return false;
      }
    }
    async #updateAuth(isFirst = true) {
      try {
        debug('开始更新Twitch授权', {
          isFirst: isFirst
        });
        const logStatus = echoLog({
          text: I18n('updatingAuth', 'Twitch'),
          before: '[Twitch]'
        });
        return await new Promise((resolve => {
          const newTab = GM_openInTab('https://www.twitch.tv/', {
            active: true,
            insert: true,
            setParent: true
          });
          newTab.name = 'ATv4_twitchAuth';
          newTab.onclose = async () => {
            const auth = GM_getValue('twitchAuth');
            if (auth) {
              debug('成功获取新的Twitch授权');
              this.#auth = auth;
              logStatus.success();
              resolve(await this.#verifyAuth(isFirst));
            } else {
              debug('获取Twitch授权失败');
              logStatus.error('Error: Update twitch auth failed!');
              resolve(false);
            }
          };
        }));
      } catch (error) {
        debug('更新Twitch授权时发生错误', {
          error: error
        });
        throwError(error, 'Twitch.updateAuth');
        return false;
      }
    }
    async #toggleChannel({name: name, doTask: doTask = true}) {
      try {
        debug('开始处理Twitch频道任务', {
          name: name,
          doTask: doTask
        });
        if (!doTask && this.whiteList.channels.includes(name)) {
          debug('Twitch频道在白名单中,跳过取消关注', {
            name: name
          });
          echoLog({
            type: 'whiteList',
            text: 'Twitch.unfollowChannel',
            id: name,
            before: '[Twitch]'
          });
          return true;
        }
        const channelId = await this.#getChannelId(name);
        if (!channelId) {
          return false;
        }
        const logStatus = echoLog({
          type: `${doTask ? '' : 'un'}followingTwitchChannel`,
          text: name,
          before: '[Twitch]'
        });
        const followData = `[{"operationName":"FollowButton_FollowUser","variables":{"input":{"disableNotifications":false,"targetID":"${channelId}` + '"}},"extensions":{"persistedQuery":{"version":1,"sha256Hash":"800e7346bdf7e5278a3c1d3f21b2b56e2639928f86815677a7126b093b2fdd08"}}}]';
        const unfollowData = `[{"operationName":"FollowButton_UnfollowUser","variables":{"input":{"targetID":"${channelId}"}},` + '"extensions":{"persistedQuery":{"version":1,"sha256Hash":"f7dae976ebf41c755ae2d758546bfd176b4eeb856656098bb40e0a672ca0d880"}}}]';
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://gql.twitch.tv/gql',
          method: 'POST',
          dataType: 'json',
          anonymous: true,
          headers: {
            Origin: 'https://www.twitch.tv',
            Referer: 'https://www.twitch.tv/',
            Authorization: `OAuth ${this.#auth.authToken}`,
            'Client-Id': this.#auth.clientId,
            'Client-Version': this.#auth.clientVersion,
            'X-Device-Id': this.#auth.deviceId,
            'Client-Session-Id': this.#auth.clientSessionId,
            'Client-Integrity': this.#integrityToken
          },
          data: doTask ? followData : unfollowData
        });
        if (result !== 'Success') {
          debug('Twitch频道操作请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200 || data.response?.[0] && data.response[0].errors) {
          debug('Twitch频道操作状态错误', {
            status: data?.status,
            statusText: data?.statusText,
            errors: data?.response?.[0].errors
          });
          logStatus.error(`Error:${data?.response?.[0].errors?.[0]?.message || `${data?.statusText}(${data?.status})`}`);
          return false;
        }
        debug('Twitch频道操作成功', {
          name: name,
          doTask: doTask
        });
        logStatus.success();
        if (doTask) {
          this.tasks.channels = unique([ ...this.tasks.channels, name ]);
        }
        return true;
      } catch (error) {
        debug('处理Twitch频道任务时发生错误', {
          error: error
        });
        throwError(error, 'Twitch.toggleChannel');
        return false;
      }
    }
    async #getChannelId(name) {
      try {
        debug('开始获取Twitch频道ID', {
          name: name
        });
        const logStatus = echoLog({
          type: 'gettingTwitchChannelId',
          text: name,
          before: '[Twitch]'
        });
        const cachedChannelId = this.#cache[name];
        if (cachedChannelId) {
          debug('从缓存获取到Twitch频道ID', {
            name: name,
            id: cachedChannelId
          });
          logStatus.success();
          return cachedChannelId;
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://gql.twitch.tv/gql',
          method: 'POST',
          headers: {
            Authorization: `OAuth ${this.#auth.authToken}`,
            'Client-Id': this.#auth.clientId
          },
          responseType: 'json',
          data: `[{"operationName":"ActiveWatchParty","variables":{"channelLogin":"${name}"},` + '"extensions":{"persistedQuery":{"version":1,"sha256Hash":"4a8156c97b19e3a36e081cf6d6ddb5dbf9f9b02ae60e4d2ff26ed70aebc80a30"}}}]'
        });
        if (result !== 'Success') {
          debug('获取Twitch频道ID请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('获取Twitch频道ID状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        const newChannelId = data.response?.[0]?.data?.user?.id;
        if (!newChannelId) {
          debug('未找到Twitch频道ID', {
            name: name
          });
          logStatus.error(`Error:${data?.statusText || 'Unknown'}(${data?.status || 'Unknown'})`);
          return false;
        }
        debug('成功获取Twitch频道ID', {
          name: name,
          id: newChannelId
        });
        this.#setCache(name, newChannelId);
        logStatus.success();
        return newChannelId;
      } catch (error) {
        debug('获取Twitch频道ID时发生错误', {
          error: error
        });
        throwError(error, 'Twitch.getChannelId');
        return false;
      }
    }
    async toggle({doTask: doTask = true, channelLinks: channelLinks = []}) {
      try {
        debug('开始处理Twitch链接任务', {
          doTask: doTask,
          channelLinksCount: channelLinks.length
        });
        if (!this.#initialized) {
          debug('Twitch模块未初始化');
          echoLog({
            text: I18n('needInit'),
            before: '[Twitch]'
          });
          return false;
        }
        const prom = [];
        if (doTask && !globalOptions.doTask.twitch.channels || !doTask && !globalOptions.undoTask.twitch.channels) {
          debug('根据全局选项跳过Twitch任务', {
            doTask: doTask
          });
          echoLog({
            type: 'globalOptionsSkip',
            text: 'twitch.channels',
            before: '[Twitch]'
          });
        } else {
          const realChannels = this.getRealParams('channels', channelLinks, doTask, (link => link.match(/https:\/\/(www\.)?twitch\.tv\/(.+)/)?.[2]));
          debug('处理后的Twitch频道列表', {
            count: realChannels.length,
            channels: realChannels
          });
          if (realChannels.length > 0) {
            for (const channel of realChannels) {
              prom.push(this.#toggleChannel({
                name: channel,
                doTask: doTask
              }));
              await delay(1e3);
            }
          }
        }
        return Promise.all(prom).then((() => true));
      } catch (error) {
        debug('处理Twitch链接任务时发生错误', {
          error: error
        });
        throwError(error, 'Twitch.toggle');
        return false;
      }
    }
    #setCache(name, id) {
      try {
        debug('设置Twitch频道ID缓存', {
          name: name,
          id: id
        });
        this.#cache[name] = id;
        GM_setValue('twitchCache', this.#cache);
      } catch (error) {
        debug('设置Twitch频道ID缓存时发生错误', {
          error: error
        });
        throwError(error, 'Twitch.setCache');
      }
    }
  }
  const encodeSha256 = async data => {
    const encoder = new TextEncoder;
    const dataBuffer = encoder.encode(data);
    const hashBuffer = await crypto.subtle.digest('SHA-256', dataBuffer);
    return Array.from(new Uint8Array(hashBuffer));
  };
  const encodeBase64 = data => {
    let binary = '';
    const bytes = new Uint8Array(data);
    const len = bytes.byteLength;
    for (let i = 0; i < len; i++) {
      binary += String.fromCharCode(bytes[i]);
    }
    return btoa(binary).replace(/=/g, '');
  };
  const decodeBase64 = data => {
    const binaryString = atob(data);
    const len = binaryString.length;
    const bytes = new Uint8Array(len);
    for (let i = 0; i < len; i++) {
      bytes[i] = binaryString.charCodeAt(i);
    }
    return Array.from(bytes);
  };
  const generateTransactionId = async (method, path, key, animationKey) => {
    const DEFAULT_KEYWORD = 'obfiowerehiring';
    const ADDITIONAL_RANDOM_NUMBER = 3;
    const timeNow = Math.floor((Date.now() - 1682924400 * 1e3) / 1e3);
    const timeNowBytes = [ timeNow & 255, timeNow >> 8 & 255, timeNow >> 16 & 255, timeNow >> 24 & 255 ];
    const data = `${method}!${path}!${timeNow}${DEFAULT_KEYWORD}${animationKey}`;
    const hashBytes = await encodeSha256(data);
    const keyBytes = decodeBase64(key);
    const randomNum = Math.floor(Math.random() * 256);
    const bytesArr = [ ...keyBytes, ...timeNowBytes, ...hashBytes.slice(0, 16), ADDITIONAL_RANDOM_NUMBER ];
    const out = new Uint8Array(bytesArr.length + 1);
    out[0] = randomNum;
    bytesArr.forEach(((item, index) => {
      out[index + 1] = item ^ randomNum;
    }));
    return encodeBase64(out);
  };
  const url = 'https://raw.githubusercontent.com/fa0311/x-client-transaction-id-pair-dict/refs/heads/main/pair.json';
  const getTID = async () => {
    const res = await fetch(url);
    const json = await res.json();
    return async (method, path) => {
      const randomPair = json[Math.floor(Math.random() * json.length)];
      const {animationKey: animationKey, verification: verification} = randomPair;
      const tid = await generateTransactionId(method, path, verification, animationKey);
      return tid;
    };
  };
  const parseResponseHeaders = headerStr => {
    const headers = {};
    if (!headerStr) {
      return headers;
    }
    headerStr.split('\r\n').forEach((line => {
      if (line) {
        const parts = line.split(':');
        const key = parts.shift()?.trim();
        const value = parts.join(':').trim();
        if (key) {
          if (key.toLowerCase() === 'set-cookie') {
            if (headers[key]) {
              if (Array.isArray(headers[key])) {
                headers[key].push(value);
              } else {
                headers[key] = [ headers[key], value ];
              }
            } else {
              headers[key] = value;
            }
          } else {
            headers[key] = value;
          }
        }
      }
    }));
    return headers;
  };
  const axiosGM = function(config) {
    const finalConfig = {
      ...axiosGM.defaults,
      ...config
    };
    const retries = finalConfig.retry ?? 0;
    const retryDelay = finalConfig.retryDelay ?? 0;
    const requestAttempt = attempt => new Promise(((resolve, reject) => {
      GM_xmlhttpRequest({
        method: finalConfig.method ? finalConfig.method.toUpperCase() : 'GET',
        url: finalConfig.url,
        headers: finalConfig.headers,
        data: finalConfig.data,
        responseType: finalConfig.responseType || 'json',
        timeout: finalConfig.timeout,
        fetch: finalConfig.fetch ?? true,
        onload(response) {
          const axiosResponse = {
            data: response.response || response.responseText,
            status: response.status,
            statusText: response.statusText,
            headers: parseResponseHeaders(response.responseHeaders),
            config: finalConfig,
            request: response
          };
          resolve(axiosResponse);
        },
        onerror(error) {
          if (attempt < retries) {
            setTimeout((() => {
              requestAttempt(attempt + 1).then(resolve).catch(reject);
            }), retryDelay);
          } else {
            reject(error);
          }
        },
        ontimeout() {
          if (attempt < retries) {
            setTimeout((() => {
              requestAttempt(attempt + 1).then(resolve).catch(reject);
            }), retryDelay);
          } else {
            reject('Error: timeout');
          }
        }
      });
    }));
    return requestAttempt(0);
  };
  axiosGM.defaults = {};
  axiosGM.get = function(url, config = {}) {
    return axiosGM({
      ...config,
      url: url,
      method: 'GET'
    });
  };
  axiosGM.post = function(url, data, config = {}) {
    return axiosGM({
      ...config,
      url: url,
      data: data,
      method: 'POST'
    });
  };
  axiosGM.head = function(url, config = {}) {
    return axiosGM({
      ...config,
      url: url,
      method: 'HEAD'
    });
  };
  axiosGM.create = function(instanceDefaults = {}) {
    const instance = config => {
      const mergedConfig = {
        ...axiosGM.defaults,
        ...instanceDefaults,
        ...config
      };
      return axiosGM(mergedConfig);
    };
    instance.defaults = {
      ...axiosGM.defaults,
      ...instanceDefaults
    };
    instance.get = function(url, config = {}) {
      return instance({
        ...config,
        url: url,
        method: 'GET'
      });
    };
    instance.post = function(url, data, config = {}) {
      return instance({
        ...config,
        url: url,
        data: data,
        method: 'POST'
      });
    };
    instance.head = function(url, config = {}) {
      return instance({
        ...config,
        url: url,
        method: 'HEAD'
      });
    };
    instance.create = axiosGM.create;
    return instance;
  };
  const getFwdForSdkUrl = async () => {
    const rawHtml = await axiosGM({
      url: 'https://x.com',
      method: 'GET'
    });
    return [ ...rawHtml.data.matchAll(/"(loader\.FwdForSdk)":"([^"]+?)"/g) ];
  };
  const fwdForSdkExpoter = async url => {
    const {data: data} = await axiosGM.get(url);
    const regex = /Uint8Array\(\w\)\.set\(\[(.*?)\]\)/;
    if (!regex.test(data)) {
      return false;
    }
    const json = `[${data.match(regex)?.[1]}]`;
    const obj = JSON.parse(json);
    return new Uint8Array(obj);
  };
  const getWasmData = async () => {
    const fwdForSdkUrl = await getFwdForSdkUrl();
    for (const url of fwdForSdkUrl) {
      const sdkData = await fwdForSdkExpoter(`https://abs.twimg.com/responsive-web/client-web/${url[1]}.${url[2]}a.js`);
      if (sdkData) {
        return sdkData;
      }
    }
    return false;
  };
  const getFwdForSdk = async () => {
    debug('开始获取 XFwdForSdk');
    const wasmData = await getWasmData();
    debug('获取 wasmData 成功', {
      wasmData: wasmData
    });
    const go = new Go;
    const wasmModule = await WebAssembly.instantiate(wasmData, {
      ...go.importObject,
      env: {
        ...go.importObject.env,
        memory: new WebAssembly.Memory({
          initial: 10
        }),
        table: new WebAssembly.Table({
          initial: 0,
          element: 'anyfunc'
        })
      }
    });
    debug('初始化 wasmModule 成功');
    go.run(wasmModule.instance);
    debug('运行 wasmModule 成功');
    const {str: str, expiryTimeMillis: expiryTimeMillis} = await globalThis.getForwardedForStr();
    debug('获取 XFwdForSdk 成功', {
      str: str,
      expiryTimeMillis: expiryTimeMillis
    });
    return {
      str: str,
      expiryTimeMillis: parseInt(expiryTimeMillis, 10)
    };
  };
  const generateSecCHUA = () => {
    if (navigator.userAgentData && navigator.userAgentData.brands) {
      return navigator.userAgentData.brands.map((brand => `"${brand.brand}";v="${brand.version}"`)).join(', ');
    }
    return '"Google Chrome";v="125", "Chromium";v="125", "Not-A.Brand";v="99"';
  };
  class Twitter extends Social {
    tasks;
    whiteList;
    #verifyId=globalOptions.other.twitterVerifyId;
    #auth=GM_getValue('twitterAuth') || {};
    #cache=GM_getValue('twitterCache') || {};
    #initialized=false;
    #getTID;
    #FwdForSdk;
    #headers={};
    constructor() {
      super();
      const defaultTasksTemplate = {
        users: [],
        retweets: [],
        likes: []
      };
      debug('初始化Twitter实例');
      this.tasks = defaultTasksTemplate;
      this.whiteList = {
        ...defaultTasksTemplate,
        ...GM_getValue('whiteList')?.twitter || {}
      };
      this.#headers = {
        authorization: 'Bearer AAAAAAAAAAAAAAAAAAAAANRILgAAAAAAnNwIzUejRCOuH5E6I8xnZz4puTs%3D1Zv7ttfk8LF81IUq16cHjhLTvJu4FA33AGWWjCpTnA',
        'X-Twitter-Auth-Type': 'OAuth2Session',
        'X-Twitter-Active-User': 'yes',
        'sec-fetch-dest': 'empty',
        'sec-fetch-mode': 'cors',
        'sec-fetch-site': 'same-origin',
        'sec-ch-ua-platform': '"Windows"',
        'sec-ch-ua-mobile': '?0',
        'sec-ch-ua': generateSecCHUA()
      };
    }
    async init() {
      try {
        debug('开始初始化Twitter模块');
        if (this.#initialized) {
          debug('Twitter模块已初始化');
          return true;
        }
        debug('获取Twitter授权信息');
        if (!await this.#updateAuth()) {
          return false;
        }
        debug('创建Twitter会话和SDK');
        this.#getTID = await getTID();
        this.#FwdForSdk = await getFwdForSdk();
        const isVerified = await this.#verifyAuth();
        if (isVerified) {
          debug('Twitter授权验证成功');
          echoLog({
            before: '[Twitter]'
          }).success(I18n('initSuccess', 'Twitter'));
          this.#initialized = true;
          return true;
        }
        debug('Twitter授权失效,尝试重新获取');
        GM_setValue('twitterAuth', null);
        if (await this.#updateAuth()) {
          debug('Twitter重新授权成功');
          echoLog({
            before: '[Twitter]'
          }).success(I18n('initSuccess', 'Twitter'));
          this.#initialized = true;
          return true;
        }
        debug('Twitter初始化失败');
        echoLog({
          before: '[Twitter]'
        }).error(I18n('initFailed', 'Twitter'));
        return false;
      } catch (error) {
        debug('Twitter初始化发生错误', {
          error: error
        });
        throwError(error, 'Twitter.init');
        return false;
      }
    }
    async #verifyAuth() {
      try {
        debug('开始验证Twitter授权');
        return await this.#toggleUser({
          name: 'verify',
          doTask: true,
          verify: true
        });
      } catch (error) {
        debug('Twitter授权验证发生错误', {
          error: error
        });
        throwError(error, 'Twitter.verifyAuth');
        return false;
      }
    }
    async #updateAuth() {
      try {
        debug('开始更新Twitter授权');
        const logStatus = echoLog({
          text: I18n('updatingAuth', 'Twitter'),
          before: '[Twitter]'
        });
        return await new Promise((resolve => {
          GM_cookie.list({
            url: 'https://x.com/settings/account'
          }, (async (cookies, error) => {
            if (!error) {
              const ct0 = cookies.find((cookie => cookie.name === 'ct0'))?.value;
              const isLogin = cookies.find((cookie => cookie.name === 'twid'))?.value;
              if (isLogin && ct0) {
                debug('成功获取Twitter授权信息');
                GM_setValue('twitterAuth', {
                  ct0: ct0
                });
                this.#auth = {
                  ct0: ct0
                };
                this.#headers['x-csrf-token'] = ct0;
                this.#headers['x-twitter-client-language'] = cookies.find((cookie => cookie.name === 'lang'))?.value || 'en';
                logStatus.success();
                resolve(true);
              } else {
                debug('获取Twitter授权失败:未登录');
                logStatus.error(I18n('needLogin'));
                resolve(false);
              }
            } else {
              debug('获取Twitter授权失败', {
                error: error
              });
              logStatus.error('Error: Update twitter auth failed!');
              resolve(false);
            }
          }));
        }));
      } catch (error) {
        debug('更新Twitter授权时发生错误', {
          error: error
        });
        throwError(error, 'Twitter.updateToken');
        return false;
      }
    }
    async #toggleUser({name: name, doTask: doTask = true, verify: verify = false, retry: retry = false}) {
      try {
        debug('开始处理Twitter用户任务', {
          name: name,
          doTask: doTask,
          verify: verify,
          retry: retry
        });
        if (!doTask && !verify && this.whiteList.users.includes(name)) {
          debug('Twitter用户在白名单中,跳过取消关注', {
            name: name
          });
          echoLog({
            type: 'whiteList',
            text: 'Twitter.unfollowUser',
            id: name,
            before: '[Twitter]'
          });
          return true;
        }
        const userId = verify ? this.#verifyId : await this.userName2id(name);
        if (!userId) {
          return false;
        }
        const logStatus = verify ? echoLog({
          text: I18n('verifyingAuth', 'Twitter'),
          before: '[Twitter]'
        }) : echoLog({
          type: `${doTask ? '' : 'un'}followingTwitterUser`,
          text: name,
          before: '[Twitter]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://x.com/i/api/1.1/friendships/${doTask ? 'create' : 'destroy'}.json`,
          method: 'POST',
          headers: {
            ...this.#headers,
            'Content-Type': 'application/x-www-form-urlencoded',
            'x-client-transaction-id': await this.#getTID('POST', `/i/api/1.1/friendships/${doTask ? 'create' : 'destroy'}.json`),
            'x-xp-forwarded-for': this.#FwdForSdk.str
          },
          responseType: 'json',
          data: $.param({
            include_profile_interstitial_type: 1,
            include_blocking: 1,
            include_blocked_by: 1,
            include_followed_by: 1,
            include_want_retweets: 1,
            include_mute_edge: 1,
            include_can_dm: 1,
            include_can_media_tag: 1,
            skip_status: 1,
            id: userId
          })
        });
        if (result !== 'Success') {
          debug('Twitter用户操作请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status === 200) {
          debug('Twitter用户操作成功', {
            name: name,
            doTask: doTask
          });
          logStatus.success();
          if (doTask && !verify) {
            this.tasks.users = unique([ ...this.tasks.users, name ]);
          }
          return true;
        }
        if (verify && data?.status === 403) {
          if (data.response?.errors?.[0]?.code === 158) {
            debug('Twitter授权验证成功(已关注)');
            logStatus.success();
            return true;
          }
          if (data.response?.errors?.[0]?.code === 353 && !retry && data.responseHeaders?.['set-cookie']) {
            const newCt0 = data.responseHeaders['set-cookie']?.find((cookie => cookie.includes('ct0=')))?.split(';')?.at(0)?.split('=')?.at(-1);
            if (newCt0) {
              debug('获取到新的Twitter授权Token,重试操作');
              this.#auth.ct0 = newCt0;
              GM_setValue('twitterAuth', this.#auth);
              logStatus.warning(I18n('retry'));
              return this.#toggleUser({
                name: name,
                doTask: doTask,
                verify: verify,
                retry: true
              });
            }
          }
        }
        debug('Twitter用户操作失败', {
          status: data?.status,
          statusText: data?.statusText
        });
        logStatus.error(`Error:${data?.statusText}(${data?.status})`);
        return false;
      } catch (error) {
        debug('处理Twitter用户任务时发生错误', {
          error: error
        });
        throwError(error, 'Twitter.toggleUser');
        return false;
      }
    }
    async userName2id(name) {
      try {
        debug('开始获取Twitter用户ID', {
          name: name
        });
        const logStatus = echoLog({
          type: 'gettingTwitterUserId',
          text: name,
          before: '[Twitter]'
        });
        const cachedUserId = this.#cache[name];
        if (cachedUserId) {
          debug('从缓存获取到Twitter用户ID', {
            name: name,
            id: cachedUserId
          });
          logStatus.success();
          return cachedUserId;
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://x.com/i/api/graphql/jUKA--0QkqGIFhmfRZdWrQ/UserByScreenName' + `?variables=%7B%22screen_name%22%3A%22${name}%22%7D&features=%7B%22responsive_web_grok_bio_auto_translation_is_enabled%22%3Afalse%2C%22hidden_profile_subscriptions_enabled%22%3Atrue%2C%22payments_enabled%22%3Afalse%2C%22profile_label_improvements_pcf_label_in_post_enabled%22%3Atrue%2C%22rweb_tipjar_consumption_enabled%22%3Atrue%2C%22verified_phone_label_enabled%22%3Afalse%2C%22subscriptions_verification_info_is_identity_verified_enabled%22%3Atrue%2C%22subscriptions_verification_info_verified_since_enabled%22%3Atrue%2C%22highlights_tweets_tab_ui_enabled%22%3Atrue%2C%22responsive_web_twitter_article_notes_tab_enabled%22%3Atrue%2C%22subscriptions_feature_can_gift_premium%22%3Atrue%2C%22creator_subscriptions_tweet_preview_api_enabled%22%3Atrue%2C%22responsive_web_graphql_skip_user_profile_image_extensions_enabled%22%3Afalse%2C%22responsive_web_graphql_timeline_navigation_enabled%22%3Atrue%7D&fieldToggles=%7B%22withAuxiliaryUserLabels%22%3Atrue%7D`,
          method: 'GET',
          headers: {
            ...this.#headers,
            'content-type': 'application/json',
            referer: `https://x.com/${name}`,
            'x-client-transaction-id': await this.#getTID('GET', '/i/api/graphql/jUKA--0QkqGIFhmfRZdWrQ/UserByScreenName' + `?variables=%7B%22screen_name%22%3A%22${name}%22%7D&features=%7B%22responsive_web_grok_bio_auto_translation_is_enabled%22%3Afalse%2C%22hidden_profile_subscriptions_enabled%22%3Atrue%2C%22payments_enabled%22%3Afalse%2C%22profile_label_improvements_pcf_label_in_post_enabled%22%3Atrue%2C%22rweb_tipjar_consumption_enabled%22%3Atrue%2C%22verified_phone_label_enabled%22%3Afalse%2C%22subscriptions_verification_info_is_identity_verified_enabled%22%3Atrue%2C%22subscriptions_verification_info_verified_since_enabled%22%3Atrue%2C%22highlights_tweets_tab_ui_enabled%22%3Atrue%2C%22responsive_web_twitter_article_notes_tab_enabled%22%3Atrue%2C%22subscriptions_feature_can_gift_premium%22%3Atrue%2C%22creator_subscriptions_tweet_preview_api_enabled%22%3Atrue%2C%22responsive_web_graphql_skip_user_profile_image_extensions_enabled%22%3Afalse%2C%22responsive_web_graphql_timeline_navigation_enabled%22%3Atrue%7D&fieldToggles=%7B%22withAuxiliaryUserLabels%22%3Atrue%7D`),
            'x-xp-forwarded-for': this.#FwdForSdk.str
          },
          responseType: 'json'
        });
        if (result !== 'Success') {
          debug('获取Twitter用户ID请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('获取Twitter用户ID状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        let response = data.response || (typeof data.responseText === 'object' ? data.responseText : null);
        if (!response) {
          try {
            response = JSON.parse(data.responseText);
          } catch (error) {
            response = null;
          }
        }
        const fetchedUserId = response?.data?.user?.result?.rest_id;
        if (!fetchedUserId) {
          debug('未找到Twitter用户ID', {
            name: name
          });
          logStatus.error(`Error:${data.statusText}(${data.status})`);
          return false;
        }
        debug('成功获取Twitter用户ID', {
          name: name,
          id: fetchedUserId
        });
        this.#setCache(name, fetchedUserId);
        logStatus.success();
        return fetchedUserId;
      } catch (error) {
        debug('获取Twitter用户ID时发生错误', {
          error: error
        });
        throwError(error, 'Twitter.getUserId');
        return false;
      }
    }
    async #toggleRetweet({retweetId: retweetId, doTask: doTask = true, retry: retry = false}) {
      try {
        debug('开始处理Twitter转推任务', {
          retweetId: retweetId,
          doTask: doTask,
          retry: retry
        });
        if (!doTask && this.whiteList.retweets.includes(retweetId)) {
          debug('Twitter转推在白名单中,跳过取消', {
            retweetId: retweetId
          });
          echoLog({
            type: 'whiteList',
            text: 'Twitter.unretweet',
            id: retweetId,
            before: '[Twitter]'
          });
          return true;
        }
        const logStatus = echoLog({
          type: `${doTask ? '' : 'un'}retweetting`,
          text: retweetId,
          before: '[Twitter]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://x.com/i/api/graphql/${doTask ? 'ojPdsZsimiJrUGLR1sjUtA/CreateRetweet' : 'iQtK4dl5hBmXewYZuEOKVw/DeleteRetweet'}`,
          method: 'POST',
          headers: {
            ...this.#headers,
            'Content-Type': 'application/json',
            origin: 'https://x.com',
            referer: 'https://x.com/home',
            'x-client-transaction-id': await this.#getTID('POST', `/i/api/graphql/${doTask ? 'ojPdsZsimiJrUGLR1sjUtA/CreateRetweet' : 'iQtK4dl5hBmXewYZuEOKVw/DeleteRetweet'}`),
            'x-xp-forwarded-for': this.#FwdForSdk.str
          },
          data: `{"variables":{"${doTask ? '' : 'source_'}tweet_id":"${retweetId}","dark_request":false},"queryId":"${doTask ? 'ojPdsZsimiJrUGLR1sjUtA' : 'iQtK4dl5hBmXewYZuEOKVw'}"}`,
          responseType: 'json'
        });
        if (result !== 'Success') {
          debug('Twitter转推操作请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status === 403 && data.response?.errors?.[0]?.code === 353 && !retry && data.responseHeaders?.['set-cookie']) {
          const newCt0 = data.responseHeaders['set-cookie']?.find((cookie => cookie.includes('ct0=')))?.split(';')?.at(0)?.split('=')?.at(-1);
          if (newCt0) {
            debug('获取到新的Twitter授权Token,重试操作');
            this.#auth.ct0 = newCt0;
            GM_setValue('twitterAuth', this.#auth);
            logStatus.warning(I18n('retry'));
            return this.#toggleRetweet({
              retweetId: retweetId,
              doTask: doTask,
              retry: true
            });
          }
        }
        if (data?.status !== 200 && !(data?.status === 403 && data.response?.errors?.[0]?.code === 327)) {
          debug('Twitter转推操作状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        if (data.response?.errors && data.response?.errors?.[0]?.code !== 327) {
          debug('Twitter转推操作出错', {
            error: data.response?.errors?.[0]?.message
          });
          logStatus.error(`Error:${data.response?.errors?.[0]?.message}`);
          return false;
        }
        debug('Twitter转推操作成功', {
          retweetId: retweetId,
          doTask: doTask
        });
        logStatus.success();
        if (doTask) {
          this.tasks.retweets = unique([ ...this.tasks.retweets, retweetId ]);
        }
        return true;
      } catch (error) {
        debug('处理Twitter转推任务时发生错误', {
          error: error
        });
        throwError(error, 'Twitter.toggleRetweet');
        return false;
      }
    }
    async toggle({doTask: doTask = true, userLinks: userLinks = [], retweetLinks: retweetLinks = []}) {
      try {
        debug('开始处理Twitter链接任务', {
          doTask: doTask,
          userLinksCount: userLinks.length,
          retweetLinksCount: retweetLinks.length
        });
        if (!this.#initialized) {
          debug('Twitter模块未初始化');
          echoLog({
            text: I18n('needInit'),
            before: '[Twitter]'
          });
          return false;
        }
        if (doTask && !globalOptions.doTask.twitter.users || !doTask && !globalOptions.undoTask.twitter.users) {
          debug('根据全局选项跳过Twitter用户任务', {
            doTask: doTask
          });
          echoLog({
            type: 'globalOptionsSkip',
            text: 'twitter.users',
            before: '[Twitter]'
          });
        } else {
          const realUsers = this.getRealParams('users', userLinks, doTask, (link => link.match(/https:\/\/x\.com\/([^/]+)/)?.[1] || link.match(/https:\/\/twitter\.com\/([^/]+)/)?.[1]));
          debug('处理后的Twitter用户列表', {
            count: realUsers.length,
            users: realUsers
          });
          if (realUsers.length > 0) {
            for (const user of realUsers) {
              if (Date.now() > this.#FwdForSdk.expiryTimeMillis) {
                debug('Twitter SDK过期,重新获取', {
                  expiryTimeMillis: this.#FwdForSdk.expiryTimeMillis
                });
                this.#FwdForSdk = await getFwdForSdk();
              }
              await this.#toggleUser({
                name: user,
                doTask: doTask
              });
              await delay(1e3);
            }
          }
        }
        if (doTask && !globalOptions.doTask.twitter.retweets || !doTask && !globalOptions.undoTask.twitter.retweets) {
          debug('根据全局选项跳过Twitter转推任务', {
            doTask: doTask
          });
          echoLog({
            type: 'globalOptionsSkip',
            text: 'twitter.retweets',
            before: '[Twitter]'
          });
        } else {
          const realRetweets = this.getRealParams('retweets', retweetLinks, doTask, (link => link.match(/https:\/\/x\.com\/.*?\/status\/([\d]+)/)?.[1] || link.match(/https:\/\/twitter\.com\/.*?\/status\/([\d]+)/)?.[1]));
          debug('处理后的Twitter转推列表', {
            count: realRetweets.length,
            retweets: realRetweets
          });
          if (realRetweets.length > 0) {
            for (const retweet of realRetweets) {
              if (Date.now() > this.#FwdForSdk.expiryTimeMillis) {
                debug('Twitter SDK过期,重新获取');
                this.#FwdForSdk = await getFwdForSdk();
              }
              await this.#toggleRetweet({
                retweetId: retweet,
                doTask: doTask
              });
              await delay(1e3);
            }
          }
        }
        return true;
      } catch (error) {
        debug('处理Twitter链接任务时发生错误', {
          error: error
        });
        throwError(error, 'Twitter.toggle');
        return false;
      }
    }
    #setCache(name, id) {
      try {
        debug('设置Twitter用户ID缓存', {
          name: name,
          id: id
        });
        this.#cache[name] = id;
        GM_setValue('twitterCache', this.#cache);
      } catch (error) {
        debug('设置Twitter用户ID缓存时发生错误', {
          error: error
        });
        throwError(error, 'Twitter.setCache');
      }
    }
  }
  class Vk extends Social {
    tasks;
    whiteList;
    #username='';
    #cache=GM_getValue('vkCache') || {};
    #initialized=false;
    constructor() {
      super();
      const defaultTasksTemplate = {
        names: []
      };
      debug('初始化Vk实例');
      this.tasks = defaultTasksTemplate;
      this.whiteList = {
        ...defaultTasksTemplate,
        ...GM_getValue('whiteList')?.vk || {}
      };
    }
    async init() {
      try {
        debug('开始初始化Vk模块');
        if (this.#initialized) {
          debug('Vk模块已初始化');
          return true;
        }
        const isVerified = await this.#verifyAuth();
        if (isVerified) {
          debug('Vk授权验证成功');
          echoLog({
            before: '[Vk]'
          }).success(I18n('initSuccess', 'Vk'));
          this.#initialized = true;
          return true;
        }
        debug('Vk初始化失败');
        echoLog({
          before: '[Vk]'
        }).error(I18n('initFailed', 'Vk'));
        return false;
      } catch (error) {
        debug('Vk初始化发生错误', {
          error: error
        });
        throwError(error, 'Vk.init');
        return false;
      }
    }
    async #verifyAuth() {
      try {
        debug('开始验证Vk授权');
        const logStatus = echoLog({
          text: I18n('verifyAuth', 'Vk'),
          before: '[Vk]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://vk.com/im',
          method: 'GET'
        });
        if (result !== 'Success') {
          debug('Vk授权验证请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.finalUrl.includes('vk.com/login')) {
          debug('Vk授权验证失败:需要登录');
          logStatus.error(`Error:${I18n('loginVk')}`, true);
          return false;
        }
        if (data?.status !== 200) {
          debug('Vk授权验证状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        this.#username = data.responseText.match(/TopNavBtn__profileLink" href="\/(.*?)"/)?.[1] || '';
        debug('Vk授权验证成功');
        logStatus.success();
        return true;
      } catch (error) {
        debug('Vk授权验证发生错误', {
          error: error
        });
        throwError(error, 'Vk.verifyAuth');
        return false;
      }
    }
    async #toggleGroup(name, dataParam, doTask = true) {
      try {
        debug('开始处理Vk群组任务', {
          name: name,
          doTask: doTask
        });
        const logStatus = echoLog({
          type: doTask ? 'joiningVkGroup' : 'leavingVkGroup',
          text: name,
          before: '[Vk]'
        });
        if (dataParam.groupAct === 'enter' && !doTask || dataParam.groupAct === 'leave' && doTask) {
          debug('Vk群组操作已完成,跳过', {
            name: name,
            doTask: doTask
          });
          logStatus.success();
          return true;
        }
        const reqData = {
          act: doTask ? 'enter' : 'leave',
          al: 1,
          gid: dataParam.groupId,
          hash: dataParam.groupHash
        };
        if (doTask) {
          reqData.context = '_';
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://vk.com/al_groups.php',
          method: 'POST',
          headers: {
            origin: 'https://vk.com',
            referer: `https://vk.com/${name}`,
            'content-type': 'application/x-www-form-urlencoded'
          },
          data: $.param(reqData)
        });
        if (result !== 'Success') {
          debug('Vk群组操作请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('Vk群组操作状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('Vk群组操作成功', {
          name: name,
          doTask: doTask
        });
        logStatus.success();
        if (doTask) {
          this.tasks.names = unique([ ...this.tasks.names, name ]);
        }
        return true;
      } catch (error) {
        debug('处理Vk群组任务时发生错误', {
          error: error
        });
        throwError(error, 'Vk.toggleGroup');
        return false;
      }
    }
    async #togglePublic(name, dataParam, doTask = true) {
      try {
        debug('开始处理Vk公共页面任务', {
          name: name,
          doTask: doTask
        });
        const logStatus = echoLog({
          type: doTask ? 'joiningVkPublic' : 'leavingVkPublic',
          text: name,
          before: '[Vk]'
        });
        if (dataParam.publicJoined && doTask || !dataParam.publicJoined && !doTask) {
          debug('Vk公共页面操作已完成,跳过', {
            name: name,
            doTask: doTask
          });
          logStatus.success();
          return true;
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://vk.com/al_public.php',
          method: 'POST',
          headers: {
            origin: 'https://vk.com',
            referer: `https://vk.com/${name}`,
            'content-type': 'application/x-www-form-urlencoded'
          },
          data: $.param({
            act: doTask ? 'a_enter' : 'a_leave',
            al: 1,
            pid: dataParam.publicPid,
            hash: dataParam.publicHash
          })
        });
        if (result !== 'Success') {
          debug('Vk公共页面操作请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('Vk公共页面操作状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('Vk公共页面操作成功', {
          name: name,
          doTask: doTask
        });
        logStatus.success();
        if (doTask) {
          this.tasks.names = unique([ ...this.tasks.names, name ]);
        }
        return true;
      } catch (error) {
        debug('处理Vk公共页面任务时发生错误', {
          error: error
        });
        throwError(error, 'Vk.togglePublic');
        return false;
      }
    }
    async #toggleLikeWall(name, dataParam, doTask = true) {
      try {
        debug('开始处理Vk点赞任务', {
          name: name,
          doTask: doTask
        });
        const logStatus = echoLog({
          type: doTask ? 'likingVkPublic' : 'unlikingVkPublic',
          text: name,
          before: '[Vk]'
        });
        const postData = {
          act: 'a_set_reaction',
          al: 1,
          event_subtype: 'post_modal',
          from: 'wall_page',
          hash: dataParam.hash,
          object: dataParam.object,
          track_code: dataParam.trackCode,
          wall: 2
        };
        if (doTask) {
          postData.reaction_id = 0;
        }
        const {result: resultR, statusText: statusTextR, status: statusR, data: dataR} = await httpRequest({
          url: 'https://vk.com/like.php?act=a_set_reaction',
          method: 'POST',
          headers: {
            origin: 'https://vk.com',
            referer: `https://vk.com/${name}`,
            'content-type': 'application/x-www-form-urlencoded'
          },
          data: $.param(postData)
        });
        if (resultR !== 'Success') {
          debug('Vk点赞操作请求失败', {
            result: resultR,
            statusText: statusTextR,
            status: statusR
          });
          logStatus.error(`${resultR}:${statusTextR}(${statusR})`);
          return false;
        }
        if (dataR?.status !== 200) {
          debug('Vk点赞操作状态错误', {
            status: dataR?.status,
            statusText: dataR?.statusText
          });
          logStatus.error(`Error:${dataR?.statusText}(${dataR?.status})`);
          return false;
        }
        if (dataR.response?.payload?.[1]?.[1]?.like_my !== true) {
          debug('Vk点赞操作验证失败');
          logStatus.error(`Error:${dataR?.statusText}(${dataR?.status})`);
          return false;
        }
        debug('Vk点赞操作成功', {
          name: name,
          doTask: doTask
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('处理Vk点赞任务时发生错误', {
          error: error
        });
        throwError(error, 'Vk.sendWall');
        return false;
      }
    }
    async #sendWall(name) {
      try {
        debug('开始处理Vk转发任务', {
          name: name
        });
        const logStatus = echoLog({
          type: 'sendingVkWall',
          text: name,
          before: '[Vk]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://vk.com/like.php',
          method: 'POST',
          headers: {
            origin: 'https://vk.com',
            referer: `https://vk.com/${name}`,
            'content-type': 'application/x-www-form-urlencoded'
          },
          data: $.param({
            act: 'publish_box',
            al: 1,
            object: name
          })
        });
        if (result !== 'Success') {
          debug('Vk转发操作请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('Vk转发操作状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        const hash = data.responseText.match(/shHash:[\s]*'(.*?)'/)?.[1];
        if (!hash) {
          debug('获取Vk转发hash失败');
          logStatus.error('Error: Get "hash" failed');
          return false;
        }
        const {result: resultR, statusText: statusTextR, status: statusR, data: dataR} = await httpRequest({
          url: 'https://vk.com/like.php',
          method: 'POST',
          headers: {
            origin: 'https://vk.com',
            referer: `https://vk.com/${name}`,
            'content-type': 'application/x-www-form-urlencoded'
          },
          data: $.param({
            Message: '',
            act: 'a_do_publish',
            al: 1,
            close_comments: 0,
            friends_only: 0,
            from: 'box',
            hash: hash,
            list: '',
            mark_as_ads: 0,
            mute_notifications: 0,
            object: name,
            ret_data: 1,
            to: 0
          })
        });
        if (resultR !== 'Success') {
          debug('Vk转发确认请求失败', {
            result: resultR,
            statusText: statusTextR,
            status: statusR
          });
          logStatus.error(`${resultR}:${statusTextR}(${statusR})`);
          return false;
        }
        if (dataR?.status !== 200) {
          debug('Vk转发确认状态错误', {
            status: dataR?.status,
            statusText: dataR?.statusText
          });
          logStatus.error(`Error:${dataR?.statusText}(${dataR?.status})`);
          return false;
        }
        const jsonData = JSON.parse(dataR.responseText?.replace('\x3c!--', '') || '{}');
        if (jsonData?.payload?.[1]?.[1]?.share_my !== true) {
          debug('Vk转发确认验证失败');
          logStatus.error(`Error:${dataR?.statusText}(${dataR?.status})`);
          return false;
        }
        debug('Vk转发操作成功', {
          name: name
        });
        logStatus.success();
        const postId = jsonData?.payload?.[1]?.[1]?.post_id;
        const ownerId = jsonData?.payload?.[1]?.[1]?.owner_id;
        if (postId && ownerId) {
          this.#setCache(name, `${ownerId}_${postId}`);
        }
        this.tasks.names = unique([ ...this.tasks.names, name ]);
        return true;
      } catch (error) {
        debug('处理Vk转发任务时发生错误', {
          error: error
        });
        throwError(error, 'Vk.sendWall');
        return false;
      }
    }
    async #deleteWall(name, dataParams) {
      try {
        debug('开始处理Vk删除墙任务', {
          name: name
        });
        const logStatus = echoLog({
          type: 'deletingVkWall',
          text: name,
          before: '[Vk]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://vk.com/al_wall.php?act=delete',
          method: 'POST',
          headers: {
            origin: 'https://vk.com',
            referer: `https://vk.com/${this.#username}?w=wall${this.#cache[name]}%2Fall`,
            'content-type': 'application/x-www-form-urlencoded'
          },
          data: $.param({
            act: 'delete',
            al: 1,
            confirm: 0,
            from: 'wkview',
            hash: dataParams.wallHash,
            post: this.#cache[name]
          })
        });
        if (result !== 'Success') {
          debug('Vk删除墙请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('Vk删除墙状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        const jsonData = JSON.parse(data.responseText?.replace('\x3c!--', '') || '{}');
        if (!jsonData?.payload?.[1]?.[1]) {
          debug('Vk删除墙验证失败');
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('Vk删除墙操作成功', {
          name: name
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('处理Vk删除墙任务时发生错误', {
          error: error
        });
        throwError(error, 'Vk.deleteWall');
        return false;
      }
    }
    async #getId(name, doTask) {
      try {
        debug('开始获取Vk ID', {
          name: name,
          doTask: doTask
        });
        let url = `https://vk.com/${name}`;
        if (/^wall-/.test(name)) {
          if (doTask) {
            return {
              type: 'sendWall'
            };
          }
          if (!this.#cache[name]) {
            return {
              type: 'unSupport'
            };
          }
          url = `https://vk.com/${this.#username}?w=wall${this.#cache[name]}`;
        }
        const logStatus = echoLog({
          type: 'gettingVkId',
          text: name,
          before: '[Vk]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: url,
          method: 'GET'
        });
        if (result !== 'Success') {
          debug('获取Vk ID请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('获取Vk ID状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        const [, groupAct, groupId, , groupHash] = data.responseText.match(/Groups.(enter|leave)\(.*?,.*?([\d]+?), (&#39;|')(.*?)(&#39;|')/) || [];
        const publicHash = data.responseText.match(/"enterHash":"(.*?)"/)?.[1];
        const publicPid = data.responseText.match(/"public_id":([\d]+?),/)?.[1];
        const publicJoined = !data.responseText.includes('Public.subscribe');
        if (groupAct && groupId && groupHash) {
          debug('获取到Vk群组ID', {
            groupAct: groupAct,
            groupId: groupId,
            groupHash: groupHash
          });
          logStatus.success();
          return {
            groupAct: groupAct,
            groupId: groupId,
            groupHash: groupHash,
            type: 'group'
          };
        }
        if (publicHash && publicPid) {
          debug('获取到Vk公共页面ID', {
            publicHash: publicHash,
            publicPid: publicPid,
            publicJoined: publicJoined
          });
          logStatus.success();
          return {
            publicHash: publicHash,
            publicPid: publicPid,
            publicJoined: publicJoined,
            type: 'public'
          };
        }
        if (name.includes('action=like')) {
          const hash = data.responseText.match(/data-reaction-hash="(.*?)"/)?.[1];
          const trackCode = data.responseText.match(/data-post-track-code="(.*?)"/)?.[1];
          const object = name.match(/wall-[\w_]+/)?.[0];
          if (hash && trackCode && object) {
            debug('获取到Vk点赞ID', {
              hash: hash,
              trackCode: trackCode,
              object: object
            });
            logStatus.success();
            return {
              type: 'likeWall',
              hash: hash,
              trackCode: trackCode,
              object: object
            };
          }
        }
        if (data.responseText.includes('wall.deletePost') && !doTask) {
          const wallHash = data.responseText.match(/wall\.deletePost\(this, '.*?', '(.*?)'\)/)?.[1];
          if (wallHash) {
            debug('获取到Vk删除墙ID', {
              wallHash: wallHash
            });
            logStatus.success();
            return {
              type: 'deleteWall',
              wallHash: wallHash
            };
          }
        }
        if (name.includes('wall') && doTask) {
          debug('获取到Vk墙ID');
          logStatus.success();
          return {
            type: 'sendWall'
          };
        }
        debug('未找到Vk ID参数');
        logStatus.error('Error: Parameters not found!');
        return false;
      } catch (error) {
        debug('获取Vk ID时发生错误', {
          error: error
        });
        throwError(error, 'Vk.getId');
        return false;
      }
    }
    async #toggleVk({name: name, doTask: doTask = true}) {
      try {
        debug('开始处理Vk任务', {
          name: name,
          doTask: doTask
        });
        if (!doTask && this.whiteList.names.includes(name)) {
          debug('Vk任务在白名单中,跳过', {
            name: name
          });
          echoLog({
            type: 'whiteList',
            text: 'Vk.undoTask',
            id: name,
            before: '[Vk]'
          });
          return true;
        }
        const formatName = name.replace(/\/$/, '');
        const data = await this.#getId(formatName, doTask);
        if (!data) {
          return false;
        }
        switch (data.type) {
         case 'group':
          return await this.#toggleGroup(formatName, data, doTask);

         case 'public':
          return await this.#togglePublic(formatName, data, doTask);

         case 'likeWall':
          return await this.#toggleLikeWall(formatName, data, doTask);

         case 'sendWall':
          return doTask ? await this.#sendWall(formatName) : true;

         case 'deleteWall':
          return doTask ? true : await this.#deleteWall(formatName, data);

         default:
          debug('未知的Vk任务类型', {
            type: data.type
          });
          return false;
        }
      } catch (error) {
        debug('处理Vk任务时发生错误', {
          error: error
        });
        throwError(error, 'Vk.toggleVk');
        return false;
      }
    }
    async toggle({doTask: doTask = true, nameLinks: nameLinks = []}) {
      try {
        debug('开始处理Vk链接任务', {
          doTask: doTask,
          nameLinksCount: nameLinks.length
        });
        if (!this.#initialized) {
          debug('Vk模块未初始化');
          echoLog({
            text: I18n('needInit'),
            before: '[Vk]'
          });
          return false;
        }
        const prom = [];
        if (doTask && !globalOptions.doTask.vk.names || !doTask && !globalOptions.undoTask.vk.names) {
          debug('根据全局选项跳过Vk任务', {
            doTask: doTask
          });
          echoLog({
            type: 'globalOptionsSkip',
            text: 'vk.names',
            before: '[Vk]'
          });
        } else {
          const realNames = this.getRealParams('names', nameLinks, doTask, (link => link.match(/https:\/\/vk\.com\/([^/]+)/)?.[1]));
          debug('处理后的Vk链接列表', {
            count: realNames.length,
            names: realNames
          });
          if (realNames.length > 0) {
            for (const name of realNames) {
              prom.push(this.#toggleVk({
                name: name,
                doTask: doTask
              }));
              await delay(1e3);
            }
          }
        }
        return Promise.all(prom).then((() => true));
      } catch (error) {
        debug('处理Vk链接任务时发生错误', {
          error: error
        });
        throwError(error, 'Vk.toggle');
        return false;
      }
    }
    #setCache(name, postId) {
      try {
        debug('设置Vk缓存', {
          name: name,
          postId: postId
        });
        this.#cache[name] = postId;
        GM_setValue('vkCache', this.#cache);
      } catch (error) {
        debug('设置Vk缓存时发生错误', {
          error: error
        });
        throwError(error, 'Vk.setCache');
      }
    }
  }
  const getInfo = async function(link, type) {
    try {
      debug('开始获取YouTube信息', {
        link: link,
        type: type
      });
      const logStatus = echoLog({
        text: I18n('gettingYtbToken'),
        before: '[Youtube]'
      });
      const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
        url: link,
        method: 'GET'
      });
      if (result !== 'Success') {
        debug('获取YouTube信息请求失败', {
          result: result,
          statusText: statusText,
          status: status
        });
        logStatus.error(`${result}:${statusText}(${status})`);
        return {};
      }
      if (data?.status !== 200) {
        debug('获取YouTube信息状态错误', {
          status: data?.status,
          statusText: data?.statusText
        });
        logStatus.error(`Error:${data?.statusText}(${data?.status})`);
        return {};
      }
      if (data.responseText.includes('accounts.google.com/ServiceLogin?service=youtube')) {
        debug('获取YouTube信息失败:需要登录');
        logStatus.error(`Error:${I18n('loginYtb')}`, true);
        return {
          needLogin: true
        };
      }
      const apiKey = data.responseText.match(/"INNERTUBE_API_KEY":"(.*?)"/)?.[1];
      const context = (data.responseText.match(/\(\{"INNERTUBE_CONTEXT":([\w\W]*?)\}\)/) || data.responseText.match(/"INNERTUBE_CONTEXT":([\w\W]*?\}),"INNERTUBE/))?.[1] || '{}';
      const {client: client, request: request} = JSON.parse(context);
      if (!apiKey || !client || !request) {
        debug('获取YouTube信息失败:缺少必要参数');
        logStatus.error('Error: Parameter "apiKey" not found!');
        return {};
      }
      client.hl = 'en';
      if (type === 'channel') {
        const channelId = data.responseText.match(/"channelId":"(.+?)"/)?.[1];
        if (!channelId) {
          debug('获取YouTube频道ID失败');
          logStatus.error('Error: Get "channelId" failed!');
          return {};
        }
        debug('成功获取YouTube频道信息', {
          channelId: channelId
        });
        logStatus.success();
        return {
          params: {
            apiKey: apiKey,
            client: client,
            request: request,
            channelId: channelId
          }
        };
      }
      if (type === 'likeVideo') {
        const videoId = data.responseText.match(/<link rel="shortlinkUrl" href="https:\/\/youtu\.be\/(.*?)">/)?.[1];
        const likeParams = data.responseText.match(/"likeParams":"(.*?)"/)?.[1];
        if (!videoId) {
          debug('获取YouTube视频ID失败');
          logStatus.error('Error: Get "videoId" failed!');
          return {};
        }
        debug('成功获取YouTube视频信息', {
          videoId: videoId
        });
        logStatus.success();
        return {
          params: {
            apiKey: apiKey,
            client: client,
            request: request,
            videoId: videoId,
            likeParams: likeParams
          }
        };
      }
      debug('未知的YouTube信息类型', {
        type: type
      });
      logStatus.error('Error: Unknown type');
      return {};
    } catch (error) {
      debug('获取YouTube信息时发生错误', {
        error: error
      });
      throwError(error, 'Youtube.getInfo');
      return {};
    }
  };
  class Youtube extends Social {
    tasks;
    whiteList;
    #auth=GM_getValue('youtubeAuth') || {};
    #initialized=false;
    #verifyChannel=`https://www.youtube.com/channel/${globalOptions.other.youtubeVerifyChannel}`;
    constructor() {
      super();
      const defaultTasksTemplate = {
        channels: [],
        likes: []
      };
      debug('初始化YouTube实例');
      this.tasks = defaultTasksTemplate;
      this.whiteList = {
        ...defaultTasksTemplate,
        ...GM_getValue('whiteList')?.youtube || {}
      };
    }
    async init() {
      try {
        debug('开始初始化YouTube模块');
        if (this.#initialized) {
          debug('YouTube模块已初始化');
          return true;
        }
        if (!this.#auth.PAPISID) {
          debug('YouTube授权信息不完整,需要更新授权');
          if (await this.#updateAuth()) {
            this.#initialized = true;
            return true;
          }
          return false;
        }
        const isVerified = await this.#verifyAuth();
        if (isVerified) {
          debug('YouTube授权验证成功');
          echoLog({
            before: '[Youtube]'
          }).success(I18n('initSuccess', 'Youtube'));
          this.#initialized = true;
          return true;
        }
        debug('YouTube授权失效,尝试重新获取');
        GM_setValue('youtubeAuth', null);
        if (await this.#updateAuth()) {
          debug('YouTube重新授权成功');
          echoLog({
            before: '[Youtube]'
          }).success(I18n('initSuccess', 'Youtube'));
          this.#initialized = true;
          return true;
        }
        debug('YouTube初始化失败');
        echoLog({
          before: '[Youtube]'
        }).error(I18n('initFailed', 'Youtube'));
        return false;
      } catch (error) {
        debug('YouTube初始化发生错误', {
          error: error
        });
        throwError(error, 'Youtube.init');
        return false;
      }
    }
    async #verifyAuth() {
      try {
        debug('开始验证YouTube授权');
        return await this.#toggleChannel({
          link: this.#verifyChannel,
          doTask: true,
          verify: true
        });
      } catch (error) {
        debug('YouTube授权验证发生错误', {
          error: error
        });
        throwError(error, 'Youtube.verifyAuth');
        return false;
      }
    }
    async #updateAuth() {
      try {
        debug('开始更新YouTube授权');
        const logStatus = echoLog({
          text: I18n('updatingAuth', 'Youtube'),
          before: '[Youtube]'
        });
        return await new Promise((resolve => {
          GM_cookie.list({
            url: 'https://www.youtube.com/@YouTube'
          }, (async (cookies, error) => {
            if (!error) {
              const PAPISID = cookies.find((cookie => cookie.name === '__Secure-3PAPISID'))?.value;
              if (PAPISID) {
                debug('成功获取YouTube新授权信息');
                GM_setValue('youtubeAuth', {
                  PAPISID: PAPISID
                });
                this.#auth = {
                  PAPISID: PAPISID
                };
                logStatus.success();
                resolve(await this.#verifyAuth());
              } else {
                debug('获取YouTube授权失败:未登录');
                logStatus.error(I18n('needLogin'));
                resolve(false);
              }
            } else {
              debug('获取YouTube授权失败', {
                error: error
              });
              logStatus.error('Error: Update youtube auth failed!');
              resolve(false);
            }
          }));
        }));
      } catch (error) {
        debug('更新YouTube授权时发生错误', {
          error: error
        });
        throwError(error, 'Youtube.updateAuth');
        return false;
      }
    }
    #getInfo(link, type) {
      debug('调用获取YouTube信息方法', {
        link: link,
        type: type
      });
      return getInfo(link, type);
    }
    async #toggleChannel({link: link, doTask: doTask = true, verify: verify = false}) {
      try {
        debug('开始处理YouTube频道任务', {
          link: link,
          doTask: doTask,
          verify: verify
        });
        const {params: params, needLogin: needLogin} = await this.#getInfo(link, 'channel');
        const {apiKey: apiKey, client: client, request: request, channelId: channelId} = params || {};
        if (needLogin) {
          debug('YouTube频道操作失败:需要登录');
          echoLog({
            html: I18n('loginYtb'),
            before: '[Youtube]'
          });
          return false;
        }
        if (!(apiKey && client && request && channelId)) {
          debug('YouTube频道操作失败:获取参数失败');
          echoLog({
            text: '"getYtbToken" failed',
            before: '[Youtube]'
          });
          return false;
        }
        if (!doTask && !verify && this.whiteList.channels.includes(channelId)) {
          debug('YouTube频道在白名单中,跳过取消订阅', {
            channelId: channelId
          });
          echoLog({
            type: 'whiteList',
            text: 'Youtube.unfollowChannel',
            id: channelId,
            before: '[Youtube]'
          });
          return true;
        }
        const logStatus = verify ? echoLog({
          text: I18n('verifyingAuth', 'Youtube'),
          before: '[Youtube]'
        }) : echoLog({
          type: doTask ? 'followingYtbChannel' : 'unfollowingYtbChannel',
          text: channelId,
          before: '[Youtube]'
        });
        const nowTime = parseInt(String((new Date).getTime() / 1e3), 10);
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://www.youtube.com/youtubei/v1/subscription/${doTask ? '' : 'un'}subscribe?key=${apiKey}&prettyPrint=false`,
          method: 'POST',
          headers: {
            origin: 'https://www.youtube.com',
            referer: `https://www.youtube.com/channel/${channelId}`,
            'content-type': 'application/json',
            'x-goog-authuser': '0',
            'x-goog-visitor-id': client?.visitorData,
            'x-origin': 'https://www.youtube.com',
            authorization: `SAPISIDHASH ${nowTime}_${sha1(`${nowTime} ${this.#auth.PAPISID} https://www.youtube.com`)}`
          },
          data: JSON.stringify({
            context: {
              client: client,
              request: {
                sessionId: request?.sessionId,
                internalExperimentFlags: [],
                consistencyTokenJars: []
              },
              user: {}
            },
            channelIds: [ channelId ],
            params: doTask ? 'EgIIAhgA' : 'CgIIAhgA'
          })
        });
        if (result !== 'Success') {
          debug('YouTube频道操作请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('YouTube频道操作状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        const isSubscribed = doTask && (/"subscribed":true/.test(data.responseText) || data.responseText.includes('The subscription already exists'));
        const isUnsubscribed = !doTask && /"subscribed":false/.test(data.responseText);
        const isVerified = verify && data.responseText.includes('You may not subscribe to yourself');
        if (isSubscribed || isUnsubscribed || isVerified) {
          debug('YouTube频道操作成功', {
            doTask: doTask,
            verify: verify
          });
          logStatus.success();
          if (doTask && !verify) {
            this.tasks.channels = unique([ ...this.tasks.channels, link ]);
          }
          return true;
        }
        debug('YouTube频道操作失败,需要更新授权');
        logStatus.error(I18n('tryUpdateYtbAuth'), true);
        return false;
      } catch (error) {
        debug('处理YouTube频道任务时发生错误', {
          error: error
        });
        throwError(error, 'Youtube.toggleChannel');
        return false;
      }
    }
    async #toggleLikeVideo({link: link, doTask: doTask = true}) {
      try {
        debug('开始处理YouTube视频点赞任务', {
          link: link,
          doTask: doTask
        });
        const {params: params, needLogin: needLogin} = await this.#getInfo(link, 'likeVideo');
        const {apiKey: apiKey, client: client, request: request, videoId: videoId, likeParams: likeParams} = params || {};
        if (needLogin) {
          debug('YouTube视频点赞失败:需要登录');
          echoLog({
            html: `${I18n('loginYtb')}`,
            before: '[Youtube]'
          });
          return false;
        }
        if (!(apiKey && client && request && videoId && likeParams)) {
          debug('YouTube视频点赞失败:获取参数失败');
          echoLog({
            text: '"getYtbToken" failed',
            before: '[Youtube]'
          });
          return false;
        }
        if (!doTask && this.whiteList.likes.includes(videoId)) {
          debug('YouTube视频在白名单中,跳过取消点赞', {
            videoId: videoId
          });
          echoLog({
            type: 'whiteList',
            text: 'Youtube.unlikeVideo',
            id: videoId,
            before: '[Youtube]'
          });
          return true;
        }
        const logStatus = echoLog({
          type: doTask ? 'likingYtbVideo' : 'unlikingYtbVideo',
          text: videoId,
          before: '[Youtube]'
        });
        const nowTime = parseInt(String((new Date).getTime() / 1e3), 10);
        const likeVideoData = {
          context: {
            client: client,
            request: {
              sessionId: request.sessionId,
              internalExperimentFlags: [],
              consistencyTokenJars: []
            },
            user: {}
          },
          target: {
            videoId: videoId
          }
        };
        if (doTask && !likeParams) {
          debug('YouTube视频点赞失败:缺少likeParams参数');
          logStatus.error('Empty likeParams');
          return false;
        }
        if (doTask) {
          likeVideoData.params = likeParams;
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://www.youtube.com/youtubei/v1/like/${doTask ? '' : 'remove'}like?key=${apiKey}`,
          method: 'POST',
          headers: {
            origin: 'https://www.youtube.com',
            referer: `https://www.youtube.com/watch?v=${videoId}`,
            'content-type': 'application/json',
            'x-goog-authuser': '0',
            'x-goog-visitor-id': client.visitorData,
            'x-origin': 'https://www.youtube.com',
            authorization: `SAPISIDHASH ${nowTime}_${sha1(`${nowTime} ${this.#auth.PAPISID} https://www.youtube.com`)}`
          },
          data: JSON.stringify(likeVideoData)
        });
        if (result !== 'Success') {
          debug('YouTube视频点赞请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('YouTube视频点赞状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        const isLiked = doTask && data.responseText.includes('Added to Liked videos');
        const isUnliked = !doTask && (data.responseText.includes('Removed from Liked videos') || data.responseText.includes('Dislike removed'));
        if (isLiked || isUnliked) {
          debug('YouTube视频点赞操作成功', {
            doTask: doTask
          });
          logStatus.success();
          if (doTask) {
            this.tasks.likes = unique([ ...this.tasks.likes, link ]);
          }
          return true;
        }
        debug('YouTube视频点赞失败,需要更新授权');
        logStatus.error(I18n('tryUpdateYtbAuth'), true);
        return false;
      } catch (error) {
        debug('处理YouTube视频点赞任务时发生错误', {
          error: error
        });
        throwError(error, 'Youtube.toggleLikeVideo');
        return false;
      }
    }
    async toggle({doTask: doTask = true, channelLinks: channelLinks = [], videoLinks: videoLinks = []}) {
      try {
        debug('开始处理YouTube链接任务', {
          doTask: doTask,
          channelLinksCount: channelLinks.length,
          videoLinksCount: videoLinks.length
        });
        if (!this.#initialized) {
          debug('YouTube模块未初始化');
          echoLog({
            text: I18n('needInit'),
            before: '[Youtube]'
          });
          return false;
        }
        const prom = [];
        const shouldProcessChannels = doTask && globalOptions.doTask.youtube.channels || !doTask && globalOptions.undoTask.youtube.channels;
        const shouldProcessVideos = doTask && globalOptions.doTask.youtube.likes || !doTask && globalOptions.undoTask.youtube.likes;
        if (!shouldProcessChannels) {
          debug('根据全局选项跳过YouTube频道任务', {
            doTask: doTask
          });
          echoLog({
            type: 'globalOptionsSkip',
            text: 'youtube.channels',
            before: '[Youtube]'
          });
        } else {
          const realChannels = this.getRealParams('channels', channelLinks, doTask, (link => {
            if (/^https:\/\/(www\.)?google\.com.*?\/url\?.*?url=https:\/\/www\.youtube\.com\/.*/.test(link)) {
              return link.match(/url=(https:\/\/www\.youtube\.com\/.*)/)?.[1];
            }
            return link;
          }));
          debug('处理后的YouTube频道链接列表', {
            count: realChannels.length,
            channels: realChannels
          });
          if (realChannels.length > 0) {
            for (const channel of realChannels) {
              prom.push(this.#toggleChannel({
                link: channel,
                doTask: doTask
              }));
              await delay(1e3);
            }
          }
        }
        if (!shouldProcessVideos) {
          debug('根据全局选项跳过YouTube视频任务', {
            doTask: doTask
          });
          echoLog({
            type: 'globalOptionsSkip',
            text: 'youtube.likes',
            before: '[Youtube]'
          });
        } else {
          const realLikes = this.getRealParams('likes', videoLinks, doTask, (link => {
            if (/^https:\/\/(www\.)?google\.com.*?\/url\?.*?url=https:\/\/www\.youtube\.com\/.*/.test(link)) {
              return link.match(/url=(https:\/\/www\.youtube\.com\/.*)/)?.[1];
            }
            return link;
          }));
          debug('处理后的YouTube视频链接列表', {
            count: realLikes.length,
            videos: realLikes
          });
          if (realLikes.length > 0) {
            for (const video of realLikes) {
              prom.push(this.#toggleLikeVideo({
                link: video,
                doTask: doTask
              }));
              await delay(1e3);
            }
          }
        }
        return Promise.all(prom).then((() => true));
      } catch (error) {
        debug('处理YouTube链接任务时发生错误', {
          error: error
        });
        throwError(error, 'Youtube.toggle');
        return false;
      }
    }
  }
  class SteamASF {
    #asfOptions;
    #botName='asf';
    #groupInfo;
    #steamWebApiKey;
    #steamId;
    constructor({AsfIpcUrl: AsfIpcUrl, AsfIpcPassword: AsfIpcPassword, AsfBotname: AsfBotname, steamWebApiKey: steamWebApiKey}) {
      debug('初始化SteamASF实例', {
        AsfIpcUrl: AsfIpcUrl,
        AsfBotname: AsfBotname
      });
      const asfCommandsUrl = new URL('/Api/Command/', AsfIpcUrl);
      this.#asfOptions = {
        url: asfCommandsUrl.href,
        method: 'POST',
        responseType: 'json',
        headers: {
          accept: 'application/json',
          'Content-Type': 'application/json',
          Host: asfCommandsUrl.host,
          Origin: asfCommandsUrl.origin,
          Referer: asfCommandsUrl.href,
          Authentication: AsfIpcPassword
        }
      };
      if (AsfBotname) {
        this.#botName = AsfBotname;
      }
      if (steamWebApiKey) {
        this.#steamWebApiKey = steamWebApiKey;
      }
      debug('SteamASF实例初始化完成', {
        botName: this.#botName
      });
    }
    async init() {
      try {
        debug('开始初始化ASF');
        const logStatus = echoLog({
          text: I18n('initingASF'),
          before: '[ASF]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          ...this.#asfOptions,
          data: '{"Command":"!stats"}'
        });
        if (result !== 'Success') {
          debug('ASF初始化请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.response?.Success === true && data.response.Message === 'OK' && data.response.Result) {
          debug('ASF初始化成功');
          logStatus.success();
          return true;
        }
        if (data?.response?.Result || data?.response?.Message) {
          debug('ASF初始化失败', {
            result: data?.response?.Result,
            message: data?.response?.Message
          });
          logStatus.error(data?.response?.Result || data.response.Message);
          return false;
        }
        debug('ASF初始化失败', {
          statusText: data?.statusText,
          status: data?.status
        });
        logStatus.error(`Error:${data?.statusText}(${data?.status})`);
        return false;
      } catch (error) {
        debug('ASF初始化发生错误', {
          error: error
        });
        throwError(error, 'SteamASF.init');
        return false;
      }
    }
    async joinGroup(groupName) {
      try {
        debug('开始加入Steam组', {
          groupName: groupName
        });
        const logStatus = echoLog({
          type: 'joiningSteamGroup',
          text: groupName,
          before: '[ASF]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          ...this.#asfOptions,
          data: JSON.stringify({
            Command: `!JOINGROUP ${this.#botName} ${groupName}`
          })
        });
        if (result !== 'Success') {
          debug('加入Steam组请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status === 200 && [ '已加入', '已申请', 'Joined', 'Applied', 'Присоединился', 'costs' ].find((text => data.response?.Result?.includes(text)))) {
          debug('成功加入Steam组', {
            groupName: groupName
          });
          logStatus.success();
          return true;
        }
        debug('加入Steam组失败', {
          result: data?.response?.Result,
          message: data?.response?.Message
        });
        logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
        return false;
      } catch (error) {
        debug('加入Steam组时发生错误', {
          error: error,
          groupName: groupName
        });
        throwError(error, 'SteamASF.joinGroup');
        return false;
      }
    }
    joinOfficialGroup=this.joinGroup;
    leaveOfficialGroup=this.leaveGroup;
    async leaveGroup(groupName) {
      try {
        debug('开始退出Steam组', {
          groupName: groupName
        });
        if (!this.#groupInfo) {
          debug('未找到组信息,尝试获取组ID');
          if (!await this.#getGroupId()) {
            return false;
          }
        }
        const groupId = await this.#groupInfo[groupName];
        if (!groupId) {
          debug('未找到组ID', {
            groupName: groupName
          });
          return false;
        }
        const logStatus = echoLog({
          type: 'leavingSteamGroup',
          text: groupName,
          before: '[ASF]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          ...this.#asfOptions,
          data: JSON.stringify({
            Command: `!LEAVEGROUP ${this.#botName} ${groupId}`
          })
        });
        if (result !== 'Success') {
          debug('退出Steam组请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status === 200 && [ '成功', 'Success', 'Успех' ].find((text => data.response?.Result?.includes(text)))) {
          debug('成功退出Steam组', {
            groupName: groupName
          });
          logStatus.success();
          return true;
        }
        debug('退出Steam组失败', {
          result: data?.response?.Result,
          message: data?.response?.Message
        });
        logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
        return false;
      } catch (error) {
        debug('退出Steam组时发生错误', {
          error: error,
          groupName: groupName
        });
        throwError(error, 'SteamASF.leaveGroup');
        return false;
      }
    }
    async #getGroupId() {
      try {
        debug('开始获取Steam组ID列表');
        const logStatus = echoLog({
          type: 'gettingSteamGroupId',
          text: 'All',
          before: '[ASF]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          ...this.#asfOptions,
          data: JSON.stringify({
            Command: `!GROUPLIST ${this.#botName}`
          })
        });
        if (result !== 'Success') {
          debug('获取Steam组ID列表请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status === 200 && data.response?.Result?.includes('|')) {
          this.#groupInfo = Object.fromEntries(data.response.Result.split('\n').map((line => {
            const [, name, id] = line.trim().split('|');
            if (name && id) {
              return [ name, id ];
            }
            return null;
          })).filter((ele => ele)));
          debug('成功获取Steam组ID列表', {
            groupCount: Object.keys(this.#groupInfo).length
          });
          logStatus.success();
          return true;
        }
        debug('获取Steam组ID列表失败', {
          result: data?.response?.Result,
          message: data?.response?.Message
        });
        logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
        return false;
      } catch (error) {
        debug('获取Steam组ID列表时发生错误', {
          error: error
        });
        throwError(error, 'SteamASF.getGroupID');
        return false;
      }
    }
    async addToWishlist(gameId) {
      try {
        debug('开始添加游戏到愿望单', {
          gameId: gameId
        });
        const logStatus = echoLog({
          type: 'addingToWishlist',
          text: gameId,
          before: '[ASF]'
        });
        const gameStatus = await this.#checkGame(gameId);
        if (gameStatus.wishlist === true) {
          debug('游戏已在愿望单中', {
            gameId: gameId
          });
          logStatus.success();
          return true;
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          ...this.#asfOptions,
          data: JSON.stringify({
            Command: `!ADDWISHLIST ${this.#botName} ${gameId}`
          })
        });
        if (result !== 'Success') {
          debug('添加游戏到愿望单请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status === 200 && [ '成功', 'Success', 'Успех' ].find((text => data.response?.Result?.includes(text)))) {
          debug('成功添加游戏到愿望单', {
            gameId: gameId
          });
          logStatus.success();
          return true;
        }
        debug('添加游戏到愿望单失败', {
          result: data?.response?.Result,
          message: data?.response?.Message
        });
        logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
        return false;
      } catch (error) {
        debug('添加游戏到愿望单时发生错误', {
          error: error,
          gameId: gameId
        });
        throwError(error, 'SteamASF.addToWishlist');
        return false;
      }
    }
    async removeFromWishlist(gameId) {
      try {
        debug('开始从愿望单移除游戏', {
          gameId: gameId
        });
        const logStatus = echoLog({
          type: 'removingFromWishlist',
          text: gameId,
          before: '[ASF]'
        });
        const gameStatus = await this.#checkGame(gameId);
        if (gameStatus.wishlist === false) {
          debug('游戏已不在愿望单中', {
            gameId: gameId
          });
          logStatus.success();
          return true;
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          ...this.#asfOptions,
          data: JSON.stringify({
            Command: `!REMOVEWISHLIST ${this.#botName} ${gameId}`
          })
        });
        if (result !== 'Success') {
          debug('从愿望单移除游戏请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status === 200 && [ '成功', 'Success', 'Успех' ].find((text => data.response?.Result?.includes(text)))) {
          debug('成功从愿望单移除游戏', {
            gameId: gameId
          });
          logStatus.success();
          return true;
        }
        debug('从愿望单移除游戏失败', {
          result: data?.response?.Result,
          message: data?.response?.Message
        });
        logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
        return false;
      } catch (error) {
        debug('从愿望单移除游戏时发生错误', {
          error: error,
          gameId: gameId
        });
        throwError(error, 'SteamASF.removeFromWishlist');
        return false;
      }
    }
    async toggleFollowGame(gameId, doTask) {
      try {
        debug('开始处理游戏关注状态', {
          gameId: gameId,
          doTask: doTask
        });
        const logStatus = echoLog({
          type: `${doTask ? '' : 'un'}followingGame`,
          text: gameId,
          before: '[ASF]'
        });
        const gameStatus = await this.#checkGame(gameId);
        if (doTask && gameStatus.followed === true || !doTask && gameStatus.followed === false) {
          debug('游戏关注状态已符合要求', {
            gameId: gameId,
            doTask: doTask,
            currentStatus: gameStatus.followed
          });
          logStatus.success();
          return true;
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          ...this.#asfOptions,
          data: JSON.stringify({
            Command: `!${doTask ? '' : 'UN'}FOLLOWGAME ${this.#botName} ${gameId}`
          })
        });
        if (result !== 'Success') {
          debug('处理游戏关注状态请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status === 200 && [ '成功', 'Success', 'Успех' ].find((text => data.response?.Result?.includes(text)))) {
          debug('成功处理游戏关注状态', {
            gameId: gameId,
            doTask: doTask
          });
          logStatus.success();
          return true;
        }
        debug('处理游戏关注状态失败', {
          result: data?.response?.Result,
          message: data?.response?.Message
        });
        logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
        return false;
      } catch (error) {
        debug('处理游戏关注状态时发生错误', {
          error: error,
          gameId: gameId,
          doTask: doTask
        });
        throwError(error, 'SteamASF.toggleFollowGame');
        return false;
      }
    }
    async #checkGame(gameId) {
      try {
        debug('开始检查游戏状态', {
          gameId: gameId
        });
        const {result: result, data: data} = await httpRequest({
          ...this.#asfOptions,
          data: JSON.stringify({
            Command: `!CHECK ${this.#botName} ${gameId}`
          })
        });
        if (result !== 'Success') {
          debug('检查游戏状态请求失败', {
            result: result
          });
          return {};
        }
        if (data?.status !== 200 || !data.response?.Result?.includes(gameId)) {
          debug('检查游戏状态响应无效', {
            status: data?.status
          });
          return {};
        }
        const matchedResult = data.response.Result.split('\n').find((result => result.includes(gameId)))?.split('|');
        if (!matchedResult || matchedResult.length <= 3) {
          debug('未找到游戏状态信息', {
            gameId: gameId
          });
          return {};
        }
        const status = {
          wishlist: matchedResult.at(-3).trim() === '√' || matchedResult.at(-2).trim() === '√',
          followed: matchedResult.at(-1).trim() === '√'
        };
        debug('成功获取游戏状态', {
          gameId: gameId,
          status: status
        });
        return status;
      } catch (error) {
        debug('检查游戏状态时发生错误', {
          error: error,
          gameId: gameId
        });
        throwError(error, 'SteamASF.checkGame');
        return {};
      }
    }
    async toggleCurator(curatorId, doTask = true) {
      try {
        debug('开始处理鉴赏家关注状态', {
          curatorId: curatorId,
          doTask: doTask
        });
        const logStatus = echoLog({
          type: doTask ? 'followingCurator' : 'unfollowingCurator',
          text: curatorId,
          before: '[ASF]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          ...this.#asfOptions,
          data: JSON.stringify({
            Command: `!${doTask ? '' : 'UN'}FOLLOWCURATOR ${this.#botName} ${curatorId}`
          })
        });
        if (result !== 'Success') {
          debug('处理鉴赏家关注状态请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status === 200 && [ '成功', 'Success', 'Успех' ].find((text => data.response?.Result?.includes(text)))) {
          debug('成功处理鉴赏家关注状态', {
            curatorId: curatorId,
            doTask: doTask
          });
          logStatus.success();
          return true;
        }
        if (data?.status === 200) {
          debug('处理鉴赏家关注状态失败', {
            result: data?.response?.Result,
            message: data?.response?.Message
          });
          logStatus.error(I18n('curatorLimitNotice'));
          return false;
        }
        debug('处理鉴赏家关注状态失败', {
          result: data?.response?.Result,
          message: data?.response?.Message
        });
        logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
        return false;
      } catch (error) {
        debug('处理鉴赏家关注状态时发生错误', {
          error: error,
          curatorId: curatorId,
          doTask: doTask
        });
        throwError(error, 'Steam.toggleCurator');
        return false;
      }
    }
    async addLicense(id) {
      try {
        debug('开始添加许可证', {
          id: id
        });
        const [type, ids] = id.split('-');
        const idsArr = ids.split(',');
        if (type === 'appid') {
          const logStatus = echoLog({
            type: 'addingFreeLicense',
            text: ids,
            before: '[ASF]'
          });
          const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
            ...this.#asfOptions,
            data: JSON.stringify({
              Command: `!addlicense ${this.#botName} ${idsArr.map((id => `app/${id}`)).join(',')}`
            })
          });
          if (result !== 'Success') {
            debug('添加应用许可证请求失败', {
              result: result,
              statusText: statusText,
              status: status
            });
            logStatus.error(`${result}:${statusText}(${status})`);
            return false;
          }
          if (data?.status === 200 && [ 'AlreadyPurchased', 'OK' ].find((text => data.response?.Result?.includes(text)))) {
            debug('成功添加应用许可证', {
              ids: ids
            });
            logStatus.success();
            return true;
          }
          debug('添加应用许可证失败', {
            result: data?.response?.Result,
            message: data?.response?.Message
          });
          logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
          return false;
        }
        if (type === 'subid') {
          const logStatus = echoLog({
            type: 'addingFreeLicenseSubid',
            text: ids,
            before: '[ASF]'
          });
          const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
            ...this.#asfOptions,
            data: JSON.stringify({
              Command: `!addlicense ${this.#botName} ${idsArr.map((id => `sub/${id}`)).join(',')}`
            })
          });
          if (result !== 'Success') {
            debug('添加订阅许可证请求失败', {
              result: result,
              statusText: statusText,
              status: status
            });
            logStatus.error(`${result}:${statusText}(${status})`);
            return false;
          }
          if (data?.status === 200 && data.response?.Result) {
            const resultLines = data.response.Result.split('\n');
            debug('处理订阅许可证结果', {
              resultLines: resultLines
            });
            idsArr.forEach((subid => {
              const targetLine = resultLines.find((text => text.includes(subid)));
              if (targetLine && [ '成功', 'Success', 'Успех' ].find((text => targetLine.includes(text)))) {
                debug('成功添加订阅许可证', {
                  subid: subid
                });
                echoLog({
                  before: '[ASF]'
                }).success(targetLine);
              } else {
                debug('添加订阅许可证失败', {
                  subid: subid,
                  targetLine: targetLine
                });
                echoLog({
                  before: '[ASF]'
                }).error(targetLine);
              }
            }));
            return true;
          }
          debug('添加订阅许可证失败', {
            result: data?.response?.Result,
            message: data?.response?.Message
          });
          logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
          return false;
        }
        debug('无效的许可证类型', {
          type: type
        });
        return false;
      } catch (error) {
        debug('添加许可证时发生错误', {
          error: error,
          id: id
        });
        throwError(error, 'SteamASF.addLicense');
        return false;
      }
    }
    async requestPlayTestAccess(id) {
      try {
        debug('开始请求游戏试玩权限', {
          id: id
        });
        const logStatus = echoLog({
          type: 'requestingPlayTestAccess',
          text: id,
          before: '[ASF]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          ...this.#asfOptions,
          data: JSON.stringify({
            Command: `!REQUESTACCESS ${this.#botName} ${id}`
          })
        });
        if (result !== 'Success') {
          debug('请求游戏试玩权限请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status === 200 && [ '成功', 'Success', 'Успех' ].find((text => data.response?.Result?.includes(text)))) {
          debug('成功请求游戏试玩权限', {
            id: id
          });
          logStatus.success();
          return true;
        }
        debug('请求游戏试玩权限失败', {
          result: data?.response?.Result,
          message: data?.response?.Message
        });
        logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
        return false;
      } catch (error) {
        debug('请求游戏试玩权限时发生错误', {
          error: error,
          id: id
        });
        throwError(error, 'SteamASF.requestPlayTestAccess');
        return false;
      }
    }
    async playGames(ids) {
      try {
        debug('开始挂游戏时长', {
          ids: ids
        });
        const logStatus = echoLog({
          text: I18n('playingGames', ids),
          before: '[ASF]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          ...this.#asfOptions,
          data: JSON.stringify({
            Command: `!play ${this.#botName} ${ids}`
          })
        });
        if (result !== 'Success') {
          debug('挂游戏时长请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status === 200 && [ '正在运行', '正在掛', 'Playing', 'Играет' ].find((text => data.response?.Result?.includes(text)))) {
          debug('成功开始挂游戏时长', {
            ids: ids
          });
          logStatus.success();
          return true;
        }
        debug('开始挂游戏时长失败', {
          result: data?.response?.Result,
          message: data?.response?.Message
        });
        logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
        return false;
      } catch (error) {
        debug('挂游戏时长时发生错误', {
          error: error,
          ids: ids
        });
        throwError(error, 'SteamASF.playGames');
        return false;
      }
    }
    async getSteamIdASF() {
      try {
        debug('开始获取Steam ID');
        const logStatus = echoLog({
          text: I18n('gettingSteamId'),
          before: '[ASF]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          ...this.#asfOptions,
          data: JSON.stringify({
            Command: `!steamid ${this.#botName}`
          })
        });
        if (result !== 'Success' || data?.status !== 200) {
          debug('获取Steam ID请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return '';
        }
        if (data.response?.Result) {
          const steamId = data.response.Result.trim()?.split(/\s+/)?.at(-1);
          if (steamId) {
            debug('成功获取Steam ID', steamId);
            logStatus.success();
            return steamId;
          }
        }
        debug('获取Steam ID失败', {
          result: data?.response?.Result,
          message: data?.response?.Message
        });
        logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
        return '';
      } catch (error) {
        debug('获取Steam ID时发生错误', {
          error: error
        });
        throwError(error, 'SteamASF.getSteamIdASF');
        return '';
      }
    }
    async getSteamIdWeb() {
      try {
        debug('开始获取Steam ID');
        const logStatus = echoLog({
          text: I18n('gettingSteamId'),
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://store.steampowered.com',
          method: 'GET',
          headers: {
            host: 'store.steampowered.com'
          }
        });
        if (result !== 'Success' || data?.status !== 200) {
          debug('获取Steam ID请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return '';
        }
        const steamId = data.responseText.match(/steamid&quot;:&quot;(\d+)/)?.[1];
        if (steamId) {
          debug('成功获取Steam ID', steamId);
          logStatus.success();
          return steamId;
        }
        debug('获取Steam ID失败', {
          data: data
        });
        logStatus.error(`${result}:${statusText}(${status})`);
        return '';
      } catch (error) {
        debug('获取Steam ID时发生错误', {
          error: error
        });
        throwError(error, 'SteamASF.getSteamIdWeb');
        return '';
      }
    }
    async getSteamId() {
      const steamId = await this.getSteamIdWeb();
      if (steamId) {
        return steamId;
      }
      return this.getSteamIdASF();
    }
    async checkPlayStatus(ids) {
      try {
        debug('开始检查挂游戏时长状态');
        if (!this.#steamWebApiKey) {
          debug('未设置Steam Web API Key');
          return 'skip';
        }
        if (!this.#steamId) {
          const steamId = await this.getSteamId();
          if (!steamId) {
            debug('未获取到Steam ID');
            return 'skip';
          }
          this.#steamId = steamId;
        }
        const logStatus = echoLog({
          text: I18n('checkingPlayStatus'),
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://api.steampowered.com/ISteamUser/GetPlayerSummaries/v0002/?key=${this.#steamWebApiKey}&steamids=${this.#steamId}`,
          method: 'GET',
          headers: {
            'Content-Type': 'application/json'
          }
        });
        if (result !== 'Success') {
          debug('检查挂游戏时长状态请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status === 200) {
          debug('挂游戏时长状态正常', {
            data: data
          });
          const playedIds = new Set(data.responseText?.match(/\d+/g));
          const neededIds = new Set(ids.match(/\d+/g));
          if (neededIds.intersection(playedIds).size > 0) {
            logStatus.success();
            return true;
          }
          logStatus.warning(I18n('noPlayStatus'));
          return false;
        }
        debug('挂游戏时长状态异常', {
          data: data
        });
        logStatus.error(`Error:${data?.statusText}(${data?.status})`);
        return false;
      } catch (error) {
        debug('检查挂游戏时长状态时发生错误', {
          error: error
        });
        throwError(error, 'SteamASF.checkPlayStatus');
        return false;
      }
    }
    async stopPlayGames() {
      try {
        debug('开始停止挂游戏时长');
        const logStatus = echoLog({
          text: I18n('stoppingPlayGames'),
          before: '[ASF]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          ...this.#asfOptions,
          data: JSON.stringify({
            Command: `!resume ${this.#botName}`
          })
        });
        if (result !== 'Success') {
          debug('停止挂游戏时长请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status === 200 && [ '已经恢复', '已恢复', '已經繼續', '已繼續', 'resumed', 'возобновлён' ].find((text => data.response?.Result?.includes(text)))) {
          debug('成功停止挂游戏时长');
          logStatus.success();
          return true;
        }
        debug('停止挂游戏时长失败', {
          result: data?.response?.Result,
          message: data?.response?.Message
        });
        logStatus.error(`Error:${data?.response?.Result || data?.response?.Message || data?.statusText}(${data?.status})`);
        return false;
      } catch (error) {
        debug('停止挂游戏时长时发生错误', {
          error: error
        });
        throwError(error, 'SteamASF.stopPlayGames');
        return false;
      }
    }
    async #unsupportted(name) {
      try {
        debug('尝试使用不支持的功能', {
          name: name
        });
        echoLog({
          before: '[ASF]'
        }).warning(I18n('ASFNotSupportted', name));
        return false;
      } catch (error) {
        debug('处理不支持的功能时发生错误', {
          error: error,
          name: name
        });
        throwError(error, 'SteamASF.unsupportted');
        return false;
      }
    }
    async toggleForum() {
      return this.#unsupportted('toggleForum');
    }
    async toggleFavoriteWorkshop() {
      return this.#unsupportted('toggleFavoriteWorkshop');
    }
    async voteUpWorkshop() {
      return this.#unsupportted('voteUpWorkshop');
    }
    async likeAnnouncement() {
      return this.#unsupportted('likeAnnouncement');
    }
  }
  class SteamWeb {
    #cache={
      ...{
        group: {},
        officialGroup: {},
        forum: {},
        workshop: {},
        curator: {}
      },
      ...GM_getValue('steamCache')
    };
    #auth={};
    #storeInitialized=false;
    #communityInitialized=false;
    #area='CN';
    #oldArea;
    #areaStatus='end';
    constructor() {
      debug('初始化SteamWeb实例');
    }
    async init(type = 'all') {
      try {
        debug('开始初始化SteamWeb', {
          type: type
        });
        const initStoreResult = await this.initStore();
        debug('Steam商店初始化完成', {
          initStoreResult: initStoreResult
        });
        if (type === 'store') {
          return initStoreResult;
        }
        const initCommunityResult = await this.initCommunity(initStoreResult);
        debug('Steam社区初始化完成', {
          initCommunityResult: initCommunityResult
        });
        return initCommunityResult;
      } catch (error) {
        debug('SteamWeb初始化发生错误', {
          error: error,
          type: type
        });
        throwError(error, 'SteamWeb.init');
        return false;
      }
    }
    async initStore() {
      try {
        debug('开始初始化Steam商店');
        if (this.#storeInitialized) {
          return true;
        }
        let storeInitialized = await this.#updateStoreAuth();
        if (!storeInitialized) {
          storeInitialized = await this.#updateStoreAuthTab();
        }
        this.#storeInitialized = storeInitialized;
        if (!this.#storeInitialized) {
          echoLog({
            before: '[Web]'
          }).error(I18n('initFailed', 'Steam'));
          return false;
        }
        echoLog({
          before: '[Web]'
        }).success(I18n('initSuccess', 'SteamStore'));
        debug('Steam商店初始化完成');
        return true;
      } catch (error) {
        debug('Steam商店初始化发生错误', {
          error: error
        });
        throwError(error, 'SteamWeb.initStore');
        return false;
      }
    }
    async initCommunity(initStoreResult) {
      try {
        debug('开始初始化Steam社区');
        if (this.#communityInitialized) {
          return true;
        }
        let communityInitialized = await this.#updateCommunityAuth(initStoreResult);
        if (!communityInitialized) {
          communityInitialized = await this.#updateCommunityAuthTab();
          GM_setValue('steamCommunityAuth', null);
        }
        this.#communityInitialized = communityInitialized;
        if (!this.#communityInitialized) {
          echoLog({
            before: '[Web]'
          }).error(I18n('initFailed', 'Steam'));
          return false;
        }
        echoLog({
          before: '[Web]'
        }).success(I18n('initSuccess', 'SteamCommunity'));
        debug('Steam社区初始化完成');
        return true;
      } catch (error) {
        debug('Steam社区初始化发生错误', {
          error: error
        });
        throwError(error, 'SteamWeb.initCommunity');
        return false;
      }
    }
    async #refreshToken(type = 'steamStore') {
      try {
        debug('开始刷新令牌', {
          type: type
        });
        const host = {
          steamStore: 'store.steampowered.com',
          steamCommunity: 'steamcommunity.com'
        };
        const logStatus = echoLog({
          text: I18n('refreshingToken', I18n(type)),
          before: '[Web]'
        });
        debug('准备刷新令牌请求数据');
        const formData = new FormData;
        formData.append('redir', `https://${host[type]}/`);
        debug('发送刷新令牌请求');
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://login.steampowered.com/jwt/ajaxrefresh',
          method: 'POST',
          responseType: 'json',
          headers: {
            Host: 'login.steampowered.com',
            Origin: `https://${host[type]}`,
            Referer: `https://${host[type]}/`
          },
          data: formData
        });
        debug('收到刷新令牌响应', {
          result: result,
          status: status,
          statusText: statusText
        });
        if (result !== 'Success') {
          debug('请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (!data?.response?.success) {
          debug('响应不成功', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('开始设置新令牌');
        if (!await this.#setToken(data.response, type)) {
          debug('设置新令牌失败');
          logStatus.error('Error');
          return false;
        }
        debug('成功刷新令牌');
        logStatus.success();
        return true;
      } catch (error) {
        debug('刷新令牌时发生错误', {
          error: error
        });
        throwError(error, 'SteamWeb.refreshToken');
        return false;
      }
    }
    async #setToken(param, type) {
      try {
        const host = {
          steamStore: 'store.steampowered.com',
          steamCommunity: 'steamcommunity.com'
        };
        debug('开始设置Steam令牌', {
          type: type
        });
        const logStatus = echoLog({
          text: I18n('settingToken', I18n(type)),
          before: '[Web]'
        });
        debug('准备表单数据');
        const formData = new FormData;
        formData.append('steamID', param.steamID);
        formData.append('nonce', param.nonce);
        formData.append('redir', param.redir);
        formData.append('auth', param.auth);
        debug('表单数据准备完成', {
          steamID: param.steamID,
          nonce: param.nonce,
          redir: param.redir
        });
        debug('发送设置令牌请求');
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://${host[type]}/login/settoken`,
          method: 'POST',
          headers: {
            Accept: 'application/json, text/plain, */*',
            Host: host[type],
            Origin: `https://${host[type]}`
          },
          data: formData
        });
        debug('收到设置令牌响应', {
          result: result,
          status: status,
          statusText: statusText
        });
        if (result !== 'Success') {
          debug('请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('响应状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('成功设置令牌');
        logStatus.success();
        return true;
      } catch (error) {
        debug('设置令牌时发生错误', {
          error: error,
          type: type
        });
        throwError(error, 'SteamWeb.setToken');
        return false;
      }
    }
    async #updateStoreAuth(retry = false) {
      try {
        debug('开始更新Steam商店身份验证');
        const logStatus = echoLog({
          text: I18n('updatingAuth', I18n('steamStore')),
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://store.steampowered.com/',
          method: 'GET',
          headers: {
            Accept: 'text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9',
            'Sec-Fetch-Dest': 'document',
            'Sec-Fetch-Mode': 'navigate',
            'Upgrade-Insecure-Requests': '1'
          },
          redirect: 'manual'
        });
        debug('收到Steam商店身份验证响应', {
          result: result,
          statusText: statusText,
          status: status
        });
        if (data?.status !== 200) {
          if (![ 301, 302 ].includes(data?.status)) {
            debug('Steam商店身份验证状态错误', {
              status: data?.status
            });
            logStatus.error(`${result}:${statusText}(${status})`);
            return false;
          }
          if (!await this.#refreshToken('steamStore')) {
            debug('Steam商店身份验证刷新失败');
            logStatus.error(`Error:${I18n('needLoginSteamStore')}`, true);
            return false;
          }
          if (retry) {
            debug('Steam商店身份验证重试失败');
            logStatus.error(`Error:${I18n('needLoginSteamStore')}`, true);
            return false;
          }
          debug('Steam商店身份验证重试中');
          logStatus.warning(I18n('retry'));
          return this.#updateStoreAuth(true);
        }
        if (!data.responseText.includes('data-miniprofile=')) {
          if (await this.#refreshToken('steamStore')) {
            debug('Steam商店身份验证需要重试');
            logStatus.warning(I18n('retry'));
            if (retry) {
              debug('Steam商店身份验证重试次数超限');
              logStatus.error(`Error:${I18n('needLoginSteamStore')}`, true);
              return false;
            }
            return this.#updateStoreAuth(true);
          }
          debug('Steam商店身份验证失败:需要登录');
          logStatus.error(`Error:${I18n('needLoginSteamStore')}`, true);
          return false;
        }
        const storeSessionID = data.responseText.match(/g_sessionID = "(.+?)";/)?.[1];
        if (!storeSessionID) {
          debug('Steam商店身份验证失败:获取sessionID失败');
          logStatus.error('Error: Get "sessionID" failed');
          return false;
        }
        this.#auth.storeSessionID = storeSessionID;
        debug('Steam商店身份验证更新成功', {
          storeSessionID: storeSessionID
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('更新Steam商店身份验证时发生错误', {
          error: error
        });
        throwError(error, 'SteamWeb.updateStoreAuth');
        return false;
      }
    }
    async #updateStoreAuthTab() {
      try {
        debug('开始通过新标签页更新Steam商店身份验证');
        const logStatus = echoLog({
          text: I18n('updatingAuth', I18n('steamStoreTab')),
          before: '[Web]'
        });
        return await new Promise((resolve => {
          GM_deleteValue('steamStoreAuth');
          GM_setValue('ATv4_updateStoreAuth', true);
          const newTab = GM_openInTab('https://store.steampowered.com/', {
            active: true,
            setParent: true
          });
          debug('打开Steam商店新标签页');
          newTab.name = 'ATv4_updateStoreAuth';
          const listenerId = GM_addValueChangeListener('steamStoreAuth', ((key, oldValue, newValue) => {
            debug('监听到Steam商店身份验证值变化', {
              oldValue: oldValue,
              newValue: newValue
            });
            GM_removeValueChangeListener(listenerId);
            GM_deleteValue('ATv4_updateStoreAuth');
            newTab?.close();
            window.focus();
            if (newValue && JSON.stringify(newValue) !== JSON.stringify(oldValue)) {
              this.#auth.storeSessionID = newValue.storeSessionID;
              debug('Steam商店身份验证更新成功', {
                storeSessionID: newValue.storeSessionID
              });
              logStatus.success();
              resolve(true);
              return;
            }
            debug('Steam商店身份验证更新失败');
            logStatus.error('Failed');
            resolve(false);
          }));
          newTab.onclose = () => {
            debug('Steam商店新标签页已关闭');
            GM_deleteValue('ATv4_updateStoreAuth');
          };
        }));
      } catch (error) {
        debug('通过新标签页更新Steam商店身份验证时发生错误', {
          error: error
        });
        throwError(error, 'SteamWeb.updateStoreAuthTab');
        return false;
      }
    }
    async #updateCommunityAuthTab() {
      try {
        debug('开始通过新标签页更新Steam社区身份验证');
        const logStatus = echoLog({
          text: I18n('updatingAuth', I18n('steamCommunityTab')),
          before: '[Web]'
        });
        return await new Promise((resolve => {
          GM_deleteValue('steamCommunityAuth');
          GM_setValue('ATv4_updateCommunityAuth', true);
          const newTab = GM_openInTab('https://steamcommunity.com/my', {
            active: true,
            setParent: true
          });
          debug('打开Steam社区新标签页');
          newTab.name = 'ATv4_updateCommunityAuth';
          const listenerId = GM_addValueChangeListener('steamCommunityAuth', ((key, oldValue, newValue) => {
            debug('监听到Steam社区身份验证值变化', {
              oldValue: oldValue,
              newValue: newValue
            });
            GM_removeValueChangeListener(listenerId);
            GM_deleteValue('ATv4_updateCommunityAuth');
            newTab?.close();
            window.focus();
            if (newValue && JSON.stringify(newValue) !== JSON.stringify(oldValue)) {
              this.#auth.steam64Id = newValue.steam64Id;
              this.#auth.communitySessionID = newValue.communitySessionID;
              debug('Steam社区身份验证更新成功', {
                steam64Id: newValue.steam64Id,
                communitySessionID: newValue.communitySessionID
              });
              logStatus.success();
              resolve(true);
              return;
            }
            debug('Steam社区身份验证更新失败');
            logStatus.error('Failed');
            resolve(false);
          }));
          newTab.onclose = () => {
            debug('Steam社区新标签页已关闭');
            GM_deleteValue('ATv4_updateCommunityAuth');
          };
        }));
      } catch (error) {
        debug('通过新标签页更新Steam社区身份验证时发生错误', {
          error: error
        });
        throwError(error, 'SteamWeb.updateCommunityAuthTab');
        return false;
      }
    }
    async #updateCommunityAuth(initStoreResult, retry = false) {
      try {
        debug('开始更新Steam社区身份验证');
        const logStatus = echoLog({
          text: I18n('gettingUserInfo'),
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://steamcommunity.com/my',
          method: 'GET',
          headers: {
            Accept: 'text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7',
            Host: 'steamcommunity.com',
            'Sec-Fetch-Dest': 'document',
            'Sec-Fetch-Mode': 'navigate'
          },
          redirect: 'follow'
        });
        debug('收到Steam社区身份验证响应', {
          result: result,
          statusText: statusText,
          status: status
        });
        if (data?.status !== 200) {
          debug('Steam社区身份验证状态错误', {
            status: data?.status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data.finalUrl.includes('https://steamcommunity.com/login/home')) {
          if (initStoreResult) {
            if (await this.#refreshToken('steamCommunity')) {
              debug('Steam社区身份验证需要重试');
              logStatus.warning(I18n('retry'));
              if (retry) {
                debug('Steam社区身份验证重试次数超限');
                logStatus.error(`Error:${I18n('needLoginSteamCommunity')}`, true);
                return false;
              }
              return this.#updateCommunityAuth(initStoreResult, retry);
            }
          }
          debug('Steam社区身份验证失败:需要登录');
          logStatus.error(`Error:${I18n('needLoginSteamCommunity')}`, true);
          return false;
        }
        const steam64Id = data.responseText.match(/g_steamID = "(.+?)";/)?.[1];
        const communitySessionID = data.responseText.match(/g_sessionID = "(.+?)";/)?.[1];
        if (!steam64Id || !communitySessionID) {
          debug('Steam社区身份验证失败:获取身份信息失败');
          logStatus.error('Error: Get "sessionID" failed');
          return false;
        }
        this.#auth.steam64Id = steam64Id;
        this.#auth.communitySessionID = communitySessionID;
        debug('Steam社区身份验证更新成功', {
          steam64Id: steam64Id,
          communitySessionID: communitySessionID
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('更新Steam社区身份验证时发生错误', {
          error: error
        });
        throwError(error, 'SteamWeb.updateCommunityAuth');
        return false;
      }
    }
    async #getAreaInfo() {
      try {
        debug('开始获取Steam地区信息');
        const logStatus = echoLog({
          text: I18n('gettingAreaInfo'),
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://store.steampowered.com/cart/',
          method: 'GET'
        });
        debug('获取地区信息请求结果', {
          result: result,
          statusText: statusText,
          status: status
        });
        if (result !== 'Success' || data?.status !== 200) {
          debug('获取地区信息失败', {
            result: result,
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(result === 'Success' ? `Error:${data?.statusText}(${data?.status})` : `${result}:${statusText}(${status})`);
          return {};
        }
        const cartConfigRaw = data.responseText.match(/data-cart_config="(.*?)"/)?.[1];
        debug('cartConfigRaw提取结果', {
          cartConfigRaw: cartConfigRaw
        });
        const temp = document.createElement('div');
        temp.innerHTML = cartConfigRaw || '{}';
        const cartConfigStr = temp.textContent || temp.innerText;
        let cartConfig;
        try {
          cartConfig = JSON.parse(cartConfigStr);
          debug('cartConfig解析成功', {
            cartConfig: cartConfig
          });
        } catch (error) {
          debug('cartConfig解析失败', {
            error: error
          });
          logStatus.error('Error: get country info filed');
          console.error(error);
          return {};
        }
        if (!cartConfig.rgUserCountryOptions) {
          debug('未找到可更换地区');
          logStatus.warning('Warning: Area cannot be changed');
          return {};
        }
        const userInfoRaw = data.responseText.match(/data-userinfo="(.*?)"/)?.[1];
        debug('userInfoRaw提取结果', {
          userInfoRaw: userInfoRaw
        });
        const temp1 = document.createElement('div');
        temp1.innerHTML = userInfoRaw || '{}';
        const userInfoStr = temp1.textContent || temp1.innerText;
        let userInfo;
        try {
          userInfo = JSON.parse(userInfoStr);
          debug('userInfo解析成功', {
            userInfo: userInfo
          });
        } catch (error) {
          debug('userInfo解析失败', {
            error: error
          });
          logStatus.error('Error: get country info filed');
          console.error(error);
          return {};
        }
        const currentArea = userInfo.country_code;
        const areas = Object.keys(cartConfig.rgUserCountryOptions).filter((area => area !== 'help'));
        debug('地区信息提取', {
          currentArea: currentArea,
          areas: areas
        });
        if (!currentArea || areas.length === 0) {
          debug('未获取到当前地区或可更换地区为空', {
            currentArea: currentArea,
            areas: areas
          });
          logStatus.error('Error: get country info filed');
          return {};
        }
        this.#area = currentArea;
        debug('获取地区信息成功', {
          currentArea: currentArea,
          areas: areas
        });
        logStatus.success();
        return {
          currentArea: currentArea,
          areas: areas
        };
      } catch (error) {
        debug('获取地区信息时发生异常', {
          error: error
        });
        throwError(error, 'SteamWeb.getAreaInfo');
        return {};
      }
    }
    async #changeArea(area) {
      try {
        debug('开始更换Steam地区', {
          area: area
        });
        if (this.#areaStatus === 'waiting') {
          debug('当前地区状态为waiting,等待状态改变');
          await new Promise((resolve => {
            const checker = setInterval((() => {
              if (this.#areaStatus !== 'waiting') {
                clearInterval(checker);
                resolve(true);
              }
            }));
          }));
        }
        if (this.#area === area || !area && this.#area !== 'CN') {
          debug('无需更换地区', {
            currentArea: this.#area,
            targetArea: area
          });
          return true;
        }
        this.#areaStatus = 'waiting';
        let aimedArea = area;
        if (!aimedArea) {
          debug('未指定目标地区,自动获取可用地区');
          const {currentArea: currentArea, areas: areas} = await this.#getAreaInfo();
          debug('获取到地区信息', {
            currentArea: currentArea,
            areas: areas
          });
          if (!currentArea || !areas) {
            debug('获取地区信息失败', {
              currentArea: currentArea,
              areas: areas
            });
            this.#areaStatus = 'error';
            return false;
          }
          if (currentArea !== 'CN') {
            debug('当前地区不是CN,无需更换', {
              currentArea: currentArea
            });
            this.#areaStatus = 'skip';
            echoLog({
              text: I18n('notNeededChangeArea'),
              before: '[Web]'
            });
            return 'skip';
          }
          const anotherArea = areas.filter((area => area && area !== 'CN'));
          debug('可更换的其他地区', {
            anotherArea: anotherArea
          });
          if (!anotherArea || anotherArea.length === 0) {
            debug('没有可用的其他地区');
            this.#areaStatus = 'noAnotherArea';
            echoLog({
              text: I18n('noAnotherArea'),
              before: '[Web]'
            });
            return false;
          }
          [aimedArea] = anotherArea;
          debug('选定目标地区', {
            aimedArea: aimedArea
          });
        }
        const logStatus = echoLog({
          text: I18n('changingArea', aimedArea),
          before: '[Web]'
        });
        debug('发送更换地区请求', {
          aimedArea: aimedArea
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://store.steampowered.com/country/setcountry',
          method: 'POST',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: $.param({
            cc: aimedArea,
            sessionid: this.#auth.storeSessionID
          })
        });
        debug('更换地区请求结果', {
          result: result,
          statusText: statusText,
          status: status
        });
        if (result !== 'Success' || data?.status !== 200 || data.responseText !== 'true') {
          debug('更换地区失败', {
            result: result,
            status: data?.status,
            statusText: data?.statusText,
            responseText: data?.responseText
          });
          this.#areaStatus = 'error';
          logStatus.error(result === 'Success' ? `Error:${data?.statusText}(${data?.status})` : `${result}:${statusText}(${status})`);
          return 'CN';
        }
        const {currentArea: currentArea} = await this.#getAreaInfo();
        debug('更换后获取到的当前地区', {
          currentArea: currentArea
        });
        if (currentArea) {
          this.#area = currentArea;
          if (!this.#oldArea) {
            this.#oldArea = currentArea;
          }
        }
        if (currentArea !== aimedArea) {
          debug('更换后当前地区与目标地区不符', {
            currentArea: currentArea,
            aimedArea: aimedArea
          });
          this.#areaStatus = 'error';
          logStatus.error('Error: change country filed');
          return 'CN';
        }
        this.#areaStatus = 'success';
        debug('更换地区成功', {
          currentArea: currentArea
        });
        logStatus.success();
        return currentArea;
      } catch (error) {
        debug('更换地区时发生异常', {
          error: error
        });
        this.#areaStatus = 'error';
        throwError(error, 'SteamWeb.changeArea');
        return false;
      }
    }
    async joinGroup(groupName) {
      try {
        debug('开始加入Steam组', {
          groupName: groupName
        });
        const logStatus = echoLog({
          type: 'joiningSteamGroup',
          text: groupName,
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://steamcommunity.com/groups/${groupName}`,
          method: 'POST',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: $.param({
            action: 'join',
            sessionID: this.#auth.communitySessionID
          })
        });
        if (result !== 'Success') {
          debug('加入Steam组请求失败', {
            result: result
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200 || data.responseText.includes('grouppage_join_area')) {
          debug('加入Steam组失败', {
            status: data?.status
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('成功加入Steam组', {
          groupName: groupName
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('加入Steam组时发生错误', {
          error: error,
          groupName: groupName
        });
        throwError(error, 'SteamWeb.joinGroup');
        return false;
      }
    }
    async leaveGroup(groupName) {
      try {
        debug('开始退出Steam组', {
          groupName: groupName
        });
        const groupId = await this.#getGroupId(groupName);
        if (!groupId) {
          return false;
        }
        const logStatus = echoLog({
          type: 'leavingSteamGroup',
          text: groupName,
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://steamcommunity.com/profiles/${this.#auth.steam64Id}/home_process`,
          method: 'POST',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: $.param({
            sessionID: this.#auth.communitySessionID,
            action: 'leaveGroup',
            groupId: groupId
          })
        });
        if (result !== 'Success') {
          debug('退出Steam组请求失败', {
            result: result
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200 || !data.finalUrl.includes('groups')) {
          debug('退出Steam组失败', {
            status: data?.status
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        const hasGroupLink = $(data.responseText.replace(/<img.*?>/g, '').toLowerCase()).find(`a[href='https://steamcommunity.com/groups/${groupName.toLowerCase()}']`).length > 0;
        if (hasGroupLink) {
          debug('Error: Group link still exists');
          return false;
        }
        debug('成功退出Steam组', {
          groupName: groupName
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('退出Steam组时发生错误', {
          error: error,
          groupName: groupName
        });
        throwError(error, 'SteamWeb.leaveGroup');
        return false;
      }
    }
    async #getGroupId(groupName) {
      try {
        debug('开始获取Steam组ID', {
          groupName: groupName
        });
        const logStatus = echoLog({
          type: 'gettingSteamGroupId',
          text: groupName,
          before: '[Web]'
        });
        const cachedGroupId = this.#cache.group[groupName];
        if (cachedGroupId) {
          debug('从缓存中获取到组ID', {
            groupName: groupName,
            cachedGroupId: cachedGroupId
          });
          logStatus.success();
          return cachedGroupId;
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://steamcommunity.com/groups/${groupName}`,
          method: 'GET',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          }
        });
        debug('获取组ID请求结果', {
          result: result,
          statusText: statusText,
          status: status
        });
        if (result !== 'Success') {
          debug('获取组ID请求失败', {
            result: result
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('获取组ID响应状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        const matchedGroupId = data.responseText.match(/OpenGroupChat\( '([0-9]+)'/)?.[1];
        debug('正则提取组ID结果', {
          matchedGroupId: matchedGroupId
        });
        if (!matchedGroupId) {
          debug('未能提取到组ID', {
            groupName: groupName
          });
          logStatus.error(`Error:${data.statusText}(${data.status})`);
          return false;
        }
        this.#setCache('group', groupName, matchedGroupId);
        debug('组ID已缓存', {
          groupName: groupName,
          matchedGroupId: matchedGroupId
        });
        logStatus.success();
        return matchedGroupId;
      } catch (error) {
        debug('获取组ID时发生异常', {
          error: error,
          groupName: groupName
        });
        throwError(error, 'SteamWeb.getGroupID');
        return false;
      }
    }
    async joinOfficialGroup(gameId) {
      try {
        debug('开始加入Steam官方组', {
          gameId: gameId
        });
        const logStatus = echoLog({
          type: 'joiningSteamOfficialGroup',
          text: gameId,
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://steamcommunity.com/games/${gameId}?action=join&sessionID=${this.#auth.communitySessionID}`,
          method: 'GET',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          }
        });
        if (result !== 'Success') {
          debug('加入Steam官方组请求失败', {
            result: result
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200 || data.responseText.includes('id="publicGroupJoin"')) {
          debug('加入Steam官方组失败', {
            status: data?.status
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        const groupId = data.responseText.match(/steam:\/\/friends\/joinchat\/([0-9]+)/)?.[1];
        if (groupId) {
          this.#setCache('officialGroup', gameId, groupId);
        }
        debug('成功加入Steam官方组', {
          gameId: gameId
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('加入Steam官方组时发生错误', {
          error: error,
          gameId: gameId
        });
        throwError(error, 'SteamWeb.joinOfficialGroup');
        return false;
      }
    }
    async leaveOfficialGroup(gameId) {
      try {
        debug('开始退出Steam官方组', {
          gameId: gameId
        });
        const groupId = await this.#getOfficialGroupId(gameId);
        if (!groupId) {
          return false;
        }
        const logStatus = echoLog({
          type: 'leavingSteamOfficialGroup',
          text: gameId,
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://steamcommunity.com/profiles/${this.#auth.steam64Id}/home_process`,
          method: 'POST',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: $.param({
            sessionID: this.#auth.communitySessionID,
            action: 'leaveGroup',
            groupId: groupId
          })
        });
        if (result !== 'Success') {
          debug('退出Steam官方组请求失败', {
            result: result
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('退出Steam官方组失败', {
            status: data?.status
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        const {result: resultR, statusText: statusTextR, status: statusR, data: dataR} = await httpRequest({
          url: `https://steamcommunity.com/games/${gameId}`,
          method: 'GET',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          }
        });
        if (resultR !== 'Success') {
          debug('退出Steam官方组时发生错误', {
            error: resultR,
            status: statusR,
            statusText: statusTextR
          });
          logStatus.error(`${resultR}:${statusTextR}(${statusR})`);
          return false;
        }
        if (dataR?.status !== 200 || !dataR.responseText.includes('id="publicGroupJoin"')) {
          debug('退出Steam官方组失败', {
            status: dataR?.status
          });
          logStatus.error(`Error:${dataR?.statusText}(${dataR?.status})`);
          return false;
        }
        debug('成功退出Steam官方组', {
          gameId: gameId
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('退出Steam官方组时发生错误', {
          error: error,
          gameId: gameId
        });
        throwError(error, 'SteamWeb.leaveOfficialGroup');
        return false;
      }
    }
    async #getOfficialGroupId(gameId) {
      try {
        debug('开始获取Steam官方群组ID', {
          gameId: gameId
        });
        const logStatus = echoLog({
          type: 'gettingSteamOfficialGroupId',
          text: gameId,
          before: '[Web]'
        });
        const cachedGroupId = this.#cache.officialGroup[gameId];
        if (cachedGroupId) {
          debug('从缓存中获取到官方群组ID', {
            gameId: gameId,
            cachedGroupId: cachedGroupId
          });
          logStatus.success();
          return cachedGroupId;
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://steamcommunity.com/games/${gameId}`,
          method: 'GET',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          }
        });
        debug('获取官方群组ID请求结果', {
          result: result,
          statusText: statusText,
          status: status
        });
        if (result !== 'Success') {
          debug('获取官方群组ID请求失败', {
            result: result
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('获取官方群组ID响应状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        const matchedGroupId = data.responseText.match(/steam:\/\/friends\/joinchat\/([0-9]+)/)?.[1];
        debug('正则提取官方群组ID结果', {
          matchedGroupId: matchedGroupId
        });
        if (!matchedGroupId) {
          debug('未能提取到官方群组ID', {
            gameId: gameId
          });
          logStatus.error(`Error:${data.statusText}(${data.status})`);
          return false;
        }
        this.#setCache('officialGroup', gameId, matchedGroupId);
        debug('官方群组ID已缓存', {
          gameId: gameId,
          matchedGroupId: matchedGroupId
        });
        logStatus.success();
        return matchedGroupId;
      } catch (error) {
        debug('获取官方群组ID时发生异常', {
          error: error,
          gameId: gameId
        });
        throwError(error, 'SteamWeb.getGroupID');
        return false;
      }
    }
    async addToWishlist(gameId) {
      try {
        debug('开始添加游戏到愿望单', {
          gameId: gameId
        });
        const logStatus = echoLog({
          type: 'addingToWishlist',
          text: gameId,
          before: '[Web]'
        });
        const {result: result, data: data} = await httpRequest({
          url: 'https://store.steampowered.com/api/addtowishlist',
          method: 'POST',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: $.param({
            sessionid: this.#auth.storeSessionID,
            appid: gameId
          }),
          dataType: 'json'
        });
        if (result === 'Success' && data?.status === 200 && data.response?.success === true) {
          debug('成功添加游戏到愿望单', {
            gameId: gameId
          });
          logStatus.success();
          return true;
        }
        const {result: resultR, statusText: statusTextR, status: statusR, data: dataR} = await httpRequest({
          url: `https://store.steampowered.com/app/${gameId}`,
          method: 'GET'
        });
        if (resultR !== 'Success') {
          debug('添加游戏到愿望单请求失败', {
            result: resultR,
            status: statusR,
            statusText: statusTextR
          });
          logStatus.error(`${resultR}:${statusTextR}(${statusR})`);
          return false;
        }
        if (dataR?.status !== 200) {
          debug('添加游戏到愿望单失败', {
            status: dataR?.status
          });
          logStatus.error(`Error:${dataR?.statusText}(${dataR?.status})`);
          return false;
        }
        if (this.#area === 'CN' && dataR.responseText.includes('id="error_box"')) {
          debug('changeAreaNotice');
          if (!await this.#changeArea()) {
            return false;
          }
          return await this.addToWishlist(gameId);
        }
        if (dataR.responseText.includes('class="queue_actions_ctn"') && dataR.responseText.includes('class="already_in_library"')) {
          debug('成功添加游戏到愿望单', {
            gameId: gameId
          });
          logStatus.success();
          return true;
        }
        if (dataR.responseText.includes('class="queue_actions_ctn"') && dataR.responseText.includes('id="add_to_wishlist_area_success" style="display: none;') || !dataR.responseText.includes('class="queue_actions_ctn"')) {
          debug('添加游戏到愿望单失败', {
            status: dataR.statusText
          });
          logStatus.error(`Error:${dataR.statusText}(${dataR.status})`);
          return false;
        }
        debug('成功添加游戏到愿望单', {
          gameId: gameId
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('添加游戏到愿望单时发生错误', {
          error: error,
          gameId: gameId
        });
        throwError(error, 'SteamWeb.addToWishlist');
        return false;
      }
    }
    async removeFromWishlist(gameId) {
      try {
        debug('开始从愿望单移除游戏', {
          gameId: gameId
        });
        const logStatus = echoLog({
          type: 'removingFromWishlist',
          text: gameId,
          before: '[Web]'
        });
        const {result: result, data: data} = await httpRequest({
          url: 'https://store.steampowered.com/api/removefromwishlist',
          method: 'POST',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: $.param({
            sessionid: this.#auth.storeSessionID,
            appid: gameId
          }),
          dataType: 'json'
        });
        if (result === 'Success' && data?.status === 200 && data.response?.success === true) {
          debug('成功从愿望单移除游戏', {
            gameId: gameId
          });
          logStatus.success();
          return true;
        }
        const {result: resultR, statusText: statusTextR, status: statusR, data: dataR} = await httpRequest({
          url: `https://store.steampowered.com/app/${gameId}`,
          method: 'GET'
        });
        if (resultR !== 'Success') {
          debug('从愿望单移除游戏请求失败', {
            result: resultR,
            status: statusR,
            statusText: statusTextR
          });
          logStatus.error(`${resultR}:${statusTextR}(${statusR})`);
          return false;
        }
        if (dataR?.status !== 200) {
          debug('从愿望单移除游戏失败', {
            status: dataR?.status
          });
          logStatus.error(`Error:${dataR?.statusText}(${dataR?.status})`);
          return false;
        }
        if (this.#area === 'CN' && dataR.responseText.includes('id="error_box"')) {
          debug('changeAreaNotice');
          const result = await this.#changeArea();
          if (!result || result === 'CN' || result === 'skip') {
            return false;
          }
          return await this.removeFromWishlist(gameId);
        }
        if (dataR.responseText.includes('class="queue_actions_ctn"') && (dataR.responseText.includes('ds_owned_flag ds_flag') || dataR.responseText.includes('add_to_wishlist_area'))) {
          debug('成功从愿望单移除游戏', {
            gameId: gameId
          });
          logStatus.success();
          return true;
        }
        debug('从愿望单移除游戏请求失败', {
          result: resultR,
          status: statusR,
          statusText: statusTextR
        });
        logStatus.error(`Error:${dataR.statusText}(${dataR.status})`);
        return false;
      } catch (error) {
        debug('从愿望单移除游戏时发生错误', {
          error: error,
          gameId: gameId
        });
        throwError(error, 'SteamWeb.removeFromWishlist');
        return false;
      }
    }
    async toggleFollowGame(gameId, doTask) {
      try {
        debug('开始处理游戏关注状态', {
          gameId: gameId,
          doTask: doTask
        });
        const logStatus = echoLog({
          type: `${doTask ? '' : 'un'}followingGame`,
          text: gameId,
          before: '[Web]'
        });
        const requestData = {
          sessionid: this.#auth.storeSessionID,
          appid: gameId
        };
        if (!doTask) {
          requestData.unfollow = '1';
        }
        const {result: result, data: data} = await httpRequest({
          url: 'https://store.steampowered.com/explore/followgame/',
          method: 'POST',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: $.param(requestData)
        });
        if (result === 'Success' && data?.status === 200 && data.responseText === 'true') {
          debug('成功处理游戏关注状态', {
            gameId: gameId,
            doTask: doTask
          });
          logStatus.success();
          return true;
        }
        const followed = await this.#isFollowedGame(gameId);
        if (this.#area === 'CN' && followed === 'areaLocked') {
          debug('changeAreaNotice');
          if (!await this.#changeArea()) {
            return false;
          }
          return await this.toggleFollowGame(gameId, doTask);
        }
        if (doTask === followed) {
          debug('成功处理游戏关注状态', {
            gameId: gameId,
            doTask: doTask
          });
          logStatus.success();
          return true;
        }
        debug('处理游戏关注状态请求失败', {
          result: result
        });
        logStatus.error(`Error:${data?.statusText}(${data?.status})`);
        return false;
      } catch (error) {
        debug('处理游戏关注状态时发生错误', {
          error: error,
          gameId: gameId,
          doTask: doTask
        });
        throwError(error, 'SteamWeb.toggleFollowGame');
        return false;
      }
    }
    async #isFollowedGame(gameId) {
      try {
        debug('开始判断Steam游戏是否已关注', {
          gameId: gameId
        });
        const {result: result, data: data} = await httpRequest({
          url: `https://store.steampowered.com/app/${gameId}`,
          method: 'GET'
        });
        debug('获取游戏页面请求结果', {
          result: result,
          status: data?.status
        });
        if (result !== 'Success') {
          debug('请求失败', {
            result: result
          });
          return false;
        }
        if (data?.status !== 200) {
          debug('响应状态错误', {
            status: data?.status
          });
          return false;
        }
        if (this.#area === 'CN' && data.responseText.includes('id="error_box"')) {
          debug('地区锁定,返回areaLocked', {
            area: this.#area
          });
          return 'areaLocked';
        }
        const isFollowed = $(data.responseText.replace(/<img.*?>/g, '')).find('.queue_control_button.queue_btn_follow>.btnv6_blue_hoverfade.btn_medium.queue_btn_active').css('display') !== 'none';
        debug('关注状态判断结果', {
          isFollowed: isFollowed
        });
        return isFollowed;
      } catch (error) {
        debug('判断游戏关注状态时发生异常', {
          error: error,
          gameId: gameId
        });
        throwError(error, 'SteamWeb.isFollowedGame');
        return false;
      }
    }
    async toggleForum(gameId, doTask = true) {
      try {
        debug('开始处理论坛订阅状态', {
          gameId: gameId,
          doTask: doTask
        });
        const forumId = await this.#getForumId(gameId);
        if (!forumId) {
          return false;
        }
        const logStatus = echoLog({
          type: `${doTask ? '' : 'un'}subscribingForum`,
          text: gameId,
          before: '[Web]'
        });
        const [id, feature] = forumId.split('_');
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://steamcommunity.com/forum/${id}/General/${doTask ? '' : 'un'}subscribe/${feature || '0'}/`,
          method: 'POST',
          responseType: 'json',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: $.param({
            sessionid: this.#auth.communitySessionID
          })
        });
        if (result !== 'Success') {
          debug('处理论坛订阅状态请求失败', {
            result: result
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return true;
        }
        if (data?.status !== 200 || data.response?.success !== 1 && data.response?.success !== 29) {
          debug('处理论坛订阅状态失败', {
            status: data?.status
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return true;
        }
        debug('成功处理论坛订阅状态', {
          gameId: gameId,
          doTask: doTask
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('处理论坛订阅状态时发生错误', {
          error: error,
          gameId: gameId,
          doTask: doTask
        });
        throwError(error, 'SteamWeb.toggleForum');
        return false;
      }
    }
    async #getForumId(gameId) {
      try {
        debug('开始获取Steam论坛ID', {
          gameId: gameId
        });
        const logStatus = echoLog({
          type: 'gettingForumId',
          text: gameId,
          before: '[Web]'
        });
        const cachedForumId = this.#cache.forum[gameId];
        if (cachedForumId) {
          debug('从缓存中获取到论坛ID', {
            gameId: gameId,
            cachedForumId: cachedForumId
          });
          logStatus.success();
          return cachedForumId;
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://steamcommunity.com/app/${gameId}/discussions/`,
          method: 'GET'
        });
        debug('获取论坛ID请求结果', {
          result: result,
          statusText: statusText,
          status: status
        });
        if (result !== 'Success') {
          debug('获取论坛ID请求失败', {
            result: result
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('获取论坛ID响应状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        const matchedForumId = data.responseText?.match(/General_([\d]+(_[\d]+)?)/)?.[1];
        debug('正则提取论坛ID结果', {
          matchedForumId: matchedForumId
        });
        if (!matchedForumId) {
          debug('未能提取到论坛ID', {
            gameId: gameId
          });
          logStatus.error(`Error:${data.statusText}(${data.status})`);
          return false;
        }
        this.#setCache('forum', gameId, matchedForumId);
        debug('论坛ID已缓存', {
          gameId: gameId,
          matchedForumId: matchedForumId
        });
        logStatus.success();
        return matchedForumId;
      } catch (error) {
        debug('获取论坛ID时发生异常', {
          error: error,
          gameId: gameId
        });
        throwError(error, 'SteamWeb.getForumId');
        return false;
      }
    }
    async toggleFavoriteWorkshop(id, doTask = true) {
      try {
        debug('开始处理创意工坊收藏状态', {
          id: id,
          doTask: doTask
        });
        const appid = await this.#getWorkshopAppId(id);
        if (!appid) {
          return false;
        }
        const logStatus = echoLog({
          type: doTask ? 'favoritingWorkshop' : 'unfavoritingWorkshop',
          text: id,
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://steamcommunity.com/sharedfiles/${doTask ? '' : 'un'}favorite`,
          method: 'POST',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: $.param({
            id: id,
            appid: appid,
            sessionid: this.#auth.communitySessionID
          })
        });
        if (result !== 'Success') {
          debug('处理创意工坊收藏状态请求失败', {
            result: result
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200 || data.responseText) {
          debug('处理创意工坊收藏状态失败', {
            status: data?.status
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('成功处理创意工坊收藏状态', {
          id: id,
          doTask: doTask
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('处理创意工坊收藏状态时发生错误', {
          error: error,
          id: id,
          doTask: doTask
        });
        throwError(error, 'SteamWeb.toggleFavoriteWorkshop');
        return false;
      }
    }
    async #getWorkshopAppId(id) {
      try {
        debug('开始获取Steam创意工坊AppId', {
          id: id
        });
        const logStatus = echoLog({
          type: 'gettingWorkshopAppId',
          text: id,
          before: '[Web]'
        });
        const cachedAppId = this.#cache.workshop[id];
        if (cachedAppId) {
          debug('从缓存中获取到AppId', {
            id: id,
            cachedAppId: cachedAppId
          });
          logStatus.success();
          return cachedAppId;
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://steamcommunity.com/sharedfiles/filedetails/?id=${id}`,
          method: 'GET'
        });
        debug('获取创意工坊AppId请求结果', {
          result: result,
          statusText: statusText,
          status: status
        });
        if (result !== 'Success') {
          debug('获取创意工坊AppId请求失败', {
            result: result
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('获取创意工坊AppId响应状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        const matchedAppId = data.responseText.match(/<input type="hidden" name="appid" value="([\d]+?)" \/>/)?.[1];
        debug('正则提取AppId结果', {
          matchedAppId: matchedAppId
        });
        if (!matchedAppId) {
          debug('未能提取到AppId', {
            id: id
          });
          logStatus.error('Error: getWorkshopAppId failed');
          return false;
        }
        debug('AppId已缓存', {
          id: id,
          matchedAppId: matchedAppId
        });
        return matchedAppId;
      } catch (error) {
        debug('获取创意工坊AppId时发生异常', {
          error: error,
          id: id
        });
        throwError(error, 'SteamWeb.getWorkshopAppId');
        return false;
      }
    }
    async voteUpWorkshop(id) {
      try {
        debug('开始点赞创意工坊物品', {
          id: id
        });
        const logStatus = echoLog({
          type: 'votingUpWorkshop',
          text: id,
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://steamcommunity.com/sharedfiles/voteup',
          method: 'POST',
          responseType: 'json',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: $.param({
            id: id,
            sessionid: this.#auth.communitySessionID
          })
        });
        if (result !== 'Success') {
          debug('点赞创意工坊物品请求失败', {
            result: result
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return true;
        }
        if (data?.status !== 200 || data.response?.success !== 1) {
          debug('点赞创意工坊物品失败', {
            status: data?.status
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return true;
        }
        debug('成功点赞创意工坊物品', {
          id: id
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('点赞创意工坊物品时发生错误', {
          error: error,
          id: id
        });
        throwError(error, 'SteamWeb.voteUpWorkshop');
        return false;
      }
    }
    async toggleCurator(curatorId, doTask = true) {
      try {
        debug('开始处理鉴赏家关注状态', {
          curatorId: curatorId,
          doTask: doTask
        });
        const logStatus = echoLog({
          type: doTask ? 'followingCurator' : 'unfollowingCurator',
          text: curatorId,
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://store.steampowered.com/curators/ajaxfollow',
          method: 'POST',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: $.param({
            clanid: curatorId,
            sessionid: this.#auth.storeSessionID,
            follow: doTask
          }),
          dataType: 'json'
        });
        if (result !== 'Success') {
          debug('处理鉴赏家关注状态请求失败', {
            result: result
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.response?.success?.success === 25) {
          debug('处理鉴赏家关注状态失败', {
            status: data?.status,
            success: data?.response?.success,
            message: data?.response?.msg
          });
          logStatus.error(I18n('curatorLimitNotice'));
          return false;
        }
        if (data?.status !== 200 || data.response?.success?.success !== 1) {
          debug('处理鉴赏家关注状态失败', {
            status: data?.status,
            success: data?.response?.success
          });
          logStatus.error(`Error:${data?.statusText}(${data?.response?.success}` || `${data?.status})`);
          return false;
        }
        debug('成功处理鉴赏家关注状态', {
          curatorId: curatorId,
          doTask: doTask
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('处理鉴赏家关注状态时发生错误', {
          error: error,
          curatorId: curatorId,
          doTask: doTask
        });
        throwError(error, 'SteamWeb.toggleCurator');
        return false;
      }
    }
    async #getAnnouncementParams(appId, viewId) {
      try {
        debug('开始获取Steam公告参数', {
          appId: appId,
          viewId: viewId
        });
        const logStatus = echoLog({
          type: 'gettingAnnouncementParams',
          text: appId,
          id: viewId,
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://store.steampowered.com/events/ajaxgetpartnerevent?appid=${appId}&announcement_gid=${viewId}&lang_list=6_0&last_modified_time=0&origin=https:%2F%2Fstore.steampowered.com&for_edit=false`,
          method: 'GET',
          responseType: 'json',
          headers: {
            Host: 'store.steampowered.com',
            Referer: `https://store.steampowered.com/news/app/${appId}/view/${viewId}`
          }
        });
        debug('获取公告参数请求结果', {
          result: result,
          statusText: statusText,
          status: status
        });
        if (result !== 'Success') {
          debug('获取公告参数请求失败', {
            result: result
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return {};
        }
        if (data?.status !== 200 || data?.response?.success !== 1) {
          debug('获取公告参数响应状态错误', {
            status: data?.status,
            statusText: data?.statusText,
            response: data?.response
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return {};
        }
        const {clanid: clanid, gid: gid} = data.response.event?.announcement_body || {};
        debug('公告参数提取', {
          clanid: clanid,
          gid: gid
        });
        if (!clanid) {
          debug('未能提取到clanid', {
            appId: appId,
            viewId: viewId
          });
          logStatus.error(`Error:${data.statusText}(${data.status})`);
          return {};
        }
        logStatus.success();
        debug('获取公告参数成功', {
          clanId: clanid,
          gid: gid
        });
        return {
          clanId: clanid,
          gid: gid
        };
      } catch (error) {
        debug('获取公告参数时发生异常', {
          error: error,
          appId: appId,
          viewId: viewId
        });
        throwError(error, 'SteamWeb.likeAnnouncement');
        return {};
      }
    }
    async likeAnnouncement(id) {
      try {
        debug('开始点赞公告', {
          id: id
        });
        const [appId, viewId] = id.split('/');
        if (!(appId && viewId)) {
          echoLog({
            before: '[Web]'
          }).error(`${I18n('missParams')}(id)`);
          return false;
        }
        const {clanId: clanId, gid: gid} = await this.#getAnnouncementParams(appId, viewId);
        if (!clanId) {
          return false;
        }
        const logStatus = echoLog({
          type: 'likingAnnouncement',
          text: appId,
          id: viewId,
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://store.steampowered.com/updated/ajaxrateupdate/${gid || viewId}`,
          method: 'POST',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
            Host: 'store.steampowered.com',
            Origin: 'https://store.steampowered.com',
            Referer: `https://store.steampowered.com/news/app/${appId}/view/${viewId}`
          },
          data: $.param({
            sessionid: this.#auth.storeSessionID,
            voteup: 1,
            clanid: clanId,
            ajax: 1
          }),
          dataType: 'json'
        });
        if (result !== 'Success') {
          debug('点赞公告请求失败', {
            result: result
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200 || data.response.success !== 1) {
          debug('点赞公告失败', {
            status: data?.status
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('成功点赞公告', {
          id: id
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('点赞公告时发生错误', {
          error: error,
          id: id
        });
        throwError(error, 'SteamWeb.likeAnnouncement');
        return false;
      }
    }
    async #appid2subid(id) {
      try {
        debug('开始将AppId转换为SubId', {
          id: id
        });
        const logStatus = echoLog({
          type: 'gettingSubid',
          text: id,
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://store.steampowered.com/app/${id}`,
          method: 'GET',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          }
        });
        debug('获取App页面请求结果', {
          result: result,
          statusText: statusText,
          status: status
        });
        if (result !== 'Success') {
          debug('获取App页面请求失败', {
            result: result
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('获取App页面响应状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        if (data.responseText.includes('ds_owned_flag ds_flag') || data.responseText.includes('class="already_in_library"')) {
          debug('App已拥有', {
            id: id
          });
          logStatus.success(I18n('owned'));
          return true;
        }
        if (this.#area === 'CN' && data.responseText.includes('id="error_box"')) {
          debug('地区锁定,尝试更换地区', {
            area: this.#area
          });
          logStatus.warning(I18n('changeAreaNotice'));
          const result = await this.#changeArea();
          if (!result || result === 'CN' || result === 'skip') {
            debug('更换地区失败或未更换', {
              result: result
            });
            return false;
          }
          return await this.#appid2subid(id);
        }
        let subid = data.responseText.match(/name="subid" value="([\d]+?)"/)?.[1];
        debug('正则提取subid结果1', {
          subid: subid
        });
        if (subid) {
          logStatus.success();
          return subid;
        }
        subid = data.responseText.match(/AddFreeLicense\(\s*(\d+)/)?.[1];
        debug('正则提取subid结果2', {
          subid: subid
        });
        if (subid) {
          logStatus.success();
          return subid;
        }
        debug('未能提取到subid', {
          id: id
        });
        logStatus.error(`Error:${I18n('noSubid')}`);
        return false;
      } catch (error) {
        debug('AppId转SubId时发生异常', {
          error: error,
          id: id
        });
        throwError(error, 'SteamWeb.appid2subid');
        return false;
      }
    }
    async #getLicenses() {
      try {
        debug('开始获取Steam用户许可证信息');
        const logStatus = echoLog({
          text: I18n('gettingLicenses'),
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://store.steampowered.com/dynamicstore/userdata/?t=${(new Date).getTime()}`,
          method: 'GET',
          responseType: 'json'
        });
        debug('获取许可证请求结果', {
          result: result,
          statusText: statusText,
          status: status
        });
        if (result !== 'Success') {
          debug('获取许可证请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('获取许可证响应状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('获取到的许可证列表', {
          licenses: data.response?.rgOwnedPackages
        });
        logStatus.remove();
        return data.response?.rgOwnedPackages;
      } catch (error) {
        debug('获取许可证时发生异常', {
          error: error
        });
        throwError(error, 'SteamWeb.getLicenses');
        return false;
      }
    }
    async addLicense(id) {
      try {
        debug('开始添加许可证', {
          id: id
        });
        const [type, ids] = id.split('-');
        debug('解析许可证ID', {
          type: type,
          ids: ids
        });
        if (type !== 'appid' && type !== 'subid') {
          debug('无效的许可证类型', {
            type: type
          });
          return false;
        }
        if (type === 'appid') {
          debug('处理appid类型许可证', {
            ids: ids
          });
          const subid = await this.#appid2subid(ids);
          debug('appid转换为subid结果', {
            appid: ids,
            subid: subid
          });
          if (!subid) {
            debug('appid转换失败', {
              appid: ids
            });
            return false;
          }
          if (subid === true) {
            debug('appid已拥有', {
              appid: ids
            });
            return true;
          }
          const logStatus = echoLog({
            type: 'addingFreeLicense',
            text: ids,
            before: '[Web]'
          });
          debug('开始添加免费许可证', {
            subid: subid
          });
          if (!await this.#addFreeLicense(subid, logStatus)) {
            debug('添加免费许可证失败', {
              subid: subid
            });
            return false;
          }
          const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
            url: `https://store.steampowered.com/app/${ids}`,
            method: 'GET'
          });
          debug('验证许可证添加状态', {
            result: result,
            status: status,
            statusText: statusText
          });
          if (result !== 'Success') {
            debug('验证请求失败', {
              result: result,
              status: status,
              statusText: statusText
            });
            logStatus.error(`${result}:${statusText}(${status})`);
            return false;
          }
          if (data?.status !== 200) {
            debug('验证响应状态错误', {
              status: data?.status,
              statusText: data?.statusText
            });
            logStatus.error(`Error:${data?.statusText}(${data?.status})`);
            return false;
          }
          if (!data.responseText.includes('ds_owned_flag ds_flag') && !data.responseText.includes('class="already_in_library"')) {
            debug('未找到游戏拥有标记', {
              status: data.status,
              statusText: data.statusText
            });
            logStatus.error(`Error:${data.statusText}(${data.status})`);
            return false;
          }
          debug('appid许可证添加成功', {
            appid: ids
          });
          logStatus.success();
          return true;
        }
        if (this.#area === 'CN') {
          debug('当前区域为CN,尝试更改区域', {
            currentArea: this.#area
          });
          echoLog({
            before: '[Web]'
          }).success(I18n('tryChangeAreaNotice'));
          await this.#changeArea();
        }
        const logStatusArr = {};
        const idsArr = ids.split(',');
        debug('处理subid类型许可证', {
          idsArr: idsArr
        });
        for (const subid of idsArr) {
          debug('开始处理单个subid', {
            subid: subid
          });
          const logStatus = echoLog({
            type: 'addingFreeLicense',
            text: subid,
            before: '[Web]'
          });
          if (!await this.#addFreeLicense(subid, logStatus)) {
            debug('添加subid许可证失败', {
              subid: subid
            });
            return false;
          }
          logStatusArr[subid] = logStatus;
        }
        const licenses = await this.#getLicenses();
        debug('获取许可证列表', {
          licenses: licenses
        });
        if (!licenses) {
          debug('获取许可证列表失败');
          return false;
        }
        for (const subid of idsArr) {
          const hasLicense = licenses.includes(parseInt(subid, 10));
          debug('验证许可证添加状态', {
            subid: subid,
            hasLicense: hasLicense
          });
          if (hasLicense) {
            logStatusArr[subid].success();
          } else {
            logStatusArr[subid].error();
          }
        }
        debug('所有subid许可证处理完成', {
          idsArr: idsArr
        });
        return true;
      } catch (error) {
        debug('添加许可证过程发生错误', {
          error: error,
          id: id
        });
        throwError(error, 'SteamWeb.addLicense');
        return false;
      }
    }
    async #addFreeLicense(id, logStatusPre) {
      try {
        debug('开始添加免费Steam游戏许可证', {
          id: id
        });
        const logStatus = logStatusPre || echoLog({
          type: 'addingFreeLicenseSubid',
          text: id,
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://store.steampowered.com/freelicense/addfreelicense/${id}`,
          method: 'POST',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
            Host: 'store.steampowered.com',
            Origin: 'https://store.steampowered.com',
            Referer: 'https://store.steampowered.com/account/licenses/'
          },
          data: $.param({
            ajax: true,
            sessionid: this.#auth.storeSessionID
          }),
          dataType: 'json'
        });
        debug('添加免费许可证请求结果', {
          result: result,
          statusText: statusText,
          status: status
        });
        if (result !== 'Success') {
          debug('添加免费许可证请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('添加免费许可证响应状态错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        if (this.#area === 'CN' && data.responseText.includes('id="error_box"')) {
          debug('地区锁定,尝试更换地区', {
            area: this.#area
          });
          logStatus.warning(I18n('changeAreaNotice'));
          const result = await this.#changeArea();
          if (!result || [ 'CN', 'skip' ].includes(result)) {
            debug('更换地区失败或未更换', {
              result: result
            });
            return false;
          }
          return await this.#addFreeLicense(id);
        }
        debug('成功添加免费许可证', {
          id: id
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('添加免费许可证时发生异常', {
          error: error,
          id: id
        });
        throwError(error, 'SteamWeb.addFreeLicense');
        return false;
      }
    }
    async requestPlayTestAccess(id) {
      debug('开始请求游戏试玩权限', {
        id: id
      });
      try {
        debug('开始请求游戏试玩权限', {
          id: id
        });
        const logStatus = echoLog({
          type: 'requestingPlayTestAccess',
          text: id,
          before: '[Web]'
        });
        debug('准备发送试玩权限请求', {
          id: id
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://store.steampowered.com/ajaxrequestplaytestaccess/${id}`,
          method: 'POST',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
            Host: 'store.steampowered.com',
            Origin: 'https://store.steampowered.com',
            Referer: `https://store.steampowered.com/app/${id}`
          },
          data: $.param({
            sessionid: this.#auth.storeSessionID
          }),
          dataType: 'json'
        });
        debug('收到试玩权限请求响应', {
          result: result,
          status: status,
          statusText: statusText,
          responseData: data
        });
        if (result !== 'Success') {
          debug('请求失败', {
            result: result,
            status: status,
            statusText: statusText
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200 || data?.response?.success !== 1) {
          debug('响应状态错误', {
            status: data?.status,
            statusText: data?.statusText,
            success: data?.response?.success
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('成功请求游戏试玩权限', {
          id: id
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('请求游戏试玩权限时发生错误', {
          error: error,
          id: id
        });
        throwError(error, 'SteamWeb.requestPlayTestAccess');
        return false;
      }
    }
    async resetArea() {
      try {
        debug('检查区域设置状态', {
          currentArea: this.#area,
          oldArea: this.#oldArea,
          needReset: Boolean(this.#oldArea && this.#area !== this.#oldArea)
        });
        if (this.#oldArea && this.#area !== this.#oldArea) {
          debug('需要重置区域', {
            fromArea: this.#area,
            toArea: this.#oldArea
          });
          echoLog({
            before: '[Web]'
          }).warning(I18n('steamFinishNotice') + this.#oldArea);
          const changeResult = await this.#changeArea(this.#oldArea);
          debug('区域重置结果', {
            success: changeResult,
            targetArea: this.#oldArea
          });
        } else {
          debug('无需重置区域', {
            currentArea: this.#area,
            oldArea: this.#oldArea
          });
        }
        debug('区域重置流程完成');
        return true;
      } catch (error) {
        debug('重置区域时发生错误', {
          error: error
        });
        throwError(error, 'SteamWeb.resetArea');
        return false;
      }
    }
    #setCache(type, name, id) {
      try {
        debug('开始设置缓存', {
          type: type,
          name: name,
          id: id
        });
        this.#cache[type][name] = id;
        GM_setValue('steamCache', this.#cache);
        debug('设置缓存成功', {
          type: type,
          name: name,
          id: id
        });
      } catch (error) {
        debug('设置缓存时发生异常', {
          error: error,
          type: type,
          name: name,
          id: id
        });
        throwError(error, 'SteamWeb.setCache');
      }
    }
  }
  class Steam extends Social {
    tasks;
    whiteList;
    #cache={
      ...{
        group: {},
        officialGroup: {},
        forum: {},
        workshop: {},
        curator: {}
      },
      ...GM_getValue('steamCache')
    };
    #TaskExecutor=[];
    constructor() {
      super();
      debug('初始化Steam实例');
      const defaultTasksTemplate = {
        groups: [],
        officialGroups: [],
        wishlists: [],
        follows: [],
        forums: [],
        workshops: [],
        workshopVotes: [],
        curators: [],
        curatorLikes: [],
        announcements: [],
        licenses: [],
        playtests: [],
        playTime: []
      };
      this.tasks = defaultTasksTemplate;
      this.whiteList = {
        ...defaultTasksTemplate,
        ...GM_getValue('whiteList')?.steam || {}
      };
      this.#TaskExecutor = this.#getTaskExecutionOrder(globalOptions.ASF.AsfEnabled, globalOptions.ASF.steamWeb, globalOptions.ASF.preferASF);
      debug('Steam实例初始化完成', {
        taskExecutorCount: this.#TaskExecutor.length
      });
    }
    async init(type = 'all') {
      try {
        debug('开始初始化Steam模块', {
          type: type
        });
        for (let i = 0; i < this.#TaskExecutor.length; i++) {
          debug(`初始化执行器 ${i + 1}/${this.#TaskExecutor.length}`);
          if (!await this.#TaskExecutor[i].init(type)) {
            debug(`执行器 ${i + 1} 初始化失败,移除该执行器`);
            this.#TaskExecutor.splice(i, 1);
          }
        }
        debug('Steam模块初始化完成', {
          remainingExecutors: this.#TaskExecutor.length
        });
        return this.#TaskExecutor.length > 0;
      } catch (error) {
        debug('Steam初始化发生错误', {
          error: error
        });
        throwError(error, 'Steam.init');
        return false;
      }
    }
    async #joinGroup(groupName) {
      try {
        debug('开始加入Steam组', {
          groupName: groupName
        });
        for (const taskExecutor of this.#TaskExecutor) {
          if (await taskExecutor.joinGroup(groupName)) {
            debug('成功加入Steam组', {
              groupName: groupName
            });
            this.tasks.groups = unique([ ...this.tasks.groups, groupName ]);
            return true;
          }
        }
        debug('加入Steam组失败', {
          groupName: groupName
        });
        return false;
      } catch (error) {
        debug('加入Steam组时发生错误', {
          error: error,
          groupName: groupName
        });
        throwError(error, 'Steam.joinGroup');
        return false;
      }
    }
    async #leaveGroup(groupName) {
      try {
        debug('开始退出Steam组', {
          groupName: groupName
        });
        if (this.whiteList.groups.includes(groupName)) {
          debug('Steam组在白名单中,跳过退出', {
            groupName: groupName
          });
          echoLog({
            type: 'whiteList',
            text: 'Steam.leaveGroup',
            id: groupName
          });
          return true;
        }
        for (const taskExecutor of this.#TaskExecutor) {
          if (await taskExecutor.leaveGroup(groupName)) {
            debug('成功退出Steam组', {
              groupName: groupName
            });
            return true;
          }
        }
        debug('退出Steam组失败', {
          groupName: groupName
        });
        return false;
      } catch (error) {
        debug('退出Steam组时发生错误', {
          error: error,
          groupName: groupName
        });
        throwError(error, 'Steam.leaveGroup');
        return false;
      }
    }
    async #joinOfficialGroup(gameId) {
      try {
        debug('开始加入Steam官方组', {
          gameId: gameId
        });
        for (const taskExecutor of this.#TaskExecutor) {
          if (await taskExecutor.joinOfficialGroup(gameId)) {
            debug('成功加入Steam官方组', {
              gameId: gameId
            });
            return true;
          }
        }
        debug('加入Steam官方组失败', {
          gameId: gameId
        });
        return false;
      } catch (error) {
        debug('加入Steam官方组时发生错误', {
          error: error,
          gameId: gameId
        });
        throwError(error, 'Steam.joinOfficialGroup');
        return false;
      }
    }
    async #leaveOfficialGroup(gameId) {
      try {
        debug('开始退出Steam官方组', {
          gameId: gameId
        });
        if (this.whiteList.officialGroups.includes(gameId)) {
          debug('Steam官方组在白名单中,跳过退出', {
            gameId: gameId
          });
          echoLog({
            type: 'whiteList',
            text: 'Steam.leaveOfficialGroup',
            id: gameId
          });
          return true;
        }
        for (const taskExecutor of this.#TaskExecutor) {
          if (await taskExecutor.leaveOfficialGroup(gameId)) {
            debug('成功退出Steam官方组', {
              gameId: gameId
            });
            this.tasks.officialGroups = unique([ ...this.tasks.officialGroups, gameId ]);
            return true;
          }
        }
        debug('退出Steam官方组失败', {
          gameId: gameId
        });
        return false;
      } catch (error) {
        debug('退出Steam官方组时发生错误', {
          error: error,
          gameId: gameId
        });
        throwError(error, 'Steam.leaveOfficialGroup');
        return false;
      }
    }
    async #addToWishlist(gameId) {
      try {
        debug('开始添加游戏到愿望单', {
          gameId: gameId
        });
        for (const taskExecutor of this.#TaskExecutor) {
          if (await taskExecutor.addToWishlist(gameId)) {
            debug('成功添加游戏到愿望单', {
              gameId: gameId
            });
            this.tasks.wishlists = unique([ ...this.tasks.wishlists, gameId ]);
            return true;
          }
        }
        debug('添加游戏到愿望单失败', {
          gameId: gameId
        });
        return false;
      } catch (error) {
        debug('添加游戏到愿望单时发生错误', {
          error: error,
          gameId: gameId
        });
        throwError(error, 'Steam.addToWishlist');
        return false;
      }
    }
    async #removeFromWishlist(gameId) {
      try {
        debug('开始从愿望单移除游戏', {
          gameId: gameId
        });
        if (this.whiteList.wishlists.includes(gameId)) {
          debug('游戏在愿望单白名单中,跳过移除', {
            gameId: gameId
          });
          echoLog({
            type: 'whiteList',
            text: 'Steam.removeFromWishlist',
            id: gameId
          });
          return true;
        }
        for (const taskExecutor of this.#TaskExecutor) {
          if (await taskExecutor.removeFromWishlist(gameId)) {
            debug('成功从愿望单移除游戏', {
              gameId: gameId
            });
            return true;
          }
        }
        debug('从愿望单移除游戏失败', {
          gameId: gameId
        });
        return false;
      } catch (error) {
        debug('从愿望单移除游戏时发生错误', {
          error: error,
          gameId: gameId
        });
        throwError(error, 'Steam.removeFromWishlist');
        return false;
      }
    }
    async #toggleFollowGame(gameId, doTask) {
      try {
        debug('开始处理游戏关注状态', {
          gameId: gameId,
          doTask: doTask
        });
        if (!doTask && this.whiteList.follows.includes(gameId)) {
          debug('游戏在关注白名单中,跳过取关', {
            gameId: gameId
          });
          echoLog({
            type: 'whiteList',
            text: 'Steam.unfollowGame',
            id: gameId
          });
          return true;
        }
        for (const taskExecutor of this.#TaskExecutor) {
          if (await taskExecutor.toggleFollowGame(gameId, doTask)) {
            if (doTask) {
              debug('成功关注游戏', {
                gameId: gameId
              });
              this.tasks.follows = unique([ ...this.tasks.follows, gameId ]);
            } else {
              debug('成功取关游戏', {
                gameId: gameId
              });
            }
            return true;
          }
        }
        debug('处理游戏关注状态失败', {
          gameId: gameId,
          doTask: doTask
        });
        return false;
      } catch (error) {
        debug('处理游戏关注状态时发生错误', {
          error: error,
          gameId: gameId,
          doTask: doTask
        });
        throwError(error, 'Steam.toggleFollowGame');
        return false;
      }
    }
    async #toggleForum(gameId, doTask = true) {
      try {
        debug('开始处理论坛订阅状态', {
          gameId: gameId,
          doTask: doTask
        });
        if (!doTask && this.whiteList.forums.includes(gameId)) {
          debug('论坛在白名单中,跳过取消订阅', {
            gameId: gameId
          });
          echoLog({
            type: 'whiteList',
            text: 'Steam.unsubscribeForum',
            id: gameId
          });
          return true;
        }
        for (const taskExecutor of this.#TaskExecutor) {
          if (await taskExecutor.toggleForum(gameId, doTask)) {
            if (doTask) {
              debug('成功订阅论坛', {
                gameId: gameId
              });
              this.tasks.forums = unique([ ...this.tasks.forums, gameId ]);
            } else {
              debug('成功取消订阅论坛', {
                gameId: gameId
              });
            }
            return true;
          }
        }
        debug('处理论坛订阅状态失败', {
          gameId: gameId,
          doTask: doTask
        });
        return false;
      } catch (error) {
        debug('处理论坛订阅状态时发生错误', {
          error: error,
          gameId: gameId,
          doTask: doTask
        });
        throwError(error, 'Steam.toggleForum');
        return true;
      }
    }
    async #toggleFavoriteWorkshop(id, doTask = true) {
      try {
        debug('开始处理创意工坊收藏状态', {
          id: id,
          doTask: doTask
        });
        if (!doTask && this.whiteList.workshops.includes(id)) {
          debug('创意工坊物品在白名单中,跳过取消收藏', {
            id: id
          });
          echoLog({
            type: 'whiteList',
            text: 'Steam.unfavoriteWorkshop',
            id: id
          });
          return true;
        }
        for (const taskExecutor of this.#TaskExecutor) {
          if (await taskExecutor.toggleFavoriteWorkshop(id)) {
            if (doTask) {
              debug('成功收藏创意工坊物品', {
                id: id
              });
              this.tasks.workshops = unique([ ...this.tasks.workshops, id ]);
            } else {
              debug('成功取消收藏创意工坊物品', {
                id: id
              });
            }
            return true;
          }
        }
        debug('处理创意工坊收藏状态失败', {
          id: id,
          doTask: doTask
        });
        return false;
      } catch (error) {
        debug('处理创意工坊收藏状态时发生错误', {
          error: error,
          id: id,
          doTask: doTask
        });
        throwError(error, 'Steam.toggleFavoriteWorkshop');
        return false;
      }
    }
    async #voteUpWorkshop(id) {
      try {
        debug('开始点赞创意工坊物品', {
          id: id
        });
        for (const taskExecutor of this.#TaskExecutor) {
          if (await taskExecutor.voteUpWorkshop(id)) {
            debug('成功点赞创意工坊物品', {
              id: id
            });
            return true;
          }
        }
        debug('点赞创意工坊物品失败', {
          id: id
        });
        return false;
      } catch (error) {
        debug('点赞创意工坊物品时发生错误', {
          error: error,
          id: id
        });
        throwError(error, 'Steam.voteupWorkshop');
        return true;
      }
    }
    async #toggleCurator(curatorId, doTask = true) {
      try {
        debug('开始处理鉴赏家关注状态', {
          curatorId: curatorId,
          doTask: doTask
        });
        if (!doTask && this.whiteList.curators.includes(curatorId)) {
          debug('鉴赏家在白名单中,跳过取关', {
            curatorId: curatorId
          });
          echoLog({
            type: 'whiteList',
            text: 'Steam.unfollowCurator',
            id: curatorId
          });
          return true;
        }
        for (const taskExecutor of this.#TaskExecutor) {
          if (await taskExecutor.toggleCurator(curatorId, doTask)) {
            if (doTask) {
              debug('成功关注鉴赏家', {
                curatorId: curatorId
              });
              this.tasks.curators = unique([ ...this.tasks.curators, curatorId ]);
            } else {
              debug('成功取关鉴赏家', {
                curatorId: curatorId
              });
            }
            return true;
          }
        }
        debug('处理鉴赏家关注状态失败', {
          curatorId: curatorId,
          doTask: doTask
        });
        return false;
      } catch (error) {
        debug('处理鉴赏家关注状态时发生错误', {
          error: error,
          curatorId: curatorId,
          doTask: doTask
        });
        throwError(error, 'Steam.toggleCurator');
        return false;
      }
    }
    async getCuratorId(path, name) {
      try {
        debug('开始获取鉴赏家ID', {
          path: path,
          name: name
        });
        const logStatus = echoLog({
          type: 'gettingCuratorId',
          text: `${path}/${name}`,
          before: '[Web]'
        });
        const curatorId = this.#cache.curator[`${path}/${name}`];
        if (curatorId) {
          debug('从缓存获取到鉴赏家ID', {
            path: path,
            name: name,
            curatorId: curatorId
          });
          logStatus.success();
          return curatorId;
        }
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://store.steampowered.com/${path}/${name}`,
          method: 'GET',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8'
          }
        });
        if (result === 'Success') {
          if (data?.status === 200) {
            const curatorId = data.responseText.match(/g_pagingData.*?"clanid":([\d]+)/)?.[1];
            if (curatorId) {
              debug('成功获取鉴赏家ID', {
                path: path,
                name: name,
                curatorId: curatorId
              });
              this.#setCache('curator', `${path}/${name}`, curatorId);
              logStatus.success();
              return curatorId;
            }
            debug('未找到鉴赏家ID', {
              path: path,
              name: name,
              status: data.status
            });
            logStatus.error(`Error:${data.statusText}(${data.status})`);
            return false;
          }
          debug('获取鉴赏家页面失败', {
            path: path,
            name: name,
            status: data?.status
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('请求鉴赏家页面失败', {
          path: path,
          name: name,
          result: result,
          status: status
        });
        logStatus.error(`${result}:${statusText}(${status})`);
        return false;
      } catch (error) {
        debug('获取鉴赏家ID时发生错误', {
          error: error,
          path: path,
          name: name
        });
        throwError(error, 'SteamWeb.getCuratorID');
        return false;
      }
    }
    async #toggleCuratorLike(link, doTask = true) {
      try {
        debug('开始处理鉴赏家点赞状态', {
          link: link,
          doTask: doTask
        });
        const [path, name] = link.split('/');
        if (!(path && name)) {
          debug('无效的鉴赏家链接', {
            link: link
          });
          echoLog({
            text: I18n('errorLink', link),
            before: '[Web]'
          });
          return false;
        }
        const curatorId = await this.getCuratorId(path, name);
        if (curatorId) {
          debug('获取到鉴赏家ID,开始处理点赞', {
            curatorId: curatorId,
            doTask: doTask
          });
          return await this.#toggleCurator(curatorId, doTask);
        }
        debug('未获取到鉴赏家ID', {
          link: link
        });
        return false;
      } catch (error) {
        debug('处理鉴赏家点赞状态时发生错误', {
          error: error,
          link: link,
          doTask: doTask
        });
        throwError(error, 'Steam.toggleCuratorLike');
        return false;
      }
    }
    async #likeAnnouncement(id) {
      try {
        debug('开始点赞公告', {
          id: id
        });
        for (const taskExecutor of this.#TaskExecutor) {
          if (await taskExecutor.likeAnnouncement(id)) {
            debug('成功点赞公告', {
              id: id
            });
            return true;
          }
        }
        debug('点赞公告失败', {
          id: id
        });
        return false;
      } catch (error) {
        debug('点赞公告时发生错误', {
          error: error,
          id: id
        });
        throwError(error, 'Steam.likeAnnouncement');
        return false;
      }
    }
    async #addLicense(id) {
      try {
        debug('开始添加许可证', {
          id: id
        });
        for (const taskExecutor of this.#TaskExecutor) {
          if (await taskExecutor.addLicense(id)) {
            debug('成功添加许可证', {
              id: id
            });
            return true;
          }
        }
        debug('添加许可证失败', {
          id: id
        });
        return false;
      } catch (error) {
        debug('添加许可证时发生错误', {
          error: error,
          id: id
        });
        throwError(error, 'Steam.addLicense');
        return false;
      }
    }
    async #requestPlayTestAccess(id) {
      try {
        debug('开始请求游戏试玩权限', {
          id: id
        });
        for (const taskExecutor of this.#TaskExecutor) {
          if (await taskExecutor.requestPlayTestAccess(id)) {
            debug('成功请求游戏试玩权限', {
              id: id
            });
            return true;
          }
        }
        debug('请求游戏试玩权限失败', {
          id: id
        });
        return false;
      } catch (error) {
        debug('请求游戏试玩权限时发生错误', {
          error: error,
          id: id
        });
        throwError(error, 'Steam.requestPlayTestAccess');
        return false;
      }
    }
    async #getDemoAppid(id) {
      try {
        debug('开始获取游戏试玩ID', {
          id: id
        });
        const logStatus = echoLog({
          type: 'gettingDemoAppid',
          text: id,
          before: '[Web]'
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://store.steampowered.com/app/${id}`,
          method: 'GET',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
            Host: 'store.steampowered.com',
            Origin: 'https://store.steampowered.com',
            Referer: `https://store.steampowered.com/app/${id}`
          }
        });
        if (result === 'Success') {
          if (data?.status === 200) {
            const demoAppid = data.responseText.match(/steam:\/\/(install|run)\/(\d+)/)?.[2];
            debug('成功获取游戏试玩ID', {
              id: id,
              demoAppid: demoAppid
            });
            logStatus.success();
            return demoAppid || false;
          }
          debug('获取游戏页面失败', {
            id: id,
            status: data?.status
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('请求游戏页面失败', {
          id: id,
          result: result,
          status: status
        });
        logStatus.error(`${result}:${statusText}(${status})`);
        return false;
      } catch (error) {
        debug('获取游戏试玩ID时发生错误', {
          error: error,
          id: id
        });
        throwError(error, 'Steam.getDemoAppid');
        return false;
      }
    }
    async #playGames(ids, playTime, doTask = true) {
      try {
        debug('开始处理游戏挂时长', {
          ids: ids,
          playTime: playTime,
          doTask: doTask
        });
        if (playTime <= 0) {
          debug('游戏时长小于等于0,跳过挂时长');
          return true;
        }
        const asf = this.#TaskExecutor.find((e => e instanceof SteamASF));
        if (!asf) {
          debug('未找到ASF实例');
          echoLog({}).warning(I18n('noASFInstance'));
          return false;
        }
        if (!doTask) {
          debug('停止挂时长');
          return await asf.stopPlayGames();
        }
        const idsArr = await Promise.all(ids.split(',').map((async id => {
          try {
            const demoAppid = await this.#getDemoAppid(id);
            return demoAppid ? `${id},${demoAppid}` : id;
          } catch (error) {
            debug('获取游戏试玩ID失败', {
              error: error,
              id: id
            });
            return id;
          }
        })));
        const uniqueIds = unique(idsArr.join(',').split(','));
        debug('处理后的游戏ID列表', {
          uniqueIds: uniqueIds
        });
        debug('开始尝试入库游戏', {
          uniqueIds: uniqueIds
        });
        await Promise.all(uniqueIds.map((async id => {
          for (const taskExecutor of this.#TaskExecutor) {
            if (await taskExecutor.addLicense(`appid-${id}`)) {
              debug('成功入库游戏', {
                id: id
              });
              return true;
            }
          }
          return false;
        })));
        await asf.playGames(uniqueIds.join(','));
        const status = await asf.checkPlayStatus(uniqueIds.join(','));
        if (status !== true) {
          await delay(3e3);
          await asf.playGames(uniqueIds.join(','));
          const status = await asf.checkPlayStatus(uniqueIds.join(','));
          if (!status) {
            debug('启动游戏失败');
            return false;
          }
        }
        const stopPlayTime = Date.now() + (playTime + 10) * 60 * 1e3;
        const stopPlayTimeOld = GM_getValue('stopPlayTime', 0) || 0;
        GM_setValue('stopPlayTime', Math.max(stopPlayTime, stopPlayTimeOld));
        const playedGames = GM_getValue('playedGames', []) || [];
        GM_setValue('playedGames', unique([ ...playedGames, ...uniqueIds ]));
        const taskLink = GM_getValue('taskLink', []) || [];
        GM_setValue('taskLink', unique([ ...taskLink, window.location.href ]));
        debug('游戏挂时长状态更新完成');
        return true;
      } catch (error) {
        debug('处理游戏挂时长时发生错误', {
          error: error,
          ids: ids,
          playTime: playTime
        });
        throwError(error, 'Steam.playGames');
        return false;
      }
    }
    async toggle({doTask: doTask = true, groupLinks: groupLinks = [], officialGroupLinks: officialGroupLinks = [], wishlistLinks: wishlistLinks = [], followLinks: followLinks = [], forumLinks: forumLinks = [], workshopLinks: workshopLinks = [], workshopVoteLinks: workshopVoteLinks = [], curatorLinks: curatorLinks = [], curatorLikeLinks: curatorLikeLinks = [], announcementLinks: announcementLinks = [], licenseLinks: licenseLinks = [], playtestLinks: playtestLinks = [], playTimeLinks: playTimeLinks = []}) {
      try {
        debug('开始处理Steam任务', {
          doTask: doTask,
          linksCount: {
            groups: groupLinks.length,
            officialGroups: officialGroupLinks.length,
            wishlists: wishlistLinks.length,
            follows: followLinks.length,
            forums: forumLinks.length,
            workshops: workshopLinks.length,
            workshopVotes: workshopVoteLinks.length,
            curators: curatorLinks.length,
            curatorLikes: curatorLikeLinks.length,
            announcements: announcementLinks.length,
            licenses: licenseLinks.length,
            playtests: playtestLinks.length,
            playTime: playTimeLinks.length
          }
        });
        const allLinks = [ ...groupLinks, ...officialGroupLinks, ...forumLinks, ...workshopLinks, ...workshopVoteLinks, ...wishlistLinks, ...followLinks, ...curatorLinks, ...curatorLikeLinks, ...announcementLinks, ...licenseLinks, ...playtestLinks, ...playTimeLinks ];
        if (allLinks.length > 0 && this.#TaskExecutor.length === 0) {
          debug('Steam模块未初始化');
          echoLog({
            text: I18n('needInit')
          });
          return false;
        }
        const tasks = [];
        if (this.shouldProcessTask('groups', doTask)) {
          debug('开始处理群组任务');
          const realGroups = this.getRealParams('groups', groupLinks, doTask, (link => link.match(/groups\/(.+)\/?/)?.[1]?.split('/')?.[0]));
          debug('处理后的群组列表', {
            count: realGroups.length,
            groups: realGroups
          });
          for (const group of realGroups) {
            tasks.push(doTask ? this.#joinGroup(group) : this.#leaveGroup(group));
            await delay(1e3);
          }
        }
        if (this.shouldProcessTask('officialGroups', doTask)) {
          const realOfficialGroups = this.getRealParams('officialGroups', officialGroupLinks, doTask, (link => link.match(/games\/(.+)\/?/)?.[1]));
          for (const officialGroup of realOfficialGroups) {
            tasks.push(doTask ? this.#joinOfficialGroup(officialGroup) : this.#leaveOfficialGroup(officialGroup));
            await delay(1e3);
          }
        }
        if (this.shouldProcessTask('wishlists', doTask)) {
          const realWishlists = this.getRealParams('wishlists', wishlistLinks, doTask, (link => link.match(/app\/([\d]+)/)?.[1]));
          for (const game of realWishlists) {
            tasks.push(doTask ? this.#addToWishlist(game) : this.#removeFromWishlist(game));
            await delay(1e3);
          }
        }
        if (this.shouldProcessTask('follows', doTask)) {
          const realFollows = this.getRealParams('follows', followLinks, doTask, (link => link.match(/app\/([\d]+)/)?.[1]));
          for (const game of realFollows) {
            tasks.push(this.#toggleFollowGame(game, doTask));
            await delay(1e3);
          }
        }
        if (this.shouldProcessTask('playTime', doTask)) {
          const realGames = this.getRealParams('playTime', playTimeLinks, doTask, (link => `${link.split('-')[0]}-${link.match(/app\/([\d]+)/)?.[1] || ''}`));
          if (realGames.length > 0) {
            const maxTime = Math.max(...realGames.map((info => parseInt(info.split('-')[0], 10) || 0)));
            const games = realGames.filter((info => {
              const [time, game] = info.split('-');
              return (parseInt(time, 10) || 0) > 0 && game;
            })).map((info => info.split('-')[1]));
            tasks.push(this.#playGames(games.join(','), maxTime, doTask));
            await delay(1e3);
          }
        }
        if (this.shouldProcessTask('forums', doTask)) {
          const realForums = this.getRealParams('forums', forumLinks, doTask, (link => link.match(/app\/([\d]+)/)?.[1]));
          for (const forum of realForums) {
            tasks.push(this.#toggleForum(forum, doTask));
            await delay(1e3);
          }
        }
        if (this.shouldProcessTask('workshops', doTask)) {
          const realWorkshops = this.getRealParams('workshops', workshopLinks, doTask, (link => link.match(/\?id=([\d]+)/)?.[1]));
          for (const workshop of realWorkshops) {
            tasks.push(this.#toggleFavoriteWorkshop(workshop, doTask));
            await delay(1e3);
          }
        }
        if (doTask && globalOptions.doTask.steam.workshopVotes) {
          const realworkshopVotes = this.getRealParams('workshopVotes', workshopVoteLinks, doTask, (link => link.match(/\?id=([\d]+)/)?.[1]));
          for (const workshop of realworkshopVotes) {
            tasks.push(this.#voteUpWorkshop(workshop));
            await delay(1e3);
          }
        }
        if (this.shouldProcessTask('curators', doTask)) {
          const realCurators = this.getRealParams('curators', curatorLinks, doTask, (link => link.match(/curator\/([\d]+)/)?.[1]));
          const realCuratorLikes = this.getRealParams('curatorLikes', curatorLikeLinks, doTask, (link => link.match(/https?:\/\/store\.steampowered\.com\/(.*?)\/([^/?]+)/)?.slice(1, 3).join('/')));
          for (const curator of realCurators) {
            tasks.push(this.#toggleCurator(curator, doTask));
            await delay(1e3);
          }
          for (const curatorLike of realCuratorLikes) {
            tasks.push(this.#toggleCuratorLike(curatorLike, doTask));
            await delay(1e3);
          }
        }
        if (doTask && globalOptions.doTask.steam.announcements) {
          const realAnnouncements = this.getRealParams('announcements', announcementLinks, doTask, (link => {
            if (link.includes('store.steampowered.com')) {
              return link.match(/store\.steampowered\.com\/news\/app\/([\d]+)\/view\/([\d]+)/)?.slice(1, 3).join('/');
            }
            return link.match(/steamcommunity\.com\/games\/([\d]+)\/announcements\/detail\/([\d]+)/)?.slice(1, 3).join('/');
          }));
          for (const id of realAnnouncements) {
            tasks.push(this.#likeAnnouncement(id));
            await delay(1e3);
          }
        }
        if (doTask && globalOptions.doTask.steam.licenses && licenseLinks.length > 0) {
          for (const ids of licenseLinks) {
            const [type, idsStr] = ids.split('-');
            const idsArr = idsStr.split(',');
            for (const id of idsArr) {
              tasks.push(this.#addLicense(`${type}-${id}`));
              await delay(1e3);
            }
          }
        }
        if (doTask && globalOptions.doTask.steam.playtests) {
          const realPlaytests = this.getRealParams('playtests', playtestLinks, doTask, (link => link.match(/app\/([\d]+)/)?.[1]));
          for (const id of realPlaytests) {
            tasks.push(this.#requestPlayTestAccess(id));
            await delay(1e3);
          }
        }
        debug('开始执行所有任务');
        const results = await Promise.all(tasks);
        this.#TaskExecutor.find((e => e instanceof SteamWeb))?.resetArea();
        debug('所有任务执行完成', {
          success: results.every((result => result))
        });
        return results.every((result => result));
      } catch (error) {
        debug('处理Steam任务时发生错误', {
          error: error
        });
        throwError(error, 'Steam.toggle');
        return false;
      }
    }
    shouldProcessTask(taskType, doTask) {
      debug('检查是否处理任务', {
        taskType: taskType,
        doTask: doTask
      });
      if (doTask) {
        const result = globalOptions.doTask.steam[taskType];
        debug('检查doTask配置', {
          taskType: taskType,
          result: result
        });
        return globalOptions.doTask.steam[taskType];
      }
      const undoTaskType = taskType;
      return undoTaskType in globalOptions.undoTask.steam && globalOptions.undoTask.steam[undoTaskType];
    }
    #setCache(type, name, id) {
      try {
        this.#cache[type][name] = id;
        GM_setValue('steamCache', this.#cache);
      } catch (error) {
        throwError(error, 'SteamWeb.setCache');
      }
    }
    #getTaskExecutionOrder(asfEnabled, steamWebEnabled, preferASF) {
      if (!asfEnabled) {
        return [ new SteamWeb ];
      }
      if (!steamWebEnabled) {
        return [ new SteamASF(globalOptions.ASF) ];
      }
      return preferASF ? [ new SteamASF(globalOptions.ASF), new SteamWeb ] : [ new SteamWeb, new SteamASF(globalOptions.ASF) ];
    }
  }
  class Website {
    undoneTasks;
    socialTasks;
    giveawayId;
    socialInitialized={
      discord: false,
      instagram: false,
      reddit: false,
      twitch: false,
      twitter: false,
      vk: false,
      youtube: false,
      steamStore: false,
      steamCommunity: false
    };
    initialized=false;
    steamTaskType={
      steamStore: false,
      steamCommunity: false
    };
    social={};
    async #bind(name, init) {
      try {
        debug('开始绑定社交媒体', {
          name: name
        });
        const result = await init;
        debug('绑定结果', {
          name: name,
          result: result
        });
        return {
          name: name,
          result: result
        };
      } catch (error) {
        debug('绑定失败', {
          name: name,
          error: error
        });
        throwError(error, 'Website.bind');
        return {
          name: name,
          result: false
        };
      }
    }
    async initSocial(action) {
      try {
        debug('开始初始化社交媒体', {
          action: action
        });
        const pro = [];
        const tasks = action === 'do' ? this.undoneTasks : this.socialTasks;
        if (tasks.discord) {
          const hasDiscord = Object.values(tasks.discord).reduce(((total, arr) => [ ...total, ...arr ])).length > 0;
          debug('检查 Discord 任务', {
            hasDiscord: hasDiscord
          });
          if (hasDiscord && (!this.socialInitialized.discord || !this.social.discord)) {
            debug('初始化 Discord');
            this.social.discord = new Discord;
            pro.push(this.#bind('discord', this.social.discord.init(action)));
          }
        }
        if (tasks.reddit) {
          const hasReddit = Object.values(tasks.reddit).reduce(((total, arr) => [ ...total, ...arr ])).length > 0;
          debug('检查 Reddit 任务', {
            hasReddit: hasReddit
          });
          if (hasReddit && (!this.socialInitialized.reddit || !this.social.reddit)) {
            debug('初始化 Reddit');
            this.social.reddit = new Reddit;
            pro.push(this.#bind('reddit', this.social.reddit.init()));
          }
        }
        if (tasks.twitch) {
          const hasTwitch = Object.values(tasks.twitch).reduce(((total, arr) => [ ...total, ...arr ])).length > 0;
          debug('检查 Twitch 任务', {
            hasTwitch: hasTwitch
          });
          if (hasTwitch && (!this.socialInitialized.twitch || !this.social.twitch)) {
            debug('初始化 Twitch');
            this.social.twitch = new Twitch;
            pro.push(this.#bind('twitch', this.social.twitch.init()));
          }
        }
        if (tasks.twitter) {
          const hasTwitter = Object.values(tasks.twitter).reduce(((total, arr) => [ ...total, ...arr ])).length > 0;
          debug('检查 Twitter 任务', {
            hasTwitter: hasTwitter
          });
          if (hasTwitter && (!this.socialInitialized.twitter || !this.social.twitter)) {
            debug('初始化 Twitter');
            this.social.twitter = new Twitter;
            pro.push(this.#bind('twitter', this.social.twitter.init()));
          }
        }
        if (tasks.vk) {
          const hasVk = Object.values(tasks.vk).reduce(((total, arr) => [ ...total, ...arr ])).length > 0;
          debug('检查 VK 任务', {
            hasVk: hasVk
          });
          if (hasVk && (!this.socialInitialized.vk || !this.social.vk)) {
            debug('初始化 VK');
            this.social.vk = new Vk;
            pro.push(this.#bind('vk', this.social.vk.init()));
          }
        }
        if (tasks.youtube) {
          const hasYoutube = Object.values(tasks.youtube).reduce(((total, arr) => [ ...total, ...arr ])).length > 0;
          debug('检查 YouTube 任务', {
            hasYoutube: hasYoutube
          });
          if (hasYoutube && (!this.socialInitialized.youtube || !this.social.youtube)) {
            debug('初始化 YouTube');
            this.social.youtube = new Youtube;
            pro.push(this.#bind('youtube', this.social.youtube.init()));
          }
        }
        if (tasks.steam) {
          const steamLength = Object.values(tasks.steam).reduce(((total, arr) => [ ...total, ...arr ])).length;
          debug('检查 Steam 任务', {
            steamLength: steamLength
          });
          if (steamLength > 0) {
            if (!this.social.steam) {
              debug('创建 Steam 实例');
              this.social.steam = new Steam;
            }
            const steamCommunityLength = Object.keys(tasks.steam).map((type => [ 'groupLinks', 'officialGroupLinks', 'forumLinks', 'workshopLinks', 'workshopVoteLinks' ].includes(type) ? tasks.steam?.[type]?.length || 0 : 0)).reduce(((total, number) => total + number), 0);
            debug('Steam 社区任务数量', {
              steamCommunityLength: steamCommunityLength
            });
            if (steamLength - steamCommunityLength > 0) {
              this.steamTaskType.steamStore = true;
              if (!this.socialInitialized.steamStore) {
                debug('初始化 Steam 商店');
                pro.push(this.#bind('steamStore', this.social.steam.init('store')));
              }
            }
            if (steamCommunityLength > 0) {
              if (!this.socialInitialized.steamCommunity) {
                this.steamTaskType.steamCommunity = true;
                debug('初始化 Steam 社区');
                pro.push(this.#bind('steamCommunity', this.social.steam.init('community')));
              }
            }
          }
        }
        if (tasks.links && tasks.links.length > 0) {
          debug('初始化链接访问', {
            linksCount: tasks.links.length
          });
          this.social.visitLink = visitLink;
        }
        debug('等待所有社交媒体初始化完成');
        return await Promise.all(pro).then((result => {
          let checked = true;
          for (const data of result) {
            if (data.result) {
              debug('社交媒体初始化成功', {
                name: data.name
              });
              this.socialInitialized[data.name] = data.result;
            } else {
              debug('社交媒体初始化失败', {
                name: data.name
              });
              checked = false;
            }
          }
          debug('社交媒体初始化完成', {
            allSuccess: checked
          });
          return checked;
        }));
      } catch (error) {
        debug('初始化社交媒体失败', {
          error: error
        });
        throwError(error, 'Website.initSocial');
        return false;
      }
    }
    uniqueTasks(allTasks) {
      try {
        debug('开始去重任务');
        const result = {};
        for (const [social, types] of Object.entries(allTasks)) {
          debug('处理社交媒体任务', {
            social: social
          });
          result[social] = {};
          for (const [type, tasks] of Object.entries(types)) {
            debug('处理任务类型', {
              social: social,
              type: type
            });
            result[social][type] = unique(tasks);
          }
        }
        debug('任务去重完成');
        return result;
      } catch (error) {
        debug('任务去重失败', {
          error: error
        });
        throwError(error, 'Website.uniqueTasks');
        return allTasks;
      }
    }
    async toggleTask(action) {
      try {
        debug('开始切换任务状态', {
          action: action
        });
        if (!this.initialized && !this.init()) {
          debug('初始化失败');
          return false;
        }
        if (!await this.classifyTask(action)) {
          debug('任务分类失败');
          return false;
        }
        debug('初始化社交媒体');
        await this.initSocial(action);
        const pro = [];
        const doTask = action === 'do';
        const tasks = doTask ? this.undoneTasks : this.socialTasks;
        if (this.socialInitialized.discord === true && this.social.discord) {
          debug('处理 Discord 任务');
          pro.push(this.social.discord.toggle({
            doTask: doTask,
            ...tasks.discord
          }));
        }
        if (this.socialInitialized.reddit === true && this.social.reddit) {
          debug('处理 Reddit 任务');
          pro.push(this.social.reddit.toggle({
            doTask: doTask,
            ...tasks.reddit
          }));
        }
        if (this.socialInitialized.twitch === true && this.social.twitch) {
          debug('处理 Twitch 任务');
          pro.push(this.social.twitch.toggle({
            doTask: doTask,
            ...tasks.twitch
          }));
        }
        if (this.socialInitialized.twitter === true && this.social.twitter) {
          debug('处理 Twitter 任务');
          pro.push(this.social.twitter.toggle({
            doTask: doTask,
            ...tasks.twitter
          }));
        }
        if (this.socialInitialized.vk === true && this.social.vk) {
          debug('处理 VK 任务');
          pro.push(this.social.vk.toggle({
            doTask: doTask,
            ...tasks.vk
          }));
        }
        if (this.socialInitialized.youtube === true && this.social.youtube) {
          debug('处理 YouTube 任务');
          pro.push(this.social.youtube.toggle({
            doTask: doTask,
            ...tasks.youtube
          }));
        }
        if ((this.steamTaskType.steamCommunity ? this.socialInitialized.steamCommunity === true : true) && (this.steamTaskType.steamStore ? this.socialInitialized.steamStore === true : true) && this.social.steam) {
          debug('处理 Steam 任务');
          pro.push(this.social.steam.toggle({
            doTask: doTask,
            ...tasks.steam
          }));
        }
        if (this.social.visitLink && tasks.links && doTask) {
          debug('处理链接任务', {
            linksCount: tasks.links.length
          });
          for (const link of tasks.links) {
            pro.push(this.social.visitLink(link));
          }
        }
        if (doTask && tasks.extra && this.extraDoTask) {
          const hasExtra = Object.values(tasks.extra).reduce(((total, arr) => [ ...total, ...arr ])).length > 0;
          if (hasExtra) {
            debug('处理额外任务');
            pro.push(this.extraDoTask(tasks.extra));
          }
        }
        debug('等待所有任务完成');
        await Promise.all(pro);
        debug('所有任务完成');
        echoLog({}).success(I18n('allTasksComplete'));
        return true;
      } catch (error) {
        debug('切换任务失败', {
          error: error
        });
        throwError(error, 'Website.toggleTask');
        return false;
      }
    }
    async doTask() {
      try {
        debug('开始执行任务');
        const result = await this.toggleTask('do');
        debug('任务执行完成', {
          success: result
        });
        return result;
      } catch (error) {
        debug('执行任务失败', {
          error: error
        });
        throwError(error, 'Website.doTask');
        return false;
      }
    }
    async undoTask() {
      try {
        debug('开始撤销任务');
        const result = await this.toggleTask('undo');
        debug('任务撤销完成', {
          success: result
        });
        return result;
      } catch (error) {
        debug('撤销任务失败', {
          error: error
        });
        throwError(error, 'Website.undoTask');
        return false;
      }
    }
  }
  const defaultTasksTemplate$6 = {
    steam: {
      groupLinks: [],
      wishlistLinks: [],
      curatorLinks: [],
      followLinks: [],
      playTimeLinks: []
    },
    discord: {
      serverLinks: []
    },
    vk: {
      nameLinks: []
    },
    youtube: {
      channelLinks: []
    },
    extra: {
      website: []
    }
  };
  const defaultTasks$8 = JSON.stringify(defaultTasksTemplate$6);
  class FreeAnyWhere extends Website {
    name='FreeAnyWhere';
    tasks=[];
    socialTasks=JSON.parse(defaultTasks$8);
    undoneTasks=JSON.parse(defaultTasks$8);
    games;
    buttons=[ 'doTask', 'undoTask', 'verifyTask', 'getKey' ];
    static test() {
      const isMatch = window.location.host === 'freeanywhere.net';
      debug('检查网站匹配', {
        host: window.location.host,
        isMatch: isMatch
      });
      return isMatch;
    }
    async init() {
      try {
        debug('初始化 FreeAnyWhere', {
          url: window.location.href
        });
        const logStatus = echoLog({
          text: I18n('initing')
        });
        debug('检测登录状态');
        if ($('div.header__login a[href*=logout]').length === 0) {
          debug('未登录,准备跳转到登录页面');
          window.open('https://freeanywhere.net/game.php?steam_login', '_self');
          logStatus.warning(I18n('needLogin'));
          return false;
        }
        debug('检测是否为登录页面');
        if (window.location.href.includes('/login')) {
          logStatus.warning(I18n('needLogin'));
          return false;
        }
        if (!await this.#checkLeftKey()) {
          debug('检查剩余密钥失败');
          echoLog({}).warning(I18n('checkLeftKeyFailed'));
        }
        const giveawayIdSuccess = this.#getGiveawayId();
        debug('获取抽奖ID结果', {
          success: giveawayIdSuccess,
          id: this.giveawayId
        });
        this.initialized = true;
        logStatus.success();
        return true;
      } catch (error) {
        debug('初始化失败', {
          error: error
        });
        throwError(error, 'Freeanywhere.init');
        return false;
      }
    }
    async classifyTask(action) {
      try {
        debug('开始分类任务', {
          action: action
        });
        const logStatus = echoLog({
          text: I18n('getTasksInfo')
        });
        if (action === 'undo') {
          debug('获取已保存的任务信息');
          this.socialTasks = GM_getValue(`fawTasks-${this.giveawayId}`)?.tasks || JSON.parse(defaultTasks$8);
        }
        const tasks = $('div.game__content-tasks__task').map(((index, element) => ({
          id: $(element).attr('data-id'),
          social: $(element).find('div.task-img img').attr('alt'),
          link: $(element).find('div.task-link a').attr('href'),
          title: $(element).find('div.task-link').text().trim(),
          type: $(element).attr('data-type'),
          data: $(element).attr('data-data'),
          isSuccess: $(element).hasClass('done')
        }))).toArray();
        debug('获取到的任务列表', {
          tasksCount: tasks.length,
          tasks: tasks
        });
        if (tasks.length === 0) {
          logStatus.success();
          return false;
        }
        if (action === 'verify') {
          this.tasks = [];
        }
        for (const task of tasks) {
          await this.#processTask(task, action);
        }
        logStatus.success();
        this.undoneTasks = this.uniqueTasks(this.undoneTasks);
        this.socialTasks = this.uniqueTasks(this.socialTasks);
        debug('任务分类结果', {
          undoneTasks: this.undoneTasks,
          socialTasks: this.socialTasks
        });
        GM_setValue(`fawTasks-${this.giveawayId}`, {
          tasks: this.socialTasks,
          time: (new Date).getTime()
        });
        return true;
      } catch (error) {
        debug('任务分类失败', {
          error: error
        });
        throwError(error, 'Freeanywhere.classifyTask');
        return false;
      }
    }
    async #processTask(task, action) {
      try {
        debug('处理任务', {
          task: task,
          action: action
        });
        const {id: id, social: social, title: title, type: type, link: link, data: data, isSuccess: isSuccess} = task;
        const taskInfo = {
          id: id,
          title: title,
          social: social,
          type: type,
          data: data
        };
        if (action === 'verify' && !isSuccess) {
          debug('添加到验证任务列表', taskInfo);
          this.tasks.push(taskInfo);
          return;
        }
        debug('处理特定类型任务', {
          type: type,
          action: action,
          isSuccess: isSuccess
        });
        switch (type) {
         case 'steam_account_verify':
         case 'site_email_verify':
          debug('跳过任务', {
            type: type
          });
          break;

         case 'steam_game_sub':
          if (action === 'undo' && link) {
            this.socialTasks.steam.followLinks.push(link);
          }
          if (action === 'do' && !isSuccess && link) {
            this.undoneTasks.steam.followLinks.push(link);
          }
          break;

         case 'steam_game_wishlist':
          if (action === 'undo' && link) {
            this.socialTasks.steam.wishlistLinks.push(link);
          }
          if (action === 'do' && !isSuccess && link) {
            this.undoneTasks.steam.wishlistLinks.push(link);
          }
          break;

         case 'steam_group_sub':
          if (action === 'undo' && link) {
            this.socialTasks.steam.groupLinks.push(link);
          }
          if (action === 'do' && !isSuccess && link) {
            this.undoneTasks.steam.groupLinks.push(link);
          }
          break;

         case 'site_visit':
          if (action === 'do' && !isSuccess) {
            this.undoneTasks.extra.website.push(`id=${id}&type=${type}&task=true`);
          }
          break;

         case 'vk_community_sub':
          if (action === 'undo' && link) {
            this.socialTasks.vk.nameLinks.push(link);
          }
          if (action === 'do' && !isSuccess && link) {
            this.undoneTasks.vk.nameLinks.push(link);
          }
          break;

         case 'vk_post_like':
          if (action === 'undo' && link) {
            this.socialTasks.vk.nameLinks.push(`${link}&action=like`);
          }
          if (action === 'do' && !isSuccess && link) {
            this.undoneTasks.vk.nameLinks.push(`${link}&action=like`);
          }
          break;

         case 'discord_server_sub':
          if (action === 'undo' && link) {
            this.socialTasks.discord.serverLinks.push(link);
          }
          if (action === 'do' && !isSuccess && link) {
            this.undoneTasks.discord.serverLinks.push(link);
          }
          break;

         case 'youtube_channel_sub':
          if (action === 'undo' && link) {
            this.socialTasks.youtube.channelLinks.push(link);
          }
          if (action === 'do' && !isSuccess && link) {
            this.undoneTasks.youtube.channelLinks.push(link);
          }
          break;

         case 'steam_game_playtime':
          if (action === 'undo' && link) {
            this.socialTasks.steam.playTimeLinks.push(`${title.match(/(\d+)\s*min/)?.[1] || '0'}-${link}`);
          }
          if (action === 'do' && !isSuccess && link) {
            this.undoneTasks.steam.playTimeLinks.push(`${title.match(/(\d+)\s*min/)?.[1] || '0'}-${link}`);
          }
          break;

         case 'telegram_channel_sub':
          debug('跳过 Telegram 任务');
          echoLog({}).warning(`${I18n('tgTaskNotice')}`);
          break;

         case 'none':
          debug('跳过未连接的任务', {
            type: type
          });
          echoLog({}).warning(`${I18n('notConnect', type)}`);
          break;

         default:
          debug('未知任务类型', {
            type: type
          });
          echoLog({}).warning(`${I18n('unKnownTaskType', type)}`);
          break;
        }
      } catch (error) {
        debug('处理任务失败', {
          error: error
        });
        throwError(error, 'FreeAnyWhere.processTask');
      }
    }
    async verifyTask() {
      try {
        debug('开始验证任务');
        if (!this.initialized && !await this.init()) {
          debug('未初始化');
          return false;
        }
        if (this.tasks.length === 0 && !await this.classifyTask('verify')) {
          debug('任务列表为空', this.tasks);
          return false;
        }
        debug('开始验证任务列表', {
          tasks: this.tasks
        });
        const pro = [];
        for (const task of this.tasks) {
          pro.push(this.#verify(task));
          await delay(1e3);
        }
        const result = await Promise.allSettled(pro);
        debug('任务验证结果', {
          result: result
        });
        echoLog({}).success(I18n('allTasksComplete'));
        if (result.every((item => item.status === 'fulfilled' && item.value === true))) {
          return !!await this.getKey(true);
        }
        return false;
      } catch (error) {
        debug('验证任务失败', {
          error: error
        });
        throwError(error, 'Freeanywhere.verifyTask');
        return false;
      }
    }
    async extraDoTask({website: website}) {
      try {
        debug('执行额外任务', {
          website: website
        });
        const promises = website.map((link => this.#doVisitWebsite(link)));
        const results = await Promise.allSettled(promises);
        debug('额外任务执行结果', {
          results: results
        });
        return true;
      } catch (error) {
        debug('执行额外任务失败', {
          error: error
        });
        throwError(error, 'FreeAnyWhere.extraDoTask');
        return false;
      }
    }
    async #doVisitWebsite(link) {
      try {
        debug('访问网站', {
          link: link
        });
        const logStatus = echoLog({
          text: I18n('visitingLink')
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://freeanywhere.net/php/task_site_visit_done.php',
          method: 'POST',
          headers: {
            'content-type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: link
        });
        if (result !== 'Success') {
          debug('访问失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.responseText.indexOf('bad') !== -1 || data?.responseText.length > 50) {
          debug('访问响应异常', {
            responseText: data?.responseText
          });
          logStatus.error(data?.responseText);
          return false;
        }
        debug('访问成功');
        logStatus.success();
        return true;
      } catch (error) {
        debug('访问网站失败', {
          error: error
        });
        throwError(error, 'FreeAnyWhere.doVisitWebsite');
        return false;
      }
    }
    async getKey(initialized) {
      try {
        debug('开始获取密钥', {
          initialized: initialized
        });
        if (!initialized && !this.initialized && !await this.init()) {
          debug('未初始化');
          return false;
        }
        const logStatus = echoLog({
          text: I18n('gettingKey')
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://freeanywhere.net/php/user_get_key.php',
          method: 'POST'
        });
        if (result !== 'Success') {
          debug('获取密钥失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.responseText.indexOf('bad') !== -1 || data?.responseText.length > 50) {
          debug('密钥响应异常', {
            responseText: data?.responseText
          });
          logStatus.error(data?.responseText);
          return false;
        }
        debug('获取密钥成功', {
          key: data.responseText
        });
        logStatus.success();
        echoLog({}).success(data.responseText);
        return data.responseText;
      } catch (error) {
        debug('获取密钥失败', {
          error: error
        });
        throwError(error, 'FreeAnyWhere.getKey');
        return false;
      }
    }
    async #verify(task) {
      try {
        if ($('.task-check-extension').length > 0) {
          return this.#verifyWithExtension(task);
        }
        return this.#verifyWithoutExtension(task);
      } catch (error) {
        debug('验证任务失败', {
          error: error
        });
        throwError(error, 'Freeanywhere.verify');
        return false;
      }
    }
    async #verifyWithExtension(task) {
      try {
        await this.#updateUserData();
        debug('验证任务', {
          task: task
        });
        const logStatus = echoLog({
          text: `${I18n('verifyingTask')}${task.title.trim()}...`
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://freeanywhere.net/php/extension/user_task_update.php',
          method: 'POST',
          headers: {
            'content-type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: `id=${task.id}&type=${task.type}${task.data && task.data !== 'none' ? `&data=${task.data}` : ''}`
        });
        if (result !== 'Success' || !data?.responseText) {
          debug('验证请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        const response = data.responseText.trim();
        if (response !== 'good') {
          debug('验证响应异常', {
            response: response,
            statusText: data?.statusText,
            status: data?.status
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('验证成功');
        logStatus.success();
        return true;
      } catch (error) {
        debug('验证任务失败', {
          error: error
        });
        throwError(error, 'Freeanywhere.verifyWithExtension');
        return false;
      }
    }
    async #verifyWithoutExtension(task) {
      try {
        debug('验证任务', {
          task: task
        });
        const logStatus = echoLog({
          text: `${I18n('verifyingTask')}${task.title.trim()}...`
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://freeanywhere.net/php/user_task_update.php',
          method: 'POST',
          headers: {
            'content-type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: `id=${task.id}&type=${task.type}${task.data && task.data !== 'none' ? `&data=${task.data}` : ''}`
        });
        if (result !== 'Success' || !data?.responseText) {
          debug('验证请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        const response = data.responseText.trim();
        if (response !== 'good') {
          debug('验证响应异常', {
            response: response,
            statusText: data?.statusText,
            status: data?.status
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('验证成功');
        logStatus.success();
        return true;
      } catch (error) {
        debug('验证任务失败', {
          error: error
        });
        throwError(error, 'Freeanywhere.verifyWithoutExtension');
        return false;
      }
    }
    async #updateUserData() {
      try {
        let postData = '';
        const userData = GM_getValue('FAW_STORAGE') || {};
        if (Object.keys(userData).length === 0 || !userData.tasks || !userData.user || !userData.games || !userData.settings) {
          if (!this.games) {
            await this.#userGamesGet();
          }
          if (!this.games) {
            debug('获取用户游戏失败');
            return false;
          }
          postData = `extension=${encodeURIComponent(JSON.stringify({
            games: this.games,
            settings: {
              game_update: Math.floor(Date.now() / 1e3)
            },
            tasks: {},
            user: {
              avatar: $('header.games_for_farm_site').attr('data-avatar'),
              lang: $('header.games_for_farm_site').attr('data-lang'),
              name: $('header.games_for_farm_site').attr('data-name'),
              steam: $('header.games_for_farm_site').attr('data-steam')
            }
          }))}`;
        } else {
          postData = `extension=${encodeURIComponent(JSON.stringify(userData))}`;
        }
        debug('更新用户数据');
        const logStatus = echoLog({
          text: `${I18n('updatingUserData')}`
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://freeanywhere.net/php/extension/user_data_update.php',
          method: 'POST',
          headers: {
            'content-type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: postData
        });
        if (data?.status !== 200) {
          debug('验证请求失败', {
            result: result,
            statusText: statusText,
            status: status,
            data: data
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        debug('验证成功');
        logStatus.success();
        return true;
      } catch (error) {
        debug('验证任务失败', {
          error: error
        });
        throwError(error, 'Freeanywhere.updateUserData');
        return false;
      }
    }
    async #userGamesGet() {
      try {
        debug('获取用户游戏');
        const logStatus = echoLog({
          text: `${I18n('gettingUserGames')}`
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://freeanywhere.net/php/extension/user_games_get.php',
          method: 'POST',
          headers: {
            'content-type': 'application/x-www-form-urlencoded; charset=UTF-8'
          },
          data: `steam=${$('header.games_for_farm_site').attr('data-steam')}`,
          dataType: 'json'
        });
        if (result !== 'Success' || data?.status !== 200 || !data?.responseText) {
          debug('验证请求失败', {
            result: result,
            statusText: statusText,
            status: status,
            data: data
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        debug('验证成功');
        this.games = data.response;
        logStatus.success();
        return true;
      } catch (error) {
        debug('验证任务失败', {
          error: error
        });
        throwError(error, 'Freeanywhere.userGamesGet');
        return false;
      }
    }
    async #checkLeftKey() {
      try {
        debug('检查剩余密钥');
        if (!globalOptions.other.checkLeftKey) {
          debug('跳过密钥检查');
          return true;
        }
        const giveawayStatus = $('div.card-info__lable-info').text()?.includes('Giveaway ended');
        debug('Giveaway状态', {
          giveawayStatus: giveawayStatus
        });
        if (!giveawayStatus) {
          return true;
        }
        debug('没有剩余密钥,显示确认对话框');
        const {value: value} = await Swal.fire({
          icon: 'warning',
          title: I18n('notice'),
          text: I18n('giveawayEnded'),
          confirmButtonText: I18n('confirm'),
          cancelButtonText: I18n('cancel'),
          showCancelButton: true
        });
        if (value) {
          debug('用户确认关闭窗口');
          window.close();
        }
        return true;
      } catch (error) {
        debug('检查剩余密钥失败', {
          error: error
        });
        throwError(error, 'FreeAnyWhere.checkLeftKey');
        return false;
      }
    }
    #getGiveawayId() {
      try {
        debug('开始获取抽奖ID');
        const giveawayId = $('link[rel="canonical"]').attr('href')?.match(/n=([\d]+)/)?.[1];
        if (giveawayId) {
          this.giveawayId = giveawayId;
          debug('获取抽奖ID成功', {
            giveawayId: giveawayId
          });
          return true;
        }
        debug('获取抽奖ID失败');
        echoLog({}).error(I18n('getFailed', 'GiveawayId'));
        return false;
      } catch (error) {
        debug('获取抽奖ID出错', {
          error: error
        });
        throwError(error, 'FreeAnyWhere.getGiveawayId');
        return false;
      }
    }
  }
  const defaultTasks$7 = {
    steam: {
      groupLinks: [],
      wishlistLinks: [],
      curatorLinks: [],
      curatorLikeLinks: [],
      followLinks: [],
      forumLinks: [],
      announcementLinks: [],
      workshopVoteLinks: [],
      playtestLinks: [],
      playTimeLinks: []
    },
    discord: {
      serverLinks: []
    },
    vk: {
      nameLinks: []
    },
    twitch: {
      channelLinks: []
    },
    reddit: {
      redditLinks: []
    },
    youtube: {
      channelLinks: [],
      likeLinks: []
    },
    twitter: {
      userLinks: [],
      retweetLinks: []
    }
  };
  class GiveawaySu extends Website {
    name='GiveawaySu';
    socialTasks=defaultTasks$7;
    undoneTasks=defaultTasks$7;
    buttons=[ 'doTask', 'undoTask' ];
    static test() {
      const url = window.location.href;
      const isMatch = /^https?:\/\/giveaway\.su\/giveaway\/view\/[\d]+/.test(url);
      debug('检查网站匹配', {
        url: url,
        isMatch: isMatch
      });
      return isMatch;
    }
    async after() {
      try {
        debug('开始执行后续操作');
        if (!this.#checkLogin()) {
          debug('登录检查失败');
          echoLog({}).warning(I18n('checkLoginFailed'));
        }
        if (!await this.#checkLeftKey()) {
          debug('检查剩余密钥失败');
          echoLog({}).warning(I18n('checkLeftKeyFailed'));
        }
        debug('显示网站通知');
        echoLog({}).warning(I18n('gsNotice'));
      } catch (error) {
        debug('后续操作失败', {
          error: error
        });
        throwError(error, 'Giveawaysu.after');
      }
    }
    init() {
      try {
        debug('初始化 GiveawaySu');
        const logStatus = echoLog({
          text: I18n('initing')
        });
        if ($('a.steam-login').length > 0) {
          debug('发现未登录状态,重定向到 Steam 登录');
          window.open('/steam/redirect', '_self');
          logStatus.warning(I18n('needLogin'));
          return false;
        }
        const giveawayIdResult = this.#getGiveawayId();
        if (!giveawayIdResult) {
          debug('获取抽奖ID失败');
          return false;
        }
        this.initialized = true;
        debug('初始化完成');
        logStatus.success();
        return true;
      } catch (error) {
        debug('初始化失败', {
          error: error
        });
        throwError(error, 'Giveawaysu.init');
        return false;
      }
    }
    async classifyTask(action) {
      try {
        debug('开始分类任务', {
          action: action
        });
        const logStatus = echoLog({
          text: I18n('getTasksInfo')
        });
        if (action === 'undo') {
          debug('恢复已保存的任务信息');
          this.socialTasks = GM_getValue(`gasTasks-${this.giveawayId}`)?.tasks || defaultTasks$7;
          return true;
        }
        const tasks = $('#actions tr');
        if (!tasks.length) {
          debug('未找到任务');
          logStatus.warning(I18n('noTasks'));
          return true;
        }
        debug('检查并处理 Discord 和 Twitch 绑定');
        if ($('div.bind-discord').is(':visible')) {
          debug('点击 Discord 绑定按钮');
          $('div.bind-discord a')[0]?.click();
        }
        if ($('div.bind-twitch').is(':visible')) {
          debug('点击 Twitch 绑定按钮');
          $('div.bind-twitch a')[0]?.click();
        }
        const processTask = async task => {
          const td = $(task).find('td:not(".hidden")');
          const colorfulTask = td.eq(1).find('a:not([data-trigger="link"])');
          const colorlessTask = td.eq(2).find('a:not([data-trigger="link"])');
          const taskDes = colorfulTask.length > 0 ? colorfulTask : colorlessTask;
          if (!taskDes.length) {
            debug('跳过无效任务');
            return true;
          }
          const taskIcon = td.eq(0).find('i').attr('class') || '';
          const taskName = taskDes.text().trim();
          const taskHref = taskDes.attr('href');
          debug('处理任务', {
            taskIcon: taskIcon,
            taskName: taskName,
            taskHref: taskHref
          });
          if (taskIcon.includes('ban') || /disable adblock/gi.test(taskName)) {
            debug('跳过禁用任务');
            return true;
          }
          if (!taskHref) {
            debug('任务链接为空');
            return false;
          }
          try {
            debug('获取重定向链接');
            const taskLink = await getRedirectLink(taskHref);
            if (!taskLink) {
              debug('获取重定向链接失败');
              return false;
            }
            debug('分类任务', {
              taskLink: taskLink,
              taskIcon: taskIcon,
              taskName: taskName
            });
            this.#classifyTaskByType(taskLink, taskIcon, taskName);
            return true;
          } catch (error) {
            debug('获取重定向链接失败', {
              error: error
            });
            throwError(error, 'Giveawaysu.classifyTask->getRedirectLink');
            return false;
          }
        };
        debug('开始处理所有任务');
        const results = await Promise.all(Array.from(tasks).map(processTask));
        const success = results.some((result => result));
        if (!success) {
          debug('所有任务处理失败');
          logStatus.error(I18n('allTasksFailed'));
          return false;
        }
        debug('任务处理完成');
        logStatus.success();
        this.undoneTasks = this.uniqueTasks(this.undoneTasks);
        this.socialTasks = this.undoneTasks;
        debug('保存任务信息');
        GM_setValue(`gasTasks-${this.giveawayId}`, {
          tasks: this.socialTasks,
          time: (new Date).getTime()
        });
        return true;
      } catch (error) {
        debug('任务分类失败', {
          error: error
        });
        throwError(error, 'Giveawaysu.classifyTask');
        return false;
      }
    }
    static TASK_PATTERNS={
      wishlist: /wishlist.*game|add.*wishlist/gim,
      follow: /follow.*button/gim,
      twitter: /(on twitter)|(Follow.*on.*Facebook)/gim,
      vkGroup: /join.*vk.*group/gim,
      youtubeVideo: /(watch|like).*video/gim,
      youtubeChannel: /subscribe.*youtube.*channel/gim,
      watchArt: /watch.*art/gim,
      reddit: /subscribe.*subreddit|follow.*reddit/gim,
      twitchChannel: /follow.*twitch.*channel/gim,
      instagram: /follow.*instagram/gim,
      discord: /join.*discord/gim,
      playtest: /request.*playtest/gim,
      steamForum: /subscribe.*steam.*forum/gim,
      curator: /(follow|subscribe).*curator/gim,
      curatorLink: /^https?:\/\/store\.steampowered\.com\/curator\//,
      announcement: /like.*announcement/gim,
      steamGroup: /join/gi
    };
    #classifyTaskByType(taskLink, taskIcon, taskName) {
      try {
        debug('开始分类任务', {
          taskLink: taskLink,
          taskIcon: taskIcon,
          taskName: taskName
        });
        const {TASK_PATTERNS: TASK_PATTERNS} = GiveawaySu;
        if (taskIcon.includes('steam') && TASK_PATTERNS.steamGroup.test(taskName)) {
          debug('添加 Steam 组任务');
          this.undoneTasks.steam.groupLinks.push(taskLink);
          return;
        }
        if (TASK_PATTERNS.announcement.test(taskName)) {
          debug('添加 Steam 公告任务');
          this.undoneTasks.steam.announcementLinks.push(taskLink);
          return;
        }
        if (TASK_PATTERNS.curator.test(taskName) && TASK_PATTERNS.curatorLink.test(taskLink)) {
          debug('添加 Steam 鉴赏家关注任务');
          this.undoneTasks.steam.curatorLinks.push(taskLink);
          return;
        }
        if (taskIcon.includes('steam') && /follow|subscribe/gim.test(taskName)) {
          debug('添加 Steam 鉴赏家点赞任务');
          this.undoneTasks.steam.curatorLikeLinks.push(taskLink);
          return;
        }
        if (TASK_PATTERNS.steamForum.test(taskName)) {
          debug('添加 Steam 论坛任务');
          this.undoneTasks.steam.forumLinks.push(taskLink);
          return;
        }
        if (taskIcon.includes('thumbs-up') && /^https?:\/\/steamcommunity\.com\/sharedfiles\/filedetails\/\?id=[\d]+/.test(taskLink)) {
          debug('添加 Steam 创意工坊投票任务');
          this.undoneTasks.steam.workshopVoteLinks.push(taskLink);
          return;
        }
        if (taskIcon.includes('plus') && TASK_PATTERNS.playtest.test(taskName)) {
          debug('添加 Steam 游戏测试任务');
          this.undoneTasks.steam.playtestLinks.push(taskLink);
          return;
        }
        if (taskIcon.includes('discord') || TASK_PATTERNS.discord.test(taskName)) {
          debug('添加 Discord 服务器任务');
          this.undoneTasks.discord.serverLinks.push(taskLink);
          return;
        }
        if (taskIcon.includes('instagram') || TASK_PATTERNS.instagram.test(taskName)) {
          debug('跳过 Instagram 任务');
          return;
        }
        if (taskIcon.includes('twitch') || TASK_PATTERNS.twitchChannel.test(taskName)) {
          debug('添加 Twitch 频道任务');
          this.undoneTasks.twitch.channelLinks.push(taskLink);
          return;
        }
        if (taskIcon.includes('reddit') || TASK_PATTERNS.reddit.test(taskName)) {
          debug('添加 Reddit 任务');
          this.undoneTasks.reddit.redditLinks.push(taskLink);
          return;
        }
        if (TASK_PATTERNS.watchArt.test(taskName)) {
          debug('添加创意工坊物品任务');
          this.undoneTasks.steam.workshopVoteLinks.push(taskLink);
          return;
        }
        if (TASK_PATTERNS.youtubeChannel.test(taskName)) {
          debug('添加 YouTube 频道任务');
          this.undoneTasks.youtube.channelLinks.push(taskLink);
          return;
        }
        if (TASK_PATTERNS.youtubeVideo.test(taskName) || (taskIcon.includes('youtube') || taskIcon.includes('thumbs-up')) && TASK_PATTERNS.youtubeVideo.test(taskName)) {
          debug('添加 YouTube 视频任务');
          this.undoneTasks.youtube.likeLinks.push(taskLink);
          return;
        }
        if (taskIcon.includes('vk') || TASK_PATTERNS.vkGroup.test(taskName)) {
          debug('添加 VK 任务');
          this.undoneTasks.vk.nameLinks.push(taskLink);
          return;
        }
        if (TASK_PATTERNS.twitter.test(taskName)) {
          debug('跳过 Twitter 任务');
          return;
        }
        if (TASK_PATTERNS.wishlist.test(taskName)) {
          debug('添加 Steam 愿望单任务');
          this.undoneTasks.steam.wishlistLinks.push(taskLink);
        }
        if (TASK_PATTERNS.follow.test(taskName)) {
          debug('添加 Steam 关注任务');
          this.undoneTasks.steam.followLinks.push(taskLink);
          return;
        }
        debug('未识别的任务类型', {
          taskLink: taskLink,
          taskIcon: taskIcon,
          taskName: taskName
        });
      } catch (error) {
        debug('任务分类失败', {
          error: error
        });
        throwError(error, 'Giveawaysu.classifyTaskByType');
      }
    }
    #checkLogin() {
      try {
        debug('检查登录状态');
        if (!globalOptions.other.checkLogin) {
          debug('跳过登录检查');
          return true;
        }
        const needLogin = $('a.steam-login').length > 0;
        if (needLogin) {
          debug('未登录,重定向到 Steam 登录');
          window.open('/steam/redirect', '_self');
        }
        debug('登录检查完成', {
          needLogin: needLogin
        });
        return true;
      } catch (error) {
        debug('登录检查失败', {
          error: error
        });
        throwError(error, 'Giveawaysu.checkLogin');
        return false;
      }
    }
    async #checkLeftKey() {
      try {
        debug('检查剩余密钥');
        if (!globalOptions.other.checkLeftKey) {
          debug('跳过密钥检查');
          return true;
        }
        const isEnded = $('.giveaway-ended').length > 0;
        const hasNoKeys = $('.giveaway-key').length === 0;
        debug('检查抽奖状态', {
          isEnded: isEnded,
          hasNoKeys: hasNoKeys
        });
        if (!(isEnded && hasNoKeys)) {
          return true;
        }
        debug('没有剩余密钥,显示确认对话框');
        const {value: value} = await Swal.fire({
          icon: 'warning',
          title: I18n('notice'),
          text: I18n('noKeysLeft'),
          confirmButtonText: I18n('confirm'),
          cancelButtonText: I18n('cancel'),
          showCancelButton: true
        });
        if (value) {
          debug('用户确认关闭窗口');
          window.close();
        }
        return true;
      } catch (error) {
        debug('检查剩余密钥失败', {
          error: error
        });
        throwError(error, 'Giveawaysu.checkLeftKey');
        return false;
      }
    }
    #getGiveawayId() {
      try {
        debug('从URL获取抽奖ID');
        const giveawayId = window.location.href.match(/\/view\/([\d]+)/)?.[1];
        if (giveawayId) {
          this.giveawayId = giveawayId;
          debug('获取抽奖ID成功', {
            giveawayId: giveawayId
          });
          return true;
        }
        debug('获取抽奖ID失败');
        echoLog({
          text: I18n('getFailed', 'GiveawayId')
        });
        return false;
      } catch (error) {
        debug('获取抽奖ID出错', {
          error: error
        });
        throwError(error, 'Giveawaysu.getGiveawayId');
        return false;
      }
    }
  }
  class Indiedb {
    name='Indiedb';
    buttons=[ 'doTask' ];
    static test() {
      const {host: host} = window.location;
      const isMatch = host === 'www.indiedb.com';
      debug('检查网站匹配', {
        host: host,
        isMatch: isMatch
      });
      return isMatch;
    }
    async after() {
      try {
        debug('开始执行后续操作');
        if (!this.#checkLogin()) {
          debug('检查登录失败');
          echoLog({}).warning(I18n('checkLoginFailed'));
        }
        if (!await this.#checkLeftKey()) {
          debug('检查剩余密钥失败');
          echoLog({}).warning(I18n('checkLeftKeyFailed'));
        }
      } catch (error) {
        debug('后续操作失败', {
          error: error
        });
        throwError(error, 'Indiedb.after');
      }
    }
    async doTask() {
      try {
        debug('开始执行任务');
        if (!await this.#join()) {
          debug('加入抽奖失败');
          return false;
        }
        return await this.#do();
      } catch (error) {
        debug('执行任务失败', {
          error: error
        });
        throwError(error, 'Indiedb.doTask');
        return false;
      }
    }
    async #join() {
      try {
        debug('开始加入抽奖');
        if ($('a.buttonenter:contains(Register to join)').length > 0) {
          debug('需要登录');
          echoLog({}).error(I18n('needLogin'));
          return false;
        }
        const currentoption = $('a.buttonenter.buttongiveaway');
        const buttonText = currentoption.text();
        debug('检查按钮状态', {
          buttonText: buttonText
        });
        if (/success/gim.test($('a.buttonenter.buttongiveaway').text())) {
          debug('已成功加入抽奖');
          return true;
        }
        if (!/join giveaway/gim.test(buttonText)) {
          debug('需要加入抽奖');
          echoLog({}).warning(I18n('needJoinGiveaway'));
          return false;
        }
        const logStatus = echoLog({
          text: `${I18n('joiningGiveaway')}...`
        });
        debug('发送加入请求');
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: currentoption.attr('href'),
          method: 'POST',
          data: 'ajax=t',
          dataType: 'json',
          headers: {
            'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
            Accept: 'application/json, text/javascript, */*; q=0.01',
            Origin: window.location.origin,
            referer: window.location.href
          }
        });
        if (result !== 'Success') {
          debug('请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('尝试备用加入方法');
          if (await this.#join2()) {
            debug('备用加入方法成功');
            logStatus.success('Success');
            return true;
          }
          debug('加入失败', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        if (!data.response?.success) {
          debug('响应失败', {
            text: data.response?.text
          });
          logStatus.error(`Error${data.response?.text ? `:${data.response?.text}` : ''}`);
          return false;
        }
        debug('加入成功');
        currentoption.addClass('buttonentered').text('Success - Giveaway joined');
        $('#giveawaysjoined').slideDown();
        $('#giveawaysrecommend').slideDown();
        logStatus.success(`Success${data.response?.text ? `:${data.response?.text}` : ''}`);
        return true;
      } catch (error) {
        debug('加入抽奖失败', {
          error: error
        });
        throwError(error, 'Indiedb.join');
        return false;
      }
    }
    async #join2() {
      try {
        debug('开始备用加入方法');
        return await new Promise((resolve => {
          const targetNode = document.getElementById('giveawaysjoined');
          const config = {
            attributes: true
          };
          const observer = new MutationObserver((() => {
            if ($('#giveawaysjoined').is(':visible')) {
              debug('检测到加入成功');
              resolve(true);
              observer.disconnect();
            }
          }));
          observer.observe(targetNode, config);
          debug('点击加入按钮');
          $('a.buttonenter.buttongiveaway')[0]?.click();
          setTimeout((() => {
            debug('加入超时');
            resolve(false);
            observer.disconnect();
          }), 3e4);
        }));
      } catch (error) {
        debug('备用加入方法失败', {
          error: error
        });
        throwError(error, 'Indiedb.join2');
        return false;
      }
    }
    async #do() {
      try {
        debug('开始执行任务');
        const id = $('script').map(((index, script) => {
          if (!/\$\(document\)/gim.test(script.innerHTML)) {
            return null;
          }
          return [ script.innerHTML.match(/"\/[\d]+"/gim)?.[0]?.match(/[\d]+/)?.[0], script.innerHTML.match(/"\/newsletter\/ajax\/subscribeprofile\/optin\/[\d]+"/gim)?.[0]?.match(/[\d]+/)?.[0] ];
        }));
        if (id.length < 2) {
          debug('获取任务ID失败');
          echoLog({}).error(I18n('getFailed', 'TaskId'));
          return false;
        }
        const pro = [];
        const tasks = $('#giveawaysjoined a[class*=promo]');
        debug('找到任务', {
          count: tasks.length
        });
        for (const task of tasks) {
          const promo = $(task);
          if (promo.hasClass('buttonentered')) {
            debug('跳过已完成任务');
            continue;
          }
          const taskText = promo.parents('p').text();
          debug('处理任务', {
            taskText: taskText
          });
          const status = echoLog({
            text: `${I18n('doing')}:${taskText}...`
          });
          if (/the-challenge-of-adblock/gim.test(promo.attr('href'))) {
            debug('跳过未知任务类型');
            status.error(`Error:${I18n('unKnownTaskType')}`);
            continue;
          }
          if (/facebookpromo|twitterpromo|visitpromo/gim.test(task.className)) {
            let text = '';
            if (promo.hasClass('facebookpromo')) {
              text = 'facebookpromo';
            } else if (promo.hasClass('twitterpromo')) {
              text = 'twitterpromo';
            } else {
              text = 'visitpromo';
            }
            debug('处理社交媒体任务', {
              type: text
            });
            pro.push(this.#handleSocialPromo(text, id[0], status, promo));
          } else if (promo.hasClass('emailoptinpromo')) {
            debug('处理邮件订阅任务');
            pro.push(this.#handleEmailPromo(id[1], status, promo));
          } else if (promo.hasClass('watchingpromo')) {
            debug('处理关注任务');
            pro.push(this.#handleWatchingPromo(promo, status));
          } else {
            debug('处理默认任务');
            pro.push(this.#handleDefaultPromo(promo, status));
          }
        }
        await Promise.all(pro);
        debug('所有任务完成');
        echoLog({}).success(I18n('allTasksComplete'));
        return true;
      } catch (error) {
        debug('执行任务失败', {
          error: error
        });
        throwError(error, 'Indiedb.do');
        return false;
      }
    }
    async #handleSocialPromo(text, id, status, promo) {
      try {
        debug('处理社交媒体任务', {
          text: text,
          id: id
        });
        return await new Promise((resolve => {
          $.ajax({
            type: 'POST',
            url: urlPath(`/giveaways/ajax/${text}/${id}`),
            timeout: 6e4,
            dataType: 'json',
            data: {
              ajax: 't'
            },
            error(response, error, exception) {
              debug('请求失败', {
                response: response,
                error: error,
                exception: exception
              });
              if (window.DEBUG) {
                console.log('%cAuto-Task[Debug]:', 'color:red', {
                  response: response,
                  error: error,
                  exception: exception
                });
              }
              status.error('Error:An error has occurred performing the action requested. Please try again shortly.');
              resolve(true);
            },
            success(response) {
              if (response.success) {
                debug('任务完成', {
                  response: response
                });
                status.success(`Success:${response.text}`);
                promo.addClass('buttonentered').closest('p').html(promo.closest('p').find('span').html());
              } else {
                debug('任务失败', {
                  response: response
                });
                status.error(`Error:${response.text}`);
              }
              resolve(true);
            }
          });
        }));
      } catch (error) {
        debug('处理社交媒体任务失败', {
          error: error
        });
        throwError(error, 'Indiedb.handleSocialPromo');
        return false;
      }
    }
    async #handleEmailPromo(id, status, promo) {
      try {
        debug('处理邮件订阅任务', {
          id: id
        });
        return await new Promise((resolve => {
          $.ajax({
            type: 'POST',
            url: urlPath(`/newsletter/ajax/subscribeprofile/optin/${id}`),
            timeout: 6e4,
            dataType: 'json',
            data: {
              ajax: 't',
              emailsystoggle: 4
            },
            error(response, error, exception) {
              debug('请求失败', {
                response: response,
                error: error,
                exception: exception
              });
              if (window.DEBUG) {
                console.log('%cAuto-Task[Debug]:', 'color:red', {
                  response: response,
                  error: error,
                  exception: exception
                });
              }
              status.error('Error:An error has occurred performing the action requested. Please try again shortly.');
              resolve(true);
            },
            success(response) {
              if (response.success) {
                debug('任务完成', {
                  response: response
                });
                status.success(`Success:${response.text}`);
                promo.toggleClass('buttonentered').closest('p').html(promo.closest('p').find('span').html());
              } else {
                debug('任务失败', {
                  response: response
                });
                status.error(`Error:${response.text}`);
              }
              resolve(true);
            }
          });
        }));
      } catch (error) {
        debug('处理邮件订阅任务失败', {
          error: error
        });
        throwError(error, 'Indiedb.handleEmailPromo');
        return false;
      }
    }
    async #handleWatchingPromo(promo, status) {
      try {
        debug('处理关注任务');
        return await new Promise((resolve => {
          const href = promo.attr('href');
          if (!href) {
            debug('无效的链接');
            status.error('Error: Invalid href');
            resolve(true);
            return;
          }
          const data = getUrlQuery(href);
          data.ajax = 't';
          const [baseUrl] = href.split(/[?#]/);
          if (!baseUrl) {
            debug('无效的URL');
            status.error('Error: Invalid URL');
            resolve(true);
            return;
          }
          debug('发送请求', {
            url: baseUrl,
            data: data
          });
          $.ajax({
            type: 'POST',
            url: urlPath(baseUrl),
            timeout: 6e4,
            dataType: 'json',
            data: data,
            error(response, error, exception) {
              debug('请求失败', {
                response: response,
                error: error,
                exception: exception
              });
              if (window.DEBUG) {
                console.log('%cAuto-Task[Debug]:', 'color:red', {
                  response: response,
                  error: error,
                  exception: exception
                });
              }
              status.error('Error:An error has occurred performing the action requested. Please try again shortly.');
              resolve(true);
            },
            success(response) {
              if (response.success) {
                debug('任务完成', {
                  response: response
                });
                status.success(`Success:${response.text}`);
                promo.toggleClass('buttonentered').closest('p').html(promo.closest('p').find('span').html());
              } else {
                debug('任务失败', {
                  response: response
                });
                status.error(`Error:${response.text}`);
              }
              resolve(true);
            }
          });
        }));
      } catch (error) {
        debug('处理关注任务失败', {
          error: error
        });
        throwError(error, 'Indiedb.handleWatchingPromo');
        return false;
      }
    }
    async #handleDefaultPromo(promo, status) {
      try {
        debug('处理默认任务');
        return await new Promise((resolve => {
          const href = promo.attr('href');
          if (!href) {
            debug('无效的链接');
            status.error('Error: Invalid href');
            resolve(true);
            return;
          }
          debug('发送请求', {
            url: href
          });
          $.ajax({
            type: 'POST',
            url: urlPath(href),
            timeout: 6e4,
            dataType: 'json',
            data: {
              ajax: 't'
            },
            error(response, error, exception) {
              debug('请求失败', {
                response: response,
                error: error,
                exception: exception
              });
              if (window.DEBUG) {
                console.log('%cAuto-Task[Debug]:', 'color:red', {
                  response: response,
                  error: error,
                  exception: exception
                });
              }
              status.error('Error:An error has occurred performing the action requested. Please try again shortly.');
              resolve(true);
            },
            success(response) {
              if (response.success) {
                debug('任务完成', {
                  response: response
                });
                status.success(`Success:${response.text}`);
                promo.toggleClass('buttonentered').closest('p').html(promo.closest('p').find('span').html());
              } else {
                debug('任务失败', {
                  response: response
                });
                status.error(`Error:${response.text}`);
              }
              resolve(true);
            }
          });
        }));
      } catch (error) {
        debug('处理默认任务失败', {
          error: error
        });
        throwError(error, 'Indiedb.handleDefaultPromo');
        return false;
      }
    }
    #checkLogin() {
      try {
        debug('检查登录状态');
        if (!globalOptions.other.checkLogin) {
          debug('跳过登录检查');
          return true;
        }
        if ($('a.buttonenter:contains(Register to join)').length > 0) {
          debug('未登录,重定向到登录页面');
          window.open('/members/login', '_self');
        }
        debug('登录检查完成');
        return true;
      } catch (error) {
        debug('检查登录失败', {
          error: error
        });
        throwError(error, 'Indiedb.checkLogin');
        return false;
      }
    }
    async #checkLeftKey() {
      try {
        debug('检查剩余密钥');
        if (!globalOptions.other.checkLeftKey) {
          debug('跳过密钥检查');
          return true;
        }
        const hasEndedButton = $('a.buttonenter:contains("next time"), a.buttonenter:contains("Giveaway is closed")').length > 0;
        debug('检查抽奖状态', {
          hasEndedButton: hasEndedButton
        });
        if (!hasEndedButton) {
          return true;
        }
        debug('抽奖已结束,显示确认对话框');
        const {value: value} = await Swal.fire({
          icon: 'warning',
          title: I18n('notice'),
          text: I18n('giveawayEnded'),
          confirmButtonText: I18n('confirm'),
          cancelButtonText: I18n('cancel'),
          showCancelButton: true
        });
        if (value) {
          debug('用户确认关闭窗口');
          window.close();
        }
        return true;
      } catch (error) {
        debug('检查剩余密钥失败', {
          error: error
        });
        throwError(error, 'Indiedb.checkLeftKey');
        return false;
      }
    }
  }
  const defaultTasksTemplate$5 = {
    steam: {
      groupLinks: [],
      officialGroupLinks: [],
      wishlistLinks: [],
      curatorLinks: []
    },
    discord: {
      serverLinks: []
    },
    extra: {
      videoTasks: []
    },
    links: []
  };
  const defaultTasks$6 = JSON.stringify(defaultTasksTemplate$5);
  class Keyhub extends Website {
    name='Keyhub';
    socialTasks=JSON.parse(defaultTasks$6);
    undoneTasks=JSON.parse(defaultTasks$6);
    buttons=[ 'doTask', 'undoTask' ];
    static test() {
      const {host: host} = window.location;
      const isMatch = host === 'key-hub.eu';
      debug('检查网站匹配', {
        host: host,
        isMatch: isMatch
      });
      return isMatch;
    }
    async after() {
      try {
        debug('开始执行后续操作');
        if (!this.#checkLogin()) {
          debug('检查登录失败');
          echoLog({}).warning(I18n('checkLoginFailed'));
        }
        if (!await this.#checkLeftKey()) {
          debug('检查剩余密钥失败');
          echoLog({}).warning(I18n('checkLeftKeyFailed'));
        }
        debug('隐藏 NSFW 内容');
        $('.NSFW').hide();
      } catch (error) {
        debug('后续操作失败', {
          error: error
        });
        throwError(error, 'Keyhub.after');
      }
    }
    init() {
      try {
        debug('开始初始化');
        const logStatus = echoLog({
          text: I18n('initing')
        });
        if ($('a[href*="/connect/steam"]').length > 0) {
          debug('需要登录 Steam');
          window.open('/connect/steam', '_self');
          logStatus.warning(I18n('needLogin'));
          return false;
        }
        if (!this.#getGiveawayId()) {
          debug('获取抽奖ID失败');
          return false;
        }
        debug('隐藏 VPN 覆盖层');
        $('#VPNoverlay').hide();
        $('#mainArticleSection').show();
        this.initialized = true;
        debug('初始化完成');
        logStatus.success();
        return true;
      } catch (error) {
        debug('初始化失败', {
          error: error
        });
        throwError(error, 'Keyhub.init');
        return false;
      }
    }
    async classifyTask(action) {
      try {
        debug('开始分类任务', {
          action: action
        });
        const logStatus = echoLog({
          text: I18n('getTasksInfo')
        });
        if (action === 'undo') {
          debug('恢复已保存的任务信息');
          this.socialTasks = GM_getValue(`khTasks-${this.giveawayId}`)?.tasks || JSON.parse(defaultTasks$6);
        }
        const tasks = $('.task:not(".googleads")').filter(((index, element) => action === 'do' ? $(element).find('i.fa-check-circle:visible').length === 0 : true)).find('a');
        debug('找到任务', {
          count: tasks.length
        });
        for (const task of tasks) {
          let link = $(task).attr('href');
          const taskDes = $(task).text().trim();
          debug('处理任务', {
            taskDes: taskDes,
            link: link
          });
          if (!link) {
            debug('跳过无链接任务');
            continue;
          }
          if (/\/away\?data=/.test(link) || /steamcommunity\.com\/gid\//.test(link)) {
            debug('获取重定向链接');
            link = await getRedirectLink(link) || link;
          }
          if (/https?:\/\/key-hub\.eu\/connect\/discord/.test(link)) {
            debug('处理 Discord 连接任务');
            GM_openInTab(link, {
              active: true
            });
            continue;
          }
          if (/steamcommunity\.com\/groups\//.test(link)) {
            debug('处理 Steam 组任务');
            if (action === 'undo') {
              this.socialTasks.steam.groupLinks.push(link);
            }
            if (action === 'do') {
              this.undoneTasks.steam.groupLinks.push(link);
            }
            continue;
          }
          if (/steamcommunity\.com\/games\/[\d]+/.test(link)) {
            debug('处理 Steam 官方组任务');
            if (action === 'undo') {
              this.socialTasks.steam.officialGroupLinks.push(link);
            }
            if (action === 'do') {
              this.undoneTasks.steam.officialGroupLinks.push(link);
            }
            continue;
          }
          if (/store\.steampowered\.com\/app\//.test(link) && /wishlist/gim.test(taskDes)) {
            debug('处理 Steam 愿望单任务');
            if (action === 'undo') {
              this.socialTasks.steam.wishlistLinks.push(link);
            }
            if (action === 'do') {
              this.undoneTasks.steam.wishlistLinks.push(link);
            }
            continue;
          }
          if (/store\.steampowered\.com\/curator\//.test(link)) {
            debug('处理 Steam 鉴赏家任务');
            if (action === 'undo') {
              this.socialTasks.steam.curatorLinks.push(link);
            }
            if (action === 'do') {
              this.undoneTasks.steam.curatorLinks.push(link);
            }
            continue;
          }
          if (/^https?:\/\/discord\.com\/invite\//.test(link)) {
            debug('处理 Discord 服务器任务');
            if (action === 'undo') {
              this.socialTasks.discord.serverLinks.push(link);
            }
            if (action === 'do') {
              this.undoneTasks.discord.serverLinks.push(link);
            }
            continue;
          }
          if (/^javascript:videoTask.+/.test(link)) {
            debug('处理视频任务');
            if (action === 'do') {
              const taskData = link.match(/javascript:videoTask\('.+?','(.+?)'/)?.[1];
              if (taskData) {
                debug('添加视频任务', {
                  taskData: taskData
                });
                this.undoneTasks.extra.videoTasks.push(taskData);
              }
            }
            continue;
          }
          if (this.#isSkippableLink(link)) {
            debug('跳过可忽略的链接', {
              link: link
            });
            continue;
          }
          debug('未知任务类型', {
            taskDes: taskDes,
            link: link
          });
          echoLog({}).warning(`${I18n('unKnownTaskType')}: ${taskDes}(${link})`);
        }
        debug('任务分类完成');
        logStatus.success();
        this.undoneTasks = this.uniqueTasks(this.undoneTasks);
        this.socialTasks = this.uniqueTasks(this.socialTasks);
        if (window.DEBUG) {
          console.log('%cAuto-Task[Debug]:', 'color:blue', JSON.stringify(this));
        }
        debug('保存任务信息');
        GM_setValue(`khTasks-${this.giveawayId}`, {
          tasks: this.socialTasks,
          time: (new Date).getTime()
        });
        return true;
      } catch (error) {
        debug('任务分类失败', {
          error: error
        });
        throwError(error, 'Keyhub.classifyTask');
        return false;
      }
    }
    #isSkippableLink(link) {
      return /^https?:\/\/www\.instagram\.com\/.*/.test(link) || /^https?:\/\/twitter\.com\/.*/.test(link) || /^https?:\/\/www\.twitch\.tv\/.*/.test(link) || /^https?:\/\/www\.facebook\.com\/.*/.test(link) || /^https?:\/\/www\.youtube\.com\/.*/.test(link) || /^https?:\/\/store\.steampowered\.com\/developer\//.test(link) || /^https?:\/\/.*?\.itch\.io\/.*/.test(link) || /^https?:\/\/key-hub\.eu.*/.test(link) || /^https?:\/\/store\.steampowered\.com\/app\/.*/.test(link) || /^https?:\/\/qr\.streamelements\.com\/.*/.test(link) || /^https?:\/\/store\.steampowered\.com\/news\/app\/.*/.test(link);
    }
    async #doScriptTask(data) {
      try {
        debug('执行脚本任务', {
          data: data
        });
        const logStatus = echoLog({
          text: I18n('doingKeyhubTask')
        });
        const {result: result, statusText: statusText, status: status, data: response} = await httpRequest({
          url: `/away?data=${data}`,
          method: 'GET',
          headers: {
            origin: 'https://key-hub.eu',
            referer: 'https://key-hub.eu/'
          }
        });
        if (result !== 'Success') {
          debug('请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (response?.status !== 200) {
          debug('响应错误', {
            status: response?.status,
            statusText: response?.statusText
          });
          logStatus.error(`Error:${response?.statusText}(${response?.status})`);
          return false;
        }
        debug('任务完成');
        logStatus.success();
        return true;
      } catch (error) {
        debug('执行脚本任务失败', {
          error: error
        });
        throwError(error, 'Keyhub.doScriptTask');
        return false;
      }
    }
    async extraDoTask({videoTasks: videoTasks}) {
      try {
        debug('开始执行额外任务', {
          count: videoTasks.length
        });
        const pro = [];
        for (const data of videoTasks) {
          pro.push(this.#doScriptTask(data));
        }
        return Promise.all(pro).then((() => {
          debug('所有额外任务完成');
          return true;
        }));
      } catch (error) {
        debug('执行额外任务失败', {
          error: error
        });
        throwError(error, 'Keyhub.extraDoTask');
        return false;
      }
    }
    #getGiveawayId() {
      try {
        debug('获取抽奖ID');
        const giveawayId = window.location.href.match(/giveaway\/([\d]+)/)?.[1];
        if (giveawayId) {
          this.giveawayId = giveawayId;
          debug('获取抽奖ID成功', {
            giveawayId: giveawayId
          });
          return true;
        }
        debug('获取抽奖ID失败');
        echoLog({}).error(I18n('getFailed', 'GiveawayId'));
        return false;
      } catch (error) {
        debug('获取抽奖ID出错', {
          error: error
        });
        throwError(error, 'Keyhub.getGiveawayId');
        return false;
      }
    }
    async #checkLeftKey() {
      try {
        debug('检查剩余密钥');
        if (!globalOptions.other.checkLeftKey) {
          debug('跳过密钥检查');
          return true;
        }
        const leftKey = $('#keysleft').text().trim();
        debug('检查剩余密钥数量', {
          leftKey: leftKey
        });
        if (leftKey !== '0') {
          return true;
        }
        debug('没有剩余密钥,显示确认对话框');
        const {value: value} = await Swal.fire({
          icon: 'warning',
          title: I18n('notice'),
          text: I18n('noKeysLeft'),
          confirmButtonText: I18n('confirm'),
          cancelButtonText: I18n('cancel'),
          showCancelButton: true
        });
        if (value) {
          debug('用户确认关闭窗口');
          window.close();
        }
        return true;
      } catch (error) {
        debug('检查剩余密钥失败', {
          error: error
        });
        throwError(error, 'Keyhub.checkLeftKey');
        return false;
      }
    }
    #checkLogin() {
      try {
        debug('检查登录状态');
        if (!globalOptions.other.checkLogin) {
          debug('跳过登录检查');
          return true;
        }
        if ($('a[href*="/connect/steam"]').length > 0) {
          debug('未登录,重定向到 Steam 登录页面');
          window.open('/connect/steam', '_self');
        }
        debug('登录检查完成');
        return true;
      } catch (error) {
        debug('检查登录失败', {
          error: error
        });
        throwError(error, 'Keyhub.checkLogin');
        return false;
      }
    }
  }
  const defaultTasksTemplate$4 = {
    steam: {
      groupLinks: [],
      wishlistLinks: [],
      curatorLinks: [],
      curatorLikeLinks: []
    },
    twitter: {
      userLinks: []
    },
    vk: {
      nameLinks: []
    },
    discord: {
      serverLinks: []
    }
  };
  const defaultTasks$5 = JSON.stringify(defaultTasksTemplate$4);
  class Givekey extends Website {
    name='Givekey';
    tasks=[];
    socialTasks=JSON.parse(defaultTasks$5);
    undoneTasks=JSON.parse(defaultTasks$5);
    userId;
    buttons=[ 'doTask', 'undoTask', 'verifyTask' ];
    static test() {
      const url = window.location.host;
      const isMatch = url === 'givekey.ru';
      debug('检查网站匹配', {
        url: url,
        isMatch: isMatch
      });
      return isMatch;
    }
    async after() {
      try {
        debug('开始执行后续操作');
        await new Promise((resolve => {
          const checker = setInterval((() => {
            if ($('#navbarDropdown').length > 0) {
              debug('导航栏元素已加载');
              clearInterval(checker);
              resolve(true);
            }
          }), 500);
        }));
        if (!await this.#checkLeftKey()) {
          debug('检查剩余密钥失败');
          echoLog({}).warning(I18n('checkLeftKeyFailed'));
        }
      } catch (error) {
        debug('后续操作失败', {
          error: error
        });
        throwError(error, 'Givekey.after');
      }
    }
    init() {
      try {
        debug('初始化 Givekey');
        const logStatus = echoLog({
          text: I18n('initing')
        });
        if ($('a[href*="/auth/steam"]').length > 0) {
          debug('未登录,重定向到 Steam 登录页面');
          window.open('/auth/steam', '_self');
          logStatus.warning(I18n('needLogin'));
          return false;
        }
        if (!this.#getGiveawayId()) {
          debug('获取抽奖ID失败');
          return false;
        }
        const userId = $('meta[name="user-id"]').attr('content');
        if (!userId) {
          debug('获取用户ID失败');
          logStatus.error(I18n('getFailed', I18n('userId')));
          return false;
        }
        this.userId = userId;
        this.initialized = true;
        debug('初始化完成', {
          userId: userId
        });
        logStatus.success();
        return true;
      } catch (error) {
        debug('初始化失败', {
          error: error
        });
        throwError(error, 'Givekey.init');
        return false;
      }
    }
    async classifyTask(action) {
      try {
        debug('开始分类任务', {
          action: action
        });
        const logStatus = echoLog({
          text: I18n('getTasksInfo')
        });
        if (action === 'undo') {
          debug('恢复已保存的任务信息');
          this.socialTasks = GM_getValue(`gkTasks-${this.giveawayId}`)?.tasks || JSON.parse(defaultTasks$5);
        }
        const tasks = $('.card-body:has("button") .row');
        debug('找到任务元素', {
          count: tasks.length
        });
        for (const task of tasks) {
          const taskEle = $(task);
          const button = taskEle.find('button');
          const isSuccess = /Complete/i.test(button.text().trim());
          debug('处理任务', {
            isSuccess: isSuccess
          });
          if (isSuccess && action !== 'undo') {
            debug('跳过已完成的任务');
            continue;
          }
          const checkButton = taskEle.find('#task_check');
          const taskId = checkButton.attr('data-id');
          if (taskId) {
            debug('添加任务ID', {
              taskId: taskId
            });
            this.tasks.push(taskId);
          }
          if (action === 'verify') {
            continue;
          }
          const taskLink = taskEle.find('a');
          let href = taskLink.attr('href');
          if (!href) {
            debug('任务链接为空');
            continue;
          }
          const text = taskLink.text().trim();
          if (!text) {
            debug('任务描述为空');
            continue;
          }
          if (/^https?:\/\/givekey\.ru\/giveaway\/[\d]+\/execution_task/.test(href)) {
            debug('获取重定向链接', {
              href: href
            });
            href = await getRedirectLink(href);
          }
          if (!href) {
            debug('获取重定向链接失败');
            continue;
          }
          const icon = taskEle.find('i');
          await this.#classifyTaskByType(href, text, icon, isSuccess, action);
        }
        debug('任务分类完成');
        logStatus.success();
        this.tasks = unique(this.tasks);
        this.undoneTasks = this.uniqueTasks(this.undoneTasks);
        this.socialTasks = this.uniqueTasks(this.socialTasks);
        debug('保存任务信息');
        GM_setValue(`gkTasks-${this.giveawayId}`, {
          tasks: this.socialTasks,
          time: (new Date).getTime()
        });
        return true;
      } catch (error) {
        debug('任务分类失败', {
          error: error
        });
        throwError(error, 'Givekey.classifyTask');
        return false;
      }
    }
    async verifyTask() {
      try {
        debug('开始验证任务');
        if (!this.initialized && !this.init()) {
          debug('初始化失败');
          return false;
        }
        if (this.tasks.length === 0 && !await this.classifyTask('verify')) {
          debug('任务分类失败');
          return false;
        }
        echoLog({}).warning(I18n('giveKeyNoticeBefore'));
        const taskLength = this.tasks.length;
        debug('开始验证任务', {
          taskCount: taskLength
        });
        for (let i = 0; i < taskLength; i++) {
          await this.#verify(this.tasks[i]);
          if (i < taskLength - 1) {
            debug('等待15秒');
            await delay(15e3);
          }
        }
        debug('所有任务验证完成');
        echoLog({}).success(I18n('allTasksComplete'));
        echoLog({
          html: `<li><font class="warning">${I18n('giveKeyNoticeAfter')}</font></li>`
        });
        return true;
      } catch (error) {
        debug('任务验证失败', {
          error: error
        });
        throwError(error, 'Givekey.verifyTask');
        return false;
      }
    }
    async #verify(task) {
      try {
        debug('验证任务', {
          taskId: task
        });
        const logStatus = echoLog({
          html: `<li>${I18n('verifyingTask')}${task}...<font></font></li>`
        });
        const csrfToken = $('meta[name="csrf-token"]').attr('content');
        if (!csrfToken) {
          debug('CSRF token 未找到');
          logStatus.error('CSRF token not found');
          return false;
        }
        debug('发送验证请求');
        const response = await $.ajax({
          url: 'https://givekey.ru/giveaway/task',
          method: 'POST',
          data: `id=${task}&user_id=${this.userId}`,
          dataType: 'json',
          headers: {
            'X-CSRF-TOKEN': csrfToken
          }
        });
        if (!response) {
          debug('未收到响应');
          logStatus.error('No response received');
          return false;
        }
        debug('处理响应', {
          response: response
        });
        if (response.btn) {
          $(`button[data-id=${this.userId}]`).html(response.btn);
        }
        if (response.status === 'ok') {
          $(`.task_check_${response.id}`).html(`<button class="btn btn-success mb-2 btn-block" disabled>${response.btn}</button>`);
          debug('任务验证成功');
          logStatus.success();
          return true;
        }
        if (response.status === 'end') {
          debug('获得密钥');
          logStatus.success();
          echoLog({}).success(response.key);
          return true;
        }
        debug('验证失败', {
          error: response.msg
        });
        logStatus.error(`Error:${response.msg}`);
        return false;
      } catch (error) {
        debug('验证过程出错', {
          error: error
        });
        throwError(error, 'Givekey.verify');
        return false;
      }
    }
    #getGiveawayId() {
      try {
        debug('从URL获取抽奖ID');
        const giveawayId = window.location.href.match(/giveaway\/([\d]+)/)?.[1];
        if (giveawayId) {
          this.giveawayId = giveawayId;
          debug('获取抽奖ID成功', {
            giveawayId: giveawayId
          });
          return true;
        }
        debug('获取抽奖ID失败');
        echoLog({
          text: I18n('getFailed', 'GiveawayId')
        });
        return false;
      } catch (error) {
        debug('获取抽奖ID出错', {
          error: error
        });
        throwError(error, 'Givekey.getGiveawayId');
        return false;
      }
    }
    async #checkLeftKey() {
      try {
        debug('检查剩余密钥');
        if (!globalOptions.other.checkLeftKey) {
          debug('跳过密钥检查');
          return true;
        }
        const keysCount = $('#keys_count').text();
        debug('检查密钥数量', {
          keysCount: keysCount
        });
        if (keysCount) {
          return true;
        }
        debug('没有剩余密钥,显示确认对话框');
        const {value: value} = await Swal.fire({
          icon: 'warning',
          title: I18n('notice'),
          text: I18n('noKeysLeft'),
          confirmButtonText: I18n('confirm'),
          cancelButtonText: I18n('cancel'),
          showCancelButton: true
        });
        if (value) {
          debug('用户确认关闭窗口');
          window.close();
        }
        return true;
      } catch (error) {
        debug('检查剩余密钥失败', {
          error: error
        });
        throwError(error, 'Givekey.checkLeftKey');
        return false;
      }
    }
    async #classifyTaskByType(href, text, icon, isSuccess, action) {
      try {
        debug('开始分类任务类型', {
          href: href,
          text: text,
          isSuccess: isSuccess,
          action: action
        });
        if (/^https?:\/\/vk\.com\//.test(href)) {
          debug('添加 VK 任务');
          this.socialTasks.vk.nameLinks.push(href);
          if (action === 'do' && !isSuccess) {
            this.undoneTasks.vk.nameLinks.push(href);
          }
          return;
        }
        if (/^https?:\/\/steamcommunity\.com\/groups/.test(href)) {
          debug('添加 Steam 组任务');
          this.socialTasks.steam.groupLinks.push(href);
          if (action === 'do' && !isSuccess) {
            this.undoneTasks.steam.groupLinks.push(href);
          }
          return;
        }
        if (/^https?:\/\/store\.steampowered\.com\/app\//.test(href)) {
          debug('添加 Steam 愿望单任务');
          this.socialTasks.steam.wishlistLinks.push(href);
          if (action === 'do' && !isSuccess) {
            this.undoneTasks.steam.wishlistLinks.push(href);
          }
          return;
        }
        if (/Subscribe/gi.test(text) && icon.hasClass('fa-steam-square')) {
          if (/^https?:\/\/store\.steampowered\.com\/curator\//.test(href)) {
            debug('添加 Steam 鉴赏家关注任务');
            this.socialTasks.steam.curatorLinks.push(href);
            if (action === 'do' && !isSuccess) {
              this.undoneTasks.steam.curatorLinks.push(href);
            }
          } else {
            debug('添加 Steam 鉴赏家点赞任务');
            this.socialTasks.steam.curatorLikeLinks.push(href);
            if (action === 'do' && !isSuccess) {
              this.undoneTasks.steam.curatorLikeLinks.push(href);
            }
          }
          return;
        }
        if (/^https?:\/\/twitter\.com\//.test(href) && /Subscribe/gi.test(text)) {
          debug('添加 Twitter 关注任务');
          this.socialTasks.twitter.userLinks.push(href);
          if (action === 'do' && !isSuccess) {
            this.undoneTasks.twitter.userLinks.push(href);
          }
          return;
        }
        if (icon.hasClass('fa-discord') || /^https?:\/\/discord\.com\/invite\//.test(href)) {
          debug('添加 Discord 服务器任务');
          this.socialTasks.discord.serverLinks.push(href);
          if (action === 'do' && !isSuccess) {
            this.undoneTasks.discord.serverLinks.push(href);
          }
          return;
        }
        debug('未识别的任务类型', {
          href: href,
          text: text
        });
        echoLog({}).warning(`${I18n('unKnownTaskType')}: ${text}(${href})`);
      } catch (error) {
        debug('任务类型分类失败', {
          error: error
        });
        throwError(error, 'Givekey.classifyTaskByType');
      }
    }
  }
  class GiveeClub extends GiveawaySu {
    name='GiveeClub';
    buttons=[ 'doTask', 'undoTask', 'verifyTask' ];
    static test() {
      const url = window.location.href;
      const isMatch = /^https?:\/\/givee\.club\/.*?\/event\/[\d]+/.test(url);
      debug('检查网站匹配', {
        url: url,
        isMatch: isMatch
      });
      return isMatch;
    }
    async after() {
      try {
        debug('开始执行后续操作');
        if (!this.#checkLogin()) {
          debug('登录检查失败');
          echoLog({}).warning(I18n('checkLoginFailed'));
        }
        if (!await this.#checkLeftKey()) {
          debug('检查剩余密钥失败');
          echoLog({}).warning(I18n('checkLeftKeyFailed'));
        }
      } catch (error) {
        debug('后续操作失败', {
          error: error
        });
        throwError(error, 'GiveeClub.after');
      }
    }
    init() {
      try {
        debug('初始化 GiveeClub');
        const logStatus = echoLog({
          text: I18n('initing')
        });
        if (!this.#checkLogin()) {
          debug('登录检查失败');
          logStatus.warning(I18n('needLogin'));
          return false;
        }
        const giveawayIdResult = this.#getGiveawayId();
        if (!giveawayIdResult) {
          debug('获取抽奖ID失败');
          return false;
        }
        this.initialized = true;
        debug('初始化完成');
        logStatus.success();
        return true;
      } catch (error) {
        debug('初始化失败', {
          error: error
        });
        throwError(error, 'GiveeClub.init');
        return false;
      }
    }
    async classifyTask(action) {
      try {
        debug('开始分类任务', {
          action: action
        });
        const logStatus = echoLog({
          text: I18n('getTasksInfo')
        });
        if (action === 'undo') {
          debug('恢复已保存的任务信息');
          this.socialTasks = GM_getValue(`gcTasks-${this.giveawayId}`)?.tasks || defaultTasks$7;
          return true;
        }
        debug('初始化未完成任务列表');
        this.undoneTasks = defaultTasks$7;
        const tasks = $('.event-actions tr');
        const processTask = async task => {
          const taskDes = $(task).find('.event-action-label a');
          const taskIcon = $(task).find('.event-action-icon i').attr('class') || '';
          const taskName = taskDes.text().trim();
          const taskType = $(task).find('button[data-type]')?.attr('data-type') || '';
          const taskFinished = $(task).find('.event-action-buttons .btn-success')?.length;
          const appId = taskDes.attr('data-steam-wishlist-appid');
          debug('处理任务', {
            taskName: taskName,
            taskType: taskType,
            taskIcon: taskIcon,
            taskFinished: taskFinished,
            appId: appId
          });
          if (taskIcon.includes('ban') || /AdBlock/i.test(taskName) || taskIcon.includes('envelope') || taskFinished) {
            debug('跳过无效或已完成任务');
            return true;
          }
          const taskHref = taskDes.attr('href');
          if (!taskHref) {
            debug('任务链接为空');
            return false;
          }
          try {
            debug('获取重定向链接', {
              taskHref: taskHref
            });
            const taskLink = await getRedirectLink(taskHref, taskType.includes('steam'));
            if (!taskLink) {
              debug('获取重定向链接失败');
              return false;
            }
            if (taskType === 'steam.game.wishlist' && appId) {
              debug('添加 Steam 愿望单任务', {
                appId: appId
              });
              this.undoneTasks.steam.wishlistLinks.push(`https://store.steampowered.com/app/${appId}`);
              return true;
            }
            debug('分类任务', {
              taskLink: taskLink,
              taskType: taskType
            });
            this.#classifyTaskByType(taskLink, taskType, taskIcon, taskName, taskDes);
            return true;
          } catch (error) {
            debug('获取重定向链接失败', {
              error: error
            });
            throwError(error, 'GiveeClub.classifyTask->getRedirectLink');
            return false;
          }
        };
        debug('开始处理所有任务');
        await Promise.all(Array.from(tasks).map(processTask));
        debug('任务处理完成');
        logStatus.success();
        this.undoneTasks = this.uniqueTasks(this.undoneTasks);
        this.socialTasks = this.undoneTasks;
        debug('保存任务信息');
        GM_setValue(`gcTasks-${this.giveawayId}`, {
          tasks: this.socialTasks,
          time: (new Date).getTime()
        });
        return true;
      } catch (error) {
        debug('任务分类失败', {
          error: error
        });
        throwError(error, 'GiveeClub.classifyTask');
        return false;
      }
    }
    #classifyTaskByType(taskLink, taskType, taskIcon, taskName, taskDes) {
      try {
        debug('开始分类任务', {
          taskLink: taskLink,
          taskType: taskType,
          taskIcon: taskIcon,
          taskName: taskName
        });
        if (taskType === 'steam.group.join' && /^https?:\/\/steamcommunity\.com\/groups/.test(taskLink)) {
          debug('添加 Steam 组任务');
          this.undoneTasks.steam.groupLinks.push(taskLink);
          return;
        }
        if (/like.*announcement/gi.test(taskName)) {
          debug('添加 Steam 公告任务');
          this.undoneTasks.steam.announcementLinks.push(taskLink);
          return;
        }
        if (taskType === 'steam.game.wishlist' && /^https?:\/\/store\.steampowered\.com\/app\//.test(taskLink)) {
          debug('添加 Steam 愿望单任务');
          this.undoneTasks.steam.wishlistLinks.push(taskLink);
          return;
        }
        if (taskType === 'steam.game.wishlist' && taskDes.attr('data-steam-wishlist-appid')) {
          debug('添加 Steam 愿望单任务(通过 appId)');
          this.undoneTasks.steam.wishlistLinks.push(`https://store.steampowered.com/app/${taskDes.attr('data-steam-wishlist-appid')}`);
          return;
        }
        if (taskType === 'steam.game.follow' && /^https?:\/\/store\.steampowered\.com\/app\//.test(taskLink)) {
          debug('添加 Steam 游戏关注任务');
          this.undoneTasks.steam.followLinks.push(taskLink);
          return;
        }
        if (/^https?:\/\/store\.steampowered\.com\/curator\//.test(taskLink)) {
          debug('添加 Steam 鉴赏家关注任务');
          this.undoneTasks.steam.curatorLinks.push(taskLink);
          return;
        }
        if (taskIcon.includes('steam') && /follow|subscribe/gim.test(taskName)) {
          debug('添加 Steam 鉴赏家点赞任务');
          this.undoneTasks.steam.curatorLikeLinks.push(taskLink);
          return;
        }
        if (/subscribe.*steam.*forum/gim.test(taskName)) {
          debug('添加 Steam 论坛任务');
          this.undoneTasks.steam.forumLinks.push(taskLink);
          return;
        }
        if (taskType === 'steam.game.playtime' && /^https?:\/\/store\.steampowered\.com\/app\//.test(taskLink)) {
          const time = taskDes.text().match(/(\d+)(?:\.\d+)?/gim)?.[0] || '0';
          debug('添加 Steam 游戏时长任务', {
            time: time
          });
          this.undoneTasks.steam.playTimeLinks.push(`${time}-${taskLink}`);
          return;
        }
        if (taskIcon.includes('discord')) {
          debug('添加 Discord 服务器任务');
          this.undoneTasks.discord.serverLinks.push(taskLink);
          return;
        }
        if (taskIcon.includes('instagram')) {
          debug('跳过 Instagram 任务');
          return;
        }
        if (taskIcon.includes('twitch')) {
          debug('添加 Twitch 频道任务');
          this.undoneTasks.twitch.channelLinks.push(taskLink);
          return;
        }
        if (taskIcon.includes('reddit')) {
          debug('添加 Reddit 任务');
          this.undoneTasks.reddit.redditLinks.push(taskLink);
          return;
        }
        if (/watch.*art/gim.test(taskName)) {
          debug('添加创意工坊物品任务');
          this.undoneTasks.steam.workshopVoteLinks.push(taskLink);
          return;
        }
        if (/subscribe.*youtube.*channel/gim.test(taskName)) {
          debug('添加 YouTube 频道任务');
          this.undoneTasks.youtube.channelLinks.push(taskLink);
          return;
        }
        if (/(watch|like).*youtube.*video/gim.test(taskName) || (taskIcon.includes('youtube') || taskIcon.includes('thumbs-up')) && /(watch|like).*video/gim.test(taskName)) {
          debug('添加 YouTube 视频任务');
          this.undoneTasks.youtube.likeLinks.push(taskLink);
          return;
        }
        if (taskIcon.includes('vk') || /join.*vk.*group/gim.test(taskName)) {
          debug('添加 VK 任务');
          this.undoneTasks.vk.nameLinks.push(taskLink);
          return;
        }
        if (taskIcon.includes('twitter')) {
          if (/https?:\/\/(twitter|x)\.com\/[^/]+\/?$/gim.test(taskLink)) {
            debug('添加 Twitter 用户关注任务');
            this.undoneTasks.twitter.userLinks.push(taskLink);
            return;
          }
          if (/https?:\/\/(twitter|x)\.com\/[^/]+?\/status\/[\d]+/gim.test(taskLink)) {
            debug('添加 Twitter 转发任务');
            this.undoneTasks.twitter.retweetLinks.push(taskLink);
            return;
          }
        }
        if (/(on twitter)|(Follow.*on.*Facebook)/gim.test(taskName)) {
          debug('跳过 Twitter/Facebook 任务');
          return;
        }
        if (/follow.*button/gim.test(taskName)) {
          debug('添加 Steam 关注任务');
          this.undoneTasks.steam.followLinks.push(taskLink);
          return;
        }
        debug('未识别的任务类型', {
          taskLink: taskLink,
          taskType: taskType,
          taskIcon: taskIcon,
          taskName: taskName
        });
      } catch (error) {
        debug('任务分类失败', {
          error: error
        });
        throwError(error, 'GiveeClub.classifyTaskByType');
        return;
      }
    }
    async verifyTask() {
      try {
        debug('开始验证任务');
        const logStatus = echoLog({
          text: I18n('giveeClubVerifyNotice')
        });
        const taskButtons = $('.event-actions tr button').has('i.glyphicon-refresh').not('[data-type="user.adblock"]');
        debug('找到需要验证的任务按钮', {
          count: taskButtons.length
        });
        for (const button of taskButtons) {
          debug('点击验证按钮', {
            type: $(button).attr('data-type')
          });
          button.click();
          if ($(button).attr('data-type') !== 'steam.game.wishlist') {
            debug('等待1秒');
            await delay(1e3);
          }
        }
        debug('任务验证完成');
        logStatus.warning(I18n('giveeClubVerifyFinished'));
        return true;
      } catch (error) {
        debug('任务验证失败', {
          error: error
        });
        throwError(error, 'Givekey.verifyTask');
        return false;
      }
    }
    #checkLogin() {
      try {
        debug('检查登录状态');
        if (!globalOptions.other.checkLogin) {
          debug('跳过登录检查');
          return true;
        }
        const needLogin = $('a[href*="/account/auth"]').length > 0;
        if (needLogin) {
          debug('未登录,重定向到登录页面');
          window.open($('a[href*="/account/auth"]').attr('href'), '_self');
        }
        debug('登录检查完成', {
          needLogin: needLogin
        });
        return true;
      } catch (error) {
        debug('登录检查失败', {
          error: error
        });
        throwError(error, 'GiveeClub.checkLogin');
        return false;
      }
    }
    #getGiveawayId() {
      try {
        debug('从URL获取抽奖ID');
        const giveawayId = window.location.href.match(/\/event\/([\d]+)/)?.[1];
        if (giveawayId) {
          this.giveawayId = giveawayId;
          debug('获取抽奖ID成功', {
            giveawayId: giveawayId
          });
          return true;
        }
        debug('获取抽奖ID失败');
        echoLog({
          text: I18n('getFailed', 'GiveawayId')
        });
        return false;
      } catch (error) {
        debug('获取抽奖ID出错', {
          error: error
        });
        throwError(error, 'GiveeClub.getGiveawayId');
        return false;
      }
    }
    async #checkLeftKey() {
      try {
        debug('检查剩余密钥');
        if (!globalOptions.other.checkLeftKey) {
          debug('跳过密钥检查');
          return true;
        }
        const isEnded = $('.event-ended').length > 0;
        const hasNoWinner = $('.event-winner').length === 0;
        debug('检查抽奖状态', {
          isEnded: isEnded,
          hasNoWinner: hasNoWinner
        });
        if (!(isEnded && hasNoWinner)) {
          return true;
        }
        debug('没有剩余密钥,显示确认对话框');
        const {value: value} = await Swal.fire({
          icon: 'warning',
          title: I18n('notice'),
          text: I18n('giveawayEnded'),
          confirmButtonText: I18n('confirm'),
          cancelButtonText: I18n('cancel'),
          showCancelButton: true
        });
        if (value) {
          debug('用户确认关闭窗口');
          window.close();
        }
        return true;
      } catch (error) {
        debug('检查剩余密钥失败', {
          error: error
        });
        throwError(error, 'GiveeClub.checkLeftKey');
        return false;
      }
    }
  }
  const defaultOptions$1 = {
    maxPoint: '99999999'
  };
  class OpiumPulses {
    name='OpiumPulses';
    options={
      ...defaultOptions$1,
      ...GM_getValue('OpiumPulsesOptions')
    };
    maxPoints=99999999;
    myPoints=0;
    buttons=[ 'doFreeTask', 'doPointTask' ];
    static test() {
      const {host: host} = window.location;
      const isMatch = host === 'www.opiumpulses.com';
      debug('检查网站匹配', {
        host: host,
        isMatch: isMatch
      });
      return isMatch;
    }
    async after() {
      try {
        debug('开始执行后续操作');
        if (!this.#checkLogin()) {
          debug('检查登录失败');
          echoLog({}).warning(I18n('checkLoginFailed'));
        }
        debug('解析最大积分', {
          maxPoint: this.options.maxPoint
        });
        this.maxPoints = parseInt(this.options.maxPoint, 10);
      } catch (error) {
        debug('后续操作失败', {
          error: error
        });
        throwError(error, 'OpiumPulses.after');
      }
    }
    async doFreeTask() {
      try {
        debug('开始执行免费任务');
        this.#toggleTask('FREE');
      } catch (error) {
        debug('执行免费任务失败', {
          error: error
        });
        throwError(error, 'OpiumPulses.doFreeTask');
      }
    }
    async doPointTask() {
      try {
        debug('开始执行积分任务');
        const pointsText = $('.page-header__nav-func-user-nav-items.points-items').text();
        const pointsMatch = pointsText.match(/[\d]+/gim)?.[0] || '0';
        this.myPoints = parseInt(pointsMatch, 10);
        debug('获取当前积分', {
          pointsText: pointsText,
          pointsMatch: pointsMatch,
          myPoints: this.myPoints
        });
        this.#toggleTask('points');
      } catch (error) {
        debug('执行积分任务失败', {
          error: error
        });
        throwError(error, 'OpiumPulses.doPointTask');
      }
    }
    async #toggleTask(type) {
      try {
        debug('开始切换任务', {
          type: type
        });
        const items = $(`.giveaways-page-item:contains('${type}'):not(:contains('ENTERED'))`);
        debug('找到未参与的抽奖项目', {
          count: items.length
        });
        for (const item of items) {
          const pointsText = $(item).find('.giveaways-page-item-header-points').text();
          const needPoints = parseInt(pointsText.match(/[\d]+/gim)?.[0] || '999999', 10);
          const name = $(item).find('.giveaways-page-item-footer-name').text().trim();
          debug('处理抽奖项目', {
            name: name,
            needPoints: needPoints
          });
          if (type === 'points') {
            if (needPoints > this.myPoints) {
              debug('积分不足', {
                needPoints: needPoints,
                myPoints: this.myPoints
              });
              echoLog({}).warning(`${I18n('noPoints')}: ${name}`);
              continue;
            }
            if (!needPoints) {
              debug('获取所需积分失败');
              echoLog({}).warning(`${I18n('getNeedPointsFailed')}: ${name}`);
              continue;
            }
            if (needPoints > this.maxPoints) {
              debug('超过最大积分限制', {
                needPoints: needPoints,
                maxPoints: this.maxPoints
              });
              continue;
            }
          }
          const logStatus = echoLog({
            text: `${I18n('joiningLottery')}<a href="${$(item).find('a.giveaways-page-item-img-btn-more').attr('href')}" target="_blank">${name}</a>...`
          });
          const aElement = $(item).find('a.giveaways-page-item-img-btn-enter:contains(\'enter\')');
          if (aElement?.attr('onclick')?.includes('checkUser')) {
            const giveawayId = aElement.attr('onclick')?.match(/[\d]+/)?.[0];
            if (giveawayId) {
              debug('执行用户检查', {
                giveawayId: giveawayId
              });
              checkUser(giveawayId);
            }
          }
          if (!aElement.attr('href')) {
            debug('无效的链接');
            logStatus.error('Error: No "href".');
            continue;
          }
          debug('发送加入请求', {
            url: aElement.attr('href')
          });
          const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
            url: aElement.attr('href'),
            method: 'GET'
          });
          if (result !== 'Success') {
            debug('请求失败', {
              result: result,
              statusText: statusText,
              status: status
            });
            logStatus.error(`${result}:${statusText}(${status})`);
            continue;
          }
          debug('发送最终请求', {
            url: data?.finalUrl
          });
          const {result: result0, statusText: statusText0, status: status0, data: data0} = await httpRequest({
            url: data?.finalUrl,
            method: 'GET'
          });
          if (!data0?.responseText) {
            debug('响应无效', {
              result: result0,
              statusText: statusText0,
              status: status0
            });
            logStatus.error(`${result0}:${statusText0}(${status0})`);
            continue;
          }
          if (/You're not eligible to enter/gim.test(data0.responseText)) {
            debug('用户不符合参与条件');
            logStatus.error('You\'re not eligible to enter');
            continue;
          }
          if (!/You've entered this giveaway/gim.test(data0.responseText)) {
            debug('加入抽奖失败', {
              result: result0,
              statusText: statusText0,
              status: status0
            });
            logStatus.error(`${result0}:${statusText0}(${status0})`);
            continue;
          }
          debug('加入抽奖成功');
          logStatus.success();
          if (type === 'points') {
            const points = data0.responseText.match(/Points:[\s]*?([\d]+)/)?.[1];
            if (points) {
              debug('更新用户积分', {
                points: points
              });
              this.myPoints = parseInt(points, 10);
            }
          }
        }
        debug('任务处理完成');
        echoLog({
          text: '-----END-----'
        });
      } catch (error) {
        debug('切换任务失败', {
          error: error
        });
        throwError(error, 'OpiumPulses.toggleTask');
      }
    }
    init() {
      debug('初始化完成');
      return true;
    }
    classifyTask() {
      debug('任务分类完成');
      return true;
    }
    #checkLogin() {
      try {
        debug('检查登录状态');
        if (!globalOptions.other.checkLogin) {
          debug('跳过登录检查');
          return true;
        }
        if ($('a[href*="/site/login"]').length > 1) {
          debug('未登录,重定向到登录页面');
          window.open('/site/login', '_self');
        }
        debug('登录检查完成');
        return true;
      } catch (error) {
        debug('检查登录失败', {
          error: error
        });
        throwError(error, 'OpiumPulses.checkLogin');
        return false;
      }
    }
  }
  const leftKeyChecker = {
    async classify(link) {
      try {
        debug('开始分类链接', {
          link: link
        });
        let result = false;
        if (/^https?:\/\/giveaway\.su\/giveaway\/view\/[\d]+/.test(link)) {
          debug('匹配到 giveaway.su 链接');
          result = await this.giveawaySu(link);
        } else if (/^https?:\/\/givee\.club\/[\w]+?\/event\/[\d]+/.test(link)) {
          debug('匹配到 givee.club 链接');
          result = await this.giveeClub(link);
        } else if (/^https?:\/\/gleam\.io\/.+?\/.+/.test(link)) {
          debug('匹配到 gleam.io 链接');
          result = await this.gleam(link);
        } else if (/^https?:\/\/www\.indiedb\.com\/giveaways\/.+/.test(link)) {
          debug('匹配到 indiedb.com 链接');
          result = await this.indieDb(link);
        } else if (/^https?:\/\/key-hub\.eu\/giveaway\/[\d]+/.test(link)) {
          debug('匹配到 key-hub.eu 链接');
          result = await this.keyhub(link);
        } else if (/^https?:\/\/opquests\.com\/quests\/[\d]+/.test(link)) {
          debug('匹配到 opquests.com 链接');
          result = await this.opquests(link);
        } else if (/^https?:\/\/itch\.io\/s\/[\d]+?\/.*/.test(link)) {
          debug('匹配到 itch.io 链接');
          result = await this.itch(link);
        } else if (/^https?:\/\/freeanywhere\.net\/game\?n=[\d]+/.test(link)) {
          debug('匹配到 freeanywhere.net 链接');
          result = await this.freeanywhere(link);
        } else {
          debug('未匹配到支持的链接格式');
        }
        debug('链接分类完成', {
          result: result
        });
        return result;
      } catch (error) {
        debug('链接分类出错', {
          error: error
        });
        throwError(error, 'leftKeyChecker.classify');
        return false;
      }
    },
    async giveawaySu(link) {
      try {
        debug('开始检查 giveaway.su 链接', {
          link: link
        });
        const {result: result, data: data} = await httpRequest({
          url: link,
          method: 'GET'
        });
        if (result !== 'Success' || data?.status !== 200) {
          debug('请求失败', {
            result: result,
            status: data?.status
          });
          return false;
        }
        if (data.responseText.includes('class="steam-login"')) {
          debug('检测到未登录状态');
          return false;
        }
        if (data.responseText.includes('class="giveaway-ended"')) {
          debug('检测到抽奖已结束');
          return 'Ended';
        }
        debug('检测到抽奖进行中');
        return 'Active';
      } catch (error) {
        debug('检查 giveaway.su 链接出错', {
          error: error
        });
        throwError(error, 'leftKeyChecker.giveawaySu');
        return false;
      }
    },
    async giveeClub(link) {
      try {
        debug('开始检查 givee.club 链接', {
          link: link
        });
        const {result: result, data: data} = await httpRequest({
          url: link,
          method: 'GET'
        });
        if (result !== 'Success' || data?.status !== 200) {
          debug('请求失败', {
            result: result,
            status: data?.status
          });
          return false;
        }
        if (data.responseText.includes('class="event-winner"')) {
          debug('检测到已中奖');
          return 'Won';
        }
        if (data.responseText.includes('class="event-ended"')) {
          debug('检测到活动已结束');
          return 'Ended';
        }
        debug('检测到活动进行中');
        return 'Active';
      } catch (error) {
        debug('检查 givee.club 链接出错', {
          error: error
        });
        throwError(error, 'leftKeyChecker.giveeClub');
        return false;
      }
    },
    async gleam(link) {
      try {
        debug('开始检查 gleam.io 链接', {
          link: link
        });
        const {result: result, data: data} = await httpRequest({
          url: link,
          method: 'GET'
        });
        if (result !== 'Success' || data?.status !== 200) {
          debug('请求失败', {
            result: result,
            status: data?.status
          });
          return false;
        }
        if (/incentives&quot;:{&quot;[\d]+?&quot;:\[&quot;.+?&quot;\]/.test(data.responseText)) {
          debug('检测到已中奖');
          return 'Won';
        }
        const campaignDiv = data.responseText.match(/<div class='popup-blocks-container'[\w\W]+?'>/)?.[0];
        if (!campaignDiv) {
          debug('未找到活动信息');
          return false;
        }
        const campaignString = $(campaignDiv).attr('ng-init')?.match(/initCampaign\(([\w\W]+?)\)$/)?.[1];
        if (!campaignString) {
          debug('未找到活动初始化数据');
          return false;
        }
        const {campaign: campaign} = JSON.parse(campaignString);
        debug('解析活动数据', {
          campaign: campaign
        });
        if (campaign.banned) {
          debug('检测到活动已被禁止');
          return 'Banned';
        }
        if (campaign.finished) {
          debug('检测到活动已结束');
          return 'Ended';
        }
        if (campaign.paused) {
          debug('检测到活动已暂停');
          return 'Paused';
        }
        if ((new Date).getTime() < campaign.starts_at * 1e3) {
          debug('检测到活动未开始');
          return 'NotStart';
        }
        debug('检测到活动进行中');
        return 'Active';
      } catch (error) {
        debug('检查 gleam.io 链接出错', {
          error: error
        });
        throwError(error, 'leftKeyChecker.gleam');
        return false;
      }
    },
    async indieDb(link) {
      try {
        debug('开始检查 indiedb.com 链接', {
          link: link
        });
        const {result: result, data: data} = await httpRequest({
          url: link,
          method: 'GET'
        });
        if (result !== 'Success' || data?.status !== 200) {
          debug('请求失败', {
            result: result,
            status: data?.status
          });
          return false;
        }
        if (data.responseText.includes('Congrats you WON')) {
          debug('检测到已中奖');
          return 'Won';
        }
        if (data.responseText.includes('Giveaway is closed') || data.responseText.includes('next time')) {
          debug('检测到抽奖已结束');
          return 'Ended';
        }
        debug('检测到抽奖进行中');
        return 'Active';
      } catch (error) {
        debug('检查 indiedb.com 链接出错', {
          error: error
        });
        throwError(error, 'leftKeyChecker.indieDb');
        return false;
      }
    },
    async keyhub(link) {
      try {
        debug('开始检查 key-hub.eu 链接', {
          link: link
        });
        const {result: result, data: data} = await httpRequest({
          url: link,
          method: 'GET'
        });
        if (result !== 'Success' || data?.status !== 200) {
          debug('请求失败', {
            result: result,
            status: data?.status
          });
          return false;
        }
        const keysleft = data.responseText.match(/<span id="keysleft">([\d]+?)<\/span>/)?.[1];
        if (!keysleft) {
          debug('未找到剩余密钥信息');
          return false;
        }
        debug('检测到剩余密钥数量', {
          keysleft: keysleft
        });
        if (keysleft === '0') {
          debug('检测到密钥已用完');
          return 'Ended';
        }
        debug('检测到活动进行中');
        return `Active(${keysleft})`;
      } catch (error) {
        debug('检查 key-hub.eu 链接出错', {
          error: error
        });
        throwError(error, 'leftKeyChecker.keyhub');
        return false;
      }
    },
    async opquests(link) {
      try {
        debug('开始检查 opquests.com 链接', {
          link: link
        });
        const {result: result, data: data} = await httpRequest({
          url: link,
          method: 'GET'
        });
        if (data?.status === 404) {
          debug('检测到活动不存在');
          return 'Ended';
        }
        if (result !== 'Success' || data?.status !== 200) {
          debug('请求失败', {
            result: result,
            status: data?.status
          });
          return false;
        }
        const keysleft = data.responseText.match(/<div class="">[\s]*?([\d]+?)[\s]*?of/)?.[1];
        if (!keysleft) {
          debug('未找到剩余密钥信息');
          return false;
        }
        debug('检测到剩余密钥数量', {
          keysleft: keysleft
        });
        if (keysleft === '0') {
          debug('检测到密钥已用完');
          return 'Ended';
        }
        debug('检测到活动进行中');
        return `Active(${keysleft})`;
      } catch (error) {
        debug('检查 opquests.com 链接出错', {
          error: error
        });
        throwError(error, 'leftKeyChecker.opquests');
        return false;
      }
    },
    async itch(link) {
      try {
        debug('开始检查 itch.io 链接', {
          link: link
        });
        const {result: result, data: data} = await httpRequest({
          url: link,
          method: 'GET'
        });
        if (result !== 'Success' || data?.status !== 200) {
          debug('请求失败', {
            result: result,
            status: data?.status
          });
          return false;
        }
        const endDate = data.responseText.match(/{"start_date":"[0-9A-Z-:]+?".*?"end_date":"([0-9A-Z-:]+?)".*?}/)?.[1];
        if (!endDate) {
          debug('未找到结束日期信息');
          return false;
        }
        debug('检测到活动结束日期', {
          endDate: endDate
        });
        if ((new Date).getTime() > new Date(endDate).getTime()) {
          debug('检测到活动已结束');
          return 'Ended';
        }
        const formattedEndDate = dayjs(endDate).format('YYYY-MM-DD HH:mm:ss');
        debug('检测到活动进行中', {
          formattedEndDate: formattedEndDate
        });
        return `Active(${formattedEndDate})`;
      } catch (error) {
        debug('检查 itch.io 链接出错', {
          error: error
        });
        throwError(error, 'leftKeyChecker.itch');
        return false;
      }
    },
    async freeanywhere(link) {
      try {
        debug('开始检查 freeanywhere.net 链接', {
          link: link
        });
        const {result: result, data: data} = await httpRequest({
          url: link,
          method: 'GET'
        });
        if (result !== 'Success' || data?.status !== 200) {
          debug('请求失败', {
            result: result,
            status: data?.status
          });
          return false;
        }
        const giveawayStatus = data.responseText.includes('Giveaway ended');
        if (giveawayStatus) {
          debug('检测到活动已结束');
          return 'Ended';
        }
        debug('检测到活动进行中');
        return 'Active';
      } catch (error) {
        debug('检查 freeanywhere.net 链接出错', {
          error: error
        });
        throwError(error, 'leftKeyChecker.freeanywhere');
        return false;
      }
    }
  };
  const defaultTasksTemplate$3 = {
    steam: {
      groupLinks: [],
      wishlistLinks: [],
      curatorLinks: [],
      curatorLikeLinks: [],
      followLinks: [],
      forumLinks: [],
      announcementLinks: [],
      workshopVoteLinks: [],
      licenseLinks: []
    },
    discord: {
      serverLinks: []
    },
    vk: {
      nameLinks: []
    },
    twitch: {
      channelLinks: []
    },
    reddit: {
      redditLinks: []
    },
    twitter: {
      userLinks: [],
      retweetLinks: []
    },
    youtube: {
      channelLinks: [],
      likeLinks: []
    }
  };
  const defaultTasks$4 = JSON.stringify(defaultTasksTemplate$3);
  class Keylol extends Website {
    name='Keylol';
    socialTasks=JSON.parse(defaultTasks$4);
    undoneTasks=JSON.parse(defaultTasks$4);
    buttons=[ 'doTask', 'undoTask', 'selectAll', 'selectNone', 'invertSelect' ];
    static CONFIG={
      LINK_PATTERNS: {
        DISCORD: /^https?:\/\/discord\.com\/invite\/.+/,
        REDDIT: /^https?:\/\/www\.reddit\.com\/(r|user)\/.+/,
        INSTAGRAM: /^https:\/\/www\.instagram\.com\/.+/,
        TWITTER: /^https:\/\/(twitter|x)\.com\/.+/,
        TWITTER_RETWEET: /https:\/\/(twitter|x)\.com\/.*?\/status\/[\d]+/,
        TWITCH: /^https:\/\/(www\.)?twitch\.tv\/.+/,
        VK: /^https:\/\/vk\.com\/.+/,
        STEAM_CURATOR: /curator\/[\d]+/,
        STEAM_PUBLISHER: /(publisher|developer|franchise)\/.+/,
        STEAM_NEWS: /news(hub)?\/app\/[\d]+\/view\/[\d]+/,
        STEAM_APP: /app\/[\d]+/,
        STEAM_GROUP: /groups\/.+/,
        STEAM_ANNOUNCEMENT: /announcements\/detail\/[\d]+/,
        YOUTUBE: /youtube\.com/
      },
      SELECTORS: {
        MAIN_POST: {
          KEYLOL: '#postlist>div[id^="post_"]:first',
          DEFAULT: 'div.container'
        }
      }
    };
    static test() {
      const {host: host} = window.location;
      const link = $('.subforum_left_title_left_up a').eq(3).attr('href');
      const isMatch = host === 'keylol.com' && (!!link?.includes('319') || !!link?.includes('234'));
      debug('检查网站匹配', {
        host: host,
        link: link,
        isMatch: isMatch
      });
      return isMatch;
    }
    init() {
      debug('初始化 Keylol');
      return true;
    }
    after() {
      try {
        debug('开始处理页面链接');
        const mainPost = $(this.name === 'Keylol' ? Keylol.CONFIG.SELECTORS.MAIN_POST.KEYLOL : Keylol.CONFIG.SELECTORS.MAIN_POST.DEFAULT);
        const allLinks = mainPost.find('a');
        debug('找到所有链接', {
          count: allLinks.length
        });
        allLinks.each(((_, link) => {
          const $link = $(link);
          const href = $link.attr('href');
          if (!href) {
            return;
          }
          this.#classifyAndProcessLink($link, href);
        }));
        debug('开始处理抽奖链接');
        this.#processGiveawayLinks(mainPost);
        if (this.name === 'Keylol') {
          debug('开始处理 Keylol 特定链接');
          this.#processKeylolSpecificLinks(mainPost);
        }
        debug('设置 MutationObserver');
        this.#setupMutationObserver();
      } catch (error) {
        debug('处理页面链接失败', {
          error: error
        });
        throwError(error, 'keylol.after');
      }
    }
    #classifyAndProcessLink($link, href) {
      debug('分类处理链接', {
        href: href
      });
      const {LINK_PATTERNS: LINK_PATTERNS} = Keylol.CONFIG;
      switch (true) {
       case LINK_PATTERNS.DISCORD.test(href):
        debug('发现 Discord 链接');
        this.#addBtn($link[0], 'discord', 'serverLinks', href);
        break;

       case LINK_PATTERNS.REDDIT.test(href):
        debug('发现 Reddit 链接');
        this.#addBtn($link[0], 'reddit', 'redditLinks', href);
        break;

       case LINK_PATTERNS.TWITTER.test(href):
        if (LINK_PATTERNS.TWITTER_RETWEET.test(href)) {
          debug('发现 Twitter 转发链接');
          this.#addBtn($link[0], 'twitter', 'retweetLinks', href);
        } else {
          debug('发现 Twitter 用户链接');
          this.#addBtn($link[0], 'twitter', 'userLinks', href);
        }
        break;

       case LINK_PATTERNS.TWITCH.test(href):
        debug('发现 Twitch 链接');
        this.#addBtn($link[0], 'twitch', 'channelLinks', href);
        break;

       case LINK_PATTERNS.VK.test(href):
        debug('发现 VK 链接');
        this.#addBtn($link[0], 'vk', 'nameLinks', href);
        break;

       case href.includes('store.steampowered.com'):
        debug('发现 Steam 商店链接');
        this.#processSteamStoreLink($link[0], href);
        break;

       case href.includes('steamcommunity.com'):
        debug('发现 Steam 社区链接');
        this.#processSteamCommunityLink($link[0], href);
        break;

       case LINK_PATTERNS.YOUTUBE.test(href):
        debug('发现 YouTube 链接');
        this.#addBtn($link[0], 'youtube', 'channelLinks', href);
        this.#addBtn($link[0], 'youtube', 'likeLinks', href);
        break;
      }
    }
    #processSteamStoreLink(element, href) {
      debug('处理 Steam 商店链接', {
        href: href
      });
      const {LINK_PATTERNS: LINK_PATTERNS} = Keylol.CONFIG;
      if (LINK_PATTERNS.STEAM_CURATOR.test(href)) {
        debug('发现 Steam 鉴赏家链接');
        this.#addBtn(element, 'steam', 'curatorLinks', href);
      } else if (LINK_PATTERNS.STEAM_PUBLISHER.test(href)) {
        debug('发现 Steam 发行商链接');
        this.#addBtn(element, 'steam', 'curatorLikeLinks', href);
      } else if (LINK_PATTERNS.STEAM_NEWS.test(href)) {
        debug('发现 Steam 新闻链接');
        this.#addBtn(element, 'steam', 'announcementLinks', href);
      } else if (LINK_PATTERNS.STEAM_APP.test(href)) {
        debug('发现 Steam 应用链接');
        this.#addBtn(element, 'steam', 'followLinks', href);
        this.#addBtn(element, 'steam', 'wishlistLinks', href);
      }
    }
    #processSteamCommunityLink(element, href) {
      debug('处理 Steam 社区链接', {
        href: href
      });
      const {LINK_PATTERNS: LINK_PATTERNS} = Keylol.CONFIG;
      if (LINK_PATTERNS.STEAM_GROUP.test(href)) {
        debug('发现 Steam 组链接');
        this.#addBtn(element, 'steam', 'groupLinks', href);
      } else if (LINK_PATTERNS.STEAM_ANNOUNCEMENT.test(href)) {
        debug('发现 Steam 公告链接');
        this.#addBtn(element, 'steam', 'announcementLinks', href);
      }
    }
    #processGiveawayLinks(mainPost) {
      debug('开始处理抽奖链接');
      const giveawayLinks = mainPost.find('a[href*="giveaway.su/giveaway/view/"],' + 'a[href*="givee.club/"],' + 'a[href*="gleam.io/"],' + 'a[href*="www.indiedb.com/giveaways/"],' + 'a[href*="key-hub.eu/giveaway/"],' + 'a[href*="opquests.com/quests/"],' + 'a[href*="freeanywhere.net/game?n="],' + 'a[href*="itch.io/s/"]:visible');
      debug('找到抽奖链接', {
        count: giveawayLinks.length
      });
      giveawayLinks.each(((_, link) => {
        const href = $(link).attr('href');
        if (!href) {
          return;
        }
        debug('检查抽奖链接状态', {
          href: href
        });
        leftKeyChecker.classify(href).then((status => {
          if (!status) {
            return;
          }
          const statusClass = /^Active/.test(status) ? 'active' : 'not-active';
          const statusTitle = /^Active/.test(status) ? I18n('Active') : I18n(status);
          debug('更新抽奖链接状态', {
            href: href,
            status: status,
            statusClass: statusClass
          });
          $(`a[href="${href}"]`).after(`<font class="auto-task-giveaway-status ${statusClass}" title="${statusTitle}">${status}</font>`);
        })).catch((error => {
          debug('检查抽奖链接状态失败', {
            href: href,
            error: error
          });
          throwError(error, 'keylol.after -> leftKeyChecker');
        }));
      }));
    }
    #processKeylolSpecificLinks(mainPost) {
      debug('开始处理 Keylol 特定链接');
      const asfLinks = mainPost.find('a[href^="#asf"]:visible');
      debug('找到 ASF 链接', {
        count: asfLinks.length
      });
      asfLinks.each(((_, link) => {
        const href = $(link).attr('href');
        if (!href) {
          return;
        }
        debug('处理 ASF 链接', {
          href: href
        });
        const $link = $(`a[href="${href}"]`);
        $link.after('<span style="color: #ccc; margin: 0 -5px 0 5px"> | </span>');
        this.#addBtn($link.next()[0], 'steam', 'licenseLinks', `appid-${href.replace('#asf', '')}`);
      }));
      const steamDbLinks = mainPost.find('a[href*="steamdb.info/sub/"]:visible');
      debug('找到 SteamDB 链接', {
        count: steamDbLinks.length
      });
      steamDbLinks.each(((_, link) => {
        const href = $(link).attr('href');
        if (!href) {
          return;
        }
        const subid = href.match(/^https:\/\/steamdb\.info\/sub\/([\d]+)/)?.[1];
        if (!subid) {
          return;
        }
        debug('处理 SteamDB 链接', {
          href: href,
          subid: subid
        });
        this.#addBtn(link, 'steam', 'licenseLinks', `subid-${subid}`);
      }));
      const asfBlocks = mainPost.find('.blockcode:contains("addlicense"):visible');
      debug('找到 ASF 代码块', {
        count: asfBlocks.length
      });
      asfBlocks.each(((_, block) => {
        const appid = [ ...block.innerText.matchAll(/a(pp)?\/([\d]+)/g) ].map((matched => matched?.[2])).filter((id => id));
        if (appid.length > 0) {
          debug('处理 ASF 代码块 appid', {
            appid: appid
          });
          this.#addBtn($(block).children('em')[0], 'steam', 'licenseLinks', `appid-${appid.join(',')}`);
        }
        const subid = block.innerText.match(/[\d]+/g)?.filter((matched => !appid.includes(matched)));
        if (subid?.length) {
          debug('处理 ASF 代码块 subid', {
            subid: subid
          });
          this.#addBtn($(block).children('em')[0], 'steam', 'licenseLinks', `subid-${subid.join(',')}`);
        }
      }));
    }
    #setupMutationObserver() {
      debug('设置 MutationObserver');
      if ($('#threadindex').length > 0) {
        const [elementTargetNode] = $('#postlist').children('div[id^="post_"]');
        const elementObserver = new MutationObserver((() => {
          debug('检测到 DOM 变化,重新处理页面');
          elementObserver.disconnect();
          this.after();
        }));
        elementObserver.observe(elementTargetNode, {
          childList: true
        });
        debug('MutationObserver 设置完成');
      }
    }
    classifyTask(action) {
      try {
        debug('开始分类任务', {
          action: action
        });
        this.socialTasks = JSON.parse(defaultTasks$4);
        this.undoneTasks = JSON.parse(defaultTasks$4);
        const selectedBtns = $('.auto-task-keylol[selected="selected"]:visible').get();
        debug('找到选中的按钮', {
          count: selectedBtns.length
        });
        for (const btn of selectedBtns) {
          const social = btn.getAttribute('data-social');
          const type = btn.getAttribute('data-type');
          const link = btn.getAttribute('data-link');
          debug('处理任务按钮', {
            social: social,
            type: type,
            link: link
          });
          if (!(social && type && link)) {
            debug('跳过无效任务按钮');
            continue;
          }
          if (!(social in this.undoneTasks)) {
            debug('跳过未知社交平台', {
              social: social
            });
            continue;
          }
          if (action === 'do' && type in this.undoneTasks[social]) {
            debug('添加到未完成任务', {
              social: social,
              type: type,
              link: link
            });
            this.undoneTasks[social][type].push(link);
          }
          if (action === 'undo' && type in this.socialTasks[social]) {
            debug('添加到社交任务', {
              social: social,
              type: type,
              link: link
            });
            this.socialTasks[social][type].push(link);
          }
        }
        this.undoneTasks = this.uniqueTasks(this.undoneTasks);
        this.socialTasks = this.uniqueTasks(this.socialTasks);
        debug('任务分类完成', {
          undoneTasks: this.undoneTasks,
          socialTasks: this.socialTasks
        });
        return true;
      } catch (error) {
        debug('任务分类失败', {
          error: error
        });
        throwError(error, 'Keylol.classifyTask');
        return false;
      }
    }
    selectAll() {
      try {
        debug('选择所有可见任务');
        const tasks = $('.auto-task-keylol:visible');
        tasks.attr('selected', 'selected');
        debug('选择完成', {
          count: tasks.length
        });
      } catch (error) {
        debug('选择所有任务失败', {
          error: error
        });
        throwError(error, 'Keylol.selectAll');
      }
    }
    selectNone() {
      try {
        debug('取消选择所有可见任务');
        const tasks = $('.auto-task-keylol:visible');
        tasks.removeAttr('selected');
        debug('取消选择完成', {
          count: tasks.length
        });
      } catch (error) {
        debug('取消选择所有任务失败', {
          error: error
        });
        throwError(error, 'Keylol.selectNone');
      }
    }
    invertSelect() {
      try {
        debug('反转选择所有可见任务');
        const tasks = $('.auto-task-keylol:visible');
        tasks.each(((_, element) => {
          const $element = $(element);
          if ($element.attr('selected')) {
            $element.removeAttr('selected');
          } else {
            $element.attr('selected', 'selected');
          }
        }));
        debug('反转选择完成', {
          count: tasks.length
        });
      } catch (error) {
        debug('反转选择任务失败', {
          error: error
        });
        throwError(error, 'Keylol.invertSelect');
      }
    }
    #addBtn(before, social, linkType, link) {
      try {
        debug('添加任务按钮', {
          social: social,
          linkType: linkType,
          link: link
        });
        if (!before || !social || !linkType || !link) {
          debug('跳过无效按钮参数');
          return;
        }
        const button = $('<a>', {
          href: 'javascript:void(0);',
          class: 'auto-task-keylol',
          target: '_self',
          'data-social': social,
          'data-type': linkType,
          'data-link': link,
          text: linkType.replace('Links', ''),
          onclick: 'this.getAttribute("selected") ? this.removeAttribute("selected") : this.setAttribute("selected", "selected")'
        });
        $(before).after(button);
        debug('按钮添加成功');
      } catch (error) {
        debug('添加按钮失败', {
          error: error,
          social: social,
          linkType: linkType
        });
        throwError(error, `keylol.addBtn: ${social}/${linkType}`);
      }
    }
  }
  const defaultTasks$3 = {
    steam: {
      groupLinks: [],
      wishlistLinks: [],
      followLinks: [],
      curatorLikeLinks: [],
      playTimeLinks: []
    },
    twitter: {
      userLinks: [],
      retweetLinks: []
    },
    discord: {
      serverLinks: []
    }
  };
  class Opquests extends Website {
    name='Opquests';
    undoneTasks={
      ...defaultTasks$3
    };
    buttons=[ 'doTask', 'verifyTask', 'getKey' ];
    static test() {
      const {host: host} = window.location;
      const isMatch = host === 'opquests.com';
      debug('检查网站匹配', {
        host: host,
        isMatch: isMatch
      });
      return isMatch;
    }
    async after() {
      try {
        debug('开始执行后续操作');
        if (!this.#checkLogin()) {
          debug('检查登录失败');
          echoLog({}).warning(I18n('checkLoginFailed'));
        }
        const opquestsVerifyTasks = GM_getValue('opquestsVerifyTasks') || [];
        debug('获取待验证任务', {
          count: opquestsVerifyTasks.length
        });
        if (opquestsVerifyTasks.length > 0) {
          const taskId = opquestsVerifyTasks.pop();
          debug('处理任务', {
            taskId: taskId
          });
          GM_setValue('opquestsVerifyTasks', opquestsVerifyTasks);
          const [verifyBtn] = $(`#task_id[value="${taskId}"]`).parent().find('button[type="button"],button[type="submit"]').has('i.fa-check');
          if (verifyBtn) {
            debug('点击验证按钮');
            verifyBtn.click();
            return;
          }
          debug('未找到验证按钮,继续处理下一个任务');
          this.after();
          return;
        }
        if (GM_getValue('opquestsVerifyTasks')) {
          debug('清除验证任务缓存');
          GM_deleteValue('opquestsVerifyTasks');
        }
      } catch (error) {
        debug('后续操作失败', {
          error: error
        });
        throwError(error, 'Opquests.after');
      }
    }
    init() {
      try {
        debug('开始初始化');
        const logStatus = echoLog({
          text: I18n('initing')
        });
        if ($('a[href*="/auth/redirect"]').length > 0) {
          debug('需要登录');
          window.open('/auth/redirect', '_self');
          logStatus.warning(I18n('needLogin'));
          return false;
        }
        if (!this.#getGiveawayId()) {
          debug('获取抽奖ID失败');
          return false;
        }
        this.initialized = true;
        debug('初始化完成');
        logStatus.success();
        return true;
      } catch (error) {
        debug('初始化失败', {
          error: error
        });
        throwError(error, 'Opquests.init');
        return false;
      }
    }
    async classifyTask(action) {
      try {
        debug('开始分类任务', {
          action: action
        });
        if (action === 'undo') {
          debug('不支持撤销操作');
          echoLog({
            text: I18n('cannotUndo')
          });
          return false;
        }
        const logStatus = echoLog({
          text: I18n('getTasksInfo')
        });
        const tasks = $('.w-full:contains("Validate") .items-center');
        debug('找到任务', {
          count: tasks.length
        });
        for (const task of tasks) {
          const link = $(task).find('a:contains("Open")').attr('href');
          if (!link) {
            debug('跳过无链接任务');
            continue;
          }
          const taskDes = $(task).find('div').eq(1).text().trim();
          debug('处理任务', {
            taskDes: taskDes,
            link: link
          });
          if (/steamcommunity\.com\/groups\//.test(link)) {
            debug('添加 Steam 组任务');
            this.undoneTasks.steam.groupLinks.push(link);
            continue;
          }
          if (/store\.steampowered\.com\/app\//.test(link)) {
            if (/wishlist/gim.test(taskDes)) {
              debug('添加 Steam 愿望单任务');
              this.undoneTasks.steam.wishlistLinks.push(link);
            } else if (/follow/gim.test(taskDes)) {
              debug('添加 Steam 关注任务');
              this.undoneTasks.steam.followLinks.push(link);
            } else if (/play/gim.test(taskDes)) {
              const time = parseInt(taskDes.replace(/\s/gim, '').match(/(\d+)hours/im)?.[1] || '0', 10) * 60;
              debug('添加 Steam 游戏时长任务', {
                time: time
              });
              this.undoneTasks.steam.playTimeLinks.push(`${time}-${link}`);
            }
            continue;
          }
          if (/store\.steampowered\.com\/(publisher|developer|curator)\//.test(link) && /follow/gim.test(taskDes)) {
            debug('添加 Steam 鉴赏家关注任务');
            this.undoneTasks.steam.curatorLikeLinks.push(link);
            continue;
          }
          if (link.includes('//x.com/')) {
            if (/follow/gim.test(taskDes)) {
              debug('添加 Twitter 关注任务');
              this.undoneTasks.twitter.userLinks.push(link);
              continue;
            }
            if (link.includes('status') && /Repost/gim.test(taskDes)) {
              debug('添加 Twitter 转发任务');
              this.undoneTasks.twitter.retweetLinks.push(link);
              continue;
            }
          }
          if (link.includes('//discord.com/')) {
            if (/join/gim.test(taskDes)) {
              debug('添加 Discord 加入任务');
              this.undoneTasks.discord.serverLinks.push(link);
              continue;
            }
          }
          if (link.includes('//discord.gg/') && /join/gim.test(taskDes)) {
            debug('获取重定向链接', {
              link: link
            });
            const taskLink = await getRedirectLink(link, false);
            if (!taskLink) {
              debug('获取重定向链接失败');
              continue;
            }
            debug('添加 Discord 加入任务');
            this.undoneTasks.discord.serverLinks.push(taskLink);
            continue;
          }
          if (/clash\.gg/.test(link)) {
            debug('跳过不支持的 Clash.gg 任务');
            echoLog({}).warning(`${I18n('unSupporttedTaskType')}: ${taskDes}(${link})`);
            continue;
          }
          debug('未知任务类型');
          echoLog({}).warning(`${I18n('unKnownTaskType')}: ${taskDes}(${link})`);
        }
        debug('任务分类完成');
        logStatus.success();
        this.undoneTasks = this.uniqueTasks(this.undoneTasks);
        if (window.DEBUG) {
          console.log('%cAuto-Task[Debug]:', 'color:blue', JSON.stringify(this));
        }
        return true;
      } catch (error) {
        debug('任务分类失败', {
          error: error
        });
        throwError(error, 'Opquests.classifyTask');
        return false;
      }
    }
    async verifyTask() {
      try {
        debug('开始验证任务');
        if (!this.initialized) {
          debug('未初始化,执行初始化');
          this.init();
        }
        const tasks = $.makeArray($('.items-center').has('input[name="task_id"]')).map((ele => $(ele).find('input[name="task_id"]').val()));
        debug('获取待验证任务', {
          count: tasks.length
        });
        GM_setValue('opquestsVerifyTasks', tasks);
        await this.#confirm();
        debug('执行后续操作');
        this.after();
        return true;
      } catch (error) {
        debug('验证任务失败', {
          error: error
        });
        throwError(error, 'Opquests.verifyTask');
        return false;
      }
    }
    async #confirm() {
      try {
        debug('开始确认任务');
        const logStatus = echoLog({
          html: `<li>${I18n('confirmingTask')}...<font></font></li>`
        });
        debug('发送确认请求');
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://opquests.com/quests/${this.giveawayId}?confirm=1`,
          method: 'GET',
          nochche: true,
          headers: {
            origin: 'https://opquests.com',
            referer: `https://opquests.com/warning?id=${this.giveawayId}`
          }
        });
        if (result !== 'Success') {
          debug('请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200) {
          debug('响应错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('确认成功');
        logStatus.success();
        return true;
      } catch (error) {
        debug('确认任务失败', {
          error: error
        });
        throwError(error, 'Opquests.confirm');
        return false;
      }
    }
    async getKey(isButton) {
      try {
        if ($('[name="task_id"]').length > 0) {
          debug('有任务未完成,不获取密钥');
          echoLog({}).warning(I18n('taskNotFinished'));
          return false;
        }
        debug('开始获取密钥', {
          isButton: isButton
        });
        const logStatus = echoLog({
          text: I18n('gettingKey')
        });
        debug('发送获取密钥请求');
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://opquests.com/keys',
          method: 'GET'
        });
        if (result !== 'Success') {
          debug('请求失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (!data?.responseText) {
          debug('响应无效', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        const questTitle = $('h1.font-bold').text().trim().replace(' Quest', '');
        const key = $(data.responseText).find(`div.items-center:contains("${questTitle}")`).find('div.font-bold').next().text();
        debug('查找密钥', {
          questTitle: questTitle,
          hasKey: !!key
        });
        if (!key) {
          debug('未找到密钥');
          logStatus.error('Error: Key was not found');
          if (isButton) {
            debug('重定向到密钥页面');
            window.open('https://opquests.com/keys', '_self');
          }
          return false;
        }
        debug('获取密钥成功');
        logStatus.success();
        echoLog({}).success(key);
        return true;
      } catch (error) {
        debug('获取密钥失败', {
          error: error
        });
        throwError(error, 'Opquests.getKey');
        return false;
      }
    }
    #getGiveawayId() {
      try {
        debug('开始获取抽奖ID');
        const giveawayId = window.location.href.match(/quests\/([\d]+)/)?.[1];
        if (giveawayId) {
          this.giveawayId = giveawayId;
          debug('获取抽奖ID成功', {
            giveawayId: giveawayId
          });
          return true;
        }
        debug('获取抽奖ID失败');
        echoLog({}).error(I18n('getFailed', 'GiveawayId'));
        return false;
      } catch (error) {
        debug('获取抽奖ID出错', {
          error: error
        });
        throwError(error, 'Opquests.getGiveawayId');
        return false;
      }
    }
    #checkLogin() {
      try {
        debug('检查登录状态');
        if (!globalOptions.other.checkLogin) {
          debug('跳过登录检查');
          return true;
        }
        if ($('a[href*="/auth/redirect"]').length > 0) {
          debug('未登录,重定向到登录页面');
          window.open('/auth/redirect', '_self');
        }
        debug('登录检查完成');
        return true;
      } catch (error) {
        debug('检查登录失败', {
          error: error
        });
        throwError(error, 'Opquests.checkLogin');
        return false;
      }
    }
  }
  const defaultTasksTemplate$2 = {
    steam: {
      groupLinks: [],
      wishlistLinks: [],
      followLinks: [],
      curatorLinks: [],
      curatorLikeLinks: [],
      playTimeLinks: []
    },
    twitter: {
      userLinks: [],
      retweetLinks: []
    },
    twitch: {
      channelLinks: []
    },
    discord: {
      serverLinks: []
    },
    youtube: {
      channelLinks: []
    },
    extra: {
      gleam: []
    }
  };
  const defaultTasks$2 = JSON.stringify(defaultTasksTemplate$2);
  class Gleam extends Website {
    name='Gleam';
    undoneTasks=JSON.parse(defaultTasks$2);
    socialTasks=JSON.parse(defaultTasks$2);
    buttons=[ 'doTask', 'undoTask', 'verifyTask' ];
    static test() {
      const {host: host} = window.location;
      const isMatch = host === 'gleam.io';
      debug('检查网站匹配', {
        host: host,
        isMatch: isMatch
      });
      return isMatch;
    }
    before() {
      try {
        debug('重写全局对话框函数');
        unsafeWindow.confirm = () => {};
        unsafeWindow.alert = () => {};
        unsafeWindow.prompt = () => {};
      } catch (error) {
        debug('重写全局对话框函数失败', {
          error: error
        });
        throwError(error, 'Gleam.before');
      }
    }
    async after() {
      try {
        debug('开始执行后续操作');
        if (window.location.search.includes('8b07d23f4bfa65f9')) {
          debug('检测到特殊查询参数,开始处理任务');
          const checkComplete = setInterval((() => {
            if ($('.entry-content .entry-method i.fa-check').length > 0) {
              debug('任务已完成,关闭窗口');
              clearInterval(checkComplete);
              window.close();
            }
          }));
          await this.verifyTask();
          echoLog({}).warning(I18n('gleamTaskNotice'));
        } else if (!await this.#checkLeftKey()) {
          debug('检查剩余密钥失败');
          echoLog({}).warning(I18n('checkLeftKeyFailed'));
        }
      } catch (error) {
        debug('后续操作失败', {
          error: error
        });
        throwError(error, 'Gleam.after');
      }
    }
    init() {
      try {
        debug('初始化 Gleam');
        const logStatus = echoLog({
          text: I18n('initing')
        });
        if (!this.#getGiveawayId()) {
          debug('获取抽奖ID失败');
          return false;
        }
        this.initialized = true;
        debug('初始化完成');
        logStatus.success();
        return true;
      } catch (error) {
        debug('初始化失败', {
          error: error
        });
        throwError(error, 'Gleam.init');
        return false;
      }
    }
    async classifyTask(action) {
      try {
        debug('开始分类任务', {
          action: action
        });
        const logStatus = echoLog({
          text: I18n('getTasksInfo')
        });
        if (action === 'undo') {
          debug('恢复已保存的任务信息');
          this.socialTasks = GM_getValue(`gleamTasks-${this.giveawayId}`)?.tasks || JSON.parse(defaultTasks$2);
        }
        const tasks = $('.entry-content .entry-method');
        debug('找到任务元素', {
          count: tasks.length
        });
        for (const task of tasks) {
          const $task = $(task);
          if (action === 'do' && $task.find('i.fa-question').length === 0) {
            debug('跳过已完成的任务');
            continue;
          }
          const socialIcon = $task.find('.icon-wrapper i');
          const taskInfo = $task.find('.user-links');
          const taskText = taskInfo.text().trim();
          const expandInfo = $task.find('.expandable');
          const aElements = expandInfo.find('a.btn');
          debug('处理任务', {
            taskText: taskText
          });
          if (aElements.length > 0) {
            debug('处理可点击元素', {
              count: aElements.length
            });
            for (const element of aElements) {
              const $element = $(element);
              const href = $element.attr('href');
              $element.removeAttr('href')[0].click();
              $element.attr('href', href);
            }
          }
          if (socialIcon.hasClass('fa-twitter') || socialIcon.hasClass('fa-x-twitter')) {
            const link = $task.find('a[href^="https://twitter.com/"],a[href^="https://x.com/"]').attr('href');
            if (!link) {
              continue;
            }
            if (/follow/gi.test(taskText)) {
              if (action === 'undo') {
                this.socialTasks.twitter.userLinks.push(link);
              }
              if (action === 'do') {
                this.undoneTasks.twitter.userLinks.push(link);
              }
              continue;
            }
            if (/retweet/gim.test(taskText)) {
              if (action === 'undo') {
                this.socialTasks.twitter.retweetLinks.push(link);
              }
              if (action === 'do') {
                this.undoneTasks.twitter.retweetLinks.push(link);
              }
              continue;
            }
          }
          if (socialIcon.hasClass('fa-twitch') && /follow/gim.test(taskText)) {
            const link = $task.find('a[href^="https://twitch.tv/"]').attr('href');
            if (!link) {
              continue;
            }
            if (action === 'undo') {
              this.socialTasks.twitch.channelLinks.push(link);
            }
            if (action === 'do') {
              this.undoneTasks.twitch.channelLinks.push(link);
            }
            continue;
          }
          if (socialIcon.hasClass('fa-discord') && /join/gim.test(taskText)) {
            let link = $task.find('a[href^="https://discord.com/invite/"]').attr('href');
            if (!link) {
              const ggLink = $task.find('a[href^="https://discord.gg/"]').attr('href')?.match(/discord\.gg\/([^/]+)/)?.[1];
              if (!ggLink) {
                continue;
              }
              link = `https://discord.com/invite/${ggLink}`;
            }
            if (action === 'undo') {
              this.socialTasks.discord.serverLinks.push(link);
            }
            if (action === 'do') {
              this.undoneTasks.discord.serverLinks.push(link);
            }
            continue;
          }
          if (socialIcon.hasClass('fa-external-link-square-alt')) {
            continue;
          }
          if (socialIcon.hasClass('fa-youtube') && /subscribe/gim.test(taskText)) {
            const link = $task.find('a[href^="https://www.youtube.com/channel/"]').attr('href');
            if (!link) {
              continue;
            }
            if (action === 'undo') {
              this.socialTasks.youtube.channelLinks.push(link);
            }
            if (action === 'do') {
              this.undoneTasks.youtube.channelLinks.push(link);
            }
            continue;
          }
          if (socialIcon.attr('class')?.includes('steam')) {
            if (/join.*group/gi.test(taskText)) {
              const link = $task.find('a[href^="https://steamcommunity.com/groups/"]').attr('href');
              if (!link) {
                continue;
              }
              if (action === 'undo') {
                this.socialTasks.steam.groupLinks.push(link);
              }
              if (action === 'do') {
                this.undoneTasks.steam.groupLinks.push(link);
              }
              continue;
            }
            if (/follow.*curator/gi.test(taskText)) {
              const link = $task.find('a[href^="https://store.steampowered.com/curator/"]').attr('href');
              if (!link) {
                continue;
              }
              if (action === 'undo') {
                this.socialTasks.steam.curatorLinks.push(link);
              }
              if (action === 'do') {
                this.undoneTasks.steam.curatorLinks.push(link);
              }
              continue;
            }
            if (/play[\w\W]*hours/gi.test(taskText)) {
              const link = $task.find('a[href^="https://steamcommunity.com/app/"],a[href^="https://store.steampowered.com/app/"]').attr('href');
              const time = [ ...taskText.matchAll(/(\d+?(\.\d+)?)\s*?hour/gi) ];
              if (!link || !time[0]?.[1]) {
                continue;
              }
              const trueTime = parseFloat(time[0][1]) * 60;
              if (action === 'do') {
                this.undoneTasks.steam.playTimeLinks.push(`${trueTime}-${link}`);
              }
              continue;
            }
            if (/Sign up/gi.test(taskText)) {
              continue;
            }
          }
          if (socialIcon.hasClass('fa-bullhorn') && /Complete|Increase/gi.test(taskText)) {
            if (action !== 'do') {
              continue;
            }
            const gleamLink = await this.#getGleamLink(taskText);
            if (!gleamLink) {
              continue;
            }
            this.undoneTasks.extra.gleam.push(gleamLink);
            continue;
          }
          if (socialIcon.hasClass('fa-question') || socialIcon.hasClass('fa-reddit') || socialIcon.hasClass('fa-instagram') || socialIcon.hasClass('fa-facebook-f') || socialIcon.hasClass('fa-telegram-plane') || socialIcon.hasClass('fa-telegram') || socialIcon.hasClass('fa-vk') || socialIcon.hasClass('fa-envelope') || socialIcon.hasClass('fa-gift') || socialIcon.hasClass('fa-square-up-right') || socialIcon.hasClass('fa-gamepad-modern') || socialIcon.hasClass('fa-dollar-sign') || socialIcon.hasClass('fa-tiktok') || socialIcon.hasClass('fa-gamepad-alt') || socialIcon.hasClass('fa-bag-shopping') || socialIcon.hasClass('fa-swords') || socialIcon.hasClass('fa-shield') && taskText.includes('one of our giveaways') || socialIcon.hasClass('fa-shield') && taskText.includes('Check out') || socialIcon.hasClass('fa-shield') && taskText.includes('vloot.io')) {
            continue;
          }
          echoLog({}).warning(`${I18n('unKnownTaskType')}: ${taskText}`);
        }
        debug('任务分类完成');
        logStatus.success();
        this.undoneTasks = this.uniqueTasks(this.undoneTasks);
        this.socialTasks = this.uniqueTasks(this.socialTasks);
        debug('保存任务信息');
        GM_setValue(`gleamTasks-${this.giveawayId}`, {
          tasks: this.socialTasks,
          time: (new Date).getTime()
        });
        return true;
      } catch (error) {
        debug('任务分类失败', {
          error: error
        });
        throwError(error, 'Gleam.classifyTask');
        return false;
      }
    }
    async extraDoTask({gleam: gleam}) {
      try {
        debug('开始执行额外任务', {
          count: gleam.length
        });
        const pro = [];
        for (const link of gleam) {
          pro.push(this.#doGleamTask(link));
        }
        return Promise.all(pro).then((() => true));
      } catch (error) {
        debug('执行额外任务失败', {
          error: error
        });
        throwError(error, 'Gleam.extraDoTask');
        return false;
      }
    }
    async #checkCampaign() {
      try {
        debug('检测人机验证');
        let logStatus;
        if ($('[campaign-key="campaign.key"]').length > 0) {
          logStatus = echoLog({
            text: I18n('campaign')
          });
          debug('检测到人机验证');
          await delay(3e3);
          logStatus.warning(I18n('retry'));
          await this.#checkCampaign();
          return true;
        }
        logStatus?.success();
        return false;
      } catch (error) {
        debug('检测人机验证失败', {
          error: error
        });
        throwError(error, 'Gleam.checkCampaign');
        return false;
      }
    }
    async verifyTask() {
      try {
        debug('开始验证任务');
        echoLog({
          text: `${I18n('verifyingTask')}...`
        });
        const tasks = $('.entry-content .entry-method');
        unsafeWindow._OxA = '_OxA';
        for (const task of tasks) {
          const campaign = await this.#checkCampaign();
          if (campaign) {
            return this.verifyTask();
          }
          const $task = $(task);
          if ($task.find('i.fa-check').length > 0) {
            debug('跳过已完成的任务');
            continue;
          }
          debug('处理任务验证');
          const taskInfo = $task.find('.user-links');
          taskInfo[0].click();
          const aElements = $task.find('.expandable').find('a.btn');
          if (aElements.length > 0) {
            debug('处理可点击元素', {
              count: aElements.length
            });
            for (const element of aElements) {
              const $element = $(element);
              const href = $element.attr('href');
              $element.removeAttr('href')[0].click();
              $element.attr('href', href);
            }
          }
          debug('处理计时器');
          unsafeWindow.$hookTimer?.setSpeed(1e3);
          const visitBtn = $task.find('.expandable').find('span:contains(more seconds),button:contains(more seconds)').filter(':visible');
          if (visitBtn.length > 0 && unsafeWindow.$hookTimer) {
            debug('处理访问按钮');
            const newTab = GM_openInTab('', {
              active: true
            });
            await delay(1e3);
            newTab?.close();
            window.focus();
          }
          await delay(3e3);
          unsafeWindow.$hookTimer?.setSpeed(1);
          const expandInfo = $task.find('.expandable');
          const [input] = expandInfo.find('input');
          if (input) {
            debug('处理输入框');
            const evt = new Event('input', {
              bubbles: true,
              cancelable: true,
              composed: true
            });
            const valuelimit = [ ...expandInfo.text().matchAll(/"(.+?)"/g) ].at(-1)?.[1];
            input.value = valuelimit || 'vloot';
            input.dispatchEvent(evt);
            await delay(1e3);
          }
          await this.#checkSync();
          const continueBtn = $task.find('.expandable').find('span:contains(Continue),button:contains(Continue),a:contains(Continue)');
          for (const button of continueBtn) {
            debug('点击继续按钮');
            button.click();
            await delay(500);
            await this.#checkSync();
          }
        }
        debug('任务验证完成');
        echoLog({
          text: I18n('verifiedGleamTasks')
        });
        return true;
      } catch (error) {
        debug('任务验证失败', {
          error: error
        });
        throwError(error, 'Gleam.verifyTask');
        return false;
      }
    }
    async #checkSync() {
      try {
        debug('开始检查同步状态');
        return await new Promise((resolve => {
          const checker = setInterval((() => {
            if ($('.entry-content .entry-method i.fa-sync').length === 0) {
              debug('同步完成');
              clearInterval(checker);
              resolve(true);
            }
          }), 500);
        }));
      } catch (error) {
        debug('检查同步状态失败', {
          error: error
        });
        throwError(error, 'Gleam.checkSync');
        return false;
      }
    }
    async #doGleamTask(link) {
      try {
        debug('执行 Gleam 任务', {
          link: link
        });
        const logStatus = echoLog({
          text: I18n('doingGleamTask')
        });
        return await new Promise((resolve => {
          GM_openInTab(`${link}?8b07d23f4bfa65f9`, {
            active: true,
            insert: true,
            setParent: true
          }).onclose = () => {
            debug('任务完成');
            logStatus.success();
            resolve(true);
          };
        }));
      } catch (error) {
        debug('执行 Gleam 任务失败', {
          error: error
        });
        throwError(error, 'Gleam.doGleamTask');
        return false;
      }
    }
    #getGiveawayId() {
      try {
        debug('获取抽奖ID');
        const giveawayId = window.location.pathname;
        if (giveawayId) {
          this.giveawayId = giveawayId;
          debug('获取抽奖ID成功', {
            giveawayId: giveawayId
          });
          return true;
        }
        debug('获取抽奖ID失败');
        echoLog({
          text: I18n('getFailed', 'GiveawayId')
        });
        return false;
      } catch (error) {
        debug('获取抽奖ID出错', {
          error: error
        });
        throwError(error, 'Gleam.getGiveawayId');
        return false;
      }
    }
    async #getGleamLink(title) {
      try {
        debug('获取 Gleam 链接', {
          title: title
        });
        const logStatus = echoLog({
          text: I18n('gettingGleamLink')
        });
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: 'https://www.vloot.io/api/v1/giveaways',
          method: 'GET',
          responseType: 'json'
        });
        if (result === 'Success') {
          if (data?.status === 200 && data?.response?.Success === true && data?.response?.Data) {
            const {link: link} = data.response.Data.find((giveaway => title.replace(/[\s]/g, '').toLowerCase().includes(giveaway.title.replace(/[\s]/g, '').toLowerCase()))) || {};
            if (link) {
              debug('获取链接成功', {
                link: link
              });
              logStatus.success();
              return link;
            }
            debug('获取链接失败');
            logStatus.error(`Error:${I18n('getLinkFailed')}`);
            return false;
          }
          debug('API响应错误', {
            status: data?.status,
            statusText: data?.statusText
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('请求失败', {
          result: result,
          status: status,
          statusText: statusText
        });
        logStatus.error(`${result}:${statusText}(${status})`);
        return false;
      } catch (error) {
        debug('获取 Gleam 链接失败', {
          error: error
        });
        throwError(error, 'Gleam.getGleamLink');
        return false;
      }
    }
    async #checkLeftKey() {
      try {
        debug('检查剩余密钥');
        if (!globalOptions.other.checkLeftKey) {
          debug('跳过密钥检查');
          return true;
        }
        const campaignString = $('div.popup-blocks-container').attr('ng-init')?.match(/initCampaign\(([\w\W]+?)\)$/)?.[1];
        if (!campaignString) {
          debug('未找到活动配置信息');
          return false;
        }
        const {campaign: campaign, incentive: incentive} = JSON.parse(campaignString);
        const controllerString = $('div.campaign.reward').attr('ng-init')?.match(/initContestant\(([\w\W]+?)\);/)?.[1];
        let ownedKey = false;
        if (controllerString) {
          if (JSON.parse(controllerString).contestant?.claims?.incentives?.[incentive.id]?.length) {
            debug('用户已拥有密钥');
            ownedKey = true;
          }
        }
        const isGiveawayInvalid = campaign.banned || campaign.finished && !ownedKey || campaign.paused || (new Date).getTime() < campaign.starts_at * 1e3;
        debug('检查抽奖状态', {
          banned: campaign.banned,
          finished: campaign.finished,
          ownedKey: ownedKey,
          paused: campaign.paused,
          notStarted: (new Date).getTime() < campaign.starts_at * 1e3
        });
        if (!isGiveawayInvalid) {
          return true;
        }
        debug('抽奖无效,显示确认对话框');
        const {value: value} = await Swal.fire({
          icon: 'warning',
          title: I18n('notice'),
          text: I18n('giveawayNotWork'),
          confirmButtonText: I18n('confirm'),
          cancelButtonText: I18n('cancel'),
          showCancelButton: true
        });
        if (value) {
          debug('用户确认关闭窗口');
          window.close();
        }
        return true;
      } catch (error) {
        debug('检查剩余密钥失败', {
          error: error
        });
        throwError(error, 'Gleam.checkLeftKey');
        return false;
      }
    }
  }
  const defaultOptions = {
    username: '',
    email: ''
  };
  class SweepWidget extends Website {
    name='SweepWidget';
    options={
      ...defaultOptions,
      ...GM_getValue('SweepWidgetOptions')
    };
    buttons=[ 'doTask' ];
    static test() {
      const {host: host} = window.location;
      const isMatch = /^sweepwidget\.com$/.test(host);
      debug('检查网站匹配', {
        host: host,
        isMatch: isMatch
      });
      return /^https?:\/\/sweepwidget\.com\/view\/[\d]+/.test(window.location.href);
    }
    async after() {
      try {
        debug('开始执行后续操作');
        if (!this.#checkLogin()) {
          debug('检查登录失败');
          echoLog({}).warning(I18n('checkLoginFailed'));
        }
      } catch (error) {
        debug('后续操作失败', {
          error: error
        });
        throwError(error, 'SweepWidget.after');
      }
    }
    init() {
      try {
        debug('开始初始化');
        const logStatus = echoLog({
          text: I18n('initing')
        });
        if (!this.#checkLogin()) {
          debug('需要登录');
          logStatus.warning(I18n('needLogin'));
          return false;
        }
        if (!this.#getGiveawayId()) {
          debug('获取抽奖ID失败');
          return false;
        }
        this.initialized = true;
        debug('初始化完成');
        logStatus.success();
        return true;
      } catch (error) {
        debug('初始化失败', {
          error: error
        });
        throwError(error, 'SweepWidget.init');
        return false;
      }
    }
    classifyTask() {
      debug('任务分类完成');
      return true;
    }
    async doTask() {
      try {
        debug('开始执行任务');
        if ($('#unlock_rewards_main_wrapper').length === 0) {
          debug('未显示奖励界面,尝试登录');
          if ($('input[name="sw__login_name"]:visible').length > 0) {
            debug('填写用户名', {
              username: this.options.username
            });
            $('input[name="sw__login_name"]').val(this.options.username);
          }
          if ($('input[name="sw__login_email"]:visible').length > 0) {
            debug('填写邮箱', {
              email: this.options.email
            });
            $('input[name="sw__login_email"]').val(this.options.email);
          }
          if ($('#sw_login_button:visible').length > 0) {
            debug('点击登录按钮');
            $('#sw_login_button')[0].click();
          }
          const isEntered = await this.#checkEnter();
          if (!isEntered) {
            debug('进入抽奖失败');
            return false;
          }
        }
        const logStatus = echoLog({
          text: I18n('SweepWidgetNotice')
        });
        const tasks = $('#sw_inner_entry_methods_l2_wrapper>div.sw_entry');
        debug('找到任务列表', {
          count: tasks.length
        });
        for (const task of tasks) {
          const $task = $(task);
          if ($task.find('i.fa-check:visible').length > 0) {
            debug('跳过已完成的任务');
            continue;
          }
          const title = $task.find('.sw_text_inner');
          const aElement = $task.find('a.sw_link');
          const link = aElement.attr('href');
          if (!link) {
            debug('跳过无效链接的任务');
            continue;
          }
          debug('处理任务', {
            title: title.text(),
            link: link
          });
          title[0].click();
          aElement.attr('href', '#a').attr('target', '_self');
          aElement[0]?.click();
          await delay(300);
          aElement.attr('href', link).attr('target', '_blank');
          debug('填写测试文本');
          $task.find('input[type="text"]').val('test');
          const verifyBtn = $task.find('input.sw_verify');
          if (verifyBtn.prop('disabled') === true) {
            debug('验证按钮被禁用,尝试重新激活');
            title[0].click();
            await delay(300);
            title[0].click();
            await delay(300);
          }
          debug('点击验证按钮');
          $task.find('input.sw_verify').removeAttr('disabled')[0]?.click();
          await this.#checkFinish($task);
          const randomDelay = parseInt(`${Math.random() * (3e3 - 1e3 + 1) + 1e3}`, 10);
          debug('等待随机延迟', {
            delay: randomDelay
          });
          await delay(randomDelay);
        }
        debug('所有任务执行完成');
        logStatus.success();
        return true;
      } catch (error) {
        debug('执行任务失败', {
          error: error
        });
        throwError(error, 'SweepWidget.doTask');
        return false;
      }
    }
    #checkLogin() {
      try {
        debug('检查登录状态');
        if ($('#twitter_login_button').length > 0) {
          debug('点击 Twitter 登录按钮');
          $('#twitter_login_button')[0].click();
        }
        debug('登录检查完成');
        return true;
      } catch (error) {
        debug('检查登录失败', {
          error: error
        });
        throwError(error, 'SweepWidget.checkLogin');
        return false;
      }
    }
    #getGiveawayId() {
      try {
        debug('开始获取抽奖ID');
        const giveawayId = window.location.href.match(/\/view\/([\d]+)/)?.[1];
        if (!giveawayId) {
          debug('获取抽奖ID失败');
          echoLog({
            text: I18n('getFailed', 'GiveawayId')
          });
          return false;
        }
        this.giveawayId = giveawayId;
        debug('获取抽奖ID成功', {
          giveawayId: giveawayId
        });
        return true;
      } catch (error) {
        debug('获取抽奖ID出错', {
          error: error
        });
        throwError(error, 'SweepWidget.getGiveawayId');
        return false;
      }
    }
    async #checkEnter() {
      try {
        debug('开始检查是否进入抽奖');
        return new Promise((resolve => {
          const checker = setInterval((() => {
            if ($('#unlock_rewards_main_wrapper').length === 0) {
              debug('等待进入抽奖...');
              return;
            }
            debug('成功进入抽奖');
            clearInterval(checker);
            resolve(true);
          }), 500);
        }));
      } catch (error) {
        debug('检查进入抽奖失败', {
          error: error
        });
        throwError(error, 'SweepWidget.checkEnter');
        return false;
      }
    }
    async #checkFinish($task) {
      try {
        debug('开始检查任务完成状态');
        return new Promise((resolve => {
          const checker = setInterval((() => {
            const isCompleted = $task.find('i.fa-check:visible').length > 0 || $task.find('.sw_entry_input:visible').length === 0;
            if (!isCompleted) {
              debug('等待任务完成...');
              return;
            }
            debug('任务完成');
            clearInterval(checker);
            resolve(true);
          }), 500);
        }));
      } catch (error) {
        debug('检查任务完成状态失败', {
          error: error
        });
        throwError(error, 'SweepWidget.checkFinish');
        return false;
      }
    }
  }
  const processFormData = formData => {
    debug('开始处理表单数据', {
      formDataLength: formData.length
    });
    const data = {};
    formData.forEach((value => {
      data[value.name] = value.value;
    }));
    debug('表单数据处理完成', {
      processedData: data
    });
    return data;
  };
  const updateGlobalOption = (element, data) => {
    const name = $(element).attr('name');
    if (!name) {
      debug('元素缺少name属性', {
        element: element
      });
      return;
    }
    debug('开始更新全局选项', {
      name: name
    });
    const keys = name.split('.');
    const value = data[name];
    const processedValue = value ? value === 'on' ? true : value : value ?? false;
    debug('处理选项值', {
      keys: keys,
      originalValue: value,
      processedValue: processedValue
    });
    if (keys.length === 3) {
      globalOptions[keys[0]][keys[1]][keys[2]] = processedValue;
      debug('更新三级选项', {
        path: keys.join('.'),
        value: processedValue
      });
    } else if (keys.length === 2) {
      globalOptions[keys[0]][keys[1]] = processedValue;
      debug('更新二级选项', {
        path: keys.join('.'),
        value: processedValue
      });
    }
  };
  const generateFormRow = (type, option, data, isFirstOption, totalOptions) => {
    debug('开始生成表单行', {
      type: type,
      option: option,
      isFirstOption: isFirstOption,
      totalOptions: totalOptions
    });
    const backgroundColor = `${stringToColour(type)}44`;
    const headerBackgroundColor = `${stringToColour(type)}66`;
    if ([ 'other', 'position', 'hotKey', 'ASF' ].includes(type)) {
      const header = isFirstOption ? `<th rowspan="${totalOptions}" style="background-color: ${headerBackgroundColor}">${I18n(type)}</th>` : '';
      if (typeof data === 'boolean') {
        debug('生成布尔类型选项行', {
          type: type,
          option: option,
          value: data
        });
        return `\n        <tr style="background-color: ${backgroundColor}">\n          ${header}\n          <td>${I18n(option)}</td>\n          <td>\n            <label>\n              <input type="checkbox" name="${type}.${option}"${data ? ' checked="checked"' : ''}/>\n              <span><i></i></span>\n            </label>\n          </td>\n        </tr>`;
      }
      debug('生成文本类型选项行', {
        type: type,
        option: option,
        value: data
      });
      return `\n      <tr style="background-color: ${backgroundColor}">\n        ${header}\n        <td>${I18n(option)}</td>\n        <td>\n          <input class="editOption" type="text" name="${type}.${option}" value="${data}"/>\n        </td>\n      </tr>`;
    }
    debug('生成社交媒体选项行', {
      type: type,
      option: option,
      dataKeys: Object.keys(data)
    });
    return Object.entries(data).map((([socialType, value]) => `\n    <tr style="background-color: ${stringToColour(option)}66">\n      ${isFirstOption ? `<th rowspan="${totalOptions}" style="background-color: ${headerBackgroundColor}">${I18n(type)}</th>` : ''}\n      <td>${option}.${I18n(socialType)}</td>\n      <td>\n        <label>\n          <input type="checkbox" name="${type}.${option}.${socialType}"${value ? ' checked="checked"' : ''}/>\n          <span><i></i></span>\n        </label>\n      </td>\n    </tr>`)).join('');
  };
  const generateGlobalOptionsForm = () => {
    debug('开始生成全局选项表单');
    const formRows = Object.entries(globalOptions).map((([type, data1]) => {
      debug('处理选项类型', {
        type: type,
        optionsCount: Object.keys(data1).length
      });
      return Object.entries(data1).map((([option, data2], index) => {
        const totalOptions = [ 'other', 'position', 'hotKey', 'ASF' ].includes(type) ? Object.keys(data1).length : Object.values(data1).reduce(((acc, cur) => acc + Object.keys(cur).length), 0);
        return generateFormRow(type, option, data2, index === 0, totalOptions);
      })).join('');
    })).join('');
    debug('表单生成完成');
    return `\n    <form id="globalOptionsForm" class="auto-task-form">\n      <table class="auto-task-table">\n        <thead>\n          <tr>\n            <td>${I18n('type')}</td>\n            <td>${I18n('option')}</td>\n            <td>${I18n('value')}</td>\n          </tr>\n        </thead>\n        <tbody>\n          ${formRows}\n        </tbody>\n      </table>\n    </form>`;
  };
  const saveData = () => {
    try {
      debug('开始保存全局选项数据');
      const formData = $('#globalOptionsForm').serializeArray();
      debug('获取表单数据', {
        formDataLength: formData.length
      });
      const data = processFormData(formData);
      debug('开始更新全局选项');
      $.makeArray($('#globalOptionsForm input')).forEach((element => {
        updateGlobalOption(element, data);
      }));
      GM_setValue('globalOptions', globalOptions);
      debug('全局选项保存完成');
      Swal.fire({
        title: I18n('changeGlobalOptionsSuccess'),
        icon: 'success'
      });
    } catch (error) {
      debug('保存全局选项时发生错误', {
        error: error
      });
      throwError(error, 'saveData');
    }
  };
  const changeGlobalOptions = showType => {
    try {
      debug('开始显示全局选项配置界面', {
        showType: showType
      });
      const formHtml = generateGlobalOptionsForm();
      if (showType === 'swal') {
        debug('使用弹窗显示选项');
        Swal.fire({
          title: I18n('globalOptions'),
          html: formHtml,
          showConfirmButton: true,
          confirmButtonText: I18n('save'),
          showCancelButton: true,
          cancelButtonText: I18n('close')
        }).then((({isConfirmed: isConfirmed}) => {
          if (isConfirmed) {
            debug('用户确认保存选项');
            saveData();
          } else {
            debug('用户取消保存选项');
          }
        }));
      } else {
        debug('使用页面内显示选项');
        $('body').append(`<h2>${I18n('globalOptions')}</h2>${formHtml}`);
      }
    } catch (error) {
      debug('显示全局选项配置界面时发生错误', {
        error: error
      });
      throwError(error, 'changeGlobalOptions');
    }
  };
  const defaultWhiteList = {
    discord: {
      servers: []
    },
    instagram: {
      users: []
    },
    twitch: {
      channels: []
    },
    twitter: {
      users: [],
      retweets: [],
      likes: []
    },
    vk: {
      names: []
    },
    youtube: {
      channels: [],
      likes: []
    },
    reddit: {
      reddits: []
    },
    steam: {
      groups: [],
      officialGroups: [],
      wishlists: [],
      follows: [],
      forums: [],
      workshops: [],
      curators: [],
      workshopVotes: [],
      curatorLikes: [],
      announcements: [],
      licenses: [],
      playtests: [],
      playTime: []
    }
  };
  const REGEX_PATTERNS = {
    DISCORD_INVITE: /invite\/(.+)/,
    INSTAGRAM_USER: /https:\/\/www\.instagram\.com\/(.+)?\//,
    TWITCH_CHANNEL: /https:\/\/(www\.)?twitch\.tv\/(.+)/,
    TWITTER_USER: /https:\/\/twitter\.com\/(.+)/,
    TWITTER_STATUS: /https:\/\/twitter\.com\/.*?\/status\/([\d]+)/,
    VK_NAME: /https:\/\/vk\.com\/([^/]+)/,
    REDDIT_USER: /https?:\/\/www\.reddit\.com\/user\/([^/]*)/,
    REDDIT_SUBREDDIT: /https?:\/\/www\.reddit\.com\/r\/([^/]*)/,
    STEAM_GROUP: /groups\/(.+)\/?/,
    STEAM_APP: /app\/([\d]+)/,
    STEAM_WORKSHOP: /\?id=([\d]+)/,
    STEAM_CURATOR: /curator\/([\d]+)/,
    STEAM_STORE: /https?:\/\/store\.steampowered\.com\/(.*?)\/([^/?]+)/
  };
  const link2id = async function(type) {
    try {
      debug('开始从链接获取ID', {
        type: type
      });
      const link = $('#socialLink').val();
      let id = '';
      switch (type) {
       case 'discord.servers':
        id = REGEX_PATTERNS.DISCORD_INVITE.exec(link)?.[1] || '';
        break;

       case 'instagram.users':
        id = REGEX_PATTERNS.INSTAGRAM_USER.exec(link)?.[1] || '';
        break;

       case 'twitch.channels':
        id = REGEX_PATTERNS.TWITCH_CHANNEL.exec(link)?.[2] || '';
        break;

       case 'twitter.users':
        id = REGEX_PATTERNS.TWITTER_USER.exec(link)?.[1] || '';
        break;

       case 'twitter.retweets':
        id = REGEX_PATTERNS.TWITTER_STATUS.exec(link)?.[1] || '';
        break;

       case 'vk.names':
        id = REGEX_PATTERNS.VK_NAME.exec(link)?.[1] || '';
        break;

       case 'youtube.channels':
        id = (await getInfo(link, 'channel'))?.params?.channelId || '';
        break;

       case 'youtube.likes':
        id = (await getInfo(link, 'likeVideo'))?.params?.videoId || '';
        break;

       case 'reddit.reddits':
        {
          const userMatch = REGEX_PATTERNS.REDDIT_USER.exec(link);
          const subredditMatch = REGEX_PATTERNS.REDDIT_SUBREDDIT.exec(link);
          id = userMatch?.[1] || subredditMatch?.[1] || '';
          break;
        }

       case 'steam.groups':
        id = REGEX_PATTERNS.STEAM_GROUP.exec(link)?.[1] || '';
        break;

       case 'steam.wishlists':
       case 'steam.follows':
       case 'steam.forums':
       case 'steam.playtests':
       case 'steam.playTime':
        id = REGEX_PATTERNS.STEAM_APP.exec(link)?.[1] || '';
        break;

       case 'steam.workshops':
        id = REGEX_PATTERNS.STEAM_WORKSHOP.exec(link)?.[1] || '';
        break;

       case 'steam.curators':
        {
          if (link.includes('curator')) {
            id = REGEX_PATTERNS.STEAM_CURATOR.exec(link)?.[1] || '';
          } else {
            const storeMatch = REGEX_PATTERNS.STEAM_STORE.exec(link);
            if (!storeMatch) {
              break;
            }
            const [, param1, param2] = storeMatch;
            const steam = new Steam;
            if (await steam.init()) {
              id = await steam.getCuratorId(param1, param2) || '';
            }
          }
          break;
        }
      }
      debug('从链接获取ID结果', {
        type: type,
        id: id
      });
      return id;
    } catch (error) {
      debug('从链接获取ID时发生错误', {
        error: error
      });
      throwError(error, 'link2id');
      return I18n('getFailed', 'id');
    }
  };
  const disabledType = {
    steam: [ 'workshopVotes', 'curatorLikes', 'announcements' ],
    twitter: [ 'likes' ]
  };
  const assignWhiteList = whiteList => {
    try {
      debug('开始合并白名单');
      const newWhiteList = {};
      for (const [key, value] of Object.entries(defaultWhiteList)) {
        newWhiteList[key] = {
          ...value,
          ...whiteList[key]
        };
      }
      debug('白名单合并完成');
      return newWhiteList;
    } catch (error) {
      debug('合并白名单时发生错误', {
        error: error
      });
      throwError(error, 'assignWhiteList');
      return defaultWhiteList;
    }
  };
  const whiteListOptions = function(showType) {
    try {
      debug('开始显示白名单选项', {
        showType: showType
      });
      const whiteList = assignWhiteList(GM_getValue('whiteList') || {});
      let whiteListOptionsForm = `<form id="whiteListForm" class="auto-task-form">\n      <table class="auto-task-table">\n        <thead>\n          <tr>\n            <td>${I18n('website')}</td>\n            <td>${I18n('type')}</td>\n            <td>${I18n('edit')}</td>\n          </tr>\n        </thead>\n        <tbody>`;
      for (const [social, types] of Object.entries(whiteList)) {
        const validTypes = Object.keys(types).filter((type => !disabledType[social]?.includes(type)));
        whiteListOptionsForm += validTypes.map(((type, index) => {
          const bgColor = `${stringToColour(social)}66`;
          return `\n          <tr style="background-color: ${bgColor}">\n            ${index === 0 ? `<th rowspan="${validTypes.length}" style="background-color: ${bgColor}">${social}</th>` : ''}\n            <td>${I18n(type)}</td>\n            <td><button type="button" class="editWhiteList" data-value="${social}.${type}">${I18n('edit')}</button></td>\n          </tr>`;
        })).join('');
      }
      whiteListOptionsForm += '</tbody></table></form>';
      if (showType === 'swal') {} else {
        debug('使用页面显示白名单选项');
        $('body').append(`<h2>${I18n('whiteList')}</h2>${whiteListOptionsForm}`);
      }
      $('.editWhiteList').on('click', (function() {
        const value = $(this).attr('data-value');
        if (!value) {
          return;
        }
        const [social, type] = value.split('.');
        const currentList = whiteList[social]?.[type];
        if (!currentList) {
          debug('未找到白名单配置', {
            social: social,
            type: type
          });
          echoLog({}).warning(I18n('whiteListNotFound', value));
          return;
        }
        debug('编辑白名单', {
          social: social,
          type: type
        });
        Swal.fire({
          title: I18n('changeWhiteListOption', value),
          input: 'textarea',
          html: `\n          <input id="socialLink" class="swal2-input" placeholder="在此处输入链接获取id">\n          <button id="link2id" data-type="${value}" class="swal2-confirm swal2-styled">获取id</button>\n          <p style="margin-bottom:0 !important;">在下方填写白名单,每行一个</p>\n        `,
          inputValue: currentList.join('\n'),
          showConfirmButton: true,
          confirmButtonText: I18n('save'),
          showCancelButton: true,
          cancelButtonText: I18n('close'),
          showDenyButton: true,
          denyButtonText: I18n('return')
        }).then((({isDenied: isDenied, isConfirmed: isConfirmed, value: value}) => {
          if (isDenied) {
            debug('返回白名单选项');
            if (showType === 'swal') {}
            return;
          }
          if (isConfirmed && value) {
            debug('保存白名单更改', {
              social: social,
              type: type,
              value: value
            });
            whiteList[social][type] = value.split('\n').filter(Boolean);
            GM_setValue('whiteList', whiteList);
            Swal.fire({
              title: I18n('changeWhiteListSuccess'),
              icon: 'success'
            });
          }
        }));
        $('#link2id').on('click', (async function() {
          const type = $(this).attr('data-type');
          if (!type) {
            return;
          }
          debug('从链接获取ID按钮点击', {
            type: type
          });
          const id = await link2id(type);
          $('#socialLink').val(id);
        }));
      }));
    } catch (error) {
      debug('显示白名单选项时发生错误', {
        error: error
      });
      throwError(error, 'whiteListOptions');
    }
  };
  const setGistData = async (token, gistId, fileName, content) => {
    try {
      debug('开始设置Gist数据', {
        gistId: gistId,
        fileName: fileName
      });
      const logStatus = echoLog({
        text: I18n('settingData')
      });
      const contentData = JSON.stringify({
        files: {
          [fileName]: {
            content: JSON.stringify(content)
          }
        }
      });
      debug('准备发送的数据', {
        contentData: contentData
      });
      const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
        url: `https://api.github.com/gists/${gistId}`,
        headers: {
          Accept: 'application/vnd.github.v3+json',
          Authorization: `token ${token}`
        },
        data: contentData,
        responseType: 'json',
        method: 'POST',
        timeout: 3e4
      });
      if (result !== 'Success') {
        debug('设置Gist数据失败', {
          result: result,
          statusText: statusText,
          status: status
        });
        logStatus.error(`${result}:${statusText}(${status})`);
        return false;
      }
      const expectedContent = JSON.stringify(content);
      if (data?.status !== 200 || data?.response?.files?.[fileName]?.content !== expectedContent) {
        debug('设置Gist数据验证失败', {
          status: data?.status,
          content: data?.response?.files?.[fileName]?.content
        });
        logStatus.error(`Error:${data?.statusText}(${data?.status})`);
        return false;
      }
      debug('设置Gist数据成功');
      logStatus.success();
      return true;
    } catch (error) {
      debug('设置Gist数据发生错误', {
        error: error
      });
      throwError(error, 'setGistData');
      return false;
    }
  };
  const getGistData = async (token, gistId, fileName, test = false) => {
    try {
      debug('开始获取Gist数据', {
        gistId: gistId,
        fileName: fileName,
        test: test
      });
      const logStatus = echoLog({
        text: I18n('gettingData')
      });
      const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
        url: `https://api.github.com/gists/${gistId}`,
        headers: {
          Accept: 'application/vnd.github.v3+json',
          Authorization: `token ${token}`
        },
        responseType: 'json',
        method: 'GET',
        timeout: 3e4
      });
      if (result !== 'Success') {
        debug('获取Gist数据失败', {
          result: result,
          statusText: statusText,
          status: status
        });
        logStatus.error(`${result}:${statusText}(${status})`);
        return false;
      }
      if (data?.status !== 200) {
        debug('获取Gist数据状态码错误', {
          status: data?.status
        });
        logStatus.error(`Error:${data?.statusText}(${data?.status})`);
        return false;
      }
      const content = data.response?.files?.[fileName]?.content;
      if (!content) {
        debug('获取的Gist数据为空');
        logStatus.error(`Error:${I18n('noRemoteData')}`);
        return false;
      }
      if (test) {
        debug('Gist数据测试成功');
        logStatus.success();
        return true;
      }
      try {
        const formatedContent = JSON.parse(content);
        debug('Gist数据解析成功', {
          contentLength: Object.keys(formatedContent).length
        });
        logStatus.success();
        return formatedContent;
      } catch (error) {
        debug('Gist数据解析失败', {
          error: error
        });
        logStatus.error(`Error:${I18n('errorRemoteDataFormat')}`);
        console.log('%c%s', 'color:white;background:red', `Auto-Task[Error]: getGistData\n${error.stack}`);
        return false;
      }
    } catch (error) {
      debug('获取Gist数据发生错误', {
        error: error
      });
      throwError(error, 'getGistData');
      return false;
    }
  };
  const syncOptions = async () => {
    try {
      debug('开始同步选项配置');
      const defaultOptions = {
        TOKEN: '',
        GIST_ID: '',
        FILE_NAME: '',
        SYNC_HISTORY: true
      };
      let syncOptions = GM_getValue('gistOptions') || defaultOptions;
      debug('获取已保存的同步选项', syncOptions);
      const createForm = options => `\n      <div class="gist-options-form">\n        <p>\n          <label for="github-token">Github Token</label>\n          <input\n            id="github-token"\n            class="swal2-input"\n            placeholder="Github Token"\n            value="${options.TOKEN}"\n            autocomplete="off"\n            spellcheck="false"\n          />\n        </p>\n        <p>\n          <label for="gist-id">Gist ID</label>\n          <input\n            id="gist-id"\n            class="swal2-input"\n            placeholder="Gist ID"\n            value="${options.GIST_ID}"\n            autocomplete="off"\n            spellcheck="false"\n          />\n        </p>\n        <p>\n          <label for="file-name">${I18n('fileName')}</label>\n          <input\n            id="file-name"\n            class="swal2-input"\n            placeholder="${I18n('fileName')}"\n            value="${options.FILE_NAME}"\n            autocomplete="off"\n            spellcheck="false"\n          />\n        </p>\n        <p class="sync-history-wrapper">\n          <label for="sync-history" class="swal2-checkbox-custom">\n            <input\n              id="sync-history"\n              type="checkbox"\n              ${options.SYNC_HISTORY ? 'checked="checked"' : ''}\n            />\n            <span class="swal2-label">${I18n('syncHistory')}</span>\n          </label>\n        </p>\n        <div class="button-group">\n          <button id="upload-data" type="button" class="swal2-confirm swal2-styled" onclick="handleUpload()">\n            ${I18n('upload2gist')}\n          </button>\n          <button id="download-data" type="button" class="swal2-confirm swal2-styled" onclick="handleDownload()">\n            ${I18n('downloadFromGist')}\n          </button>\n        </div>\n      </div>\n    `;
      const showConfigDialog = async () => {
        debug('显示配置对话框');
        const result = await Swal.fire({
          title: I18n('gistOptions'),
          html: createForm(syncOptions),
          focusConfirm: false,
          showLoaderOnConfirm: true,
          footer: `<a href="https://auto-task-doc.js.org/guide/#%E6%95%B0%E6%8D%AE%E5%90%8C%E6%AD%A5" target="_blank">${I18n('help')}</a>`,
          preConfirm: async () => {
            const options = {
              TOKEN: $('#github-token').val().trim(),
              GIST_ID: $('#gist-id').val().trim(),
              FILE_NAME: $('#file-name').val().trim(),
              SYNC_HISTORY: $('#sync-history').prop('checked')
            };
            debug('保存新的同步选项', options);
            GM_setValue('gistOptions', options);
            syncOptions = options;
            return await getGistData(options.TOKEN, options.GIST_ID, options.FILE_NAME, true);
          },
          allowOutsideClick: () => !Swal.isLoading(),
          confirmButtonText: I18n('saveAndTest'),
          showCancelButton: true,
          cancelButtonText: I18n('close')
        });
        if (result.value) {
          debug('配置测试成功');
          await Swal.fire({
            icon: 'success',
            title: I18n('testSuccess'),
            timer: 2e3,
            timerProgressBar: true
          });
          await showConfigDialog();
        } else if (result.value !== undefined) {
          debug('配置测试失败');
          await Swal.fire({
            icon: 'error',
            title: I18n('testFailed'),
            timer: 2e3,
            timerProgressBar: true
          });
          await showConfigDialog();
        }
      };
      const handleUpload = async () => {
        debug('开始处理数据上传');
        const options = GM_getValue('gistOptions');
        if (!options?.TOKEN || !options?.GIST_ID || !options?.FILE_NAME) {
          debug('配置信息不完整');
          await Swal.fire({
            icon: 'error',
            title: I18n('saveAndTestNotice')
          });
          await showConfigDialog();
          return;
        }
        debug('显示数据处理提示');
        Swal.fire({
          icon: 'info',
          title: I18n('processingData'),
          allowOutsideClick: false
        });
        const data = {};
        const names = GM_listValues();
        const syncHistory = $('#sync-history').prop('checked');
        debug('开始收集数据', {
          namesCount: names.length,
          syncHistory: syncHistory
        });
        for (const name of names) {
          if (name === 'gistOptions' || /^[\w]+?Auth$/.test(name)) {
            continue;
          }
          if (!syncHistory && /^[\w]+?Tasks-/.test(name)) {
            continue;
          }
          data[name] = GM_getValue(name);
        }
        debug('数据收集完成', {
          dataKeysCount: Object.keys(data).length
        });
        Swal.update({
          icon: 'info',
          title: I18n('updatingData')
        });
        const success = await setGistData(options.TOKEN, options.GIST_ID, options.FILE_NAME, data);
        debug('数据上传完成', {
          success: success
        });
        await Swal.fire({
          icon: success ? 'success' : 'error',
          title: I18n(success ? 'syncDataSuccess' : 'syncDataFailed'),
          timer: 2e3,
          timerProgressBar: true
        });
      };
      const handleDownload = async () => {
        debug('开始处理数据下载');
        const options = GM_getValue('gistOptions');
        if (!options?.TOKEN || !options?.GIST_ID || !options?.FILE_NAME) {
          debug('配置信息不完整');
          await Swal.fire({
            icon: 'error',
            title: I18n('saveAndTestNotice')
          });
          await showConfigDialog();
          return;
        }
        debug('显示数据下载提示');
        Swal.fire({
          icon: 'info',
          title: I18n('downloadingData'),
          allowOutsideClick: false
        });
        const data = await getGistData(options.TOKEN, options.GIST_ID, options.FILE_NAME);
        if (!data || typeof data === 'boolean') {
          debug('未检测到远程数据');
          await Swal.fire({
            icon: 'error',
            title: I18n('checkedNoData')
          });
          await showConfigDialog();
          return;
        }
        debug('开始保存数据');
        Swal.update({
          icon: 'info',
          title: I18n('savingData')
        });
        const syncHistory = $('#sync-history').prop('checked');
        let savedCount = 0;
        for (const [name, value] of Object.entries(data)) {
          if (!syncHistory && /^[\w]+?Tasks-/.test(name)) {
            continue;
          }
          GM_setValue(name, value);
          savedCount += 1;
        }
        debug('数据保存完成', {
          savedCount: savedCount
        });
        await Swal.fire({
          icon: 'success',
          title: I18n('syncDataSuccess'),
          timer: 2e3,
          timerProgressBar: true
        });
      };
      unsafeWindow.handleUpload = handleUpload;
      unsafeWindow.handleDownload = handleDownload;
      await showConfigDialog();
    } catch (error) {
      debug('同步选项发生错误', {
        error: error
      });
      throwError(error, 'syncOptions');
      await Swal.fire({
        icon: 'error',
        title: I18n('error'),
        text: error instanceof Error ? error.message : 'Unknown error occurred',
        timer: 3e3,
        timerProgressBar: true
      });
    }
  };
  const VALID_SIDES_X = [ 'right', 'left' ];
  const VALID_SIDES_Y = [ 'top', 'bottom' ];
  class Setting {
    name='Setting';
    buttons=[ 'saveGlobalOptions', 'syncData', 'tasksHistory' ];
    syncData=syncOptions;
    selectors={
      body: 'body',
      autoTaskInfo: '#auto-task-info',
      autoTaskButtons: '#auto-task-buttons',
      showButtonDiv: 'div.show-button-div',
      positionInputs: 'input[name^="position"]',
      hotKeyInputs: 'input[name^="hotKey"]'
    };
    tasksHistory() {
      debug('打开任务历史记录页面');
      GM_openInTab('https://auto-task.hclonely.com/history.html', {
        active: true
      });
    }
    static test() {
      const {host: host, pathname: pathname} = window.location;
      const isMatch = [ 'auto-task.hclonely.com', 'auto-task-doc.js.org' ].includes(host) && pathname === '/setting.html';
      debug('检查设置页面匹配', {
        host: host,
        pathname: pathname,
        isMatch: isMatch
      });
      return isMatch;
    }
    before() {
      try {
        debug('开始清空页面内容');
        $(this.selectors.body).html('').addClass('auto-task-options');
        debug('页面内容已清空');
      } catch (error) {
        debug('清空页面内容失败', {
          error: error
        });
        throwError(error, 'Setting.before');
      }
    }
    async after() {
      try {
        debug('开始初始化设置页面');
        await this.#initializeEnvironment();
        this.#initializeGlobalSettings();
        this.#setupSocialButtons();
        this.#setupPositionHandlers();
        this.#setupHotKeyHandlers();
        debug('设置页面初始化完成');
      } catch (error) {
        debug('设置页面初始化失败', {
          error: error
        });
        throwError(error, 'Setting.after');
      }
    }
    saveGlobalOptions() {
      try {
        debug('开始保存全局选项');
        saveData();
        debug('全局选项保存完成');
      } catch (error) {
        debug('保存全局选项失败', {
          error: error
        });
        throwError(error, 'Setting.saveGlobalOptions');
      }
    }
    async #initializeEnvironment() {
      try {
        debug('开始初始化环境信息');
        const userAgent = await browser.getInfo();
        debug('获取用户代理信息', {
          userAgent: userAgent
        });
        const environmentHtml = this.#generateEnvironmentHtml(userAgent);
        $(this.selectors.body).append(`<h2>${I18n('environment')}</h2>${environmentHtml}`);
        debug('环境信息初始化完成');
      } catch (error) {
        debug('初始化环境信息失败', {
          error: error
        });
        throwError(error, 'Setting.initializeEnvironment');
      }
    }
    #generateEnvironmentHtml(userAgent) {
      return `\n      <form id="environmentForm" class="auto-task-form">\n        <table class="auto-task-table">\n          <thead>\n            <tr>\n              <td>${I18n('type')}</td>\n              <td>${I18n('name')}</td>\n              <td>${I18n('version')}</td>\n            </tr>\n          </thead>\n          <tbody>\n            <tr>\n              <td>${I18n('os')}</td>\n              <td>${userAgent.system}</td>\n              <td>${userAgent.systemVersion}</td>\n            </tr>\n            <tr>\n              <td>${I18n('browser')}</td>\n              <td>${userAgent.browser}</td>\n              <td>${userAgent.browserVersion}</td>\n            </tr>\n            <tr>\n              <td>${I18n('scriptManager')}</td>\n              <td>${GM_info.scriptHandler}</td>\n              <td>${GM_info.version}</td>\n            </tr>\n            <tr>\n              <td>${I18n('script')}</td>\n              <td>${GM_info.script.name}</td>\n              <td>${GM_info.script.version}</td>\n            </tr>\n          </tbody>\n        </table>\n      </form>\n    `;
    }
    #initializeGlobalSettings() {
      debug('开始初始化全局设置');
      changeGlobalOptions('page');
      whiteListOptions('page');
      debug('全局设置初始化完成');
    }
    #setupSocialButtons() {
      debug('开始设置社交媒体按钮');
      this.#addSocialButton('other.twitterVerifyId', 'getTwitterUserId', 'twitterUser');
      this.#addSocialButton('other.youtubeVerifyChannel', 'getYoutubeChannelId', 'youtubeChannel');
      debug('社交媒体按钮设置完成');
    }
    #addSocialButton(inputName, buttonId, socialType) {
      debug('添加社交媒体按钮', {
        inputName: inputName,
        buttonId: buttonId,
        socialType: socialType
      });
      $(`input[name="${inputName}"]`).after(`<button id="${buttonId}" type="button">${I18n(`get${buttonId}`)}</button>`);
      $(`#${buttonId}`).on('click', (() => this.#getId(socialType)));
      debug('社交媒体按钮添加完成');
    }
    #setupPositionHandlers() {
      debug('开始设置位置处理器');
      $(this.selectors.positionInputs).on('input', (event => {
        const input = $(event.target);
        const type = input.attr('name')?.replace('position.', '');
        if (!type) {
          debug('无效的位置类型');
          return;
        }
        debug('处理位置变化', {
          type: type
        });
        this.#handlePositionChange(type);
      }));
      debug('位置处理器设置完成');
    }
    #handlePositionChange(type) {
      debug('开始处理位置变化', {
        type: type
      });
      const config = this.#getPositionConfig(type);
      if (!config) {
        debug('获取位置配置失败');
        return;
      }
      const {distance: distance, sideX: sideX, sideY: sideY} = config;
      if (!this.#isValidPosition(distance, sideX, sideY)) {
        debug('无效的位置配置', {
          distance: distance,
          sideX: sideX,
          sideY: sideY
        });
        return;
      }
      const [x, y] = distance.split(',');
      const target = this.#getPositionTarget(type);
      if (!target) {
        debug('获取目标元素失败');
        return;
      }
      debug('更新元素位置', {
        target: target,
        sideX: sideX,
        sideY: sideY,
        x: x,
        y: y
      });
      this.#updateElementPosition(target, sideX, sideY, x, y);
    }
    #getPositionConfig(type) {
      debug('获取位置配置', {
        type: type
      });
      const baseType = type.replace(/(?:button|log|show)(?:SideX|SideY|Distance)$/, '');
      const distance = $(`input[name="position.${baseType}Distance"]`).val();
      const sideX = $(`input[name="position.${baseType}SideX"]`).val();
      const sideY = $(`input[name="position.${baseType}SideY"]`).val();
      const config = {
        distance: distance,
        sideX: sideX,
        sideY: sideY
      };
      debug('位置配置', config);
      return config;
    }
    #isValidPosition(distance, sideX, sideY) {
      const isValid = VALID_SIDES_X.includes(sideX) && VALID_SIDES_Y.includes(sideY) && /^[\d]+?,[\d]+$/.test(distance);
      debug('验证位置配置', {
        distance: distance,
        sideX: sideX,
        sideY: sideY,
        isValid: isValid
      });
      return isValid;
    }
    #getPositionTarget(type) {
      const targetMap = {
        button: this.selectors.autoTaskButtons,
        showButton: this.selectors.showButtonDiv,
        log: this.selectors.autoTaskInfo
      };
      const baseType = type.replace(/(?:SideX|SideY|Distance)$/, '');
      return targetMap[baseType];
    }
    #updateElementPosition(selector, sideX, sideY, x, y) {
      debug('更新元素位置', {
        selector: selector,
        sideX: sideX,
        sideY: sideY,
        x: x,
        y: y
      });
      const $element = $(selector);
      const oppositeX = sideX === 'right' ? 'left' : 'right';
      const oppositeY = sideY === 'top' ? 'bottom' : 'top';
      $element.css(sideX, `${x}px`).css(sideY, `${y}px`).css(oppositeX, '').css(oppositeY, '');
      debug('元素位置更新完成');
    }
    #setupHotKeyHandlers() {
      debug('开始设置热键处理器');
      $(this.selectors.hotKeyInputs).attr('readonly', 'readonly').off('keydown').on('keydown', this.#handleHotKeyPress);
      debug('热键处理器设置完成');
    }
    #handleHotKeyPress(event) {
      debug('处理热键按下事件', {
        key: event.key
      });
      const functionKeys = [];
      if (event.altKey) {
        functionKeys.push('alt');
      }
      if (event.ctrlKey) {
        functionKeys.push('ctrl');
      }
      if (event.shiftKey) {
        functionKeys.push('shift');
      }
      const key = event.key.length === 1 ? event.key.toLowerCase() : '';
      const value = functionKeys.length ? `${functionKeys.join(' + ')} + ${key}` : key;
      debug('设置热键值', {
        functionKeys: functionKeys,
        key: key,
        value: value
      });
      $(event.target).val(value);
    }
    async #getId(social) {
      try {
        debug('开始获取社交媒体ID', {
          social: social
        });
        const result = await Swal.fire({
          title: I18n('getId', I18n(social)),
          html: this.#generateIdInputHtml(social),
          showCancelButton: true,
          cancelButtonText: I18n('close'),
          showConfirmButton: false
        });
        if (!result.isDismissed) {
          debug('用户确认获取ID');
          await this.#handleIdRetrieval(social);
        } else {
          debug('用户取消获取ID');
        }
      } catch (error) {
        debug('获取社交媒体ID失败', {
          error: error
        });
        throwError(error, 'Setting.getId');
      }
    }
    #generateIdInputHtml(social) {
      return `\n      <input id="socialLink" class="swal2-input" placeholder="在此处输入链接获取id">\n      <button id="link2id" data-type="${social}" class="swal2-confirm swal2-styled">获取id</button>\n    `;
    }
    async #handleIdRetrieval(social) {
      const link = $('#socialLink').val();
      if (!link) {
        debug('链接为空');
        return;
      }
      debug('开始处理ID获取', {
        social: social,
        link: link
      });
      let id = '';
      if (social === 'twitterUser') {
        const name = link.match(/https:\/\/twitter\.com\/(.+)/)?.[1] || link;
        debug('获取Twitter用户ID', {
          name: name
        });
        id = await (new Twitter).userName2id(name) || '';
      } else if (social === 'youtubeChannel') {
        const name = this.#extractYoutubeUrl(link);
        debug('获取YouTube频道信息', {
          name: name
        });
        const info = await getInfo(name, 'channel');
        id = info?.params?.channelId || '';
      }
      debug('ID获取结果', {
        id: id
      });
      $('#socialLink').val(id);
    }
    #extractYoutubeUrl(link) {
      debug('提取YouTube URL', {
        link: link
      });
      if (/^https:\/\/(www\.)?google\.com.*?\/url\?.*?url=https:\/\/www.youtube.com\/.*/.test(link)) {
        const extractedUrl = link.match(/url=(https:\/\/www\.youtube\.com\/.*)/)?.[1] || link;
        debug('从Google搜索结果提取URL', {
          extractedUrl: extractedUrl
        });
        return extractedUrl;
      }
      return link;
    }
  }
  class History extends Keylol {
    name='History';
    buttons=[ 'doTask', 'undoTask', 'selectAll', 'selectNone', 'invertSelect', 'clearHistory' ];
    static test() {
      try {
        const {host: host, pathname: pathname} = window.location;
        const isMatch = [ 'auto-task.hclonely.com', 'auto-task-doc.js.org' ].includes(host) && pathname === '/history.html';
        debug('检查是否为历史记录页面', {
          host: host,
          pathname: pathname,
          isMatch: isMatch
        });
        return isMatch;
      } catch (error) {
        debug('检查历史记录页面时出错', {
          error: error
        });
        throwError(error, 'History.test');
        return false;
      }
    }
    before() {
      try {
        debug('开始初始化历史记录页面');
        $('body').html('<div class="container"></div>').addClass('auto-task-history');
        debug('页面基础结构已创建');
        const data = GM_listValues() || [];
        debug('获取存储的所有值', {
          count: data.length
        });
        const tasksHistory = data.map((value => /^[\w]+?Tasks-/.test(value) ? value : null)).filter((value => value));
        debug('筛选任务历史记录', {
          count: tasksHistory.length
        });
        tasksHistory.forEach((item => {
          debug('处理任务项', {
            item: item
          });
          this.#addItem(item);
        }));
        debug('历史记录页面初始化完成');
      } catch (error) {
        debug('初始化历史记录页面时出错', {
          error: error
        });
        throwError(error, 'History.before');
      }
    }
    clearHistory() {
      try {
        debug('开始清除历史记录');
        const data = GM_listValues() || [];
        debug('获取存储的所有值', {
          count: data.length
        });
        const tasksHistory = data.map((value => /^[\w]+?Tasks-/.test(value) ? value : null)).filter((value => value));
        debug('筛选要清除的任务历史记录', {
          count: tasksHistory.length
        });
        tasksHistory.forEach((item => {
          debug('删除任务项', {
            item: item
          });
          GM_deleteValue(item);
        }));
        debug('历史记录清除完成');
        Swal.fire({
          title: I18n('clearHistoryFinished'),
          icon: 'success'
        });
      } catch (error) {
        debug('清除历史记录时出错', {
          error: error
        });
        throwError(error, 'History.clearHistory');
      }
    }
    #addItem(item) {
      try {
        debug('开始添加任务项', {
          item: item
        });
        const tasksData = GM_getValue(item);
        if (!tasksData?.tasks) {
          debug('任务数据无效', {
            item: item
          });
          return;
        }
        const {title: title, link: link} = this.#getTaskInfo(item);
        if (!title || !link) {
          debug('获取任务信息失败', {
            item: item
          });
          return;
        }
        debug('生成任务HTML', {
          item: item,
          title: title,
          link: link
        });
        const html = this.#generateTaskHtml(tasksData.tasks);
        this.#appendTaskToContainer(item, title, link, html, tasksData.time);
        this.#bindDeleteEvent();
        debug('任务项添加完成', {
          item: item
        });
      } catch (error) {
        debug('添加任务项时出错', {
          error: error,
          item: item
        });
        throwError(error, 'History.addItem');
      }
    }
    #getTaskInfo(item) {
      try {
        debug('开始获取任务信息', {
          item: item
        });
        const [website, id] = item.split('-');
        debug('解析任务标识符', {
          website: website,
          id: id
        });
        const taskInfoMap = {
          fawTasks: {
            title: `Freeanywhere[${id}]`,
            link: `https://freeanywhere.net/#/giveaway/${id}`
          },
          gasTasks: {
            title: `Giveawaysu[${id}]`,
            link: `https://giveaway.su/giveaway/view/${id}`
          },
          gcTasks: {
            title: `GiveeClub[${id}]`,
            link: `https://givee.club/event/${id}`
          },
          gkTasks: {
            title: `Givekey[${id}]`,
            link: `https://givekey.ru/giveaway/${id}`
          },
          gleamTasks: {
            title: `Gleam[${id}]`,
            link: `https://gleam.io${id}`
          },
          khTasks: {
            title: `keyhub[${id}]`,
            link: `https://key-hub.eu/giveaway/${id}`
          },
          prysTasks: {
            title: `Prys[${id}]`,
            link: `https://prys.revadike.com/giveaway/?id=${id}`
          }
        };
        const result = taskInfoMap[website] || {
          title: '',
          link: ''
        };
        debug('获取任务信息结果', {
          result: result
        });
        return result;
      } catch (error) {
        debug('获取任务信息时出错', {
          error: error,
          item: item
        });
        throwError(error, 'History.getTaskInfo');
        return {
          title: '',
          link: ''
        };
      }
    }
    #generateTaskHtml(tasks) {
      try {
        debug('开始生成任务HTML');
        let html = '';
        for (const [social, types] of Object.entries(tasks)) {
          for (const [type, taskList] of Object.entries(types)) {
            for (const task of taskList) {
              debug('处理任务', {
                social: social,
                type: type,
                task: task
              });
              const displayTask = task.length > 55 ? `${task.slice(0, 55)}...` : task;
              html += `<li>\n              <font class="auto-task-capitalize">${social}.${I18n(type.replace('Link', ''))}: </font>\n              <a href="${task}" target="_blank">${displayTask}</a>\n            </li>`;
            }
          }
        }
        debug('任务HTML生成完成');
        return html;
      } catch (error) {
        debug('生成任务HTML时出错', {
          error: error
        });
        throwError(error, 'History.generateTaskHtml');
        return '';
      }
    }
    #appendTaskToContainer(item, title, link, html, time) {
      try {
        debug('开始添加任务到容器', {
          item: item,
          title: title,
          link: link
        });
        $('.container').append(`\n        <div class="card" data-name="${item}">\n          <div class="title">\n            <a href="${link}" target="_blank">${title}</a>\n            <span class="delete-task" data-name="${item}" title="${I18n('deleteTask')}">\n              <svg class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="2734" width="32" height="32">\n                <path d="M607.897867 768.043004c-17.717453 0-31.994625-14.277171-31.994625-31.994625L575.903242 383.935495c0-17.717453 14.277171-31.994625 31.994625-31.994625s31.994625 14.277171 31.994625 31.994625l0 351.94087C639.892491 753.593818 625.61532 768.043004 607.897867 768.043004z" p-id="2735" fill="#d81e06"></path>\n                <path d="M415.930119 768.043004c-17.717453 0-31.994625-14.277171-31.994625-31.994625L383.935495 383.935495c0-17.717453 14.277171-31.994625 31.994625-31.994625 17.717453 0 31.994625 14.277171 31.994625 31.994625l0 351.94087C447.924744 753.593818 433.647573 768.043004 415.930119 768.043004z" p-id="2736" fill="#d81e06"></path>\n                <path d="M928.016126 223.962372l-159.973123 0L768.043004 159.973123c0-52.980346-42.659499-95.983874-95.295817-95.983874L351.94087 63.989249c-52.980346 0-95.983874 43.003528-95.983874 95.983874l0 63.989249-159.973123 0c-17.717453 0-31.994625 14.277171-31.994625 31.994625s14.277171 31.994625 31.994625 31.994625l832.032253 0c17.717453 0 31.994625-14.277171 31.994625-31.994625S945.73358 223.962372 928.016126 223.962372zM319.946246 159.973123c0-17.545439 14.449185-31.994625 31.994625-31.994625l320.806316 0c17.545439 0 31.306568 14.105157 31.306568 31.994625l0 63.989249L319.946246 223.962372 319.946246 159.973123 319.946246 159.973123z" p-id="2737" fill="#d81e06"></path>\n                <path d="M736.048379 960.010751 288.123635 960.010751c-52.980346 0-95.983874-43.003528-95.983874-95.983874L192.139761 383.591466c0-17.717453 14.277171-31.994625 31.994625-31.994625s31.994625 14.277171 31.994625 31.994625l0 480.435411c0 17.717453 14.449185 31.994625 31.994625 31.994625l448.096758 0c17.717453 0 31.994625-14.277171 31.994625-31.994625L768.215018 384.795565c0-17.717453 14.277171-31.994625 31.994625-31.994625s31.994625 14.277171 31.994625 31.994625l0 479.231312C832.032253 916.835209 789.028725 960.010751 736.048379 960.010751z" p-id="2738" fill="#d81e06"></path>\n              </svg>\n            </span>\n          </div>\n          <ul>${html}</ul>\n          <span class="time">${I18n('lastChangeTime')}: ${dayjs(time).format('YYYY-MM-DD HH:mm:ss')}</span>\n        </div>\n      `);
        debug('任务已添加到容器', {
          item: item
        });
      } catch (error) {
        debug('添加任务到容器时出错', {
          error: error,
          item: item
        });
        throwError(error, 'History.appendTaskToContainer');
      }
    }
    #bindDeleteEvent() {
      try {
        debug('开始绑定删除事件');
        $('span.delete-task').on('click', (function() {
          const itemName = $(this).attr('data-name');
          debug('点击删除按钮', {
            itemName: itemName
          });
          if (!itemName) {
            debug('删除失败:未找到任务名称');
            Swal.fire({
              title: I18n('clearTaskFailed'),
              icon: 'error'
            });
            return;
          }
          GM_deleteValue(itemName);
          $(`div.card[data-name="${itemName}"]`).remove();
          debug('任务删除成功', {
            itemName: itemName
          });
          Swal.fire({
            title: I18n('clearTaskFinished'),
            text: itemName,
            icon: 'success'
          });
        }));
        debug('删除事件绑定完成');
      } catch (error) {
        debug('绑定删除事件时出错', {
          error: error
        });
        throwError(error, 'History.bindDeleteEvent');
      }
    }
  }
  const defaultTasksTemplate$1 = {
    steam: {
      groupLinks: [],
      wishlistLinks: [],
      followLinks: [],
      curatorLinks: [],
      curatorLikeLinks: []
    },
    twitter: {
      userLinks: [],
      retweetLinks: []
    },
    twitch: {
      channelLinks: []
    },
    discord: {
      serverLinks: []
    },
    youtube: {
      channelLinks: []
    },
    extra: {
      giveawayHopper: []
    }
  };
  const defaultTasks$1 = JSON.stringify(defaultTasksTemplate$1);
  class GiveawayHopper extends Website {
    name='GiveawayHopper';
    undoneTasks=JSON.parse(defaultTasks$1);
    socialTasks=JSON.parse(defaultTasks$1);
    tasks=[];
    buttons=[ 'doTask', 'undoTask', 'verifyTask' ];
    static test() {
      const {host: host} = window.location;
      const isMatch = host === 'giveawayhopper.com';
      debug('检查网站匹配', {
        host: host,
        isMatch: isMatch
      });
      return isMatch;
    }
    async after() {
      try {
        debug('开始执行后续操作');
        if (!this.#checkLogin()) {
          debug('登录检查失败');
          echoLog({}).warning(I18n('checkLoginFailed'));
        }
        const giveawayIdResult = this.#getGiveawayId();
        debug('获取抽奖ID', {
          success: giveawayIdResult,
          id: this.giveawayId
        });
      } catch (error) {
        debug('后续操作失败', {
          error: error
        });
        throwError(error, 'GiveawayHopper.after');
      }
    }
    async init() {
      try {
        debug('初始化 GiveawayHopper');
        const logStatus = echoLog({
          text: I18n('initing')
        });
        const leftKeyResult = await this.#checkLeftKey();
        if (!leftKeyResult) {
          debug('检查剩余密钥失败');
          echoLog({}).warning(I18n('checkLeftKeyFailed'));
        }
        this.initialized = true;
        debug('初始化完成');
        logStatus.success();
        return true;
      } catch (error) {
        debug('初始化失败', {
          error: error
        });
        throwError(error, 'GiveawayHopper.init');
        return false;
      }
    }
    async classifyTask(action) {
      try {
        debug('开始分类任务', {
          action: action
        });
        if (!this.giveawayId) {
          debug('未找到抽奖ID,尝试获取');
          await this.#getGiveawayId();
        }
        const logStatus = echoLog({
          text: I18n('getTasksInfo')
        });
        if (action === 'undo') {
          debug('恢复已保存的任务信息');
          this.socialTasks = GM_getValue(`giveawayHopperTasks-${this.giveawayId}`)?.tasks || JSON.parse(defaultTasks$1);
        }
        debug('请求任务列表');
        const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
          url: `https://giveawayhopper.com/api/v1/campaigns/${this.giveawayId}/with-auth`,
          method: 'GET',
          responseType: 'json',
          headers: {
            authorization: `Bearer ${window.sessionStorage.gw_auth}`,
            'x-xsrf-token': decodeURIComponent(document.cookie.match(/XSRF-TOKEN=([^;]+)/)?.[1])
          }
        });
        if (result !== 'Success') {
          debug('请求任务列表失败', {
            result: result,
            statusText: statusText,
            status: status
          });
          logStatus.error(`${result}:${statusText}(${status})`);
          return false;
        }
        if (data?.status !== 200 || !data?.response?.tasks) {
          debug('任务列表数据异常', {
            status: data?.status,
            response: data?.response
          });
          logStatus.error(`Error:${data?.statusText}(${data?.status})`);
          return false;
        }
        debug('获取到任务列表', {
          count: data.response.tasks.length
        });
        this.tasks = data.response.tasks;
        for (const task of data.response.tasks) {
          if (task.isDone) {
            debug('跳过已完成任务', {
              taskId: task.id,
              type: task.type
            });
            continue;
          }
          debug('处理任务', {
            taskId: task.id,
            category: task.category,
            type: task.type
          });
          await httpRequest({
            url: `https://giveawayhopper.com/api/v1/campaigns/${this.giveawayId}/tasks/${task.id}/visited`,
            method: 'GET',
            responseType: 'json',
            headers: {
              authorization: `Bearer ${window.sessionStorage.gw_auth}`,
              'x-xsrf-token': decodeURIComponent(document.cookie.match(/XSRF-TOKEN=([^;]+)/)?.[1])
            }
          });
          if (task.category === 'Steam' && task.type === 'JoinGroup') {
            debug('处理 Steam 组任务');
            const steamGroupLink = await getRedirectLink(`https://steamcommunity.com/gid/${task.group_id}`);
            if (!steamGroupLink) {
              debug('获取 Steam 组链接失败');
              continue;
            }
            debug('添加 Steam 组链接', {
              action: action,
              link: steamGroupLink
            });
            if (action === 'undo') {
              this.socialTasks.steam.groupLinks.push(steamGroupLink);
            }
            if (action === 'do') {
              this.undoneTasks.steam.groupLinks.push(steamGroupLink);
            }
            continue;
          }
          if (task.category === 'Discord' && task.type === 'JoinServer') {
            const discordLink = `https://discord.gg/${task.invite_code}`;
            debug('添加 Discord 服务器链接', {
              action: action,
              link: discordLink
            });
            if (action === 'undo') {
              this.socialTasks.discord.serverLinks.push(discordLink);
            }
            if (action === 'do') {
              this.undoneTasks.discord.serverLinks.push(discordLink);
            }
            continue;
          }
          if ([ 'TikTok', 'YouTube', 'General' ].includes(task.category)) {
            debug('跳过特殊任务类型', {
              category: task.category
            });
            continue;
          }
          debug('发现未知任务类型', {
            category: task.category,
            type: task.type
          });
          echoLog({}).warning(`${I18n('unKnownTaskType')}: ${task.category}-${task.type}`);
        }
        logStatus.success();
        this.undoneTasks = this.uniqueTasks(this.undoneTasks);
        this.socialTasks = this.uniqueTasks(this.socialTasks);
        debug('任务分类完成', {
          undoneTasks: this.undoneTasks,
          socialTasks: this.socialTasks
        });
        GM_setValue(`giveawayHopperTasks-${this.giveawayId}`, {
          tasks: this.socialTasks,
          time: (new Date).getTime()
        });
        return true;
      } catch (error) {
        debug('任务分类失败', {
          error: error
        });
        throwError(error, 'GiveawayHopper.classifyTask');
        return false;
      }
    }
    async verifyTask() {
      try {
        debug('开始验证任务');
        for (const task of this.tasks) {
          if (task.isDone) {
            debug('跳过已完成任务', {
              taskId: task.id
            });
            continue;
          }
          debug('验证任务', {
            taskId: task.id,
            name: task.displayName?.replace(':target', task.targetName) || task.name
          });
          const logStatus = echoLog({
            text: `${I18n('verifyingTask')}[${task.displayName?.replace(':target', task.targetName) || task.name}]...`
          });
          if (!task.link) {
            debug('获取任务链接');
            task.link = this.#getTaskLink(task);
          }
          if (task.link) {
            debug('访问任务链接', {
              link: task.link
            });
            await this.#visitTaskLink(task);
          }
          await delay(1e3);
          const verifyResult = await this.#verifyTask(task, logStatus);
          debug('任务验证结果', {
            taskId: task.id,
            success: verifyResult
          });
          if (!verifyResult) {
            continue;
          }
        }
        debug('所有任务验证完成');
        return true;
      } catch (error) {
        debug('任务验证失败', {
          error: error
        });
        throwError(error, 'GiveawayHopper.verifyTask');
        return false;
      }
    }
    #getTaskLink(task) {
      try {
        debug('生成任务链接', {
          category: task.category,
          type: task.type
        });
        let link = '';
        if (task.category === 'YouTube' && task.type === 'FollowAccount') {
          link = `https://www.youtube.com/@${task.targetName}`;
        } else if (task.category === 'TikTok' && task.type === 'FollowAccount') {
          link = `https://www.tiktok.com/@${task.targetName}`;
        } else if (task.category === 'Steam' && task.type === 'JoinGroup') {
          link = '';
        } else if (task.category === 'Discord' && task.type === 'JoinServer') {
          link = '';
        }
        debug('生成的任务链接', {
          link: link
        });
        return link;
      } catch (error) {
        debug('生成任务链接失败', {
          error: error
        });
        throwError(error, 'GiveawayHopper.getTaskLink');
        return '';
      }
    }
    async #visitTaskLink(task) {
      debug('访问任务链接', {
        taskId: task.id,
        link: task.link
      });
      await httpRequest({
        url: `https://giveawayhopper.com/fw?url=${encodeURIComponent(task.link)}&src=campaign&src_id=${this.giveawayId}&ref=task&ref_id=${task.id}&token=${window.sessionStorage.gw_auth}`,
        method: 'GET',
        headers: {
          authorization: `Bearer ${window.sessionStorage.gw_auth}`,
          'x-xsrf-token': decodeURIComponent(document.cookie.match(/XSRF-TOKEN=([^;]+)/)?.[1])
        }
      });
    }
    async #verifyTask(task, logStatus) {
      debug('验证任务', {
        taskId: task.id,
        category: task.category,
        type: task.type
      });
      const postData = {
        taskcategory: task.category,
        taskname: task.type
      };
      if ([ 'YouTube', 'TikTok' ].includes(task.category)) {
        postData.username = '1';
      }
      const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
        url: `https://giveawayhopper.com/api/v1/campaigns/${this.giveawayId}/tasks/${task.id}/complete`,
        method: 'POST',
        headers: {
          authorization: `Bearer ${window.sessionStorage.gw_auth}`,
          'x-xsrf-token': decodeURIComponent(document.cookie.match(/XSRF-TOKEN=([^;]+)/)?.[1]),
          'content-type': 'application/json'
        },
        dataType: 'json',
        data: JSON.stringify(postData)
      });
      if (result !== 'Success') {
        debug('任务验证请求失败', {
          result: result,
          statusText: statusText,
          status: status
        });
        logStatus.error(`${result}:${statusText}(${status})`);
        return false;
      }
      if (data?.status !== 200 || !data?.response?.completed) {
        debug('任务验证响应异常', {
          status: data?.status,
          response: data?.response
        });
        logStatus.error(`Error:${data?.statusText}(${data?.status})`);
        return false;
      }
      debug('任务验证成功', {
        taskId: task.id
      });
      logStatus.success();
      return true;
    }
    #getGiveawayId() {
      try {
        debug('从URL获取抽奖ID');
        const giveawayId = window.location.pathname.split('/').at(-1);
        if (!giveawayId) {
          debug('获取抽奖ID失败');
          echoLog({
            text: I18n('getFailed', 'GiveawayId')
          });
          return false;
        }
        this.giveawayId = giveawayId;
        debug('获取抽奖ID成功', {
          giveawayId: giveawayId
        });
        return true;
      } catch (error) {
        debug('获取抽奖ID出错', {
          error: error
        });
        throwError(error, 'GiveawayHopper.getGiveawayId');
        return false;
      }
    }
    #checkLogin() {
      try {
        debug('检查登录状态');
        if (!globalOptions.other.checkLogin) {
          debug('跳过登录检查');
          return true;
        }
        const needLogin = $('div.widget-connections-edit:contains("Log in")').length > 0;
        if (needLogin) {
          debug('未登录,自动点击登录按钮');
          $('div.widget-connections-edit:contains("Log in") a')[0].click();
        }
        debug('登录检查完成', {
          needLogin: needLogin
        });
        return true;
      } catch (error) {
        debug('登录检查失败', {
          error: error
        });
        throwError(error, 'GiveawayHopper.checkLogin');
        return false;
      }
    }
    async #checkLeftKey() {
      try {
        debug('检查剩余密钥');
        if (!globalOptions.other.checkLeftKey) {
          debug('跳过密钥检查');
          return true;
        }
        const keyCount = parseInt($('p.widget-single-prize span').text()?.match(/\d+/)?.[0] || '0', 10);
        debug('剩余密钥数量', {
          keyCount: keyCount
        });
        if (keyCount > 0) {
          return true;
        }
        debug('没有剩余密钥,显示确认对话框');
        const {value: value} = await Swal.fire({
          icon: 'warning',
          title: I18n('notice'),
          text: I18n('noKeysLeft'),
          confirmButtonText: I18n('confirm'),
          cancelButtonText: I18n('cancel'),
          showCancelButton: true
        });
        if (value) {
          debug('用户确认关闭窗口');
          window.close();
        }
        return true;
      } catch (error) {
        debug('检查剩余密钥失败', {
          error: error
        });
        throwError(error, 'GiveawayHopper.checkLeftKey');
        return false;
      }
    }
  }
  const defaultTasksTemplate = {
    steam: {
      groupLinks: [],
      curatorLinks: [],
      wishlistLinks: [],
      followLinks: []
    },
    youtube: {
      channelLinks: []
    }
  };
  const defaultTasks = JSON.stringify(defaultTasksTemplate);
  class Prys extends Website {
    name='Prys';
    socialTasks=JSON.parse(defaultTasks);
    undoneTasks=JSON.parse(defaultTasks);
    buttons=[ 'doTask', 'undoTask', 'verifyTask' ];
    static test() {
      const {host: host} = window.location;
      const isMatch = host === 'prys.revadike.com';
      debug('检查网站匹配', {
        host: host,
        isMatch: isMatch
      });
      return isMatch;
    }
    async after() {
      try {
        debug('开始执行后续操作');
        if (!this.#checkLogin()) {
          debug('检查登录失败');
          echoLog({}).warning(I18n('checkLoginFailed'));
        }
        if (!await this.#checkLeftKey()) {
          debug('检查剩余密钥失败');
          echoLog({}).warning(I18n('checkLeftKeyFailed'));
        }
      } catch (error) {
        debug('后续操作失败', {
          error: error
        });
        throwError(error, 'Prys.after');
      }
    }
    init() {
      try {
        debug('开始初始化');
        const logStatus = echoLog({
          text: I18n('initing')
        });
        if ($('button:contains("Sign")').length > 0) {
          debug('需要登录');
          logStatus.warning(I18n('needLogin'));
          return false;
        }
        if (!this.#getGiveawayId()) {
          debug('获取抽奖ID失败');
          return false;
        }
        this.initialized = true;
        debug('初始化完成');
        logStatus.success();
        return true;
      } catch (error) {
        debug('初始化失败', {
          error: error
        });
        throwError(error, 'Prys.init');
        return false;
      }
    }
    async classifyTask(action) {
      try {
        debug('开始分类任务', {
          action: action
        });
        const logStatus = echoLog({
          text: I18n('getTasksInfo')
        });
        if (action === 'undo') {
          debug('恢复已保存的任务信息');
          this.socialTasks = GM_getValue(`prysTasks-${this.giveawayId}`)?.tasks || JSON.parse(defaultTasks);
        }
        const steps = $('#steps tbody tr');
        debug('找到任务步骤', {
          count: steps.length
        });
        for (let eq = 0; eq < steps.length; eq += 1) {
          if (steps.eq(eq).find('span:contains(Success)').length === 0) {
            debug('点击检查按钮', {
              step: eq
            });
            checkClick(eq);
          }
        }
        const pro = [];
        for (const step of steps) {
          const isSuccess = $(step).find('span:contains(Success)').length > 0;
          if (isSuccess && action === 'do') {
            debug('跳过已完成的任务');
            continue;
          }
          const appLink = $(step).find('a[href*=\'store.steampowered.com/app/\']').attr('href');
          if (appLink) {
            const taskType = $(step).find('a[href*=\'store.steampowered.com/app/\']').text().includes('wishlist') ? 'wishlistLinks' : 'followLinks';
            debug('添加 Steam 应用任务', {
              type: taskType,
              link: appLink
            });
            if (action === 'undo') {
              this.socialTasks.steam[taskType].push(appLink);
            }
            if (action === 'do') {
              this.undoneTasks.steam[taskType].push(appLink);
            }
            continue;
          }
          const curatorLink = $(step).find('a[href*=\'store.steampowered.com/curator/\']').attr('href');
          if (curatorLink) {
            debug('添加 Steam 鉴赏家任务', {
              link: curatorLink
            });
            if (action === 'undo') {
              this.socialTasks.steam.curatorLinks.push(curatorLink);
            }
            if (action === 'do') {
              this.undoneTasks.steam.curatorLinks.push(curatorLink);
            }
            continue;
          }
          const groupLink = $(step).find('a[href*=\'steamcommunity.com/groups/\']').attr('href');
          if (groupLink) {
            debug('添加 Steam 组任务', {
              link: groupLink
            });
            if (action === 'undo') {
              this.socialTasks.steam.groupLinks.push(groupLink);
            }
            if (action === 'do') {
              this.undoneTasks.steam.groupLinks.push(groupLink);
            }
            continue;
          }
          const gidLink = $(step).find('a[href*=\'steamcommunity.com/gid\']').attr('href');
          if (gidLink) {
            debug('处理 Steam GID 链接', {
              link: gidLink
            });
            pro.push(getRedirectLink(gidLink).then((finalUrl => {
              if (!finalUrl || !/^https?:\/\/steamcommunity\.com\/groups\//.test(finalUrl)) {
                debug('无效的 Steam 组链接', {
                  finalUrl: finalUrl
                });
                return false;
              }
              debug('添加 Steam 组任务(从 GID)', {
                link: finalUrl
              });
              if (action === 'undo') {
                this.socialTasks.steam.groupLinks.push(finalUrl);
              }
              if (action === 'do') {
                this.undoneTasks.steam.groupLinks.push(finalUrl);
              }
            })));
          }
        }
        await Promise.allSettled(pro);
        debug('任务分类完成');
        logStatus.success();
        this.undoneTasks = this.uniqueTasks(this.undoneTasks);
        this.socialTasks = this.uniqueTasks(this.socialTasks);
        if (window.DEBUG) {
          console.log('%cAuto-Task[Debug]:', 'color:blue', JSON.stringify(this));
        }
        GM_setValue(`prysTasks-${this.giveawayId}`, {
          tasks: this.socialTasks,
          time: (new Date).getTime()
        });
        return true;
      } catch (error) {
        debug('任务分类失败', {
          error: error
        });
        throwError(error, 'Prys.classifyTask');
        return false;
      }
    }
    async verifyTask() {
      try {
        debug('开始验证任务');
        const checks = $('#steps tbody a[id^=check]');
        if (checks.length === 0) {
          debug('没有需要验证的任务');
          echoLog({}).success(I18n('allTasksComplete'));
          return;
        }
        const pro = [];
        for (const check of checks) {
          const id = $(check).attr('id')?.match(/[\d]+/)?.[0];
          if (!id) {
            debug('跳过无效任务ID');
            continue;
          }
          const taskDes = $(check).parent()?.prev()?.html()?.trim();
          debug('验证任务', {
            id: id,
            taskDes: taskDes
          });
          const status = echoLog({
            text: `${I18n('verifyingTask')}${taskDes}...`
          });
          pro.push(new Promise((resolve => {
            this.#checkStep(id, resolve, status);
          })));
        }
        await Promise.all(pro);
        debug('所有任务验证完成');
        echoLog({}).success(I18n('allTasksComplete'));
      } catch (error) {
        debug('验证任务失败', {
          error: error
        });
        throwError(error, 'Prys.verifyTask');
      }
    }
    #getGiveawayId() {
      try {
        debug('开始获取抽奖ID');
        const giveawayId = window.location.search.match(/id=([\d]+)/)?.[1];
        if (giveawayId) {
          this.giveawayId = giveawayId;
          debug('获取抽奖ID成功', {
            giveawayId: giveawayId
          });
          return true;
        }
        debug('获取抽奖ID失败');
        echoLog({}).error(I18n('getFailed', 'GiveawayId'));
        return false;
      } catch (error) {
        debug('获取抽奖ID出错', {
          error: error
        });
        throwError(error, 'Prys.getGiveawayId');
        return false;
      }
    }
    async #checkLeftKey() {
      try {
        debug('检查剩余密钥');
        if (!globalOptions.other.checkLeftKey) {
          debug('跳过密钥检查');
          return true;
        }
        const leftKey = $('#header').text().match(/([\d]+).*?prize.*?left/)?.[1];
        debug('检查剩余密钥数量', {
          leftKey: leftKey
        });
        if (leftKey !== '0') {
          return true;
        }
        debug('没有剩余密钥,显示确认对话框');
        const {value: value} = await Swal.fire({
          icon: 'warning',
          title: I18n('notice'),
          text: I18n('noKeysLeft'),
          confirmButtonText: I18n('confirm'),
          cancelButtonText: I18n('cancel'),
          showCancelButton: true
        });
        if (value) {
          debug('用户确认关闭窗口');
          window.close();
        }
        return true;
      } catch (error) {
        debug('检查剩余密钥失败', {
          error: error
        });
        throwError(error, 'Prys.checkLeftKey');
        return false;
      }
    }
    #checkLogin() {
      try {
        debug('检查登录状态');
        if (!globalOptions.other.checkLogin) {
          debug('跳过登录检查');
          return true;
        }
        if ($('button:contains("Sign")').length > 0) {
          debug('未登录');
          echoLog({}).warning(I18n('needLogin'));
        }
        debug('登录检查完成');
        return true;
      } catch (error) {
        debug('检查登录失败', {
          error: error
        });
        throwError(error, 'Prys.checkLogin');
        return false;
      }
    }
    #checkStep(step, resolve, status, captcha = null) {
      try {
        debug('开始检查步骤', {
          step: step,
          hasCaptcha: !!captcha
        });
        if (step !== 'captcha') {
          debug('更新步骤状态为检查中');
          $(`#check${step}`).replaceWith(`<span id="check${step}"><i class="fa fa-refresh fa-spin fa-fw"></i> Checking...</span>`);
        }
        debug('发送检查请求');
        $.post('/api/check_step', {
          step: step,
          id: getURLParameter('id'),
          'g-recaptcha-response': captcha
        }, (json => {
          resolve();
          debug('收到检查响应', {
            success: json.success
          });
          if (step !== 'captcha') {
            if (json.success) {
              debug('步骤检查成功');
              $(`#check${step}`).replaceWith(`<span class="text-success" id="check${step}"><i class="fa fa-check"></i> Success</span>`);
              status.success();
            } else {
              debug('步骤检查失败');
              $(`#check${step}`).replaceWith(`<a id="check${step}" href="javascript:checkStep(${step})"><i class="fa fa-question"></i> Check</a>`);
              status.error(json.response?.error || 'Error');
            }
          }
          if (!json.response) {
            return;
          }
          if (json.response.prize) {
            debug('获得奖品', {
              prize: json.response.prize
            });
            showAlert('success', `Here is your prize:<h1 role="button" align="middle" style="word-wrap: break-word;">${json.response.prize}</h2>`);
          }
          if (!json.response.captcha) {
            return;
          }
          debug('需要验证码');
          if (json.success) {
            showAlert('info', json.response.captcha);
          } else {
            showAlert('warning', json.response.captcha);
          }
          captchaCheck();
        })).fail((() => {
          resolve();
          debug('请求失败');
          $(`#check${step}`).replaceWith(`<a id="check${step}" href="javascript:checkStep(${step})"><i class="fa fa-question"></i> Check</a>`);
          status.error('Error:0');
        }));
      } catch (error) {
        debug('检查步骤失败', {
          error: error
        });
        throwError(error, 'prys.checkStep');
        resolve(false);
      }
    }
  }
  const Websites = [ FreeAnyWhere, GiveawaySu, Indiedb, Keyhub, Givekey, GiveeClub, OpiumPulses, Keylol, Opquests, Gleam, SweepWidget, Setting, History, GiveawayHopper, Prys ];
  const generateFormHtml = options => {
    debug('开始生成网站选项表单HTML', {
      options: options
    });
    const tableRows = Object.entries(options).map((([option, value]) => `\n      <tr>\n        <td>${option}</td>\n        <td>\n          <input\n            class="editOption"\n            type="text"\n            name="${option}"\n            value="${value}"\n          />\n        </td>\n      </tr>\n    `)).join('');
    const formHtml = `\n    <form id="websiteOptionsForm" class="auto-task-form">\n      <table class="auto-task-table">\n        <thead>\n          <tr>\n            <td>${I18n('option')}</td>\n            <td>${I18n('value')}</td>\n          </tr>\n        </thead>\n        <tbody>\n          ${tableRows}\n        </tbody>\n      </table>\n    </form>\n  `;
    debug('表单HTML生成完成');
    return formHtml;
  };
  const saveOptions = (website, options, formValues) => {
    debug('开始保存网站选项', {
      website: website,
      formValues: formValues
    });
    formValues.forEach((({name: name, value: value}) => {
      options[name] = value;
      debug('更新选项值', {
        name: name,
        value: value
      });
    }));
    GM_setValue(`${website}Options`, options);
    debug('选项已保存到存储', {
      website: website
    });
    Swal.fire({
      title: I18n('changeWebsiteOptionsSuccess'),
      icon: 'success'
    });
  };
  const websiteOptions = async (website, options) => {
    try {
      debug('开始设置网站选项', {
        website: website
      });
      if (!website || typeof website !== 'string') {
        debug('无效的网站参数', {
          website: website
        });
        throw new Error('Invalid website parameter');
      }
      if (!options || typeof options !== 'object') {
        debug('无效的选项参数', {
          options: options
        });
        throw new Error('Invalid options parameter');
      }
      debug('显示选项编辑对话框');
      const result = await Swal.fire({
        title: I18n('websiteOptions'),
        html: generateFormHtml(options),
        showConfirmButton: true,
        confirmButtonText: I18n('save'),
        showCancelButton: true,
        cancelButtonText: I18n('close')
      });
      if (result.isConfirmed) {
        debug('用户确认保存选项');
        const form = document.getElementById('websiteOptionsForm');
        if (!form) {
          debug('未找到表单元素');
          throw new Error('Form element not found');
        }
        const formData = $('#websiteOptionsForm').serializeArray();
        debug('获取表单数据', {
          formData: formData
        });
        saveOptions(website, options, formData);
      } else {
        debug('用户取消保存选项');
      }
    } catch (error) {
      debug('设置网站选项时发生错误', {
        error: error
      });
      throwError(error, 'websiteOptions');
    }
  };
  const UPDATE_LINKS = {
    github: 'https://github.com/HCLonely/auto-task/raw/main/',
    jsdelivr: 'https://cdn.jsdelivr.net/gh/HCLonely/auto-task@main/',
    standby: 'https://auto-task.hclonely.com/'
  };
  const checkUpdate = async (updateLink, auto) => {
    try {
      debug('开始检查更新', {
        updateLink: updateLink,
        auto: auto
      });
      const checkUrl = `${updateLink}package.json?time=${Date.now()}`;
      debug('构建检查URL', {
        checkUrl: checkUrl
      });
      const {result: result, statusText: statusText, status: status, data: data} = await httpRequest({
        url: checkUrl,
        responseType: 'json',
        method: 'GET',
        timeout: 3e4
      });
      if (result === 'Success' && data?.response?.version) {
        debug('成功获取更新信息', {
          version: data.response.version
        });
        return data.response;
      }
      if (!auto) {
        const errorMessage = data?.response?.version ? `${I18n('checkUpdateFailed')}[${data?.statusText}(${data?.status})]` : `${I18n('checkUpdateFailed')}[${result}:${statusText}(${status})]`;
        debug('检查更新失败', {
          errorMessage: errorMessage
        });
        echoLog({}).error(errorMessage);
      } else {
        debug('自动检查更新失败', {
          result: result,
          statusText: statusText,
          status: status
        });
      }
      return false;
    } catch (error) {
      debug('检查更新发生错误', {
        error: error
      });
      throwError(error, 'checkUpdate');
      return false;
    }
  };
  const hasNewVersion = (currentVersion, remoteVersion) => {
    try {
      debug('开始比较版本号', {
        currentVersion: currentVersion,
        remoteVersion: remoteVersion
      });
      const [currentRealVersion] = currentVersion.split('-');
      const [remoteRealVersion, isPreview] = remoteVersion.split('-');
      if (isPreview && !globalOptions.other.receivePreview) {
        debug('不接收预览版本', {
          isPreview: isPreview
        });
        return false;
      }
      const currentVersionParts = currentRealVersion.split('.').map(Number);
      const remoteVersionParts = remoteRealVersion.split('.').map(Number);
      debug('版本号解析', {
        currentVersionParts: currentVersionParts,
        remoteVersionParts: remoteVersionParts
      });
      for (let i = 0; i < 3; i++) {
        if (remoteVersionParts[i] > currentVersionParts[i]) {
          debug('发现新版本', {
            position: i,
            current: currentVersion,
            remote: remoteVersion
          });
          return true;
        }
        if (remoteVersionParts[i] < currentVersionParts[i]) {
          debug('远程版本较旧', {
            position: i,
            current: currentVersion,
            remote: remoteVersion
          });
          return false;
        }
      }
      debug('版本号相同');
      return false;
    } catch (error) {
      debug('比较版本号时发生错误', {
        error: error
      });
      throwError(error, 'compareVersion');
      return false;
    }
  };
  const getUpdateLink = updateSource => {
    debug('获取更新链接', {
      updateSource: updateSource
    });
    const source = updateSource.toLowerCase();
    const link = UPDATE_LINKS[source] || UPDATE_LINKS.github;
    debug('选择的更新链接', {
      source: source,
      link: link
    });
    return link;
  };
  const showUpdateInfo = (packageData, currentVersion, updateLink) => {
    debug('准备显示更新信息', {
      currentVersion: currentVersion,
      newVersion: packageData.version
    });
    if (hasNewVersion(currentVersion, packageData.version)) {
      const scriptUrl = `${updateLink}dist/${GM_info.script.name}.user.js`;
      debug('发现新版本,显示更新通知', {
        scriptUrl: scriptUrl
      });
      echoLog({
        html: `<li><font>${I18n('newVersionNotice', packageData.version, scriptUrl)}</font></li>`
      });
      const changeList = packageData.change?.map((change => `<li>${change}</li>`)).join('') || '';
      debug('显示更新日志', {
        changeListLength: packageData.change?.length
      });
      echoLog({
        html: `<li>${I18n('updateText', packageData.version)}</li><ol class="update-text">${changeList}<li>${I18n('updateHistory')}</li></ol>`
      });
    } else {
      debug('当前已是最新版本');
    }
  };
  const updateChecker = async () => {
    try {
      debug('开始检查更新流程');
      const currentVersion = GM_info.script.version;
      const updateSource = globalOptions.other.autoUpdateSource;
      debug('当前配置', {
        currentVersion: currentVersion,
        updateSource: updateSource
      });
      let packageData = false;
      if ([ 'github', 'jsdelivr', 'standby' ].includes(updateSource.toLowerCase())) {
        debug('使用指定的更新源', {
          updateSource: updateSource
        });
        const updateLink = getUpdateLink(updateSource);
        packageData = await checkUpdate(updateLink, false);
      } else {
        debug('按优先级尝试不同的更新源');
        for (const source of [ 'github', 'jsdelivr', 'standby' ]) {
          debug('尝试更新源', {
            source: source
          });
          packageData = await checkUpdate(UPDATE_LINKS[source], true);
          if (packageData) {
            debug('成功获取更新信息', {
              source: source
            });
            break;
          }
        }
      }
      if (!packageData) {
        debug('所有更新源检查失败');
        echoLog({}).error(I18n('checkUpdateFailed'));
        return;
      }
      showUpdateInfo(packageData, currentVersion, getUpdateLink(updateSource));
    } catch (error) {
      debug('更新检查过程发生错误', {
        error: error
      });
      throwError(error, 'updateChecker');
    }
  };
  const byteToHex = [];
  for (let i = 0; i < 256; ++i) {
    byteToHex.push((i + 256).toString(16).slice(1));
  }
  function unsafeStringify(arr, offset = 0) {
    return (byteToHex[arr[offset + 0]] + byteToHex[arr[offset + 1]] + byteToHex[arr[offset + 2]] + byteToHex[arr[offset + 3]] + '-' + byteToHex[arr[offset + 4]] + byteToHex[arr[offset + 5]] + '-' + byteToHex[arr[offset + 6]] + byteToHex[arr[offset + 7]] + '-' + byteToHex[arr[offset + 8]] + byteToHex[arr[offset + 9]] + '-' + byteToHex[arr[offset + 10]] + byteToHex[arr[offset + 11]] + byteToHex[arr[offset + 12]] + byteToHex[arr[offset + 13]] + byteToHex[arr[offset + 14]] + byteToHex[arr[offset + 15]]).toLowerCase();
  }
  let getRandomValues;
  const rnds8 = new Uint8Array(16);
  function rng() {
    if (!getRandomValues) {
      if (typeof crypto === 'undefined' || !crypto.getRandomValues) {
        throw new Error('crypto.getRandomValues() not supported. See https://github.com/uuidjs/uuid#getrandomvalues-not-supported');
      }
      getRandomValues = crypto.getRandomValues.bind(crypto);
    }
    return getRandomValues(rnds8);
  }
  const randomUUID = typeof crypto !== 'undefined' && crypto.randomUUID && crypto.randomUUID.bind(crypto);
  var native = {
    randomUUID: randomUUID
  };
  function v4(options, buf, offset) {
    if (native.randomUUID && true && !options) {
      return native.randomUUID();
    }
    options = options || {};
    const rnds = options.random ?? options.rng?.() ?? rng();
    if (rnds.length < 16) {
      throw new Error('Random bytes length must be >= 16');
    }
    rnds[6] = rnds[6] & 15 | 64;
    rnds[8] = rnds[8] & 63 | 128;
    return unsafeStringify(rnds);
  }
  function fawExtension() {
    const hostname = window.location.hostname;
    function IsJsonString(str) {
      try {
        JSON.parse(str);
      } catch (e) {
        return false;
      }
      return true;
    }
    $(window).on('load', (function() {
      console.log('👌 gamesforfarm extension');
      if (hostname == 'freeanywhere.net' || hostname == 'give.gamesforfarm.local' || hostname == 'gamesforfarm-testing.ru') {
        const steam = $('.games_for_farm_site').data('steam');
        const avatar = $('.games_for_farm_site').data('avatar');
        const name = $('.games_for_farm_site').data('name');
        const lang = $('.games_for_farm_site').data('lang');
        let need_update = true;
        GM_addValueChangeListener('FAW_STORAGE', (function(newValue, oldValue) {
          if (need_update == false) {
            return;
          }
          GM_getValue('FAW_STORAGE', (function(STORAGE) {
            $.ajax({
              type: 'POST',
              url: '/php/extension/user_data_update.php',
              data: {
                extension: JSON.stringify(STORAGE)
              },
              success: function(data) {}
            });
          }));
        }));
        const STORAGE = GM_getValue('FAW_STORAGE') || {};
        if (Object.keys(STORAGE).length === 0 || !STORAGE['tasks'] || !STORAGE['user'] || !STORAGE['games'] || !STORAGE['settings']) {
          GM_deleteValue('FAW_STORAGE');
          let new_storage = {};
          new_storage['tasks'] = {};
          new_storage['user'] = {};
          new_storage['games'] = {};
          new_storage['settings'] = {};
          if (steam) {
            new_storage['user']['steam'] = steam;
          }
          if (avatar) {
            new_storage['user']['avatar'] = avatar;
          }
          if (name) {
            new_storage['user']['name'] = name;
          }
          if (lang) {
            new_storage['user']['lang'] = lang;
          }
          if (steam) {
            $.ajax({
              type: 'POST',
              url: '/php/extension/user_games_get.php',
              data: {
                steam: steam
              },
              success: function(data) {
                if (data != '' && IsJsonString(data)) {
                  const user_games = JSON.parse(data);
                  new_storage['games'] = user_games;
                  new_storage['settings']['game_update'] = parseInt(Date.now() / 1e3);
                  GM_setValue('FAW_STORAGE', new_storage);
                }
              }
            });
          } else {
            GM_setValue('FAW_STORAGE', new_storage);
          }
        }
        if (STORAGE['user'] && STORAGE['settings'] && STORAGE['user']['steam']) {
          if (!STORAGE['settings']['game_update']) {
            STORAGE['settings']['game_update'] = 0;
          }
          let time_now = parseInt(Date.now() / 1e3);
          let time_update = parseInt(STORAGE['settings']['game_update']);
          if (time_now - time_update > 60 * 60 * 24) {
            $.ajax({
              type: 'POST',
              url: '/php/extension/user_games_get.php',
              data: {
                steam: STORAGE['user']['steam']
              },
              success: function(data) {
                if (data != '' && IsJsonString(data)) {
                  const games = JSON.parse(data);
                  STORAGE['games'] = games;
                  STORAGE['settings']['game_update'] = parseInt(Date.now() / 1e3);
                  GM_setValue('FAW_STORAGE', STORAGE);
                }
              }
            });
          }
        }
        if (steam && STORAGE['user']) {
          if (!STORAGE['user']['steam']) {
            STORAGE['user']['steam'] = steam;
          }
          if (STORAGE['user']['steam'] != steam) {
            $.ajax({
              type: 'POST',
              url: '/php/extension/user_data_get.php',
              data: {
                steam: steam
              },
              success: function(data) {
                if (!data) {
                  return;
                }
                const db_storage = JSON.parse(data);
                if (db_storage) {
                  GM_deleteValue('FAW_STORAGE');
                  need_update = false;
                  GM_setValue('FAW_STORAGE', db_storage);
                  setTimeout((function() {
                    need_update = true;
                  }), 100);
                }
              }
            });
          } else {
            if (avatar) {
              STORAGE['user']['avatar'] = avatar;
            }
            if (name) {
              STORAGE['user']['name'] = name;
            }
            if (lang) {
              STORAGE['user']['lang'] = lang;
            }
            GM_setValue('FAW_STORAGE', STORAGE);
          }
        }
        if (STORAGE['tasks']) {
          let update_tasks = [];
          let is_update = false;
          let time_now = parseInt(Date.now() / 1e3);
          $.each(STORAGE['tasks'], (function(index, val) {
            if (val['time'] && time_now - parseInt(val['time']) > 2 * 60 * 60) {
              is_update = true;
              return;
            }
            update_tasks.push(val);
          }));
          if (is_update == true) {
            STORAGE['tasks'] = update_tasks;
            GM_setValue('FAW_STORAGE', STORAGE);
          }
        }
        if (STORAGE['discord']) {
          if (STORAGE['discord'] && STORAGE['discord'].length > 0) {
            $.ajax({
              type: 'POST',
              url: '/php/extension/discord_levels_update.php',
              data: {
                discord: JSON.stringify(STORAGE['discord'])
              },
              success: function(data) {
                if (data.indexOf('success') != -1) {
                  alert('Данные discord уровней обновлены');
                  delete STORAGE['discord'];
                  GM_setValue('FAW_STORAGE', STORAGE);
                } else {
                  alert('Возникла ошибка при обновлении discord уровней');
                  delete STORAGE['discord'];
                  GM_setValue('FAW_STORAGE', STORAGE);
                }
              },
              error: function() {
                alert('Возникла ошибка при обновлении discord уровней');
                delete STORAGE['discord'];
                GM_setValue('FAW_STORAGE', STORAGE);
              }
            });
          }
        }
        function check_tasks_button() {
          let tasks_done = true;
          $('.game__content-tasks__task').each((function(index, el) {
            if ($(this).hasClass('done') == false) {
              tasks_done = false;
            }
          }));
          if (tasks_done == true) {
            $('.js-get-key').removeClass('inactive');
          } else {
            $('.js-get-key').addClass('inactive');
          }
        }
        if ($('.games_for_farm_extension.work').length > 0) {
          $('.games_for_farm_extension.not_work').remove();
          $('.games_for_farm_extension.work').slideDown(200);
        }
        if ($('.game__content-tasks__task .task-check-extension').length > 0) {
          $('.task-check-extension').removeClass('js-extentions-modal');
          $('.game__content-tasks__task[data-extension=\'1\'] .task-link a').removeClass('js-extentions-modal');
          $('.game__content-tasks__task .task-check-extension').on('click', (function(event) {
            event.preventDefault();
            let $button = $(this);
            if ($button.hasClass('loading')) {
              return;
            }
            let $parrent = $(this).parent('.game__content-tasks__task');
            const type = $parrent.data('type');
            const id = $parrent.data('id');
            const data = $parrent.data('data');
            const extension = $parrent.data('extension');
            if (extension == false) {
              return;
            }
            $button.addClass('loading');
            const STORAGE = GM_getValue('FAW_STORAGE') || {};
            $.ajax({
              type: 'POST',
              url: '/php/extension/user_data_update.php',
              data: {
                extension: JSON.stringify(STORAGE)
              },
              success: function(update) {
                const getTime_start = (new Date).getTime();
                $.ajax({
                  type: 'POST',
                  url: '/php/extension/user_task_update.php',
                  data: {
                    id: id,
                    type: type,
                    data: data
                  },
                  success: function(data) {
                    const getTime_end = (new Date).getTime();
                    console.log('👌 checking task in ' + (getTime_end - getTime_start) + ' ms');
                    if (data.indexOf('good') != -1) {
                      setTimeout((function() {
                        $parrent.addClass('done');
                        $parrent.removeClass('error');
                        $button.removeClass('loading');
                        check_tasks_button();
                      }), 1250);
                    } else if (data.indexOf('bad') != -1) {
                      setTimeout((function() {
                        $parrent.addClass('error');
                        $parrent.removeClass('done');
                        $button.removeClass('loading');
                        check_tasks_button();
                      }), 1250);
                    } else {
                      setTimeout((function() {
                        $parrent.removeClass('error');
                        $parrent.removeClass('done');
                        $button.removeClass('loading');
                        check_tasks_button();
                      }), 1250);
                    }
                  },
                  error: function() {
                    setTimeout((function() {
                      $parrent.removeClass('error');
                      $parrent.removeClass('done');
                      $button.removeClass('loading');
                      check_tasks_button();
                    }), 1250);
                  }
                });
              },
              error: function() {
                setTimeout((function() {
                  $parrent.removeClass('error');
                  $parrent.removeClass('done');
                  $button.removeClass('loading');
                  check_tasks_button();
                }), 1250);
              }
            });
          }));
        }
      }
      function storage_tasks_update(tasks, type, value, action) {
        let result = [];
        let is_find = false;
        $.each(tasks, (function(index, val) {
          if (!val['type'] || !val['data']) {
            return;
          }
          if (val['type'] == type && val['data'] == value) {
            is_find = true;
            return;
          }
          result.push(val);
        }));
        const obj = {};
        switch (action) {
         case 'add':
          if (is_find == true) {
            return;
          }
          const task = {};
          task['type'] = type;
          task['data'] = value;
          task['time'] = parseInt(Date.now() / 1e3);
          result.push(task);
          obj['tasks'] = result;
          GM_setValue('FAW_STORAGE', obj);
          break;

         case 'remove':
          if (is_find == false) {
            return;
          }
          obj['tasks'] = result;
          GM_setValue('FAW_STORAGE', obj);
        }
      }
      if (hostname == 'store.steampowered.com' || hostname == 'steamcommunity.com') {
        if (document.querySelector('span[id^=\'CuratorUnFollowBtn_\']')) {
          const curator_id = $('span[id^=\'CuratorUnFollowBtn_\']').attr('id').split('_')[1];
          const follow_btn = '#CuratorFollowBtn_' + curator_id;
          const unfollow_btn = '#CuratorUnFollowBtn_' + curator_id;
          const STORAGE = GM_getValue('FAW_STORAGE') || {};
          if (!STORAGE['tasks']) {
            return;
          }
          const follow = $(follow_btn).css('display');
          const unfollow = $(unfollow_btn).css('display');
          if (unfollow && unfollow == 'none') {
            storage_tasks_update(STORAGE['tasks'], 'steam_curator_sub', curator_id, 'remove');
          }
          if (follow && follow == 'none') {
            storage_tasks_update(STORAGE['tasks'], 'steam_curator_sub', curator_id, 'add');
          }
          $(follow_btn).on('click', (function(event) {
            const STORAGE = GM_getValue('FAW_STORAGE') || {};
            if (!STORAGE['tasks']) {
              return;
            }
            storage_tasks_update(STORAGE['tasks'], 'steam_curator_sub', curator_id, 'add');
          }));
          $(unfollow_btn).on('click', (function(event) {
            const STORAGE = GM_getValue('FAW_STORAGE') || {};
            if (!STORAGE['tasks']) {
              return;
            }
            storage_tasks_update(STORAGE['tasks'], 'steam_curator_sub', curator_id, 'remove');
          }));
        }
        if (document.querySelector('.followStatsBlock')) {
          const user_id = $('#HeaderUserInfoName a').attr('href').split('/').pop();
          const follow_btn = '#FollowUserOptionAdd';
          const unfollow_btn = '#FollowUserOptionFollowing, .followOption.remove';
          const STORAGE = GM_getValue('FAW_STORAGE') || {};
          if (!STORAGE['tasks']) {
            return;
          }
          const follow = $(follow_btn).css('visibility');
          const unfollow = $(unfollow_btn).css('visibility');
          if (unfollow && unfollow == 'hidden') {
            storage_tasks_update(STORAGE['tasks'], 'steam_guides_sub', user_id, 'remove');
          }
          if (follow && follow == 'hidden') {
            storage_tasks_update(STORAGE['tasks'], 'steam_guides_sub', user_id, 'add');
          }
          $(follow_btn).on('click', (function(event) {
            const STORAGE = GM_getValue('FAW_STORAGE') || {};
            if (!STORAGE['tasks']) {
              return;
            }
            storage_tasks_update(STORAGE['tasks'], 'steam_guides_sub', user_id, 'add');
          }));
          $(unfollow_btn).on('click', (function(event) {
            const STORAGE = GM_getValue('FAW_STORAGE') || {};
            if (!STORAGE['tasks']) {
              return;
            }
            storage_tasks_update(STORAGE['tasks'], 'steam_guides_sub', user_id, 'remove');
          }));
        }
        if (document.querySelector('#ScrollingItemControls')) {
          const manual_id = $('#PublishedFileFavorite input[name=\'id\']').val();
          const follow_btn = '#FavoriteItemOptionAdd';
          const unfollow_btn = '#FavoriteItemOptionFavorited, .favoriteOption.removefavorite';
          const STORAGE = GM_getValue('FAW_STORAGE') || {};
          if (!STORAGE['tasks']) {
            return;
          }
          const follow = $(follow_btn).css('visibility');
          const unfollow = $(unfollow_btn).css('visibility');
          if (unfollow && unfollow == 'hidden') {
            storage_tasks_update(STORAGE['tasks'], 'steam_manual_favourite', manual_id, 'remove');
          }
          if (follow && follow == 'hidden') {
            storage_tasks_update(STORAGE['tasks'], 'steam_manual_favourite', manual_id, 'add');
          }
          $(follow_btn).on('click', (function(event) {
            const STORAGE = GM_getValue('FAW_STORAGE') || {};
            if (!STORAGE['tasks']) {
              return;
            }
            storage_tasks_update(STORAGE['tasks'], 'steam_manual_favourite', manual_id, 'add');
          }));
          $(unfollow_btn).on('click', (function(event) {
            const STORAGE = GM_getValue('FAW_STORAGE') || {};
            if (!STORAGE['tasks']) {
              return;
            }
            storage_tasks_update(STORAGE['tasks'], 'steam_manual_favourite', manual_id, 'remove');
          }));
        }
        if (document.querySelector('#queueBtnFollow')) {
          const game_id = $('.game_page_background').data('miniprofile-appid');
          const follow_btn = '#queueBtnFollow .queue_btn_inactive';
          const unfollow_btn = '#queueBtnFollow .queue_btn_active';
          const STORAGE = GM_getValue('FAW_STORAGE') || {};
          if (!STORAGE['tasks']) {
            return;
          }
          const follow = $(follow_btn).css('display');
          const unfollow = $(unfollow_btn).css('display');
          if (unfollow && unfollow == 'none') {
            storage_tasks_update(STORAGE['tasks'], 'steam_game_sub', game_id, 'remove');
          }
          if (follow && follow == 'none') {
            storage_tasks_update(STORAGE['tasks'], 'steam_game_sub', game_id, 'add');
          }
          $(follow_btn).on('click', (function(event) {
            const STORAGE = GM_getValue('FAW_STORAGE') || {};
            if (!STORAGE['tasks']) {
              return;
            }
            storage_tasks_update(STORAGE['tasks'], 'steam_game_sub', game_id, 'add');
          }));
          $(unfollow_btn).on('click', (function(event) {
            const STORAGE = GM_getValue('FAW_STORAGE') || {};
            if (!STORAGE['tasks']) {
              return;
            }
            storage_tasks_update(STORAGE['tasks'], 'steam_game_sub', game_id, 'remove');
          }));
        }
      }
      if (hostname == 'www.youtube.com' || hostname == 'm.youtube.com') {
        setInterval((function() {
          if (document.querySelector('yt-subscribe-button-view-model')) {
            const channel_id = $('meta[itemprop="identifier"]').attr('content');
            const subscribe = [ 'Подписаться', 'Падпісацца', 'Підписатися', 'Abonnieren', 'Subscribe', 'Suscribirse', 'Mag-subscribe', 'S\'abonner', 'Iscriviti', 'Subskrybuj', 'Subscrever', 'Abonează-te', '订阅', 'チャンネル登録', '訂閱' ];
            const subscribed = [ 'Вы подписаны', 'Вы падпісаны', 'Ви підписалися', 'Abonniert', 'Subscribed', 'Suscrito', 'Naka-subscribe', 'Abonné', 'Iscritto', 'Subskrybujesz', 'Subscrito', 'Abonat(ă)', '已订阅', '登録済み', '已訂閱' ];
            const $parent = $('yt-subscribe-button-view-model');
            const text = $parent.text();
            if (subscribe.indexOf(text) != -1 || subscribed.indexOf(text) != -1) {
              if (subscribe.indexOf(text) != -1) {
                const STORAGE = GM_getValue('FAW_STORAGE') || {};
                if (!STORAGE['tasks']) {
                  return;
                }
                storage_tasks_update(STORAGE['tasks'], 'youtube_channel_sub', channel_id, 'remove');
              }
              if (subscribed.indexOf(text) != -1) {
                const STORAGE = GM_getValue('FAW_STORAGE') || {};
                if (!STORAGE['tasks']) {
                  return;
                }
                storage_tasks_update(STORAGE['tasks'], 'youtube_channel_sub', channel_id, 'add');
              }
            } else {
              if ($('.ytSubscribePlusButtonViewModelHost').length != 0) {
                const STORAGE = GM_getValue('FAW_STORAGE') || {};
                if (!STORAGE['tasks']) {
                  return;
                }
                storage_tasks_update(STORAGE['tasks'], 'youtube_channel_sub', channel_id, 'add');
              } else {
                const color = $parent.find('button').css('color');
                if (color == '#0f0f0f' || color == 'rgb(15, 15, 15)') {
                  const STORAGE = GM_getValue('FAW_STORAGE') || {};
                  if (!STORAGE['tasks']) {
                    return;
                  }
                  storage_tasks_update(STORAGE['tasks'], 'youtube_channel_sub', channel_id, 'remove');
                }
                if (color == '#f1f1f1' || color == 'rgb(241, 241, 241)') {
                  const STORAGE = GM_getValue('FAW_STORAGE') || {};
                  if (!STORAGE['tasks']) {
                    return;
                  }
                  storage_tasks_update(STORAGE['tasks'], 'youtube_channel_sub', channel_id, 'add');
                }
              }
            }
          }
          if (document.querySelector('.ytLikeButtonViewModelHost')) {
            let video_id;
            if (hostname == 'm.youtube.com') {
              video_id = $('link[rel="canonical"]').attr('href').split('v=').pop();
            } else {
              video_id = $('meta[itemprop="identifier"]').attr('content');
            }
            if ($('.ytLikeButtonViewModelHost button').attr('aria-pressed') == 'false') {
              const STORAGE = GM_getValue('FAW_STORAGE') || {};
              if (!STORAGE['tasks']) {
                return;
              }
              storage_tasks_update(STORAGE['tasks'], 'youtube_video_like', video_id, 'remove');
            }
            if ($('.ytLikeButtonViewModelHost button').attr('aria-pressed') == 'true') {
              const STORAGE = GM_getValue('FAW_STORAGE') || {};
              if (!STORAGE['tasks']) {
                return;
              }
              storage_tasks_update(STORAGE['tasks'], 'youtube_video_like', video_id, 'add');
            }
          }
        }), 600);
      }
    }));
    if (hostname == 'mee6.xyz') {
      var message = false;
      $(document).on('keydown', '', (function(event) {
        if (event.code == 'End') {
          let USERS = [];
          $('.md\\:block').each((function(index, el) {
            const user = {};
            user['name'] = $(this).find('.justify-start p').text();
            user['level'] = $(this).find('.leaderboardPlayerStat .items-center').text();
            USERS.push(user);
            if (user['level'] == 0) {
              const obj = {};
              obj['discord'] = USERS;
              GM_setValue('FAW_STORAGE', obj);
              if (message == false) {
                alert('Можно переходить на freeanywhere.net');
                message = true;
              }
              return false;
            }
          }));
        }
      }));
    }
    if (hostname == 'gamesforfarm.com') {
      const STORAGE = GM_getValue('FAW_STORAGE') || {};
      if (!STORAGE['settings']) {
        return;
      }
      if (!STORAGE['games']) {
        return;
      }
      if (STORAGE['settings']['hide_games'] && STORAGE['settings']['hide_games'] == true) {
        $('.product__item').each((function(index, el) {
          const image = $(this).find('.product__box-image img').data('src');
          const cart = $(this).find('.product__box-props a').attr('href');
          if (image && image.indexOf('/apps/') != -1) {
            const id = parseInt(image.split('/apps/')[1].split('/')[0]);
            if (id && isNaN(id) == false) {
              if (STORAGE['games'][id]) {
                $(this).css('opacity', '.2');
              }
            }
          } else if (cart && cart.indexOf('/app/') != -1) {
            const id = parseInt(cart.split('/app/')[1].split('/')[0]);
            if (id && isNaN(id) == false) {
              if (STORAGE['games'][id]) {
                $(this).css('opacity', '.2');
              }
            }
          }
        }));
      }
    }
  }
  try {
    consoleLogHook();
  } catch (error) {
    console.error('Auto-Task[Warning]: consoleLogHook 初始化失败', error);
  }
  window.STYLE = GM_addStyle(style + GM_getResourceText('style'));
  window.DEBUG = !!globalOptions.other?.debug;
  window.TRACE = !!globalOptions.other?.debug && typeof console.trace === 'function';
  const handleTwitchAuth = async () => {
    debug('开始处理Twitch认证');
    const authToken = Cookies.get('auth-token');
    const isLogin = !!Cookies.get('login');
    if (isLogin) {
      const authData = {
        authToken: authToken,
        clientVersion: window.__twilightBuildID,
        clientId: window.commonOptions?.headers?.['Client-ID'],
        deviceId: window.commonOptions?.headers?.['Device-ID'],
        clientSessionId: window.localStorage.local_storage_app_session_id.replace(/"/g, '')
      };
      GM_setValue('twitchAuth', authData);
      window.close();
      await Swal.fire('', I18n('closePageNotice'));
    } else {
      await Swal.fire('', I18n('needLogin'));
    }
  };
  const handleRedditAuth = async () => {
    debug('开始处理Reddit认证');
    const betaButton = $('#redesign-beta-optin-btn');
    if (betaButton.length > 0) {
      betaButton[0].click();
      return;
    }
    window.close();
    await Swal.fire('', I18n('closePageNotice'));
  };
  const handleDiscordAuth = async () => {
    debug('开始处理Discord认证');
    const LocalStorage = window.localStorage;
    const allLocalStorage = getAllLocalStorageAsObjects(LocalStorage);
    const discordAuth = allLocalStorage.token;
    if (discordAuth && discordAuth.length > 0) {
      const browserInfo = await browser.getInfo();
      GM_setValue('discordAuth', {
        auth: discordAuth,
        xSuperProperties: window.btoa(JSON.stringify({
          os: browserInfo.system,
          browser: browserInfo.browser,
          device: '',
          system_locale: browserInfo.language,
          ...allLocalStorage.deviceProperties || {},
          browser_user_agent: navigator.userAgent,
          browser_version: browserInfo.browserVersion,
          os_version: browserInfo.systemVersion,
          referrer: '',
          referring_domain: '',
          referrer_current: '',
          referring_domain_current: '',
          release_channel: 'stable',
          client_build_number: unsafeWindow.GLOBAL_ENV.BUILD_NUMBER,
          client_event_source: null,
          has_client_mods: false,
          client_launch_id: v4(),
          client_heartbeat_session_id: allLocalStorage.LAST_CLIENT_HEARTBEAT_SESSION?.uuid,
          client_app_state: 'focused'
        }))
      });
      window.close();
      Swal.fire('', I18n('closePageNotice'));
    } else {
      Swal.fire({
        text: I18n('getDiscordAuthFailed'),
        icon: 'error'
      });
    }
  };
  const handleSteamStoreAuth = async () => {
    debug('开始处理Steam商店认证');
    const storeSessionID = document.body.innerHTML.match(/g_sessionID = "(.+?)";/)?.[1];
    if (storeSessionID) {
      GM_deleteValue('ATv4_updateStoreAuth');
      GM_setValue('steamStoreAuth', {
        storeSessionID: storeSessionID
      });
      window.close();
      await Swal.fire('', I18n('closePageNotice'));
    } else {
      await Swal.fire({
        title: 'Error: Get "sessionID" failed',
        icon: 'error'
      });
    }
  };
  const handleSteamCommunityAuth = async () => {
    debug('开始处理Steam社区认证');
    const steam64Id = document.body.innerHTML.match(/g_steamID = "(.+?)";/)?.[1];
    const communitySessionID = document.body.innerHTML.match(/g_sessionID = "(.+?)";/)?.[1];
    if (steam64Id && communitySessionID) {
      GM_deleteValue('ATv4_updateCommunityAuth');
      GM_setValue('steamCommunityAuth', {
        steam64Id: steam64Id,
        communitySessionID: communitySessionID
      });
      window.close();
      await Swal.fire('', I18n('closePageNotice'));
    } else {
      setTimeout((async () => {
        await Swal.fire({
          title: 'Error: Get "sessionID" failed',
          icon: 'error'
        });
      }), 3e3);
    }
  };
  const initializeUI = website => {
    debug('初始化UI元素', {
      website: website.name
    });
    const $body = $('body');
    $body.append(`\n    <div id="auto-task-info"\n        style="display:${globalOptions.other.defaultShowLog ? 'block' : 'none'};\n                ${globalOptions.position.logSideX}:${globalOptions.position.logDistance.split(',')[0]}px;\n                ${globalOptions.position.logSideY}:${globalOptions.position.logDistance.split(',')[1]}px;">\n    </div>\n    <div id="auto-task-buttons"\n        style="display:${globalOptions.other.defaultShowButton ? 'block' : 'none'};\n                ${globalOptions.position.buttonSideX}:${globalOptions.position.buttonDistance.split(',')[0]}px;\n                ${globalOptions.position.buttonSideY}:${globalOptions.position.buttonDistance.split(',')[1]}px;">\n    </div>\n    <div class="show-button-div"\n        style="display:${globalOptions.other.defaultShowButton ? 'none' : 'block'};\n                ${globalOptions.position.showButtonSideX}:${globalOptions.position.showButtonDistance.split(',')[0]}px;\n                ${globalOptions.position.showButtonSideY}:${globalOptions.position.showButtonDistance.split(',')[1]}px;">\n      <a class="auto-task-website-btn"\n        href="javascript:void(0);"\n        target="_self"\n        title="${I18n('showButton')}"> </a>\n    </div>\n  `);
    const $autoTaskInfo = $('#auto-task-info');
    const $autoTaskButtons = $('#auto-task-buttons');
    const $showButtonDiv = $('div.show-button-div');
    $showButtonDiv.on('click', (() => {
      $autoTaskButtons.show();
      $showButtonDiv.hide();
    }));
    if (website.buttons && $autoTaskButtons.children().length === 0) {
      $autoTaskButtons.addClass(`${website.name}-buttons`);
      for (const button of website.buttons) {
        if (website[button]) {
          const btnElement = $(`<p><a class="auto-task-website-btn ${website.name}-button" href="javascript:void(0);" target="_self">${I18n(button)}</a></p>`);
          btnElement.find('a.auto-task-website-btn').on('click', (() => {
            website[button]();
          }));
          $autoTaskButtons.append(btnElement);
        }
      }
    }
    const hideButtonElement = $(`<p><a class="auto-task-website-btn ${website.name}-button" href="javascript:void(0);" target="_self">${I18n('hideButton')}</a></p>`);
    hideButtonElement.find('a.auto-task-website-btn').on('click', (() => {
      $autoTaskButtons.hide();
      $showButtonDiv.show();
    }));
    const toggleLogElement = $(`<p><a id="toggle-log" class="auto-task-website-btn ${website.name}-button" href="javascript:void(0);" target="_self" data-status="${globalOptions.other.defaultShowLog ? 'show' : 'hide'}">${globalOptions.other.defaultShowLog ? I18n('hideLog') : I18n('showLog')}</a></p>`);
    const toggleLog = () => {
      const $toggleLog = $('#toggle-log');
      const status = $toggleLog.attr('data-status');
      if (status === 'show') {
        $autoTaskInfo.hide();
        $toggleLog.attr('data-status', 'hide').text(I18n('showLog'));
      } else {
        $autoTaskInfo.show();
        $toggleLog.attr('data-status', 'show').text(I18n('hideLog'));
      }
    };
    toggleLogElement.find('a.auto-task-website-btn').on('click', toggleLog);
    $autoTaskButtons.append(hideButtonElement).append(toggleLogElement);
    if (website.options) {
      GM_registerMenuCommand(I18n('changeWebsiteOptions'), (() => {
        websiteOptions(website.name, website.options);
      }));
    }
  };
  const initializeHotkeys = website => {
    debug('初始化热键', {
      website: website.name
    });
    keyboardJS.bind(globalOptions.hotKey.doTaskKey, (() => {
      if (website.doTask) {
        website.doTask();
      }
    }));
    keyboardJS.bind(globalOptions.hotKey.undoTaskKey, (() => {
      if (website.undoTask) {
        website.undoTask();
      }
    }));
    keyboardJS.bind(globalOptions.hotKey.toggleLogKey, (() => {
      const $toggleLog = $('#toggle-log');
      const status = $toggleLog.attr('data-status');
      const $autoTaskInfo = $('#auto-task-info');
      if (status === 'show') {
        $autoTaskInfo.hide();
        $toggleLog.attr('data-status', 'hide').text(I18n('showLog'));
      } else {
        $autoTaskInfo.show();
        $toggleLog.attr('data-status', 'show').text(I18n('hideLog'));
      }
    }));
  };
  const checkSteamASFStatus = async () => {
    debug('检查Steam ASF状态');
    if (!globalOptions.ASF.AsfEnabled || !globalOptions.ASF.AsfIpcUrl || !globalOptions.ASF.AsfIpcPassword) {
      return;
    }
    const stopPlayTime = GM_getValue('stopPlayTime', 0) || 0;
    if (stopPlayTime === 0 || stopPlayTime >= Date.now()) {
      return;
    }
    const stopPlayTimeMinutes = Math.floor((Date.now() - stopPlayTime) / 6e4);
    const {value: value} = await Swal.fire({
      title: I18n('stopPlayTimeTitle'),
      text: I18n('stopPlayTimeText', stopPlayTimeMinutes.toString()),
      icon: 'warning',
      confirmButtonText: I18n('confirm'),
      cancelButtonText: I18n('cancel'),
      showCancelButton: true
    });
    if (!value) {
      return;
    }
    let steamASF = new SteamASF(globalOptions.ASF);
    try {
      const isInitialized = await steamASF.init();
      if (!isInitialized) {
        return;
      }
      const isGamesStopped = await steamASF.stopPlayGames();
      if (!isGamesStopped) {
        return;
      }
      const taskLink = GM_getValue('taskLink', []) || [];
      for (const link of taskLink) {
        GM_openInTab(link, {
          active: true
        });
      }
      GM_setValue('stopPlayTime', 0);
      GM_setValue('playedGames', []);
      GM_setValue('taskLink', []);
    } catch (error) {
      console.error('SteamASF operation failed:', error);
    } finally {
      steamASF = null;
    }
  };
  const checkVersionAndNotice = () => {
    debug('检查版本和通知');
    const {scriptHandler: scriptHandler} = GM_info;
    if (scriptHandler !== 'Tampermonkey') {
      debug('未知脚本管理器', {
        scriptHandler: scriptHandler
      });
      echoLog({}).warning(I18n('unknownScriptHandler'));
      return;
    }
    const [v1, v2] = GM_info.version?.split('.') || [];
    if (!(parseInt(v1, 10) >= 5 && parseInt(v2, 10) >= 2)) {
      echoLog({}).error(I18n('versionNotMatched'));
    }
    if (!GM_getValue('notice')) {
      Swal.fire({
        title: I18n('swalNotice'),
        icon: 'warning'
      }).then((() => {
        GM_openInTab(I18n('noticeLink'), {
          active: true
        });
        GM_setValue('notice', (new Date).getTime());
      }));
      echoLog({
        html: `<li><font class="warning">${I18n('echoNotice', I18n('noticeLink'))}</font></li>`
      }).font?.find('a').on('click', (() => {
        GM_setValue('notice', (new Date).getTime());
      }));
    }
  };
  const loadScript = async () => {
    debug('主程序入口 loadScript 开始');
    if (window.name === 'ATv4_twitchAuth' && window.location.hostname === 'www.twitch.tv') {
      debug('检测到Twitch认证窗口');
      await handleTwitchAuth();
      return;
    }
    if (window.name === 'ATv4_redditAuth' && window.location.hostname === 'www.reddit.com') {
      debug('检测到Reddit认证窗口');
      await handleRedditAuth();
      return;
    }
    let website;
    for (const Website of Websites) {
      if (Website.test()) {
        debug('识别到支持的网站', {
          website: Website.name
        });
        website = new Website;
        break;
      }
    }
    if (!website) {
      debug('未识别到支持的网站,脚本停止加载');
      console.log('%c%s', 'color:#ff0000', 'Auto-Task[Warning]: 脚本停止加载,当前网站不支持!');
      return;
    }
    if (website.before) {
      debug('执行网站 before 钩子');
      await website.before();
    }
    initializeUI(website);
    initializeHotkeys(website);
    if (website.after) {
      debug('执行网站 after 钩子');
      await website.after();
    }
    if (website.name !== 'Setting') {
      debug('注册全局菜单命令');
      GM_registerMenuCommand(I18n('changeGlobalOptions'), (() => {
        changeGlobalOptions('swal');
      }));
      GM_registerMenuCommand(I18n('settingPage'), (() => {
        GM_openInTab('https://auto-task.hclonely.com/setting.html', {
          active: true
        });
      }));
    }
    debug('脚本加载完成');
    console.log('%c%s', 'color:#1bbe1a', 'Auto-Task[Load]: 脚本加载完成');
    if (window.DEBUG) {
      echoLog({}).warning(I18n('debugModeNotice'));
    }
    await checkSteamASFStatus();
    checkVersionAndNotice();
    updateChecker();
  };
  try {
    debug('主程序入口开始', {
      hostname: window.location.hostname,
      windowName: window.name
    });
    if ([ 'freeanywhere.net', 'give.gamesforfarm.local', 'gamesforfarm-testing.ru', 'store.steampowered.com', 'steamcommunity.com', 'www.youtube.com', 'm.youtube.com', 'mee6.xyz', 'gamesforfarm.com' ].includes(window.location.hostname) && $('.task-check-extension').length > 0) {
      debug('检测到freeanywhere.com,加载扩展');
      fawExtension();
    }
    if (window.location.hostname === 'discord.com') {
      if (window.name === 'ATv4_discordAuth') {
        debug('检测到Discord认证窗口');
        handleDiscordAuth();
      } else {
        debug('检测到Discord主站');
        const discordAuth = window.localStorage?.getItem('token')?.replace(/^"|"$/g, '');
        if (discordAuth && discordAuth.length > 0) {
          debug('获取到Discord认证token');
          GM_setValue('discordAuth', {
            auth: discordAuth
          });
        }
      }
    } else if (window.location.hostname === 'opquests.com') {
      debug('检测到opquests.com,加载主脚本');
      loadScript();
    } else if ((window.name === 'ATv4_updateStoreAuth' || GM_getValue('ATv4_updateStoreAuth')) && window.location.host === 'store.steampowered.com') {
      debug('检测到Steam商店认证窗口');
      $((() => {
        if ($('[data-miniprofile]').length === 0) {
          return;
        }
        handleSteamStoreAuth();
      }));
    } else if ((window.name === 'ATv4_updateCommunityAuth' || GM_getValue('ATv4_updateCommunityAuth')) && window.location.host === 'steamcommunity.com') {
      debug('检测到Steam社区认证窗口');
      $((() => {
        handleSteamCommunityAuth();
      }));
    } else {
      if (window.location.hostname === 'key-hub.eu') {
        debug('检测到key-hub.eu,设置全局变量');
        unsafeWindow.keyhubtracker = 1;
        unsafeWindow.gaData = {};
      }
      debug('加载主脚本');
      $(loadScript);
    }
  } catch (error) {
    debug('主程序入口发生异常', {
      error: error
    });
  }
})(Swal, Cookies, browser, util, dayjs, keyboardJS);