CS:GO Lounge Destroyer

Spam the fuck out of the CS:GL queue system, because it's absolute crap

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==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&currency=" + 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 &#x2764;</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">&#x2715;</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);
    });
}