Torn Helper

Adds extra information to different pages all around Torn.

当前为 2017-01-15 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name        Torn Helper
// @namespace   Jebster.Torn
// @author      Jeggy
// @description Adds extra information to different pages all around Torn.
// @include     *.torn.com/profiles.php?XID=*
// @version     0.1.6
// @require     http://code.jquery.com/jquery-2.2.4.min.js
// @require     http://code.jquery.com/ui/1.12.1/jquery-ui.min.js
// @resource    jquery-ui http://ajax.googleapis.com/ajax/libs/jqueryui/1.10.3/themes/black-tie/jquery-ui.min.css
// @resource    jquery-base http://code.jquery.com/ui/1.12.1/themes/base/jquery-ui.css
// @grant       GM_addStyle
// @grant       GM_getResourceText
// ==/UserScript==
// debugger;
GM_addStyle(GM_getResourceText('jquery-base'));
GM_addStyle(GM_getResourceText('jquery-ui'));

String.prototype.format = function() {
    var formatted = this;
    for (var i = 0; i < arguments.length; i++) {
        var regexp = new RegExp('\\{'+i+'\\}', 'gi');
        formatted = formatted.replace(regexp, arguments[i]);
    }
    return formatted;
};

var icons = {
    // https://www.iconfinder.com/icons/1303892/add_circle_create_new_plus_sign_icon
    plus: '',
    // https://www.iconfinder.com/icons/1303875/circle_close_minimize_minus_remove_icon
    minus: '',
};

var data = {};

(function() {
    'use strict';

    var site = window.location.pathname;

    loadData();
    saveOwnData();

    loadAttackLog();


    if(site.indexOf('profiles.php') > 0) profileView();

})();

function loadAttackLog(){
    var selections = 'attacks';
    if('attackLogLoad' in data)
        //if(data.attackLogLoad > new Date().getTime() + (60*60*6))
        logAttacks(false);
    else
        logAttacks(true);
}

function logAttacks(full){
    var selections = 'attacks' + (full ? 'full' : '');
    var url = 'https://api.torn.com/user/'+data.me.id+'?selections='+selections+'&key='+data.apikey;
    var now = new Date().getTime();
    if(!('attackLogLoad' in data) || data.attackLogLoad < now - (60*60*1000)){ // every one hour
        data.attackLogLoad = now;

        apiCall(url, function(d) {
            if(d.error) getApiKey();
            else{
                for(var p in d.attacks){
                    if (d.attacks.hasOwnProperty(p)) {
                        var attack = d.attacks[p];
                        var defender_id = attack.defender_id;
                        if(!(defender_id in data)) data[defender_id] = {};

                        if(attack.attacker_id == data.me.id){
                            // My attack
                            if(!('attacks' in data[defender_id])) {console.log('Test'); data[defender_id].attacks = {};}
                            data[defender_id].attacks[p] = attack;
                        }else{
                            // I'm being attacked
                        }
                    }

                }

                save();
            }
        });
    }
}

function profileView(){
    var userid = getParameterByName('XID');
    var userData = data[userid];
    var content = '';
    content += profileViewSelectionPopUp(); // TODO: Figoure out how to call tampermonkey function from the injected code.
    content += '<div id="compareStats">';
    content += 'Loading...';
    content += '<br />';
    content += '</div>';

    var a = accordion('Torn Helper', content);

    $(a).insertAfter('.tutorial-cont + .profile-wrapper');

    var compareFunc = function(){
        $('#compareStats').replaceWith(compareTemplate(userid, data.me.userid));
    };
    apiUserStats(userid, compareFunc);
    apiUserStats(data.me.userid, compareFunc);
}

