// ==UserScript== // @name Steam Badges Crafter // @name:zh-CN Steam一键批量合卡合徽章 // @name:zh-TW Steam一鍵批量合卡合徽章 // @version 1.1 // @description (Steam Auto Mass Craft Trading Cards Badges in Bulk) It will automatically use up your gamecard sets for crafting badges. You can control the which card sets and how many sets to craft by using it. // @description:zh-CN 这是一个自动合卡插件,可以指定徽章合成的数量和种类 // @description:zh-TW 這是一個自動合卡挿件,可以指定徽章合成的數量和種類 // @author QHS // @include *steamcommunity.com/*/badges* // @grant GM_addStyle // @grant GM_setValue // @grant GM_getValue // @supportURL https://steamcn.com/t339531-1-1 // @icon http://pan.hriq.org/steam.green.ico // @namespace https://greasyfork.org/users/155548-黑山東雲光圈研究所 // @namespace https://steamcommunity.com/profiles/76561198132556503 // @downloadURL none // ==/UserScript== var timer_scan = GM_getValue("timer_scan", 1000), //扫描卡组间隔 Interval: between badges scans(ms)[500+ recommended] timer_craft = GM_getValue("timer_craft", 500), //合成卡牌间隔 Interval: between crafting card sets(ms)[100+ recommended] sales=["245070","762800"], //Appid for sales cards config_cap_level = GM_getValue("config_cap_level", 0), //Set 1 if you want to craft all badges up to level 1 [1 - 5] config_blacklist = GM_getValue("config_blacklist", ''); //Set 1 if you want to craft all badges up to level 1 [1 - 5] (function() { 'use strict'; GM_addStyle(`.profile_xp_block_right { text-align: center!important; } .profile_xp_block_right { display: block; width: 230px; border: 1px #aaa solid; border-radius: 4px; padding: 8px 0px; cursor: pointer; transition: .5s } .profile_xp_block_right:hover { color: #333; background: #aaa } .craft_list p { margin: 2px 10px; } .craft_list p input { width: 36px; height: 13px; background-color: #152f4a; border: 1px #fff solid; color: #fff; padding: 2px 0 0 10px; font-weight: bold; transition: .5s; background-repeat: no-repeat; background-position-x: -46px; background-image: url("") } .cannot_craft { -webkit-filter: grayscale(100%); -moz-filter: grayscale(100%); -ms-filter: grayscale(100%); -o-filter: grayscale(100%); filter: grayscale(100%); filter: gray; cursor: not-allowed; } .sum_show { width: 882px; height: 95px; position: relative; top: -115px; z-index: 10000001; padding: 10px 0; background: #1d1d1d; border-top: #383838 1px solid; } div.craft_list { padding: 10px 0 95px 0; } .craft_list.dischange input { border: 1px #827d7d solid; color: #827d7d; cursor: not-allowed; } p.do_not_craft { opacity: .5; } input.ready_to_craft.finished { background-color: #ff0909; } input.ready_to_craft.in_progress { border: 1px #ffffff solid!important; color: #ffffff!important; } #start img { padding-top: 0px; } .b_icon { position: absolute!important; padding: 12.5px!important; margin: 8px 0 0 3px!important; } .calculate { background: url("")!important } .lightning { background: url("")!important } span.slightning { background: -webkit-linear-gradient( top, #e9b50e 5%, #5f2f05 95%)!important; linear-gradient( to bottom, #e9b50e 5%, #5f2f05 95%)!important; margin-left: 30px!important; } span.scalculate { background: -webkit-linear-gradient( top, #778088 5%, #414a52 95%)!important; background: linear-gradient( to bottom, #778088 5%, #414a52 95%)!important; margin-left: 30px!important; } .rapid { background: #d8a506!important; background: -webkit-linear-gradient( top, #f3d608 5%, #902100 95%)!important; background: linear-gradient( to bottom, #f3d608 5%, #902100 95%)!important; } .btn_large { float: left; margin-top: 28px; margin-left: 28px; } .rapid:hover { background: #d8a506!important; background: -webkit-linear-gradient( top, #fbe440 5%, #eb5224 95%)!important; background: linear-gradient( to bottom,#fbe440 5%, #eb5224 95%)!important; } .setting { float: left; margin-left: 28px; padding: 17.5px; margin-top: 31px; cursor: pointer; border: transparent solid 1px; border-radius: 5px; } .setting:hover { border: #fff solid 1px; } p.setting_title { font-size: 26px; color: #fff; border-bottom: 1px solid #fff; margin-bottom: 60px; padding-bottom: 10px; } font.config_name { color: #fff; font-weight: bold; } input.property { background: #ffffff; width: 55px; height: 19px; margin: 0 7px; border: 2px solid #0fa9f3; border-radius: 5px; } ._setting { background: url("")!important; } ._save { background: url("") } .attention { display: inline-block; width: 200px; height: 200px; background: url(""); } font.appid { border: 1px solid; padding: 2px 7px; border-radius: 8px; margin: 2px 3px; display: inline-block; text-decoration: line-through; } textarea.property { background: #ffffff; width: 730px; height: 30px; margin: 0 7px; border: 2px solid #0fa9f3; border-radius: 9px; } .sum_show { text-align: center; } .newmodal_buttons { display: inline-block; position: relative; left: -14px; } .newmodal_buttons { display: inline-block; } ._confirms,.sconfirm { background: linear-gradient( to bottom, #00ea54 5%, #004c06 95%)!important; } .ss { margin-left: 30px!important; } ._confirms:hover { background: linear-gradient( to bottom, #42ff86 5%, #00ce10 95%)!important; } span.b_icon._confirm { background: url("")!important; } ._cancels, .scancel { background: linear-gradient( to bottom, #ff6a00 5%, #a70000 95%)!important; } ._cancels:hover { background: linear-gradient( to bottom, #fd9a54 5%, #ff1010 95%)!important; } span.b_icon._cancel { background: url("")!important; }`); var _border, g_sessionID, badge_cap_level, __appID, _gappid,blacklist=[]; var text = {}, lan = $J('head').html().match(/l=([^"&*]+)"/)[1]; if (lan == "schinese" || lan == "tchinese") { text.start = "批量合成徽章"; text.title = "批量合卡"; text.notice = '

