您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Greasy Fork is available in English.
自动完成 Freeanywhere,Giveawaysu,GiveeClub,Givekey,Gleam,Indiedb,keyhub,OpiumPulses,Opquests,SweepWidget 等网站的任务。
// ==UserScript== // @name auto-task // @namespace auto-task // @version 5.0.4 // @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) { result[key] = 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]+?), ('|')(.*?)('|')/) || []; 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":"(\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":{"[\d]+?":\[".+?"\]/.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 (/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.*hours/gi.test(taskText)) { const link = $task.find('a[href^="https://steamcommunity.com/app/"],a[href^="https://store.steampowered.com/app/"]').attr('href'); if (!link) { continue; } if (action === 'undo') { this.socialTasks.steam.playTimeLinks.push(link); } if (action === 'do') { this.undoneTasks.steam.playTimeLinks.push(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); await Swal.fire({ title: I18n('stopPlayTimeTitle'), text: I18n('stopPlayTimeText', stopPlayTimeMinutes.toString()), icon: 'warning', confirmButtonText: I18n('confirm') }); 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);