// Only supports one accordion on the page at the moment.
function accordion(title, block){
    var css = '<style>'+
        'button.Jaccordion {'+
        'cursor: pointer;'+
        'padding: 18px;'+
        'width: 100%;'+
        'text-align: left;'+
        'transition: 0.4s;'+
        '}'+
        'div.Jpanel {'+
        'max-height: 0;'+
        'overflow: hidden;'+
        'transition: 0.8s ease-in-out;'+
        'opacity: 0;'+
        '}'+
        'div.Jshow {'+
        'opacity: 1;'+
        'max-height: 1000px;'+
        'width: auto;'+
        '}'+
        'div.JAccordionIconShow {'+
        'opacity: 1;'+
        'max-height: 30px;'+
        'width: auto;'+
        '}'+
        '.JProfileViewAccordion{'+
        'width: 0;'+
        'float: right;'+
        'max-height: 0;'+
        'overflow: hidden;'+
        'opacity: 0;'+
        'padding-right: 7px;'+
        '}'+
        '.JAccordionIcon{'+
        'height: 18px;'+
        'padding: 6px 5px 6px 25px;'+
        'filter: brightness(0) invert(1);'+
        'cursor: pointer;'+
        '}'+
        '</style>';

    var script = '<script>$(".JAccordionIcon").click(function() {'+
        '$(".Jpanel").toggleClass("Jshow");'+
        '$(".JProfileViewAccordion").toggleClass("JshJAccordionIconShowow");'+
        '$(".JProfileViewAccordion").toggleClass("JAccordionIconShow");'+
        '});</script>';

    var show = data.profileview.show;

    var html = '<div class="Jaccordion profile-wrapper medals-wrapper m-top10">'+
        '<div class="menu-header">'+title+
        '<div class="JProfileViewAccordion '+(show ? '' : 'JAccordionIconShow')+'"><img src="'+icons.plus+'" class="JAccordionIcon" /></div>'+
        '<div class="JProfileViewAccordion '+(show ? 'JAccordionIconShow' : '')+'"><img src="'+icons.minus+'" class="JAccordionIcon" /></div>'+
        '</div>'+
        '<div class="Jpanel '+(show ? 'Jshow' : '')+'">'+
        block+
        '</div>'+
        '</div>';

    $(document).on('click','.Jaccordion', function(){
        console.log('Clicked: ');
        data.profileview.show = !data.profileview.show;
        save();
    });

    return css+script+html;
}

function profileViewSelectionPopUp(){
    var possibilities = possibleStats();

    var popupHtml = '<div>';
    var categories = {};
    for(var p in possibilities){
        var o = possibilities[p];
        var checked = inArray(p, data.profileview.display) ? 'checked' : '';

        if(o.category){
            if(!categories[o.category]) categories[o.category] = {display: o.category, html: ''};
        }else{
            if(!categories.others) categories.others = {display: 'Others', html: ''};
        }

        var cat = o.category ? o.category : 'others';

        var html = '<label id="JC'+cat+'" for="view'+p+'" style="';
        html += 'border-radius: 2px;border: 1px solid gray;background-color: lightgray;margin: 3px;';
        html += 'padding: 3px 10px 3px 5px;display: inline-block;cursor: pointer;';
        html += '"><input type="checkbox" name="view'+p+'" id="view'+p+'" '+checked+'>';
        html += o.display+'</label>';
        categories[cat].html += html;
    }

    for(var category in categories){
        if (categories.hasOwnProperty(category)) {
            popupHtml += '<fieldset style="border:1px solid black; padding: 10px; margin: 10px 0;">';
            popupHtml += '<legend onclick=\'document.getElementById("JC'+category+'").style.display = document.getElementById("JC'+category+'").style.display == "none" ? "block" : "none";\'';
            popupHtml += '><b>'+categories[category].display+'</b></legend>';
            popupHtml += categories[category].html;
            popupHtml += '</fieldset>';
        }
    }

    popupHtml += '</div>';
    var popup = popupWindow(
        {
            element: '#editProfileView',
            title: 'Edit Profile view',
            width: 640
        },
        popupHtml,
        [
            {
                'display': 'Save',
                'callback': function(){
                    var selected = [];
                    for(var p in possibleStats()){
                        var checked = $("#view"+p).is(':checked');
                        if(checked) selected.push(p);
                    }
                    data.profileview.display = selected;
                    save();
                    location.reload();
                }
            }
        ]
    );

    var button = '<div style="float:right">' +
        '<button id="editProfileView">Edit</button></div>';

    return button+popup;
}