这是一个自动合卡插件,可以指定徽章合成的数量和种类。请自行承担操作不当带来的风险

可在SteamCNGithubGreasy Fork反馈问题,也可以在我的资料页下方留言(ASF经常吞消息可能看不到私聊0.0)

可以随时关闭本标签页来停止插件的自动操作

计算模式: 先扫描出每个徽章的可合成次数,扫描完毕后可以手动调整合成数量,再通过确认进行批量合卡。

极速模式: 先显示您的黑名单设置,确认后将直接合成所有可以用来合成的徽章,不会使用黑名单中的游戏卡牌

示例GIF1 示例GIF2

'; text.button1 = "开始统计本页可合成卡组(计算模式)"; text.buttonr1 = "不管那么多了,跳过扫描直接合卡!(极速模式)"; text.button2 = "确认无误后开始合卡"; text.button2no = "不合卡的吗?"; text.calculating = "计算中(输入想要合成的次数,计算过程中可以关闭本弹出窗口)"; text.calculated = "请在每个徽章前输入想要合成的次数,并确认合成"; text.crafting = "合成中,合成过程中可以关闭本弹出窗口"; text.crafted = "合成完毕"; text.list1 = '←输入合成套数'; text.list2 = '可升'; text.list3 = '级 (当前'; text.list4 = '级) 游戏:'; text.nosets1 = '

想合卡?先去市场买来再说吧!

'; text.nosets2 = '这是一个悲伤地故事'; text.nosets3 = '虽














