try to take over the world!
// ==UserScript== // @name Steam Key Helper // @namespace http://tampermonkey.net/ // @version 2.0.0 // @description try to take over the world! // @icon http://store.steampowered.com/favicon.ico // @author Bisumaruko // @include http*://* // @exclude http*://*dailyindiegame.com/account_createtrade.html // @exclude http*://steam.depar.me/* // @require https://cdnjs.cloudflare.com/ajax/libs/jquery/3.2.1/jquery.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/mark.js/8.11.1/jquery.mark.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/limonte-sweetalert2/7.0.6/sweetalert2.min.js // @resource SweetAlert2CSS https://cdnjs.cloudflare.com/ajax/libs/limonte-sweetalert2/7.0.6/sweetalert2.min.css // @connect store.steampowered.com // @grant GM_xmlhttpRequest // @grant GM_setValue // @grant GM_getValue // @grant GM_addStyle // @grant GM_getResourceText // @run-at document-start // @noframes // ==/UserScript== /* global swal, g_AccountID, g_sessionID, g_oSuggestParams */ // setup jQuery const $ = jQuery.noConflict(true); // inject CSS GM_addStyle(` ${GM_getResourceText('SweetAlert2CSS')} .hide { display: none; } .SKH_processed { color: #57bae8; cursor: pointer; } .SKH_processed:hover { text-decoration: underline; } .SKH_activated { text-decoration: line-through; } .SKH_panel { width: 60px; height: 60px; position: fixed; top: 50%; right: 20px; transform: translateY(-50%); background-color: rgb(87, 186, 232); opacity: 0.5; text-align: center; } .SKH_panel:hover { opacity: 1; } .SKH_panel .switch { position: relative; display: inline-block; width: 40px; height: 24px; margin-top: 5px; } .SKH_panel .switch input { display: none; } .SKH_panel .slider { position: absolute; cursor: pointer; top: 0; left: 0; right: 0; bottom: 0; background-color: #ccc; transition: 0.4s; } .SKH_panel .slider:before { position: absolute; content: ""; height: 20px; width: 20px; left: 2px; bottom: 2px; background-color: white; transition: 0.4s; } .SKH_panel input:checked + .slider { background-color: #2196F3; } .SKH_panel input:focus + .slider { box-shadow: 0 0 1px #2196F3; } .SKH_panel input:checked + .slider:before { transform: translateX(16px); } .SKH_panel > span { display: inline-block; cursor: pointer; color: white; } .SKH_panel > button { width: 20px; height: 20px; position: absolute; bottom: 0; padding: 2px; background-color: transparent; background-size: 18px; background-repeat: no-repeat; background-origin: padding-box; background-position: 50% 50%; border: none; outline: none; box-sizing: border-box; cursor: pointer; } .SKH_panel > .hidePanel { left: 0; background-image: url(); filter: opacity(60%); } .SKH_panel > .disable { left: 20px; background-size: contain; background-image: url(); filter: opacity(60%); } .SKH_panel > .settings { left: 40px; background-image: url(); } .SKH_settings .name { text-align: right; vertical-align: top; white-space: nowrap; } .SKH_settings .value { text-align: left; } .SKH_settings .value > * { height: 30px; margin: 0 20px 10px; } .SKH_settings .switch { position: relative; display: inline-block; width: 60px; } .SKH_settings .switch input { display: none; } .SKH_settings .slider { position: absolute; cursor: pointer; top: 0; left: 0; right: 0; bottom: 0; background-color: #ccc; transition: 0.4s; } .SKH_settings .slider:before { position: absolute; content: ""; height: 26px; width: 26px; left: 2px; bottom: 2px; background-color: white; transition: 0.4s; } .SKH_settings input:checked + .slider { background-color: #2196F3; } .SKH_settings input:focus + .slider { box-shadow: 0 0 1px #2196F3; } .SKH_settings input:checked + .slider:before { transform: translateX(30px); } .SKH_settings input[type=text] { width: 200px; } .SKH_settings input[type=number] { width: 60px; height: 30px; box-sizing: border-box; } .SKH_settings textarea { width: 200px; min-width: 200px; height: 60px; min-height: 60px; } .SKH_settings label + p { display: none; transition: display 1s; } .SKH_settings label ~ *:last-child { display: block; transition: display 1s; } .SKH_settings .toggleOn label + p { display: block; } .SKH_settings .toggleOn label ~ *:last-child { display: none; } `); // load config const eol = "\n"; const has = Object.prototype.hasOwnProperty; const regKey = /(?:(?:([A-Z0-9])(?!\1{4})){5}-){2,5}[A-Z0-9]{5}/g; let activating = false; const activated = JSON.parse(GM_getValue('SKH_activated') || '[]'); const currentActivated = []; const config = { data: JSON.parse(GM_getValue('SKH_config') || '{}'), save() { GM_setValue('SKH_config', JSON.stringify(this.data)); }, set(key, value, callback = null) { this.data[key] = value; this.save(); if (typeof callback === 'function') callback(); }, get(key) { return has.call(this.data, key) ? this.data[key] : null; }, push(key, value) { if (Array.isArray(this.data[key])) { this.data[key].push(value); this.save(); } }, splice(key, start, count) { if (Array.isArray(this.data[key])) { this.data[key].splice(start, count); this.save(); } }, init() { if (!has.call(this.data, 'autoUpdateSessionID')) this.data.autoUpdateSessionID = true; if (!has.call(this.data, 'autoClosePopup')) this.data.autoClosePopup = false; if (!has.call(this.data, 'autoClosePopupTimer')) this.data.autoClosePopupTimer = 3; if (!has.call(this.data, 'autoReplyActivated')) this.data.autoReplyActivated = false; if (!has.call(this.data, 'enabledForever')) this.data.enabledForever = true; if (!has.call(this.data, 'enabled')) this.data.enabled = []; if (!has.call(this.data, 'disabled')) this.data.disabled = []; if (!has.call(this.data, 'hideForever')) this.data.hideForever = false; if (!has.call(this.data, 'hide')) this.data.hide = []; if (!has.call(this.data, 'off')) this.data.off = []; } }; config.init(); // text const i18n = { tchinese: { name: '繁體中文', updat###ccessTitle: '更新成功!', updat###ccess: '成功更新Steam sessionID', errorTitle: '糟糕!', errorUnexpected: '發生未知錯誤,請稍後再試', errorInvalidKey: '序號錯誤', errorUsedKey: '序號已被使用', errorRateLimited: '啟動受限', errorCountryRestricted: '地區限制', errorAlreadyOwned: '產品已擁有', errorMissingBaseGame: '未擁有主程式', errorPS3Required: '需要PS3 啟動', errorGiftWallet: '偵測到禮物卡/錢包序號', errorFailedRequest: '處理資料發生錯誤,請稍後再試', errorFailedRequestNeedUpdate: '請求發生錯誤,請稍後再試<br>或者嘗試更新SessionID', errorActivated: '你已啟動過這序號', successTitle: '啟動成功!', processingTitle: '喵~', processingMsg: '啟動序號中,請稍後', notLoggedInTitle: '未登入', notLoggedInMsg: '請登入Steam 以讓腳本紀錄SessionID', missingTitle: '未發現SessionID', missingMsg: '請問要更新SessionID 嗎?', titlehidePanel: '在這網站隱藏懸浮框', titleDisable: '在這網站禁止腳本', titleSettings: '開啟設定', settingsTitle: '設定', settingsAutoUpdateSessionID: '自動更新SessionID', settingsSessionID: '我的sessionID', settingsLanguage: '語言', settingsAutoClosePopup: '自動關閉彈窗', settingsTimerDisabled: '不關閉彈窗', settingsTimerSecond: '秒', settingsAutoReplyActivated: '自動回覆啟動序號', settingsEnabledForever: '全域運行腳本', settingsEnabled: '在這些網站運行腳本', settingsDisabled: '在這些網站禁止腳本', settingsHideForever: '全域隱藏懸浮', settingsHide: '在這些網站隱藏懸浮', settingsOff: '在這些網站暫停腳本', placeholderEnabled: '如不全域運行腳本此欄不得為空', activatedReply: '感謝大佬!已拿%KEYS%' }, schinese: { name: '简体中文', updat###ccessTitle: '更新成功', updat###ccess: '成功更新Steam sessionID', errorTitle: '糟糕!', errorUnexpected: '发生未知错误,请稍后再试', errorInvalidKey: '激活码错误', errorUsedKey: '激活码已被使用', errorRateLimited: '激活受限', errorCountryRestricted: '地区限制', errorAlreadyOwned: '产品已拥有', errorMissingBaseGame: '未拥有游戏本体', errorPS3Required: '需要PS3 激活', errorGiftWallet: '侦测到礼物卡/钱包激活码', errorFailedRequest: '处理资料发生错误,请稍后再试', errorFailedRequestNeedUpdate: '请求发生错误,请稍后再试<br>或者尝试更新SessionID', errorActivated: '你已激活过这激活码', successTitle: '激活成功!', processingTitle: '喵~', processingMsg: '激活中,请稍后', notLoggedInTitle: '未登入', notLoggedInMsg: '请登入Steam 以让脚本记录SessionID', missingTitle: '未发现SessionID', missingMsg: '请问要更新SessionID 吗?', titlehidePanel: '在这网站隐藏悬浮', titleDisable: '在这网站禁止脚本', titleSettings: '打开设置', settingsTitle: '设置', settingsAutoUpdateSessionID: '自动更新SessionID', settingsSessionID: '我的sessionID', settingsLanguage: '语言', settingsAutoClosePopup: '自动关闭弹窗', settingsTimerDisabled: '不关闭弹窗', settingsTimerSecond: '秒', settingsAutoReplyActivated: '自动回复激活KEY', settingsEnabledForever: '全域运行脚本', settingsEnabled: '在这些网站运行脚本', settingsDisabled: '在这些网站禁止脚本', settingsHideForever: '全域隐藏悬浮', settingsHide: '在这些网站隐藏悬浮', settingsOff: '在这些网站暂停脚本', placeholderEnabled: '如不全域运行脚本此栏不得为空', activatedReply: '感谢大佬!已激活%KEYS%' }, english: { name: 'English', updat###ccessTitle: 'Update Successful!', updat###ccess: 'Steam sessionID is successfully updated', errorTitle: 'Opps!', errorUnexpected: 'An unexpected error has occured, please try again later', errorInvalidKey: 'Invalid Key', errorUsedKey: 'Used Key', errorRateLimited: 'Rate Limited', errorCountryRestricted: 'Country Restricted', errorAlreadyOwned: 'Product Already Owned', errorMissingBaseGame: 'Missing Base Game', errorPS3Required: 'PS3 Activation Required', errorGiftWallet: 'Gift Card/Wallet Code Detected', errorFailedRequest: 'R###lt parse failed, please try again', errorFailedRequestNeedUpdate: 'Request failed, please try again<br>or update sessionID', errorActivated: 'You have activated this key before', successTitle: 'Activation Successful!', processingTitle: 'Nyaa~', processingMsg: 'Activating key, please wait', notLoggedInTitle: 'Not Logged-In', notLoggedInMsg: 'Please login to Steam so sessionID can be saved', missingTitle: 'Missing SessionID', missingMsg: 'Do you want to update your Steam sessionID?', titlehidePanel: 'Hide floating panel on this site', titleDisable: 'Disable script on this site', titleSettings: 'Open settings panel', settingsTitle: 'Settings', settingsAutoUpdateSessionID: 'Auto Update SessionID', settingsSessionID: 'Your sessionID', settingsLanguage: 'Language', settingsAutoClosePopup: 'Close popup in', settingsTimerDisabled: 'Don\'t close popup', settingsTimerSecond: 'second', settingsAutoReplyActivated: 'Auto Reply Activated Keys', settingsEnabledForever: 'Enable scrip all the time', settingsEnabled: 'Enable script on', settingsDisabled: 'Disable script on', settingsHideForever: 'Hide floating panel all the time', settingsHide: 'Hide floating panel on', settingsOff: 'Turn off script on', placeholderEnabled: 'Must not be empty if script not enabled all the time', activatedReply: 'Thank you for the keys! Activated %KEYS%' } }; let text = has.call(i18n, config.get('language')) ? i18n[config.get('language')] : i18n.english; // functions const getSessionID = () => { GM_xmlhttpRequest({ method: 'GET', url: 'https://store.steampowered.com/', onload: res => { if (res.status === 200) { const accountID = res.response.match(/g_AccountID = (\d+)/).pop(); const sessionID = res.response.match(/g_sessionID = "(\w+)"/).pop(); if (accountID > 0) config.set('sessionID', sessionID);else { swal({ title: text.notLoggedInTitle, text: text.notLoggedInMsg, type: 'error', showCancelButton: true }).then(() => { window.open('https://store.steampowered.com/'); }); } } } }); }; const settings = () => { const panelHTML = ` <div class="SKH_settings"> <table> <tr> <td class="name">${text.settingsAutoUpdateSessionID}</td> <td class="value"> <label class="switch"> <input type="checkbox" class="autoUpdateSessionID"> <span class="slider"></span> </label> </td> </tr> <tr> <td class="name">${text.settingsSessionID}</td> <td class="value"> <input type="text" class="sessionID" value="${config.get('sessionID')}" disabled> </td> </tr> <tr> <td class="name">${text.settingsLanguage}</td> <td class="value"> <select class="language"></select> </td> </tr> <tr> <td class="name">${text.settingsAutoClosePopup}</td> <td class="value"> <label class="switch"> <input type="checkbox" class="autoClosePopup"> <span class="slider"></span> </label> <p><input type="number" class="autoClosePopupTimer" min="0" value="${config.get('autoClosePopupTimer')}"> ${text.settingsTimerSecond}</p> <p class="timerDisabledText">${text.settingsTimerDisabled}</p> </td> </tr> <tr> <td class="name">${text.settingsAutoReplyActivated}</td> <td class="value"> <label class="switch"> <input type="checkbox" class="autoReplyActivated"> <span class="slider"></span> </label> </td> </tr> <tr> <td class="name">${text.settingsEnabled}</td> <td class="value"> <label class="switch"> <input type="checkbox" class="enabledForever"> <span class="slider"></span> </label> <p class="enabledForeverText">${text.settingsEnabledForever}</p> <textarea class="enabledList" placeholder="${text.placeholderEnabled}"></textarea> </td> </tr> <tr> <td class="name">${text.settingsDisabled}</td> <td class="value"> <textarea class="disabledList"></textarea> </td> </tr> <tr> <td class="name">${text.settingsHide}</td> <td class="value"> <label class="switch"> <input type="checkbox" class="hideForever"> <span class="slider"></span> </label> <p class="hideForeverText">${text.settingsHideForever}</p> <textarea class="hideList"></textarea> </td> </tr> <tr> <td class="name">${text.settingsOff}</td> <td class="value"> <textarea class="offList"></textarea> </td> </tr> </table> </div> `; const panelHandler = () => { // apply settings const $panel = $(swal.getContent()); const $sessionID = $panel.find('.sessionID'); const $language = $panel.find('.language'); // toggles $panel.find('input[type="checkbox"]').each((index, input) => { const $input = $(input); $input.change(e => { swal.showLoading(); const setting = e.delegateTarget.className; const state = e.delegateTarget.checked; config.set(setting, state); $(e.delegateTarget).closest('td').toggleClass('toggleOn', state); if (setting === 'autoUpdateSessionID') $sessionID.attr('disabled', state); setTimeout(swal.hideLoading, 500); }); $input.prop('checked', !!config.get(input.className)).trigger('change'); }); // sessionID input $sessionID.prop('disabled', config.get('autoUpdateSessionID')); $sessionID.change(() => { swal.showLoading(); config.set('sessionID', $sessionID.val().trim()); setTimeout(swal.hideLoading, 500); }); // language Object.keys(i18n).forEach(language => { $language.append(new Option(i18n[language].name, language)); }); $panel.find(`option[value=${config.get('language')}]`).prop('selected', true); $language.change(() => { swal.showLoading(); const newLanguage = $language.val(); config.set('language', newLanguage); text = has.call(i18n, newLanguage) ? i18n[newLanguage] : i18n.english; setTimeout(swal.hideLoading, 500); }); // timer $panel.find('.autoClosePopupTimer').change(e => { const second = parseInt(e.delegateTarget.value, 10); config.set('autoClosePopupTimer', Number.isInteger(second) ? second : 3); }); // websites list ['enabled', 'disabled', 'hide', 'off'].forEach(list => { const $list = $panel.find(`.${list}List`); $list.val(config.get(list).join(eol)); $list.change(() => { swal.showLoading(); const hostList = $list.val().split(eol).map(x => x.trim()).filter(x => x.length); config.set(list, hostList); if (list === 'enabled') { const state = !!$panel.find('.enabledForever:checked').length; // script is not enabled all the time and empty enabled website if (!state && hostList.length === 0) config.set('enabledForever', true); } setTimeout(swal.hideLoading, 500); }); }); }; swal({ title: text.settingsTitle, html: panelHTML, onBeforeOpen: panelHandler }); }; const insertPanel = callback => { const $panel = $('<div class="SKH_panel"></div>'); // add toggle switch $panel.append($(` <label class="switch"> <input type="checkbox"> <span class="slider"></span> </label> `).change(() => { const host = location.hostname; const toggle = !!$('.SKH_panel input:checked').length; const index = config.get('off').indexOf(host); if (toggle && index > -1) config.splice('off', index, 1);else if (!toggle && index === -1) config.push('off', host); })); // add hide button $panel.append($(`<button class="hidePanel" title="${text.titlehidePanel}"> </button>`).click(() => { config.push('hide', location.hostname); $panel.remove(); })); // add disabled button $panel.append($(`<button class="disable" title="${text.titleDisable}"> </button>`).click(() => { config.push('disabled', location.hostname); $panel.remove(); })); // add settings button $panel.append($(`<button class="settings" class="${text.titleSettings}"> </button>`).click(settings)); $('body').append($panel); callback(); // hide panel after 5 seconds setTimeout(() => { $panel.hide(); }, 5000); }; const getR###ltMsg = r###lt => { const msg = { title: text.errorTitle, html: text.errorUnexpected, type: 'error' }; const errors = { 14: text.errorInvalidKey, 15: text.errorUsedKey, 53: text.errorRateLimited, 13: text.errorCountryRestricted, 9: text.errorAlreadyOwned, 24: text.errorMissingBaseGame, 36: text.errorPS3Required, 50: text.errorGiftWallet }; if (r###lt.success === 1) { msg.title = text.successTitle; msg.html = ''; msg.type = 'success'; } else if (r###lt.success === 2) { const errorCode = r###lt.purchase_r###lt_details; if (has.call(errors, errorCode)) msg.html = errors[errorCode]; } const details = []; r###lt.purchase_receipt_info.line_items.forEach(item => { const detail = [`<b>${item.line_item_description}</b>`]; if (item.packageid > 0) detail.push(`sub: <a target="_blank" href="https://steamdb.info/sub/${item.packageid}/">${item.packageid}</a>`); if (item.appid > 0) detail.push(`app: <a target="_blank" href="https://steamdb.info/sub/${item.appid}/">${item.appid}</a>`); details.push(detail.join(', ')); }); if (details.length > 0) msg.html += `<br>${details.join('<br>')}`; return msg; }; const activateKey = key => { if ($('.SKH_panel input:checked').length > 0 && !activating) { activating = true; swal(text.processingTitle, text.processingMsg); swal.showLoading(); const timer = config.get('autoClosePopup') ? config.get('autoClosePopupTimer') * 1000 : null; if (activated.includes(key)) { swal.close(); if (timer !== 0) { swal({ title: text.errorTitle, text: text.errorActivated, type: 'error', timer }).catch(swal.noop); } activating = false; } else { GM_xmlhttpRequest({ method: 'POST', url: 'https://store.steampowered.com/account/ajaxregisterkey/', headers: { 'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8', Origin: 'https://store.steampowered.com', Referer: 'https://store.steampowered.com/account/registerkey' }, data: `product_key=${key}&sessionid=${config.get('sessionID')}`, onload: res => { swal.close(); if (res.status === 200) { try { const r###lt = JSON.parse(res.response); const msg = getR###ltMsg(r###lt); if (timer !== 0) { swal({ title: msg.title, html: msg.html, type: msg.type, timer }).catch(swal.noop); } // update activated if (!activated.includes(key)) { const failCode = r###lt.purchase_r###lt_details; if (r###lt.success === 1 || [14, 15, 9].includes(failCode)) { activated.push(key); GM_setValue('SKH_activated', JSON.stringify(activated)); $(`span:contains(${key}), [value=${key}]`).addClass('SKH_activated'); } } // insert activated in this session if (r###lt.success === 1) currentActivated.push(key); } catch (e) { swal(text.errorTitle, text.errorFailedRequest, 'error'); } } else { const errorMsg = []; errorMsg.push('<pre class="SKH_errorMsg">'); errorMsg.push(`sessionID: ${config.get('sessionID') + eol}`); errorMsg.push(`autoUpdate: ${config.get('autoUpdateSessionID') + eol}`); errorMsg.push(`status: ${res.status + eol}`); errorMsg.push(`response: ${res.response + eol}`); errorMsg.push('</pre>'); swal({ title: text.errorTitle, html: text.errorFailedRequestNeedUpdate + eol + errorMsg.join(''), type: 'error' }); getSessionID(); } activating = false; } }); } } }; const scanInput = (i, input) => { if (input.type === 'text' && regKey.test(input.value)) { const $input = $(input); const key = input.value.trim(); if (activated.includes(key)) $input.addClass('SKH_activated'); $input.prop('disabled', false); $input.click(() => { activateKey(key); }); } }; const init = () => { // save sessionID if (location.hostname === 'store.steampowered.com') { if (g_AccountID > 0) { const currentID = config.get('sessionID'); const sessionID = g_sessionID || ''; const language = g_oSuggestParams.l || 'english'; if (!config.get('language')) config.set('language', language); if (sessionID.length > 0) { const update = config.get('autoUpdateSessionID') && currentID !== sessionID; if (!currentID || update) { config.set('sessionID', sessionID, () => { swal({ title: text.updat###ccessTitle, text: text.updat###ccess, type: 'success', timer: 3000 }); }); } } } /* else { swal(text.notLoggedInTitle, text.notLoggedInMsg, 'error'); } */ } else { // check sessionID if (!config.get('sessionID')) getSessionID(); const host = location.hostname; let run = true; if (!config.get('enabledForever') && !config.get('enabled').includes(host)) run = false; if (config.get('disabled').includes(host)) run = false; if (run) { const markOptions = { element: 'span', exclude: ['.SKH_processed'], each: marked => { const $marked = $(marked); const key = $marked.text(); // checked if activated before if (activated.includes(key)) $marked.addClass('SKH_activated'); // bind click event $marked.click(e => { e.preventDefault(); activateKey(key); }); // append class $marked.addClass('SKH_processed'); } }; // hide floating panel if (config.get('hideForever') || config.get('hide').includes(host)) { GM_addStyle('.SKH_panel { display: none; }'); } // insert floating panel insertPanel(() => { // toggle on / off $('.SKH_panel input').prop('checked', !config.get('off').includes(host)); }); // start scanning $('body').markRegExp(regKey, markOptions); $('input[type="text"]').each(scanInput); // auto reply activated keys on SteamCN if (config.get('autoReplyActivated')) { $(window).on('unload', () => { if (currentActivated.length > 0) { $('######age').val(text.activatedReply.replace('%KEYS%', currentActivated.join())); $('#vreplysubmit').click(); } }); } // monitor new MutationObserver(mutations => { mutations.forEach(mutation => { // monitor added nodes Array.from(mutation.addedNodes).forEach(addedNode => { if (addedNode.nodeType === 1) { $(addedNode).each(scanInput).markRegExp(regKey, markOptions); } else if (addedNode.nodeType === 3) { $(addedNode.parentNode).markRegExp(regKey, markOptions); } }); // monitor text change if (mutation.type === 'characterData') { $(mutation.target.parentNode).markRegExp(regKey, markOptions); } }); }).observe(document.body, { childList: true, subtree: true, characterData: true }); } } }; $(window).on('load', init);