// ==UserScript== // @name Steam Auto Mass Craft Cards Badges in Bulk // @name:zh-CN Steam一键批量合卡合徽章 // @name:zh-TW Steam一鍵批量合卡合徽章 // @version 1.5 // @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 sales=["245070","762800"],//Appid for sales cards timer_craft = GM_getValue("timer_craft", 500), timer_scan = GM_getValue("timer_scan", 1000), config_cap_level = GM_getValue("config_cap_level", 0), config_blacklist = GM_getValue("config_blacklist", ''); (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; } .level_up > span { position: relative; top: -2px; } .target_level_icon > span { position: relative; top: -2px; } font.level_up { font-size: 24px; color: #ffc902; } .target_level_icon { display: inline-block; } .target_level{ color:#fff; } .animated { -webkit-animation-duration: .7s; animation-duration: .7s; -webkit-animation-fill-mode: both; animation-fill-mode: both } .animated.infinite { -webkit-animation-iteration-count: infinite; animation-iteration-count: infinite } .animated.hinge { -webkit-animation-duration: 2s; animation-duration: 2s } @-webkit-keyframes bounce { 0%,100%,20%,50%,80% { -webkit-transform: translateY(0); transform: translateY(0) } 40% { -webkit-transform: translateY(-10px); transform: translateY(-10px) } 60% { -webkit-transform: translateY(-5px); transform: translateY(-5px) } } @keyframes bounce { 0%,100%,20%,50%,80% { -webkit-transform: translateY(0); -ms-transform: translateY(0); transform: translateY(0) } 40% { -webkit-transform: translateY(-10px); -ms-transform: translateY(-10px); transform: translateY(-10px) } 60% { -webkit-transform: translateY(-5px); -ms-transform: translateY(-5px); transform: translateY(-5px) } } .bounce { -webkit-animation-name: bounce; animation-name: bounce } @-webkit-keyframes rubberBand { 0% { -webkit-transform: scale(1); transform: scale(1) } 30% { -webkit-transform: scaleX(1.25) scaleY(0.75); transform: scaleX(1.25) scaleY(0.75) } 40% { -webkit-transform: scaleX(0.75) scaleY(1.25); transform: scaleX(0.75) scaleY(1.25) } 60% { -webkit-transform: scaleX(1.15) scaleY(0.85); transform: scaleX(1.15) scaleY(0.85) } 100% { -webkit-transform: scale(1); transform: scale(1) } } @keyframes rubberBand { 0% { -webkit-transform: scale(1); -ms-transform: scale(1); transform: scale(1) } 30% { -webkit-transform: scaleX(1.25) scaleY(0.75); -ms-transform: scaleX(1.25) scaleY(0.75); transform: scaleX(1.25) scaleY(0.75) } 40% { -webkit-transform: scaleX(0.75) scaleY(1.25); -ms-transform: scaleX(0.75) scaleY(1.25); transform: scaleX(0.75) scaleY(1.25) } 60% { -webkit-transform: scaleX(1.15) scaleY(0.85); -ms-transform: scaleX(1.15) scaleY(0.85); transform: scaleX(1.15) scaleY(0.85) } 100% { -webkit-transform: scale(1); -ms-transform: scale(1); transform: scale(1) } } .rubberBand { -webkit-animation-name: rubberBand; animation-name: rubberBand } input.friendPlayerLevelNum { width: 28px; background: transparent; border: 0; font-size: inherit; text-align: center; overflow: visible; } span[class$='00'] input { width: 31px!important; } .calculator { border: 1px #fff solid; padding: 20px; margin-top: 30px; } .calculator legend { margin-left: 20px; } span#lvdiff input { width: 72px; background: transparent; color: #000; border: 0; text-align: center; margin: 0 10px; font-size: 16px; } span#lvdiff { background: rgba(255, 255, 255, 0.95); border-radius: 3px; padding: 10px; color: #aaa; } .calculator div { float: left; } span.levelnumber input { width: 66px; background: transparent; border: 0; text-align: center; overflow: visible; color: #ddd; font-size: 13px; } span.levelnumber { display: block; font-size: 13px; position: relative; left: 23px; margin-top: 5px; } .calculator span.friendPlayerLevel { margin: 0px 40px; } .calculator ._m { position: relative; top: 10px; } ._m span { float: left; display: block; } .friendPlayerLevel.lvl_1000 input, .friendPlayerLevel.lvl_1100 input, .friendPlayerLevel.lvl_1200 input, .friendPlayerLevel.lvl_1300 input, .friendPlayerLevel.lvl_1400 input, .friendPlayerLevel.lvl_1500 input, .friendPlayerLevel.lvl_1600 input, .friendPlayerLevel.lvl_1700 input, .friendPlayerLevel.lvl_1800 input, .friendPlayerLevel.lvl_1900 input, .friendPlayerLevel.lvl_2000 input, .friendPlayerLevel.lvl_2100 input, .friendPlayerLevel.lvl_2200 input, .friendPlayerLevel.lvl_2300 input, .friendPlayerLevel.lvl_2400 input, .friendPlayerLevel.lvl_2500 input, .friendPlayerLevel.lvl_2600 input, .friendPlayerLevel.lvl_2700 input, .friendPlayerLevel.lvl_2800 input, .friendPlayerLevel.lvl_2900 input, .friendPlayerLevel.lvl_3000 input { width: 32px; } `); var _border, g_sessionID, badge_cap_level, __appID, _gappid,blacklist=[],tar_lv,o_tar_lv; var cur_xp = $J('.profile_xp_block_xp').html().replace(/[^0-9]/g,'')*1; var lv01 = ivscc(cur_xp)*1; var lv02 = lv01+1; var text = {}, lan = $J('head').html().match(/l=([^"&*]+)"/)[1]; if (lan == "schinese" || lan == "tchinese") { text.start = "批量合成徽章"; text.title = "批量合卡"; text.notice = '
这是一个自动合卡插件,可以指定徽章合成的数量和种类。
可在SteamCN、Github、Greasy Fork反馈问题,也可以在我的资料页下方留言(ASF经常吞消息可能看不到私聊0.0)
可以随时关闭本标签页来停止插件的自动操作计算模式: 先扫描出每个徽章的可合成次数,扫描完毕后可以手动调整合成数量,再通过确认进行批量合卡。
极速模式: 先显示您的黑名单设置,确认后将直接合成所有可以用来合成的徽章,不会使用黑名单中的游戏卡牌。
'; 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 = '虽Github、Greasy 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.
'; 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 number of cards sets 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" + 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('' + sum_sets + ' sets ( ' + sum_badges + ' badges ) to craft! [ '+icon_raw(cur_lv)+' > '+icon_raw(tar_lv)+' ]
'); $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('" + 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) }); var cur_lv = o_tar_lv = tar_lv = ivscc(cur_xp); $J('#start').before('Crafted: 0 sets [ '+icon_raw(cur_lv)+' > '+icon_raw(tar_lv)+' ]
'); 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'); var tar_lv = ivscc(cur_xp+sum_crafted_r*100); if(tar_lv!=o_tar_lv){ icon($J('font.target_level_icon .friendPlayerLevel'),tar_lv); $J('font.target_level_icon').addClass('rubberBand animated infinite'); setTimeout(function(){$J('font.target_level_icon').removeClass('rubberBand animated infinite');},700); } o_tar_lv = tar_lv; if($J('font[data-app='+appid+'_'+border+']').length>0){ $J('font[data-app='+appid+'_'+border+']').html($J('font[data-app='+appid+'_'+border+']').html()*1+1) }else{ $J('.craft_list').append("Crafted 1 sets for "+gappid+" badge