/**
* Example usage:
* popupWindow(
* { 'element': '#myDialogButton', 'title': 'Hello World' },
* '<h3>Something</h3><input type="text" id="myfield" />',
* [{'display': 'Save',
* 'close': false, // default: true
* callback: function(){var something = $("#myfield").val();}
* }]);
*/
function popupWindow(e, content, buttons){
    var script = '<script>$("'+e.element+'").click(function(){$("#dialog-message").dialog({'+
        'modal: true,'+
        'draggable: true,'+
        'create: function(){$(this).css("maxHeight", $(window).height()-240);},'+
        'resizable: true,'+
        'position: [\'center\'],'+
        'show: \'blind\','+
        'hide: \'blind\','+
        'width: '+(e.width ? e.width : 400)+','+
        'buttons: [';
    var test = '';
    for(var i = 0; i < buttons.length; i++){
        script += '{';
        script += 'text: \''+buttons[i].display+'\',';
        script += 'id: \'dialogButton'+i+'\'';
        script += '}';

        // TODO: Fix this shit later!
        $(document).on('click','#dialogButton0', function(){
            buttons[0].callback();
            $("a.ui-dialog-titlebar-close")[0].click();
        });
        // TODO: Support multiple buttons!
    }
    // todo: remove last char
    script += ']}';
    script += ');});'+test;

    script += '</script>';

    var html = '<div id="dialog-message" title="'+e.title+'" style="display: none; max-height: 80%;">';
    html += content;
    html += '</div>';
    return script+html;
}

function inArray(c, a){
    // Somehow $.inArray is not working? ?
    for(var i = 0; i < a.length; i++){
        if(c == a[i]) return true;
    }
    return false;
}

function getUserValue(userid, property){
    var user = data[userid];
    if(user){
        if($.isArray(property)){
            for(var i = 0; i < property.length; i++){
                user = user[property[i]];
            }
            return user;
        }else{
            var userData = user[property];
            if(userData){
                return userData;
            }
        }
    }
    return -1;
}

function setUserValue(userid, property, value){
    if(data[userid] === undefined) data[userid] = {};
    data[userid][property] = value;
}

function save(){
    localStorage.setItem('jebster.torn', JSON.stringify(data));
}

function saveOwnData(){
    if(!('me' in data) || !('id' in data.me) || data.me.id < 1){
        var url = 'https://api.torn.com/user/'+data.me.id+'?selections=basic&key='+data.apikey;
        apiCall(url, function(d) {
            id = d.player_id;
            data.me = {'id': id};
            save();
        });
    }
}

function loadData(){
    data = localStorage.getItem('jebster.torn');
    if(data === undefined || data === null){
        // Default settings
        data = {
            profileview:{
                show: true,
                display: ['xantaken','logins','refills','useractivity']
            }
        };
    }else{
        data = JSON.parse(data);
    }

    if(data.apikey === undefined || data.apikey === ''){
        getApiKey();
    }
}

var asked = false;
function getApiKey(){
    // Add some beautiful input/popup on the page itself
    if(asked) return; asked = true;
    var key = prompt('You API Key', 'key');
    if(!('me' in data)) data.me = {};
    data.apikey = key;
    save();
}

function apiUserStats(userid, cb){
    var lastRequest = getUserValue(userid, 'lastRequest');
    var now = new Date();
    if(lastRequest === 0 || lastRequest < now.getTime() - (60*60*5*1000)){ // TODO:
        var selections = 'personalstats,basic,crimes';
        var url = 'https://api.torn.com/user/'+userid+'?selections='+selections+'&key='+data.apikey;
        apiCall(url, function(data) {
            if(data.error) getApiKey();
            else{
                setUserValue(userid, 'stats', data.personalstats);
                setUserValue(userid, 'lastRequest', now.getTime());
                setUserValue(userid, 'username', data.name);
                setUserValue(userid, 'crimes', data.crimes);
                save();
                cb(data);
            }
        });
    }else{
        cb(data[userid].stats);
    }
}