多'; text.setting_title = '设置(下面三个按钮都能保存)'; text.cap_level = '最高等级'; text.cap_level_d = '(仅针对计算模式有效!默认为0。例如设置为3:默认将小于3的徽章合成至3级。若设置0则无任何合成限制)'; text.timer_scan = '扫描时间'; text.timer_scan_d = '(仅针对计算模式有效,计算中扫描每个徽章页面的时间间隔(单位毫秒),默认为1000,请不要设置的过小)'; text.timer_craft = '合成时间间隔'; text.timer_craft_d = '(仅针对计算模式有效,计算过后合成徽章的时间间隔(单位毫秒),默认为500,请不要设置的过小)'; text.blacklist = '黑名单'; text.blacklist_d = '(将游戏的APPID填入,用逗号分隔,则该游戏的普通/闪亮徽章在任何模式下都不会合成。写法形如 550,322310,730)'; text.attention_title = '注意!
确认过后将进行合成,如果需要停止请关闭本标签页. '; text.attention_title2 = '您并没有设置黑名单,所以所有可合成徽章都将进行合成)'; text.attention_title3 = '除了这些在你黑名单中的游戏: '; text.confirm = '确认'; text.cancel = '取消'; } else { text.start = "Craft Badges in Bulk"; text.title = "Craft Badges in Bulk"; text.notice = '

SteamCNGithubGreasy Fork or comment on Steam profile(I always miss the chat message as getting command through chat by asf) for feedback.

Close this WEBPAGE when you want to stop crafting!

You can set intervals and blacklist badges in setting.

Calculation mode: Scan and calculate max badges you can craft first and you can regulating the number of card sets for specified bagdes. Then craft.

Rapidly mode: Show you the setting of blacklist. Crafting immediately after you confirm it. It will use up ALL your available gamecard sets for crafting badges except the games whose APPID is in the blacklist.

Demo1 Demo2

'; text.button1 = "Calculate how many badges you can craft in this page before craft"; text.buttonr1 = "Craft now rapidly!"; text.button2 = "Start Crafting!"; text.button2no = "No Cards for Crafting!"; text.calculating = "Calculating, you can close this pop-up window in the process of calculation"; text.calculated = "Enter the times you want to craft for the badges. Confirm and craft them by click the green button below"; text.crafting = "Crafting! You can close this pop-up window in the process of crafting"; text.crafted = "Finished!"; text.list1 = 'sets to craft'; text.list2 = 'can craft'; text.list3 = 'more (Level'; text.list4 = 'now) GAME:'; text.nosets1 = '

No sets to craft qwq

'; text.nosets2 = 'So sad'; text.nosets3 = 'T
h
e
r
e

i
s

n
o

r
e
s
u
l
t
!'; text.setting_title = 'Setting (Click the button below to save it)'; text.cap_level = 'Cap Level'; text.cap_level_d = '(Only for calculation mode! Number type with default value of 0. Set to 3 if you want to craft all badges up to level 3 [also can be 1,2,4,5]. There is no level limit if set to 0.)'; text.timer_scan = 'Scan Timer'; text.timer_scan_d = '(The interval between badges when calculating. number type with default value of 1000(ms). Unless you have a reason to edit this property, you should keep it at default.)'; text.timer_craft = 'Craft Timer'; text.timer_craft_d = '(The interval between cards sets crafting. number type with default value of 500(ms). Unless you have a reason to edit this property, you should keep it at default. Only for calculation mode.)'; text.blacklist_d = '(The cards whose APPID is in the blacklist will not being used for crafting in any mode. Separated by commas, like 550,322310,730)'; text.attention_title = 'Attention!
Please confirm that you are going to use up ALL your available gamecard sets for crafting badges. '; text.attention_title2 = 'As there is no APPID in your blacklist(you can add some in setting)'; text.attention_title3 = 'Except the games whose APPID is in the blacklist: '; text.confirm = 'Confirm'; text.cancel = 'Cancel'; text.blacklist = 'Blacklist'; } $J('body').prepend(` `); for (var i = 0; i < 20; i++) { setTimeout(function() { $J('.profile_xp_block_right').html(text.start); $J('.es_faq_cards').hide(); }, i * 500); } //hide the content that is not necessary function craft() { _save(); $J('.window_title').html(text.calculating); $J('.craft_title').slideUp(); $J('#start').html(''); var total_number = $J('a.badge_craft_button').length, each_count = 0, sum_sets = 0, sum_badges = 0; if (total_number === 0) { $J('#start').html(text.nosets1); $J('.window_title').html(text.nosets2); $J('.craft_list').append("

