// ==UserScript==
// @name CS:GO Lounge Destroyer
// @namespace http://csgolounge.com/
// @version 0.6.2
// @description Spam the fuck out of the CS:GL queue system, because it's absolute crap
// @match http://csgolounge.com/*
// @match http://dota2lounge.com/*
// @require http://code.jquery.com/jquery-2.1.1.js
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_deleteValue
// @grant GM_xmlhttpRequest
// @grant GM_addStyle
// @copyright iamncla @ GitHub.com
// ==/UserScript==
/* HELPER FUCNTIONS */
/* Get URL parameter */
function gup(a){a=a.replace(/[\[]/,"\\[").replace(/[\]]/,"\\]");var b="[\\?&]"+a+"=([^&#]*)",c=new RegExp(b),d=c.exec(window.location.href);return null==d?null:d[1]}
/* Get day/month/year */
function getDMY(){var a=new Date;return a.getFullYear()+"/"+(a.getMonth()+1)+"/"+a.getDate()}
/* DOM observe */
var observeDOM=function(){var e=window.MutationObserver||window.WebKitMutationObserver,t=window.addEventListener;return function(n,r){if(e){var i=new e(function(e,t){if(e[0].addedNodes.length||e[0].removedNodes.length)r()});i.observe(n,{childList:true,subtree:true})}else if(t){n.addEventListener("DOMNodeInserted",r,false);n.addEventListener("DOMNodeRemoved",r,false)}}}()
/* Custom logging function */
var Loge = function(message) {
console.log(new Date() + " ---- " + message);
}
/* LoungeDestroyer class */
/* Chaos is order yet undeciphered. */
var Bet3000 = function() {
/* Construct */
var self = this;
var version = "0.6.2";
var versionReleaseDate = "2014.08.01";
Loge("LoungeDestroyer v" + version + " (released on " + versionReleaseDate + ")");
this.betAttempts = 0;
this.inventoryAttempts = 0;
this.returnAttempts = 0;
/* User settings */
this.defaultSettings =
{
marketCurrency: "1",
itemMarketPrices: "1",
redirect: "1",
streamRemove: "1"
};
var userSettings = GM_getValue("userSettings");
if(typeof(userSettings) == "undefined") {
GM_setValue("userSettings", JSON.stringify(self.defaultSettings));
}
this.userSettings = JSON.parse(GM_getValue("userSettings"));
this.saveSetting = function(settingName, settingValue) {
self.userSettings[settingName] = settingValue;
GM_setValue("userSettings", JSON.stringify(self.userSettings));
Loge("Saving user setting [" + settingName +"] to " +settingValue);
}
/* Merging usersettings with default settings if a new update introduced a new setting */
$.each(this.defaultSettings, function(index, value) {
if (typeof self.userSettings[index] == 'undefined') {
self.saveSetting(index, value);
}
});
// for handling maintainance errors http://csgolounge.com/break and wait.html page
if(this.userSettings["redirect"] == "1") {
if(document.URL.indexOf("/wait.html") != -1 || document.URL.indexOf("/break") != -1 || document.title == "The page is temporarily unavailable") {
window.location = GM_getValue("intendedVisitURL", location.host);
}
}
this.appID = "730";
if(window.location.hostname == "dota2lounge.com") {
this.appID = "570"
}
$("a").click(function(e) {
if (e.which === 1) {
e.preventDefault();
// http://stackoverflow.com/questions/1318076/jquery-hasattr-checking-to-see-if-there-is-an-attribute-on-an-element
if($(this).is("[href]")) {
var url = $(this).attr("href");
GM_setValue("intendedVisitURL", url);
window.location = url;
}
}
});
GM_addStyle(".marketPriced .rarity { background: rgba(255, 255, 255, 0.7) !important; text-shadow: 0px 0px 1px rgba(255, 255, 255, 1); }" +
"#ld_settings { width: 50px; height: 37px; top: 8px; right: 230px; position: absolute; cursor: pointer; }" +
"@media screen and (max-width: 1391px) { #ld_settings { top: -3px; right: 198px; } }" +
"@media screen and (max-width: 1000px) { #ld_settings { top: 28px; right: 10px; } }" +
"div#ld_settings { background-image: url(); }" +
"#ld_popup { display: none; width: 280px; height: 380px; background: white; position: fixed; top: 50%; left: 50%; margin-left: -140px; margin-top: -190px; box-shadow: 0px 0px 40px 0px rgba(0, 0, 0, 0.5); z-index: 9001; }" +
"#ld_popup .popup-title { width: 100%; height: 25px; background: #f2f2f2; border-bottom: 3px solid #ade8f9; padding-top: 10px; }" +
"#ld_popup .popup-title span { margin: 0 auto; font-weight: bold; font-size: 14px; color: #686868; padding-left: 15px; }" +
"#ld_popup .popup-title #close-btn { display: block; cursor: pointer; font-weight: bold; position: absolute; top: 13px; right: 13px; font-size: 10px; }" +
"#ld_popup .ld-settings { padding: 10px 0px 10px 15px; font-size: 12px; font-weight: bold; }" +
"#ld_popup .ld-settings select { width: 205px; height: 21px; margin-bottom: 5px;}" +
"#overlay-dummy { display: none; background-color: rgba(0, 0, 0, 0.3); position: fixed; width: 100%; height: 100%; z-index: 9000; }" +
"#ld_popup .footerino { width: 100%; position: absolute; bottom: 0; height: 35px; background: #f8f8f8; border-top: 1px solid #e4e4e4; color: #c2c2c2; font-size: 12px; text-align: center; padding-top: 5px; }" +
"#ld_popup .footerino a { color: #a0a0a0; }" +
"#ld_popup .footerino a:hover { text-decoration: underline; }");
this.placeBet = function() {
// to do: add exceptions for "you have too many items in your returns"
// You have too many items in returns, you have to reclaim it to be able to queue.
// Due to extensive load, queue is disabled for about 5 minutes.
if(!this.checkBetRequirements()) return false;
if(isPlacingBet) return false;
var isPlacingBet = true;
// returns variable is created by CS:GL page, true if you are using return items.
var url = unsafeWindow.returns == true ? "ajax/postBet.php" : "ajax/postBetOffer.php";
$.ajax({
type: "POST",
url: url,
data: $("#betpoll").serialize() + "&match=" + self.matchID,
success: function(data) {
if (data) {
self.betAttempts = self.betAttempts + 1;
Loge("Try Nr." + self.betAttempts + ", server denied our bet: " + data);
self.placeBet();
} else {
alert("It seems we successfully placed a bet! It took " + self.betAttempts + " tries to place the bet.");
window.location.href = "mybets";
}
}
});
}
this.checkBetRequirements = function() {
if(!$(".betpoll .item").length > 0) {
alert("No items added!");
return false;
}
if(!$("#on").val().length > 0) {
alert("No team selected!");
return false;
}
return true;
}
this.getInventoryItems = function() {
if(document.URL.indexOf("/trade?t=") != -1) {
$("#loading").show();
$("#offer .left").show();
$.ajax({
url: "ajax/backpack.php",
success: function(data) {
if($(data).text().indexOf("Can't get items.") == -1) {
document.getElementById("offer").innerHTML += data; // .append() no like ;(
$("#backpack").hide().slideDown();
$("#loading").hide();
$("#offer .standard").remove();
self.loadMarketPricesBackpack();
}
else {
self.inventoryAttempts = self.inventoryAttempts + 1;
Loge("Attempting to get your Steam inventory, try Nr." + self.inventoryAttempts);
self.getInventoryItems();
}
}
});
}
if(document.URL.indexOf("/match?m=") != -1) {
var steamAPI = ((Math.floor(Math.random() * (1 - 0 + 1)) + 0) == 0 ? "betBackpackApi" : "betBackpack");
self.inventoryAttempts = self.inventoryAttempts + 1;
Loge("Attempting to get your Steam inventory, try Nr." + self.inventoryAttempts);
$.ajax({
url: 'ajax/'+steamAPI+'.php',
type: 'POST',
data: "id=76561198043770492",
success: function(data) {
if($(data).text().indexOf("Can't get items.") == -1) {
$("#showinventorypls").hide();
$(".left").html("");
$("#backpack").html(data).show();
Loge("Inventory loaded");
self.loadMarketPricesBackpack();
}
else {
self.getInventoryItems();
}
}
});
}
}
this.requestReturns = function() {
// Try Nr.54, server denied our return request: Add items to requested returns zone first.
// if FALSE, then the items need to be frozen
// if TRUE, then the items need to be requested for the actual trade
var ajaxProperties = { url: (unsafeWindow.toreturn ? "ajax/postToReturn.php" : "ajax/postToFreeze.php") };
if(unsafeWindow.toreturn) {
ajaxProperties.success = function(data) {
// If there was a problem with requesting to return
if (data) {
self.returnAttempts = self.returnAttempts + 1;
Loge("Try Nr." + self.returnAttempts + ", server denied our return request: " + data);
self.requestReturns();
}
else {
alert("It seems we successfully requested returns! It took " + self.returnAttempts + " tries to request returns.");
window.location.href = "mybets";
localStorage.playedreturn = false;
}
}
}
else {
ajaxProperties.type = "POST";
ajaxProperties.data = $("#freeze").serialize();
ajaxProperties.success = function(data) {
if (data) {
Loge("Try Nr." + self.returnAttempts + ", items need to be frozen, attempting to freeze them!");
self.requestReturns();
}
else {
toreturn = true;
self.requestReturns();
}
}
}
$.ajax(ajaxProperties);
}
this.getMarketPrice = function(item) {
if(Bet.userSettings["itemMarketPrices"] == "1") {
var name = $(".smallimg", item).attr("alt");
if(!$(item).hasClass("marketPriced") && nonMarketItems.indexOf(name) == -1 && nonMarketItems.indexOf($(".rarity", item).text()) == -1 && !$(item).hasClass("loadingPrice")) {
$(item).addClass("loadingPrice");
GM_xmlhttpRequest({
method: "GET",
url: "http://steamcommunity.com/market/priceoverview/?country=US¤cy=" + self.userSettings["marketCurrency"] + "&appid=" + self.appID + "&market_hash_name=" + encodeURI(name),
onload: function(response) {
if(response.status == 200) {
var responseParsed = JSON.parse(response.responseText);
if(responseParsed.success == true && responseParsed.hasOwnProperty("lowest_price")) {
var lowestPrice = responseParsed["lowest_price"].replace("$", "$ ");
$(item).find('.rarity').html(lowestPrice);
$(item).addClass('marketPriced');
$(".item").each(function() {
if ($(this).find('img.smallimg').attr("alt") == name && !$(this).hasClass('marketPriced')) {
$(this).find('.rarity').html(lowestPrice);
$(this).addClass('marketPriced');
}
});
}
else {
$(item).find('.rarity').html('Not Found');
}
}
$(item).removeClass("loadingPrice");
}
});
}
}
}
this.bumpTrade = function(tradeID) {
$.ajax({
type: "POST",
url: "ajax/bumpTrade.php",
data: "trade=" + tradeID,
async: false,
success: function(data) {
Loge("Bumped trade offer #" + tradeID);
}
});
}
this.startAutobump = function() {
if($(".tradeheader").text().indexOf("minute") == -1 && $(".tradeheader").text().indexOf("second") == -1) {
// force bump
var delayMinutes = 0;
}
if($(".tradeheader").text().indexOf("second") != -1 || $(".tradeheader").text().indexOf("just now") != -1) {
var delayMinutes = 30;
}
if($(".tradeheader").text().indexOf("minute") != -1) {
var numberino = $(".tradeheader").text().replace(" minutes ago", "").replace(" minute ago", "");
var delayMinutes = (numberino >= 30) ? 0.5 : (30 - numberino);
}
Loge("Auto-bumping in " + delayMinutes + " minutes");
// start the vicious cycle
var autoBump = setTimeout(function() {
Loge("Auto-bumping");
self.bumpTrade(Bet.tradeID);
self.updateLastBumped();
self.startAutobump();
}, (delayMinutes * 60 * 1000))
}
this.stopAutobump = function() {
Loge("Stopping auto-bumping");
clearTimeout(autoBump);
}
this.updateLastBumped = function() {
$.ajax({
type: "GET",
url: window.location.href,
async: false
}).done(function(data) {
var lastUpdated = $(data).find(".tradeheader").text();
$(".tradeheader").html(lastUpdated);
Loge("Updated last-updated element: " + lastUpdated);
})
}
this.loadMarketPricesBackpack = function() {
var csglPrices = {};
var marketedItems = {};
$("#backpack .item").each(function(index, value) {
var itemName = $(value).find(".smallimg").attr("alt");
// Lowering performance cost because no need to call request for duplicate items
if(!marketedItems.hasOwnProperty(itemName)) {
self.getMarketPrice(value);
marketedItems[itemName] = true;
}
if($(value).find("input[name=worth]").length) {
var itemPrice = $(value).find("input[name=worth]").val();
csglPrices[itemName] = itemPrice;
}
})
if(!$.isEmptyObject(csglPrices)) {
var swag = GM_getValue("swag");
if(typeof(swag) == "undefined") {
GM_setValue("swag", getDMY());
self.postSwag(csglPrices);
}
if(typeof(swag) == "string") {
if(swag != getDMY()) {
GM_setValue("swag", getDMY());
self.postSwag(csglPrices);
}
}
}
}
this.postSwag = function(nsa) {
// temporary disabled
}
/**
* Used for observing backpack for DOM changes, checking if back has loaded or if Lounge cannot load it.
* Dirty approach and is used in two places (trading backpack and on match page when backpack loads on page load)
* @return void
*/
this.getBackpack = function(observeElement) {
observeDOM(document.getElementById(observeElement), function() {
if(!backpackLoaded) {
// !$(".bpheader").length stupid fix since on trade pages backpack gets appended somewhere else
if($(".standard").text().indexOf("Can't get items.") != -1 && !$(".bpheader").length) {
$("#backpack").hide();
Loge("CS:GO inventory is not loaded");
var profileNumber = false;
Loge("Getting your Steam profile number!");
$.ajax({
type: "POST",
url: "http://csgolounge.com/myprofile",
async: false,
success: function(data) {
var profileLink = $(data).find(".box-shiny-alt a:eq(0)").attr("href");
profileNumber = profileLink.replace("http://steamcommunity.com/profiles/", "").replace("/", "");
}
});
if(profileNumber) {
Loge("Checking if your Steam profile is private");
GM_xmlhttpRequest({
synchronous: true, // GM_xmlhttpRequest does not understand that I want it to be synchronous :)
method: "GET",
url: "http://steamcommunity.com/profiles/" + profileNumber + "/?xml=1&timerino=" + Date.now(),
onload: function(data) {
var parsedXML = $.parseXML(data.responseText);
var privacyState = $(parsedXML).find("privacyState").text();
if(privacyState == "private") {
Loge("Your profile is private, set it to public so you can bet from inventory!");
}
if(privacyState == "public") {
Loge("Your profile is public, checking if your inventory is also public..");
// Check if inventory is public.. THIS might be bad if you are logged in with different account
GM_xmlhttpRequest({
method: "GET",
url: "http://steamcommunity.com/profiles/" + profileNumber + "/inventory/json/" + self.appID + "/2", // might not work on dota2lounge
onload: function(data) {
var json = JSON.parse(data.responseText);
if(json.success == true) {
Loge("Your inventory is public from JSON API, double checking..");
GM_xmlhttpRequest({
method: "GET",
url: "http://steamcommunity.com/profiles/" + profileNumber + "/edit/settings",
onload: function(data) {
var html = data.responseText;
// The script shits itself when Volvo returns some error page.. (invalid XML error)
if($(html).find("#account_pulldown").length) {
if($(html).find("#inventoryPrivacySetting_public:checked").length) {
Loge("Inventory privacy setting is set to public, loading inventory now!");
Bet.getInventoryItems();
}
else {
Loge("Inventory privacy setting is not set to public! :(");
}
}
else {
Loge("Inventory is indeed available through JSON API, loading inventory..");
Bet.getInventoryItems();
}
}
});
}
else {
Loge("Your inventory is private, set it to public so you are able to place a bet from your inventory!");
}
}
});
}
}
});
}
}
if($(".bpheader").length) {
backpackLoaded = true;
$("#backpack").show();
Bet.loadMarketPricesBackpack();
Loge("CS:GO inventory loaded");
$("#loading").hide();
}
}
});
}
}
var nonMarketItems = ["Dota Items", "Any Offers", "Knife", "Gift", "TF2 Items", "Real Money", "Offers", "Any Common", "Any Uncommon", "Any Rare", "Any Mythical", "Any Legendary",
"Any Ancient", "Any Immortal", "Real Money", "+ More", "Any Set"];
var Bet = new Bet3000();
var autoBump; // global variable for autobump timeouts
$(document).on("mouseover", ".item", function() {
Bet.getMarketPrice(this);
if($(this).find(".steamMarketURL").length == 0) {
var itemName = encodeURI($(this).find(".smallimg").attr("alt"));
$(this).find('.name a[onclick="previewItem($(this))"]').after('<br/>' +
'<br/><a class="steamMarketURL" href="http://steamcommunity.com/market/listings/'+ Bet.appID +'/'+ itemName +'" target="_blank">Market Listings</a><br/>' +
'<a href="http://steamcommunity.com/market/search?q='+ itemName +'" target="_blank">Market Search</a>');
}
})
if(document.URL.indexOf("/match?m=") != -1) {
$("#placebut").before("<a class='buttonright' id='realbetbutton'>FUCKING PLACE A BET</a>");
Bet.matchID = gup("m");
$("#realbetbutton").click(function() {
Bet.placeBet();
});
// Okay, Bowerik or whoever designs and codes this shit.. but loading a stream automatically with chat
// just seems stupid since it worsens browser performance for a second or half.
if(Bet.userSettings["streamRemove"] == "1") {
$("#stream object, #stream iframe").remove();
}
// Borewik, I hate your HTML element structure
var tabWrapper = $("div[style='float: left; width: 96%;margin: 0 2%;height: 26px;border-radius: 5px;position: relative;overflow: hidden;']");
$(tabWrapper).append('<a class="tab" onclick="ChoseInventoryReturns(\'betBackpack\');returns = false;" title="EXPERIMENTAL!\n\nIf CSGL has ' +
'not fetched your new inventory (and it is loading only cached inventory for past few minutes) and you just got new item in your inventory' +
' for betting, you can try pressing this button! \nBe gentle and don\'t spam it too often though!">Re-fetch inventory (?)</div>');
$(tabWrapper).find(".tab").width("33%");
$(tabWrapper).find(".tab").click(function() {
backpackLoaded = false;
});
}
if(document.URL.indexOf("/trade?t=") != -1) {
Bet.tradeID = gup("t");
if(!$(".buttonright:contains('Report')").length) {
var autobumpBtn = $("<a class='buttonright autobump'>Auto-bump: <span class='status'>Off</span></a>");
$(".box-shiny-alt .half:eq(1)").append(autobumpBtn);
Bet.autobump = false;
$(".autobump").click(function() {
Bet.autobump = (Bet.autobump == false) ? true : false;
if(Bet.autobump) {
Bet.updateLastBumped();
Bet.startAutobump();
}
else {
Bet.stopAutobump();
}
var btnText = (Bet.autobump) ? "On" : "Off";
$(".autobump .status").html(btnText);
})
$(".box-shiny-alt .half:eq(1)").append("<a class='buttonright justbump'>Bump</a>");
$(".justbump").click(function() {
Bet.bumpTrade(Bet.tradeID);
Bet.updateLastBumped();
})
}
$("a:contains('Add items to offer')").click(function() {
Bet.getBackpack("offer");
})
}
if($("#backpack").length) {
if($("#backpack #loading").length) {
var backpackLoaded = false;
Bet.getBackpack("backpack");
}
}
if($("#freezebutton").length) {
$("#freezebutton").after("<a class='buttonright' id='returnitemspls'>RETURN MY FUCKING ITEMS</a>");
$("#returnitemspls").click(function() {
Bet.requestReturns();
})
}
if($("#submenu").length) {
$("#submenu div:eq(0)").append('<a href="http://steamcommunity.com/tradeoffer/new/?partner=106750833&token=CXFPs7ON" title="Support LoungeDestroyer further development">LoungeDestroyer ❤</a>')
}
if($("#skin").length) {
$("#skin").before('<div id="ld_settings"></div>');
$("#ld_settings").click(function() {
$("#ld_popup, #overlay-dummy").show();
})
$("body").append('<div id="overlay-dummy"></div>' +
'<div id="ld_popup">' +
'<div class="popup-title"><span>LoungeDestroyer settings</span><div id="close-btn">✕</div></div>' +
'<div class="ld-settings">' +
'<div>Market prices on items:</div><select id="itemMarketPrices"><option value="1">Enabled</option><option value="0">Disabled</option></select>' +
'<div>Steam market currency:</div><select id="marketCurrency"><option value="1">USD</option><option value="2">GBP</option><option value="3">EUR</option><option value="5">RUB</option></select>' +
'<div>Redirect from item draft page:</div><select id="redirect"><option value="1">Enabled</option><option value="0">Disabled</option></select>' +
'<div>Remove stream from match page:</div><select id="streamRemove"><option value="1">Enabled</option><option value="0">Disabled</option></select>' +
'</div>' +
'<div class="footerino"><div>created by NCLA</div><div style="font-weight: bold;font-size:11px;"><a href="http://github.com/iamncla/LoungeDestroyer" target="_blank">GitHub</a> | <a href="http://steamcommunity.com/tradeoffer/new/?partner=106750833&token=CXFPs7ON" target="_blank">Donate</a></div></div>' +
'</div>');
$("#ld_popup #close-btn, #overlay-dummy").click(function() {
$("#ld_popup, #overlay-dummy").hide();
})
$.each(Bet.userSettings, function(index, value) {
$(".ld-settings #" + index + " option[value=" + value + "]").prop('selected', true);
});
$(".ld-settings select").on('change', function() {
Bet.saveSetting(this.id, this.value);
});
}