function compareTemplate(user1Id, user2Id){

    var css = '<style>'+
        '.tornHelper{' +
        'min-width:200px;' +
        '}' +
        '</style>';

    var html = css+'<div class="profile-container basic-info"><ul class="basic-list">' +
        '<li>' +
        '<div class="user-information-section left"><span class="bold"></span></div>' +
        '<div class="user-information-section left tornHelper"><span class="bold">'+getUserValue(user1Id, 'username')+'</span></div>' +
        '<div class="tornHelper"><span class="bold">'+getUserValue(user2Id, 'username')+' (You)</span></div>' +
        '</li>';

    var stats = possibleStats();
    for(var i = 0; i < data.profileview.display.length; i++){
        var display = stats[data.profileview.display[i]];
        if(stats[data.profileview.display[i]]){
            var user1Value = getUserValue(user1Id, display.apiname);
            var user2Value = getUserValue(user2Id, display.apiname);

            user1Value = user1Value ? user1Value : 0;
            user2Value = user2Value ? user2Value : 0;

            if(display.format){
                user1Value = display.format(user1Value);
                user2Value = display.format(user2Value);
            }

            html += '<li>' +
                '<div class="user-information-section left"><span class="bold">' +
                display.display +
                '</span></div>' +
                '<div class="user-information-section left tornHelper">' +
                user1Value +'</div>' +
                '<div class="tornHelper">'+user2Value+'</div>' +
                '</li>';
        }
    }
    html += '</ul><hr />';
    var attacks = {hosp: {display: 'Hosped', times: 0}, mug: {display: 'Mugged', times: 0}, left: {display: 'Left', times: 0}, lost: {display: 'Lost', times: 0}};
    if('attacks' in data[user1Id]){
        for(var p in data[user1Id].attacks){
            var attack = data[user1Id].attacks[p];

            switch(attack.result){
                case 'Mug':
                    attacks.mug.times++;
                    break;
                case 'Hospitalize':
                    attacks.hosp.times++;
                    break;
                case 'Leave':
                    attacks.left.times++;
                    break;
                case 'Lose':
                    attacks.lost.times++;
                    break;
            }
        }
    }
    html += 'Attacks you made against \''+getUserValue(user1Id, 'username')+'\'';
    html += '<ul class="basic-list">';
    for(var type in attacks){
        html += '<li>';
        html += '<div class="user-information-section left width112"><span class="bold">'+attacks[type].display+'</span></div>';
        html += attacks[type].times;
        html += '</li>';
    }
    html += '</ul></div>';

    return html;
}