" + text.nosets3 + "

"); $J('#start').addClass('start_2').removeClass('start_1'); $J('.craft_list').append("

=========END=========

"); } $J('a.badge_craft_button').each(function(i) { ///if(i>1){return false;}//a.badge_craft_button//a.badge_row_overlay var badge_link = $J(this).attr('href'), badge_level = 0, count_min = 99999; setTimeout(function() { $J.get(badge_link, function(html) { if (i === 0) { g_sessionID = html.match(/g_sessionID = "([^"]+)"/)[1]; } var gamename = $J(html).find('.badge_title').text(); var _badge = $J(html).find('.badge_current .badge_info_description>div:eq(1)'); $J(html).find('.badge_detail_tasks>.badge_card_set_card').each(function() { var count = $J(this).find('.badge_card_set_text_qty').text(); if (count) { count = parseInt(count.replace(/[()]/g, '')); if (count < count_min) { count_min = count; } } else { count_min = 0; } }); if (_badge.length) { badge_level = parseInt(_badge.text().match(/\d+/)); } var _appid = badge_link.match(/\/([0-9]{1,7})\/$/); var _appid2 = badge_link.match(/\/([0-9]{1,7})\/\?border=1$/); if (!_appid) { _border = 1; __appID = _appid2[1]; _gappid = __appID + "b1"; if ($J.inArray(__appID, sales) >= 0) { badge_cap_level = config_cap_level == 0 ? 99999 : config_cap_level; } else { badge_cap_level = 1; } } else { _border = 0; __appID = _appid[1]; _gappid = __appID + "b0"; if ($J.inArray(__appID, sales) >= 0) { badge_cap_level = config_cap_level == 0 ? 99999 : config_cap_level; } else { badge_cap_level = config_cap_level == 0 ? 5 : config_cap_level; } } if($J.inArray(__appID*1, blacklist) >= 0){count_min=0;}//blacklist.include(__appID) var upgrade_sets = Math.min(count_min, Math.max((badge_cap_level - badge_level),0)); ///Math.min(count_min, (badge_cap_level - badge_level));//2; $J('.craft_list').append("

" + text.list1 + " APPID:" + __appID + " " + text.list2 + " " + upgrade_sets + " " + text.list3 + " " + badge_level + " " + text.list4 + " " + gamename + "

"); sum_sets += upgrade_sets; sum_badges += 1; if (i == (total_number - 1)) { ///2-1 $J('#start').html('
' + text.button2 + '
'); $J('.window_title').html(text.calculated); $J('#start').addClass('start_2').removeClass('start_1'); $J('#start>div').addClass('btn_green_white_innerfade').removeClass('btn_grey_white_innerfade'); $J('#start').before('

' + sum_sets + ' sets(' + sum_badges + ' badges) to craft!

'); $J('.craft_list').append("

=========END=========

