Torn Helper

Adds extra information to different pages all around Torn.

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

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

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

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

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴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.1
// @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 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>';
    // content += '<hr />';
    // content += '<b>Notes</b><br />';
    // content += getUserValue(userid, 'notes');
    // content += '<br />';

    var block = $(
        '<div class="profile-wrapper medals-wrapper m-top10">' +
        '<div class="menu-header">Torn Helper</div>' +
        '<div class="profile-container"> ' +
        '<div class="profile-container-description">'+content+'</div>' +
        '</div>' +
        '<div class="clear"></div></div>');

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

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

function profileViewSelectionPopUp(){
    var possibilities = possibleStats();
    var popupHtml = '<div><fieldset><legend>Select:</legend>';
    for(var p in possibilities){
        var o = possibilities[p];
        var checked = inArray(p, data.profileview.display) ? 'checked' : '';

        popupHtml += '<label for="checkbox'+p+'">'+o.display+'</label>: '+
            '<input type="checkbox" name="checkbox'+p+'" id="view'+p+'" '+checked+'> ';
    }
    popupHtml += '</fieldset></div>';
    var popup = popupWindow(
        {
            element: '#editProfileView',
            title: 'Edit Profile view'
        },
        popupHtml,
        [
            {
                'display': 'Save',
                'close': false, // default: true // TODO: Not working!
                'callback': function(){
                    for(var p in possibleStats()){
                        alert(p);
                    }
                    $(this).dialog("close");
                }
            }
        ]
    );

    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,'+
        'resizable: true,'+
        'position: [\'center\'],'+
        'show: \'blind\','+
        'hide: \'blind\','+
        'width: 400,'+
        'buttons: [';
    var test = '';
    for(var i = 0; i < buttons.length; i++){
        var a = buttons[i].callback;
        script += '{';
        script += 'text: \''+buttons[i].display+'\',';
        script += 'id: \'dialogButton'+i+'\',';
        script += 'click: '+a;
        /*
        if('close' in buttons[i] || buttons[i].close)
            script += '$(this).dialog("close");';*/
        script += '}';

        // 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;">';
    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:{
                display: ['xanax','logins','refills','timeplayed']
            }
        };
    }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';
        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);
                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);

            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 {
        xanax:{apiname:['stats','xantaken'], display: 'Xanax taken'},
        refills:{apiname:['stats','refills'], display: 'Refills'},
        logins:{apiname:['stats','logins'], display: 'Logins'},
        timeplayed:{apiname:['stats','useractivity'], display: 'Time played', format: formatSeconds}
    };
}

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);
}

// 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, " "));
}