function possibleStats(){
    return {
        attackswon:{apiname:['stats','attackswon'], display: 'Attacks won', category: 'Attacking'},
        attackslost:{apiname:['stats','attackslost'], display: 'Attacks lost', category: 'Attacking'},
        attacksdraw:{apiname:['stats','attacksdraw'], display: 'Attacks Draw', category: 'Attacking'},
        // Attacks stalemated
        attacksassisted:{apiname:['stats','attacksassisted'], display: 'Attacks assisted', category: 'Attacking'},
        defendswon:{apiname:['stats','defendswon'], display: 'Defends won', category: 'Attacking'},
        defendslost:{apiname:['stats','defendslost'], display: 'Defends lost', category: 'Attacking'},
        defendsstalemated:{apiname:['stats','defendsstalemated'], display: 'Defends stalemated', category: 'Attacking'},
        yourunaway:{apiname:['stats','yourunaway'], display: 'Run Aways', category: 'Attacking'},
        theyrunaway:{apiname:['stats','theyrunaway'], display: 'Other Ran Away', category: 'Attacking'},
        bestkillstreak:{apiname:['stats','bestkillstreak'], display: 'Best Kill Streak', category: 'Attacking'},
        attackcriticalhits:{apiname:['stats','attackcriticalhits'], display: 'Attack Critical Hits', category: 'Attacking'},
        attackhits:{apiname:['stats','attackhits'], display: 'Attack Hits', category: 'Attacking'},
        attackmisses:{apiname:['stats','attackmisses'], display: 'Attack Misses', category: 'Attacking'},
        roundsfired:{apiname:['stats','roundsfired'], display: 'Rounds Fired', category: 'Attacking'},
        attacksstealthed:{apiname:['stats','attacksstealthed'], display: 'Attacks Stealthed', category: 'Attacking'},
        moneymugged:{apiname:['stats','moneymugged'], display: 'Money Mugged', category: 'Attacking'},
        largestmug:{apiname:['stats','largestmug'], display: 'Largest Mug', category: 'Attacking'},
        highestbeaten:{apiname:['stats','highestbeaten'], display: 'Highest Level Beaten', category: 'Attacking'},
        respectforfaction:{apiname:['stats','respectforfaction'], display: 'Respect For Faction', category: 'Attacking'},
        itemsbought:{apiname:['stats','itemsbought'], display: 'Items Bought', category: 'Trading'},
        auctionswon:{apiname:['stats','auctionswon'], display: 'Auctions Won', category: 'Trading'},
        auctionsells:{apiname:['stats','auctionsells'], display: 'Auction Sells', category: 'Trading'},
        itemssent:{apiname:['stats','itemssent'], display: 'Items Sent', category: 'Trading'},
        trades:{apiname:['stats','trades'], display: 'Trades', category: 'Trading'},
        weaponsbought:{apiname:['stats','weaponsbought'], display: 'Weapons Bought', category: 'Trading'},
        pointssold:{apiname:['stats','pointssold'], display: 'Points Sold', category: 'Trading'},
        pointsbought:{apiname:['stats','pointsbought'], display: 'Points Bought', category: 'Trading'},
        bazaarcustomers:{apiname:['stats','bazaarcustomers'], display: 'Bazaar Customers', category: 'Trading'},
        bazaarsales:{apiname:['stats','bazaarsales'], display: 'Bazaar Sales', category: 'Trading'},
        bazaarprofit:{apiname:['stats','bazaarprofit'], display: 'Bazaar Profit', category: 'Trading'},
        jailed:{apiname:['stats','jailed'], display: 'Jailed', category: 'Jail'},
        peoplebusted:{apiname:['stats','peoplebusted'], display: 'People Busted', category: 'Jail'},
        failedbusts:{apiname:['stats','failedbusts'], display: 'Failed Busts', category: 'Jail'},
        peoplebought:{apiname:['stats','peoplebought'], display: 'People Bought out of Jail', category: 'Jail'},
        peopleboughtspent:{apiname:['stats','peopleboughtspent'], display: 'Money Spent on buying people out of jail', category: 'Jail'}, // TODO: Some shorter display text
        hospital:{apiname:['stats','hospital'], display: 'Hospital', category: 'Hospital'},
        medicalitemsused:{apiname:['stats','medicalitemsused'], display: 'Medical Items Used', category: 'Hospital'},
        bloodwithdrawn:{apiname:['stats','bloodwithdrawn'], display: 'Blood Withdrawn'}, category: 'Hospital',
        revives:{apiname:['stats','revives'], display: 'Revives', category: 'Hospital'},
        revivesreceived:{apiname:['stats','revivesreceived'], display: 'Revives Received', category: 'Hospital'},
        medstolen:{apiname:['stats','medstolen'], display: 'Medical Items Stolen', category: 'Hospital'},
        heahits:{apiname:['stats','heahits'], display: 'Heavy artillery', category: 'Finishing Hits'},
        machits:{apiname:['stats','machits'], display: 'Machine guns', category: 'Finishing Hits'},
        rifhits:{apiname:['stats','rifhits'], display: 'Rifles', category: 'Finishing Hits'},
        smghits:{apiname:['stats','smghits'], display: 'Sub machine guns', category: 'Finishing Hits'},
        shohits:{apiname:['stats','shohits'], display: 'Shotguns', category: 'Finishing Hits'},
        pishits:{apiname:['stats','pishits'], display: 'Pistols', category: 'Finishing Hits'},
        grehits:{apiname:['stats','grehits'], display: 'Temporary weapons', category: 'Finishing Hits'},
        piehits:{apiname:['stats','piehits'], display: 'Piercing weapons', category: 'Finishing Hits'},
        slahits:{apiname:['stats','slahits'], display: 'Slashing weapons', category: 'Finishing Hits'},
        axehits:{apiname:['stats','axehits'], display: 'Clubbed weapons', category: 'Finishing Hits'},
        chahits:{apiname:['stats','chahits'], display: 'Mechanical weapons', category: 'Finishing Hits'},
        mailssent:{apiname:['stats','mailssent'], display: 'Mail Sent', category: 'Communication'},
        friendmailssent:{apiname:['stats','friendmailssent'], display: 'Friend Mail Sent', category: 'Communication'},
        factionmailssent:{apiname:['stats','factionmailssent'], display: 'Faction Mail Sent', category: 'Communication'},
        companymailssent:{apiname:['stats','companymailssent'], display: 'Company Mail Sent', category: 'Communication'},
        spousemailssent:{apiname:['stats','spousemailssent'], display: 'Spouse Mail Sent', category: 'Communication'},
        classifiedadsplaced:{apiname:['stats','classifiedadsplaced'], display: 'Classified Newspaper Ads Placed', category: 'Communication'},
        personalsplaced:{apiname:['stats','personalsplaced'], display: 'Personal Placed', category: 'Communication'},
        bountiesplaced:{apiname:['stats','bountiesplaced'], display: 'Bounties Placed', category: 'Bounties'},
        totalbountyspent:{apiname:['stats','totalbountyspent'], display: 'Total Bounty Money Spent', category: 'Bounties'},
        bountiescollected:{apiname:['stats','bountiescollected'], display: 'Bounties Collected', category: 'Bounties'},
        totalbountyreward:{apiname:['stats','totalbountyreward'], display: 'Total Bounty Money Gained', category: 'Bounties'},
        bountiesreceived:{apiname:['stats','bountiesreceived'], display: 'Bounties Received', category: 'Bounties'},
        cityfinds:{apiname:['stats','cityfinds'], display: 'City Finds', category: 'Items'},
        itemsdumped:{apiname:['stats','itemsdumped'], display: 'Items Dumped', category: 'Items'},
        dumpsearches:{apiname:['stats','dumpsearches'], display: 'Dump Searches', category: 'Items'},
        dumpfinds:{apiname:['stats','dumpfinds'], display: 'Dump Finds', category: 'Items'},
        traveltimes:{apiname:['stats','traveltimes'], display: 'Travel Times', category: 'Travel'},
        itemsboughtabroad:{apiname:['stats','itemsboughtabroad'], display: 'Items Bought Abroad', category: 'Travel'},
        argtravel:{apiname:['stats','argtravel'], display: 'Argentina Traveled', category: 'Travel'},
        mextravel:{apiname:['stats','mextravel'], display: 'Mexico Traveled', category: 'Travel'},
        dubtravel:{apiname:['stats','dubtravel'], display: 'Dubai Traveled', category: 'Travel'},
        hawtravel:{apiname:['stats','hawtravel'], display: 'Hawaii Traveled', category: 'Travel'},
        japtravel:{apiname:['stats','japtravel'], display: 'Japan Traveled', category: 'Travel'},
        lontravel:{apiname:['stats','lontravel'], display: 'London Traveled', category: 'Travel'},
        soutravel:{apiname:['stats','soutravel'], display: 'South Africa Traveled', category: 'Travel'},
        switravel:{apiname:['stats','switravel'], display: 'Switzerland Traveled', category: 'Travel'},
        chitravel:{apiname:['stats','chitravel'], display: 'China Traveled', category: 'Travel'},
        cantravel:{apiname:['stats','cantravel'], display: 'Canada Traveled', category: 'Travel'},
        caytravel:{apiname:['stats','caytravel'], display: 'Cayman Islands Traveled', category: 'Travel'},
        drugsused:{apiname:['stats','drugsused'], display: 'Drug Used', category: 'Drugs'},
        overdosed:{apiname:['stats','overdosed'], display: 'Drug Overses', category: 'Drugs'},
        cantaken:{apiname:['stats','cantaken'], display: 'Canabis Taken', category: 'Drugs'},
        exttaken:{apiname:['stats','exttaken'], display: 'Ecstasy Taken', category: 'Drugs'},
        kettaken:{apiname:['stats','kettaken'], display: 'Ketamine Taken', category: 'Drugs'},
        lsdtaken:{apiname:['stats','lsdtaken'], display: 'LSD Taken', category: 'Drugs'},
        opitaken:{apiname:['stats','opitaken'], display: 'Opium Taken', category: 'Drugs'},
        shrtaken:{apiname:['stats','shrtaken'], display: 'Shrooms Taken', category: 'Drugs'},
        spetaken:{apiname:['stats','spetaken'], display: 'Speed Taken', category: 'Drugs'},
        pcptaken:{apiname:['stats','pcptaken'], display: 'PCP Taken', category: 'Drugs'},
        xantaken:{apiname:['stats','xantaken'], display: 'Xanax Taken', category: 'Drugs'},
        victaken:{apiname:['stats','victaken'], display: 'Vicodin Taken', category: 'Drugs'},
        networth:{apiname:['stats','networth'], display: 'Networth', format: formatMoney},
        logins:{apiname:['stats','logins'], display: 'Logins'},
        useractivity:{apiname:['stats','useractivity'], display: 'Time Played', format: formatSeconds},
        meritsbought:{apiname:['stats','meritsbought'], display: 'Merits Bought'},
        refills:{apiname:['stats','refills'], display: 'Refills'},
        trainsreceived:{apiname:['stats','trainsreceived'], display: 'Trains Received'},
        spydone:{apiname:['stats','spydone'], display: 'Spies Done'},
        statenhancersused:{apiname:['stats','statenhancersused'], display: 'Stat Enhancers Used'},
        virusescoded:{apiname:['stats','virusescoded'], display: 'Viruses Coded'},
        daysbeendonator:{apiname:['stats','daysbeendonator'], display: 'Days Been Donator'},
        missionscompleted:{apiname:['stats','missionscompleted'], display: 'Missions Completed', category: 'Missions'},
        contractscompleted:{apiname:['stats','contractscompleted'], display: 'Contracts Completed', category: 'Missions'},
        dukecontractscompleted:{apiname:['stats','dukecontractscompleted'], display: 'Duke Contracts Completed', category: 'Missions'},
        missioncreditsearned:{apiname:['stats','missioncreditsearned'], display: 'Mission Credits Earned', category: 'Missions'},
        sellingillegalproducts:{apiname:['crimes','selling_illegal_products'], display: 'Illegal Products Sold', category: 'Crimes'},
        theft:{apiname:['crimes','theft'], display: 'Theft', category: 'Crimes'},
        auto_theft:{apiname:['crimes','auto_theft'], display: 'Auto Theft', category: 'Crimes'},
        drug_deals:{apiname:['crimes','drug_deals'], display: 'Drug Deals', category: 'Crimes'},
        computer_crimes:{apiname:['crimes','computer_crimes'], display: 'Computer Crimes', category: 'Crimes'},
        murder:{apiname:['crimes','murder'], display: 'Murder', category: 'Crimes'},
        fraud_crimes:{apiname:['crimes','fraud_crimes'], display: 'Fraud Crimes', category: 'Crimes'},
        other:{apiname:['crimes','other'], display: 'Other Crimes', category: 'Crimes'},
        total:{apiname:['crimes','total'], display: 'Total Crimes', category: 'Crimes'},
    };
}