"); } }); }, i * timer_scan); }); } function craft_do() { var queue = [], finished_count = {}, all_count = {}, sum_crafted = 0, sum_sets = 0; $J('.ready_to_craft').attr("disabled", true); $J('.craft_list').addClass("dischange"); $J('.ready_to_craft').each(function() { if ($J(this).val() > $J(this).attr("max")) { $J(this).val($J(this).attr("max")); } if ($J(this).val() > 0) { sum_sets += parseInt($J(this).val()); } }); $J('#start').html(''); $J('.window_title').html(text.crafting); $J('.before_c').html('Success: 0 '); $J('.ready_to_craft').each(function() { if ($J(this).val() > 0) { for (var i = 0; i < $J(this).val(); i++) { queue.push({ "border": $J(this).data("border"), "appid": $J(this).data("appid"), "times": $J(this).val() }); } } }); $J.each(queue, function(i) { setTimeout(function() { var border = queue[i].border, appid = queue[i].appid, para = appid + "b" + border; $J.ajax({ type: 'post', url: g_strProfileURL + '/ajaxcraftbadge', data: { sessionid: g_sessionID, series: 1, border_color: border, appid: appid }, timeout: 8000, complete: function(XMLHttpRequest, status) { if (i == (queue.length - 1)) { $J('#start').html('' + text.crafted + ''); $J('.window_title').html(text.crafted); } if (all_count[para]) { all_count[para]++; } else { all_count[para] = 1; } if (all_count[para] == 1) { $J('input.ready_to_craft[data-gappid="' + para + '"]').addClass('in_progress'); } if (all_count[para] == queue[i].times) { $J('input.ready_to_craft[data-gappid="' + para + '"]').addClass('finished'); } }, success: function(data) { if (data.success == 1) { sum_crafted += 1; $J('.sum_crafted').text(sum_crafted); $J('.sum_percent').text(GetPercent(sum_crafted, sum_sets)); $J('.sum_xp').text('+' + (sum_crafted * 100) + 'XP'); if (finished_count[para]) { finished_count[para]++; } else { finished_count[para] = 1; } $J('input.ready_to_craft[data-gappid="' + para + '"]').css('background-position-x', 46 * (finished_count[para] / queue[i].times - 1)); } } }); }, timer_craft * i); }); } function update_sum_sets() { var sum_badges = 0, sum_sets = 0; $J('input.ready_to_craft').each(function(i) { if ($J(this).val() > 0) { sum_sets += parseInt($J(this).val()); sum_badges += 1; if ($J(this).parent().hasClass('do_not_craft')) { $J(this).parent().removeClass('do_not_craft'); } } else { $J(this).parent().addClass('do_not_craft'); } $J('font.sum_badges').html(sum_badges); $J('font.sum_sets').html(sum_sets); }); if (sum_badges === 0) { $J('#start>div').addClass('cannot_craft'); $J('#start span').html(text.button2no); } if (sum_badges !== 0) { $J('#start>div').removeClass('cannot_craft'); $J('#start span').html(text.button2); } } $J('body').on('click', '.newmodal_close', function() { _close(); }); $J('body').on('click', '.craft_background', function() { _close(); }); $J('body').on('click', '.profile_xp_block_right', function() { _open(); }); $J('body').on('click', '.start_1 div.btn_large1', function() { _save(); craft(); }); $J('body').on('click', '.start_2 div.btn_large1', function() { craft_do(); }); $J('body').on('click', '._setting', function() { _setting(); }); $J('body').on('click', '._save', function() { _save(); }); $J('body').on('click', '._rapid', function() { _rapid(); }); $J('body').on('click', '._confirms', function() { _rapid_do(); }); $J('body').on('click', '._cancels', function() { $J('.craft_title').slideDown(); $J('.attention_list').slideUp(); $J('.button_s1').slideDown(); $J('.confirm').slideUp(); }); $J('body').on('change', '.ready_to_craft', function() { update_sum_sets(); }); function _rapid() { _save(); $J('.craft_title').slideUp(); $J('.attention_list').slideDown(); $J('.button_s1').slideUp(); $J('.confirm').slideDown(); $J("#appid").html(''); if(blacklist.length<1){ $J('.attention_bl').hide(); $J('.attention_nbl').show(); }else{ blacklist.each(function(e,i){ $J('.attention_bl').show(); $J('.attention_nbl').hide(); $J("#appid").append(''+e+''); }); } } function _rapid_do() { $J('.window_title').html(text.crafting); $J('.attention_list').slideUp(); $J('.confirm').html(''); var total_number = $J('a.badge_craft_button').length, each_count = 0, sum_sets = 0, sum_badges = 0; if (total_number === 0) { $J('.confirm').html(text.nosets1); $J('.window_title').html(text.nosets2); $J('.craft_list').append("