function apiCall(url, cb){
    console.log(url);
    $.ajax({
        url: url,
        type: 'GET',
        success: function(data) {
            cb(data);
        },
        //async: false
    });
}

function removeFirstAndLastLine(text){
    var lines = text.split('\n');
    lines.splice(0,1);
    lines.splice(-1,1);
    var newtext = lines.join('\n');
}

function formatSeconds(s){
    var minutes = Math.floor(s/60)%60;
    var hours = Math.floor(s/(60*60))%24;
    var days = Math.floor(s/(60*60*24));
    var seconds = s%60;

    return '{0}d {1}h {2}m {3}s'.format(days, hours, minutes, seconds);
}

function formatNumber(n){
    return n;
}

function formatMoney(m){
    return '$'+m.toString().replace(/(\d)(?=(\d\d\d)+(?!\d))/g, "$1,");
}

// Taken from: http://stackoverflow.com/a/15724300/1832471
function getCookieValue(name) {
    var nameEQ = name + "=";
    var ca = document.cookie.split(';');
    for(var i=0;i < ca.length;i++) {
        var c = ca[i];
        while (c.charAt(0)==' ') c = c.substring(1,c.length);
        if (c.indexOf(nameEQ) === 0) return c.substring(nameEQ.length,c.length);
    }
    return null;
}

// Taken from: http://stackoverflow.com/a/901144/1832471
function getParameterByName(name, url) {
    if (!url) {
        url = window.location.href;
    }
    name = name.replace(/[\[\]]/g, "\\$&");
    var regex = new RegExp("[?&]" + name + "(=([^&#]*)|&|#|$)"),
        results = regex.exec(url);
    if (!results) return null;
    if (!results[2]) return '';
    return decodeURIComponent(results[2].replace(/\+/g, " "));
}