" + text.nosets3 + "

"); $J('.craft_list').append("

=========END=========

"); } window.queue_r=[]; $J('a.badge_craft_button').each(function(i) { var badge_link = $J(this).attr('href'); if (i === 0) { g_sessionID = $J(":root").html().match(/g_sessionID = "([^"]+)"/)[1]; } var _appid = badge_link.match(/\/([0-9]{1,7})\/$/); var _appid2 = badge_link.match(/\/([0-9]{1,7})\/\?border=1$/); if (!_appid) { _border = 1; __appID = _appid2[1]; _gappid = __appID + " Foil "; if ($J.inArray(__appID, sales) >= 0) { badge_cap_level = 99999; } else { badge_cap_level = 1; } } else { _border = 0; __appID = _appid[1]; _gappid = __appID + ""; if ($J.inArray(__appID, sales) >= 0) { badge_cap_level = 99999; } else { badge_cap_level = 5; } } if($J.inArray(__appID*1,blacklist)<0&&badge_cap_level>0){queue_r.push({appid:__appID,border:_border,gappid:_gappid,badge_cap_level:badge_cap_level*1});}//blacklist.include(__appID) }); $J('#start').before('

Crafted: 0 sets

'); rapid_post(); } var sum_crafted_r=0; function rapid_post(){ if(queue_r.length<1){ $J('.confirm').html('' + text.crafted + ''); $J('.window_title').html(text.crafted);return;} if(queue_r[0].badge_cap_level>0){queue_r[0].badge_cap_level--;}else{queue_r.splice(0,1);rapid_post();return;} var border = queue_r[0].border, appid = queue_r[0].appid, gappid = queue_r[0].gappid; $J.ajax({ type: 'post', url: g_strProfileURL + '/ajaxcraftbadge', data: { sessionid: g_sessionID, series: 1, border_color: border, appid: appid }, timeout: 8000, complete: function(XMLHttpRequest, status) { setTimeout(function(){rapid_post();},timer_craft*0); }, success: function(data) { if (data.success == 1) { sum_crafted_r += 1; $J('.sum_crafted').text(sum_crafted_r); $J('.sum_xp').text('+' + (sum_crafted_r * 100) + 'XP'); if($J('font[data-app='+appid+']').length>0){ $J('font[data-app='+appid+']').html($J('font[data-app='+appid+']').html()*1+1) }else{ $J('.craft_list').append("Crafted 1 sets for "+gappid+" badge
"); } }else{queue_r.splice(0,1);} } }); } function _close() { $J('.craft_background').fadeOut(); $J('.craft_window').slideUp(); } function _open() { $J('.craft_background').fadeIn(); $J('.craft_window').slideDown(); } function _setting() { $J('.setting').removeClass("_setting").addClass('_save'); $J('.craft_title').slideUp(); $J('.setting_list').slideDown(); } function _save() { $J('.setting').removeClass("_save").addClass('_setting'); $J('.setting_list').slideUp(); $J('.craft_title').slideDown(); GM_setValue("config_cap_level", $J("input.config_cap_level").val()); GM_setValue("timer_scan", $J("input.timer_scan").val()); GM_setValue("timer_craft", $J("input.timer_craft").val()); GM_setValue("config_blacklist", $J("textarea.config_blacklist").val()); config_cap_level=$J("input.config_cap_level").val(); timer_scan=$J("input.timer_scan").val(); timer_craft=$J("input.timer_craft").val(); config_blacklist=$J("textarea.config_blacklist").val(); blacklist = eval('['+config_blacklist.replace(/[^0-9,]/,'')+']'); } function GetPercent(num, total) { num = parseFloat(num); total = parseFloat(total); if (isNaN(num) || isNaN(total)) { return "-"; } return total <= 0 ? "0%": (Math.round(num / total * 10000) / 100.00 + "%"); } })();