Torn Helper

Adds extra information to different pages all around Torn.

当前为 2017-02-06 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Torn Helper
  3. // @namespace Jebster.Torn
  4. // @author Jeggy
  5. // @description Adds extra information to different pages all around Torn.
  6. // @include *.torn.com/profiles.php?XID=*
  7. // @version 0.2.5
  8. // @require http://code.jquery.com/jquery-2.2.4.min.js
  9. // @require http://code.jquery.com/ui/1.12.1/jquery-ui.min.js
  10. // @resource jquery-ui http://ajax.googleapis.com/ajax/libs/jqueryui/1.10.3/themes/black-tie/jquery-ui.min.css
  11. // @resource jquery-base http://code.jquery.com/ui/1.12.1/themes/base/jquery-ui.css
  12. // @grant GM_addStyle
  13. // @grant GM_getResourceText
  14. // ==/UserScript==
  15. // debugger;
  16.  
  17. GM_addStyle(GM_getResourceText('jquery-base'));
  18. GM_addStyle(GM_getResourceText('jquery-ui'));
  19.  
  20. String.prototype.format = function() {
  21. var formatted = this;
  22. for (var i = 0; i < arguments.length; i++) {
  23. var regexp = new RegExp('\\{'+i+'\\}', 'gi');
  24. formatted = formatted.replace(regexp, arguments[i]);
  25. }
  26. return formatted;
  27. };
  28.  
  29. var data = {};
  30.  
  31. (function() {
  32. 'use strict';
  33.  
  34. $( 'head' ).append( '<link rel="stylesheet" href="//maxcdn.bootstrapcdn.com/font-awesome/4.3.0/css/font-awesome.min.css">' );
  35.  
  36. var site = window.location.pathname;
  37.  
  38. loadData();
  39. saveOwnData();
  40.  
  41. loadAttackLog();
  42.  
  43. if(site.indexOf('profiles.php') > 0) profileView();
  44.  
  45. })();
  46.  
  47. function loadAttackLog(){
  48. var selections = '';
  49. var now = new Date().getTime();
  50.  
  51. if('attackLogLoad' in data){
  52. if(data.attackLogLoad < now - (12*60*60*1000)) // More than 12 hours ago
  53. selections = 'attacksfull';
  54. else if(data.attackLogLoad < now - (2*60*1000)) // More than 2 minutes ago
  55. selection = 'attacks';
  56. }else // First time
  57. selections = 'attacksfull';
  58.  
  59. var url = 'https://api.torn.com/user/'+data.me.id+'?selections='+selections+'&key='+data.apikey;
  60. if(selections !== ''){
  61. data.attackLogLoad = now;
  62.  
  63. apiCall(url, function(d) {
  64. if(d.error) getApiKey();
  65. else{
  66. for(var p in d.attacks){
  67. if (d.attacks.hasOwnProperty(p)) {
  68. var attack = d.attacks[p];
  69. var defender_id = attack.defender_id;
  70. if(!(defender_id in data)) data[defender_id] = {};
  71.  
  72. if(attack.attacker_id == data.me.id){
  73. // My attack
  74. if(!('attacks' in data[defender_id])) data[defender_id].attacks = {};
  75. data[defender_id].attacks[p] = attack;
  76. }else if('attacker_id' in attack && attack.attacker_id in data){
  77. // I'm being attacked
  78. if(!('defends' in data[attack.attacker_id])) data[attack.attacker_id].defends = {};
  79. data[attack.attacker_id].defends[p] = attack;
  80. }
  81. }
  82. }
  83.  
  84. save();
  85. }
  86. });
  87. }
  88. }
  89.  
  90. function profileView(){
  91. var userid = getParameterByName('XID');
  92. var userData = data[userid];
  93. var content = '';
  94. content += profileViewSelectionPopUp(); // TODO: Figoure out how to call tampermonkey function from the injected code.
  95. content += '<div id="compareStats">';
  96. content += 'Loading...';
  97. content += '<br />';
  98. content += '</div>';
  99.  
  100. var acrdHtml = accordion('Torn Helper', content);
  101.  
  102. $(acrdHtml).insertAfter('.tutorial-cont + .profile-wrapper');
  103.  
  104. var compareFunc = function(){
  105. $('#compareStats').replaceWith(compareTemplate(userid, data.me.userid));
  106. };
  107. apiUserStats(userid, compareFunc);
  108. apiUserStats(data.me.userid, compareFunc);
  109. }
  110.  
  111. // Only supports one accordion on the page at the moment.
  112. function accordion(title, block){
  113. var css = '<style>'+
  114. 'button.Jaccordion {'+
  115. 'cursor: pointer;'+
  116. 'padding: 18px;'+
  117. 'width: 100%;'+
  118. 'text-align: left;'+
  119. 'transition: 0.4s;'+
  120. '}'+
  121. 'div.Jpanel {'+
  122. 'max-height: 0;'+
  123. 'overflow: hidden;'+
  124. 'transition: 0.8s ease-in-out;'+
  125. 'opacity: 0;'+
  126. '}'+
  127. 'div.Jshow {'+
  128. 'opacity: 1;'+
  129. 'max-height: 1000px;'+
  130. 'width: auto;'+
  131. '}'+
  132. 'div.JAccordionIconShow {'+
  133. 'opacity: 1;'+
  134. 'max-height: 30px;'+
  135. 'width: auto;'+
  136. '}'+
  137. '.JProfileViewAccordion{'+
  138. 'width: 0;'+
  139. 'float: right;'+
  140. 'max-height: 0;'+
  141. 'overflow: hidden;'+
  142. 'opacity: 0;'+
  143. 'padding-right: 7px;'+
  144. '}'+
  145. '.JAccordionIcon{'+
  146. 'padding: 6px 5px 6px 25px;'+
  147. 'cursor: pointer;'+
  148. '}'+
  149. '</style>';
  150.  
  151. var script = '<script>$(".JAccordionIcon").click(function() {'+
  152. '$(".Jpanel").toggleClass("Jshow");'+
  153. '$(".JProfileViewAccordion").toggleClass("JshJAccordionIconShowow");'+
  154. '$(".JProfileViewAccordion").toggleClass("JAccordionIconShow");'+
  155. '});</script>';
  156.  
  157. var show = data.profileview.show;
  158.  
  159. var html = '<div class="Jaccordion profile-wrapper medals-wrapper m-top10">'+
  160. '<div class="menu-header">'+title+
  161. '<div class="JProfileViewAccordion '+(show ? '' : 'JAccordionIconShow')+'"><i class="fa fa-plus-square JAccordionIcon" aria-hidden="true" /></div>'+
  162. '<div class="JProfileViewAccordion '+(show ? 'JAccordionIconShow' : '')+'"><i class="fa fa-minus-square JAccordionIcon" aria-hidden="true" /></div>'+
  163. '</div>'+
  164. '<div class="Jpanel '+(show ? 'Jshow' : '')+'">'+
  165. block+
  166. '</div>'+
  167. '</div>';
  168.  
  169. $(document).on('click','.JAccordionIcon', function(){
  170. data.profileview.show = !data.profileview.show;
  171. save();
  172. });
  173.  
  174. return css+script+html;
  175. }
  176.  
  177. function profileViewSelectionPopUp(){
  178. var possibilities = possibleStats();
  179.  
  180. var popupHtml = '<div>';
  181. var categories = {};
  182. for(var p in possibilities){
  183. var o = possibilities[p];
  184. var checked = inArray(p, data.profileview.display) ? 'checked' : '';
  185.  
  186. if(o.category){
  187. if(!categories[o.category]) categories[o.category] = {display: o.category, html: ''};
  188. }else{
  189. if(!categories.others) categories.others = {display: 'Others', html: ''};
  190. }
  191.  
  192. var cat = o.category ? o.category : 'others';
  193.  
  194. var html = '<label id="JC'+cat+'" for="view'+p+'" style="';
  195. html += 'border-radius: 2px;border: 1px solid gray;background-color: lightgray;margin: 3px;';
  196. html += 'padding: 3px 10px 3px 5px;display: inline-block;cursor: pointer;';
  197. html += '"><input type="checkbox" name="view'+p+'" id="view'+p+'" '+checked+'>';
  198. html += o.display+'</label>';
  199. categories[cat].html += html;
  200. }
  201.  
  202. for(var category in categories){
  203. if (categories.hasOwnProperty(category)) {
  204. popupHtml += '<fieldset style="border:1px solid black; padding: 10px; margin: 10px 0;">';
  205. popupHtml += '<legend onclick=\'document.getElementById("JC'+category+'").style.display = document.getElementById("JC'+category+'").style.display == "none" ? "block" : "none";\'';
  206. popupHtml += '><b>'+categories[category].display+'</b></legend>';
  207. popupHtml += categories[category].html;
  208. popupHtml += '</fieldset>';
  209. }
  210. }
  211.  
  212. popupHtml += '</div>';
  213. var popup = popupWindow(
  214. {
  215. element: '#editProfileView',
  216. title: 'Edit Profile view',
  217. width: 640
  218. },
  219. popupHtml,
  220. [
  221. {
  222. 'display': 'Close',
  223. 'close': true
  224. },
  225. {
  226. 'display': 'Save',
  227. 'callback': function(){
  228. var selected = [];
  229. for(var p in possibleStats()){
  230. var checked = $("#view"+p).is(':checked');
  231. if(checked) selected.push(p);
  232. }
  233. data.profileview.display = selected;
  234. save();
  235. location.reload();
  236. }
  237. }
  238. ]
  239. );
  240.  
  241. var button = '<div style="float:right">' +
  242. '<button id="editProfileView" style="';
  243. button += 'background-color: #282828;';
  244. button += 'border: none;';
  245. button += 'border-radius: 5px;';
  246. button += 'color: white;';
  247. button += 'padding: 5px 5px 5px 7px;';
  248. button += 'text-align: center;';
  249. button += 'text-decoration: none;';
  250. button += 'display: inline-block;';
  251. button += 'font-size: 16px;';
  252. button += 'margin: 4px 2px;';
  253. button += 'cursor: pointer;';
  254. button += '"><i class="fa fa-pencil-square-o" aria-hidden="true" /></button></div>';
  255.  
  256. return button+popup;
  257. }
  258.  
  259. /**
  260. * Example usage:
  261. * popupWindow(
  262. * { 'element': '#myDialogButton', 'title': 'Hello World' },
  263. * '<h3>Something</h3><input type="text" id="myfield" />',
  264. * [{'display': 'Save',
  265. * 'close': false, // default: true
  266. * callback: function(){var something = $("#myfield").val();}
  267. * }]);
  268. */
  269. function popupWindow(e, content, buttons){
  270. var script = '<script>$("'+e.element+'").click(function(){$("#dialog-message").dialog({'+
  271. 'modal: true,'+
  272. 'draggable: true,'+
  273. 'create: function(){$(this).css("maxHeight", $(window).height()-240);},'+
  274. 'resizable: true,'+
  275. 'position: [\'center\'],'+
  276. 'show: \'blind\','+
  277. 'hide: \'blind\','+
  278. 'width: '+(e.width ? e.width : 400)+','+
  279. 'buttons: [';
  280. var test = '';
  281. for(var i = 0; i < buttons.length; i++){
  282. script += '{';
  283. script += 'text: \''+buttons[i].display+'\',';
  284. script += 'id: \'dialogButton'+i+'\'';
  285. script += '},';
  286.  
  287. // this only supports up to 10 buttons.
  288. $(document).on('click','#dialogButton'+i, function(){
  289. var pressed = this.id.substr(this.id.length -1);
  290. if(buttons[pressed].callback)
  291. buttons[pressed].callback();
  292.  
  293. if(buttons[pressed].close)
  294. $("a.ui-dialog-titlebar-close")[0].click();
  295. });
  296. }
  297. script = script.slice(0, -1);
  298.  
  299. script += ']}';
  300. script += ');});'+test;
  301.  
  302. script += '</script>';
  303.  
  304. var html = '<div id="dialog-message" title="'+e.title+'" style="display: none; max-height: 80%;">';
  305. html += content;
  306. html += '</div>';
  307. return script+html;
  308. }
  309.  
  310. function inArray(c, a){
  311. // Somehow $.inArray is not working? ?
  312. for(var i = 0; i < a.length; i++){
  313. if(c == a[i]) return true;
  314. }
  315. return false;
  316. }
  317.  
  318. function getUserValue(userid, property){
  319. var user = data[userid];
  320. if(user){
  321. if($.isArray(property)){
  322. for(var i = 0; i < property.length; i++){
  323. user = user[property[i]];
  324. }
  325. return user;
  326. }else{
  327. var userData = user[property];
  328. if(userData){
  329. return userData;
  330. }
  331. }
  332. }
  333. return -1;
  334. }
  335.  
  336. function setUserValue(userid, property, value){
  337. if(data[userid] === undefined) data[userid] = {};
  338. data[userid][property] = value;
  339. }
  340.  
  341. function save(){
  342. localStorage.setItem('jebster.torn', JSON.stringify(data));
  343. }
  344.  
  345. function saveOwnData(){
  346. if(!('me' in data) || !('id' in data.me) || data.me.id < 1){
  347. var url = 'https://api.torn.com/user/'+data.me.id+'?selections=basic&key='+data.apikey;
  348. apiCall(url, function(d) {
  349. id = d.player_id;
  350. data.me = {'id': id};
  351. save();
  352. });
  353. }
  354. }
  355.  
  356. function loadData(){
  357. data = localStorage.getItem('jebster.torn');
  358. if(data === undefined || data === null){
  359. // Default settings
  360. data = {
  361. profileview:{
  362. show: true,
  363. display: ['xantaken','logins','refills','useractivity']
  364. }
  365. };
  366. }else{
  367. data = JSON.parse(data);
  368. }
  369.  
  370. if(data.apikey === undefined || data.apikey === ''){
  371. getApiKey();
  372. }
  373. }
  374.  
  375. var asked = false;
  376. function getApiKey(){
  377. if(asked) return; asked = true;
  378.  
  379.  
  380. var button = '<button id="JApiKeyBtn" style="';
  381. button += 'background-color: #282828;';
  382. button += 'border: none;';
  383. button += 'border-radius: 0 8px 8px 0;';
  384. button += 'color: white;';
  385. button += 'padding: 5px 5px 5px 6px;';
  386. button += 'text-align: center;';
  387. button += 'text-decoration: none;';
  388. button += 'display: inline-block;';
  389. button += 'font-size: 16px;';
  390. button += 'margin: 4px 0px;';
  391. button += 'cursor: pointer;';
  392. button += '"><i class="fa fa-floppy-o" aria-hidden="true"></i></button>';
  393.  
  394. var input = '<input type="text" id="JApiKeyInput" style="';
  395. input += 'border-radius: 8px 0 0 8px;';
  396. input += 'margin: 4px 0px;';
  397. input += 'padding: 5px;';
  398. input += 'font-size: 16px;';
  399. input += '" placeholder="ApiKey"></input>';
  400.  
  401. var block = '<div class="profile-wrapper medals-wrapper m-top10">';
  402. block += '<div class="menu-header">Torn Helper</div>';
  403. block += '<div class="profile-container"><div class="profile-container-description">';
  404. block += 'In order to use this script you need to enter your Torn Api Key, which you can'+
  405. 'get on your <a href="http://www.torn.com/preferences.php">preferences page</a> and under the \'API Key\' tab.<br />';
  406. block += input;
  407. block += button;
  408. block += '</div></div></div>';
  409.  
  410. $(block).insertAfter('.content-title');
  411.  
  412. $('#JApiKeyBtn').click(function(){
  413. var key = $("#JApiKeyInput").val();
  414. if(!('me' in data)) data.me = {};
  415. data.apikey = key;
  416. save();
  417. location.reload();
  418. });
  419.  
  420. }
  421.  
  422. function apiUserStats(userid, cb){
  423. var lastRequest = getUserValue(userid, 'lastRequest');
  424. var now = new Date();
  425. if(lastRequest === 0 || lastRequest < now.getTime() - (60*60*5*1000)){ // TODO:
  426. var selections = 'personalstats,basic,crimes';
  427. var url = 'https://api.torn.com/user/'+userid+'?selections='+selections+'&key='+data.apikey;
  428. apiCall(url, function(data) {
  429. if(data.error) getApiKey();
  430. else{
  431. setUserValue(userid, 'stats', data.personalstats);
  432. setUserValue(userid, 'lastRequest', now.getTime());
  433. setUserValue(userid, 'username', data.name);
  434. setUserValue(userid, 'crimes', data.crimes);
  435. save();
  436. cb(data);
  437. }
  438. });
  439. }else{
  440. cb(data[userid].stats);
  441. }
  442. }
  443.  
  444. function compareTemplate(user1Id, user2Id){
  445.  
  446. var css = '<style>'+
  447. '.tornHelper{' +
  448. 'min-width:200px;' +
  449. '}' +
  450. '</style>';
  451.  
  452. var html = css+'<div class="profile-container basic-info"><ul class="basic-list">' +
  453. '<li>' +
  454. '<div class="user-information-section left"><span class="bold"></span></div>' +
  455. '<div class="user-information-section left tornHelper"><span class="bold">'+getUserValue(user1Id, 'username')+'</span></div>' +
  456. '<div class="tornHelper"><span class="bold">'+getUserValue(user2Id, 'username')+' (You)</span></div>' +
  457. '</li>';
  458.  
  459. var stats = possibleStats();
  460. for(var i = 0; i < data.profileview.display.length; i++){
  461. var display = stats[data.profileview.display[i]];
  462. if(stats[data.profileview.display[i]]){
  463. var user1Value = 0, user2Value = 0;
  464. if(display.apiname){
  465. user1Value = getUserValue(user1Id, display.apiname);
  466. user2Value = getUserValue(user2Id, display.apiname);
  467. }else if(display.total){
  468. for(var j = 0; j < display.total.length; j++){
  469. user1Value += getUserValue(user1Id, display.total[j]);
  470. user2Value += getUserValue(user2Id, display.total[j]);
  471. }
  472. }else if(display.custom){
  473. user1Value = display.custom(user1Id);
  474. user2Value = display.custom(user2Id);
  475. }
  476. user1Value = user1Value ? user1Value : 0;
  477. user2Value = user2Value ? user2Value : 0;
  478.  
  479. if(display.format){
  480. user1Value = display.format(user1Value);
  481. user2Value = display.format(user2Value);
  482. }else{
  483. user1Value = formatNumber(user1Value);
  484. user2Value = formatNumber(user2Value);
  485. }
  486.  
  487. html += '<li>';
  488. html += '<div class="user-information-section left"><span class="bold">';
  489. html += display.display;
  490. if(display.tooltip){
  491. html += ' <i class="fa fa-question-circle" aria-hidden="true" title="'+display.tooltip+'" style="cursor: pointer;" />';
  492. }
  493. html += '</span></div>';
  494. html += '<div class="user-information-section left tornHelper">';
  495. html += user1Value +'</div>';
  496. html += '<div class="tornHelper">'+user2Value+'</div>';
  497. html += '</li>';
  498. }
  499. }
  500. html += '</ul><hr />';
  501.  
  502. var f = function(type, id){
  503. var attacks = {hosp: {display: 'Hosped', times: 0, other: 0}, mug: {display: 'Mugged', times: 0},
  504. left: {display: 'Left', times: 0}, lost: {display: 'Lost', times: 0}};
  505. if(type in data[id]){
  506. for(var p in data[id][type]){
  507. var attack = data[id][type][p];
  508. switch(attack.result){
  509. case 'Mug':
  510. attacks.mug.times++;
  511. break;
  512. case 'Hospitalize':
  513. attacks.hosp.times++;
  514. break;
  515. case 'Leave':
  516. attacks.left.times++;
  517. break;
  518. case 'Lose':
  519. attacks.lost.times++;
  520. break;
  521. }
  522. }
  523. }
  524. return attacks;
  525. };
  526. var attacks = f('attacks', user1Id);
  527. var defends = f('defends', user1Id);
  528.  
  529. var ahtml = '<ul class="basic-list">';
  530. var lis = '';
  531. var anyAttacks = false;
  532. for(var type in attacks){
  533. if(attacks[type].times > 0 || defends[type].times > 0){
  534. lis += '<li>';
  535. lis += '<div class="user-information-section left width112"><span class="bold">'+attacks[type].display+'</span></div>';
  536. lis += '<div class="user-information-section left tornHelper">'+defends[type].times+'</div>';
  537. lis += '<div class="tornHelper">'+attacks[type].times+'</div>';
  538. lis += '</li>';
  539. anyAttacks = true;
  540. }
  541. }
  542. if(anyAttacks){
  543. ahtml += '<li>';
  544. ahtml += '<div class="user-information-section left"><span class="bold">Attacks</span></div>';
  545. ahtml += '<div class="user-information-section left tornHelper"><span class="bold">He made</span></div>';
  546. ahtml += '<div class="tornHelper"><span class="bold">You\'ve done</span></div>';
  547. ahtml += '</li>';
  548. }
  549. ahtml += lis;
  550. ahtml += '</ul></div>';
  551. //html += anyAttacks ? 'Attacks you made against \''+getUserValue(user1Id, 'username')+'\'' : 'You haven\'t attacked '+getUserValue(user1Id, 'username')+' before.';
  552. html += !anyAttacks ? 'You haven\'t attacked '+getUserValue(user1Id, 'username')+' before.' : '';
  553. html += ahtml;
  554. return html;
  555. }
  556.  
  557. function possibleStats(){
  558. return {
  559. attackswon:{apiname:['stats','attackswon'], display: 'Attacks won', category: 'Attacking'},
  560. attackslost:{apiname:['stats','attackslost'], display: 'Attacks lost', category: 'Attacking'},
  561. attacksdraw:{apiname:['stats','attacksdraw'], display: 'Attacks Draw', category: 'Attacking'},
  562. attacksassisted:{apiname:['stats','attacksassisted'], display: 'Attacks assisted', category: 'Attacking'},
  563. totalattacks:{total:[['stats','attackswon'],['stats','attackslost'],['stats','attacksdraw'],['stats','attacksassisted'],['stats','yourunaway']], display: 'Total attacks', category: 'Attacking', format:formatNumber},
  564. defendswon:{apiname:['stats','defendswon'], display: 'Defends won', category: 'Attacking'},
  565. defendslost:{apiname:['stats','defendslost'], display: 'Defends lost', category: 'Attacking'},
  566. defendsstalemated:{apiname:['stats','defendsstalemated'], display: 'Defends stalemated', category: 'Attacking'},
  567. yourunaway:{apiname:['stats','yourunaway'], display: 'Run Aways', category: 'Attacking'},
  568. theyrunaway:{apiname:['stats','theyrunaway'], display: 'Other Ran Away', category: 'Attacking'},
  569. bestkillstreak:{apiname:['stats','bestkillstreak'], display: 'Best Kill Streak', category: 'Attacking'},
  570. attackcriticalhits:{apiname:['stats','attackcriticalhits'], display: 'Attack Critical Hits', category: 'Attacking'},
  571. attackhits:{apiname:['stats','attackhits'], display: 'Attack Hits', category: 'Attacking'},
  572. attackmisses:{apiname:['stats','attackmisses'], display: 'Attack Misses', category: 'Attacking'},
  573. roundsfired:{apiname:['stats','roundsfired'], display: 'Rounds Fired', category: 'Attacking'},
  574. attacksstealthed:{apiname:['stats','attacksstealthed'], display: 'Attacks Stealthed', category: 'Attacking'},
  575. moneymugged:{apiname:['stats','moneymugged'], display: 'Money Mugged', category: 'Attacking', format: formatMoney},
  576. largestmug:{apiname:['stats','largestmug'], display: 'Largest Mug', category: 'Attacking', format: formatMoney},
  577. highestbeaten:{apiname:['stats','highestbeaten'], display: 'Highest Level Beaten', category: 'Attacking'},
  578. respectforfaction:{apiname:['stats','respectforfaction'], display: 'Respect For Faction', category: 'Attacking'},
  579.  
  580. itemsbought:{apiname:['stats','itemsbought'], display: 'Items Bought', category: 'Trading'},
  581. auctionswon:{apiname:['stats','auctionswon'], display: 'Auctions Won', category: 'Trading'},
  582. auctionsells:{apiname:['stats','auctionsells'], display: 'Auction Sells', category: 'Trading'},
  583. itemssent:{apiname:['stats','itemssent'], display: 'Items Sent', category: 'Trading'},
  584. trades:{apiname:['stats','trades'], display: 'Trades', category: 'Trading'},
  585. weaponsbought:{apiname:['stats','weaponsbought'], display: 'Weapons Bought', category: 'Trading'},
  586. pointssold:{apiname:['stats','pointssold'], display: 'Points Sold', category: 'Trading'},
  587. pointsbought:{apiname:['stats','pointsbought'], display: 'Points Bought', category: 'Trading'},
  588. bazaarcustomers:{apiname:['stats','bazaarcustomers'], display: 'Bazaar Customers', category: 'Trading'},
  589. bazaarsales:{apiname:['stats','bazaarsales'], display: 'Bazaar Sales', category: 'Trading'},
  590. bazaarprofit:{apiname:['stats','bazaarprofit'], display: 'Bazaar Profit', category: 'Trading', format: formatMoney},
  591. jailed:{apiname:['stats','jailed'], display: 'Jailed', category: 'Jail'},
  592. peoplebusted:{apiname:['stats','peoplebusted'], display: 'People Busted', category: 'Jail'},
  593. failedbusts:{apiname:['stats','failedbusts'], display: 'Failed Busts', category: 'Jail'},
  594. peoplebought:{apiname:['stats','peoplebought'], display: 'People Bought out of Jail', category: 'Jail'},
  595. peopleboughtspent:{apiname:['stats','peopleboughtspent'], display: 'Money Spent on buying people out of jail', category: 'Jail', format: formatMoney}, // TODO: Some shorter display text
  596. hospital:{apiname:['stats','hospital'], display: 'Hospital', category: 'Hospital'}, // TODO:
  597. medicalitemsused:{apiname:['stats','medicalitemsused'], display: 'Medical Items Used', category: 'Hospital'},
  598. bloodwithdrawn:{apiname:['stats','bloodwithdrawn'], display: 'Blood Withdrawn', category: 'Hospital'},
  599. revives:{apiname:['stats','revives'], display: 'Revives', category: 'Hospital'},
  600. revivesreceived:{apiname:['stats','revivesreceived'], display: 'Revives Received', category: 'Hospital'},
  601. medstolen:{apiname:['stats','medstolen'], display: 'Medical Items Stolen', category: 'Hospital'},
  602. heahits:{apiname:['stats','heahits'], display: 'Heavy artillery', category: 'Finishing Hits'},
  603. machits:{apiname:['stats','machits'], display: 'Machine guns', category: 'Finishing Hits'},
  604. rifhits:{apiname:['stats','rifhits'], display: 'Rifles', category: 'Finishing Hits'},
  605. smghits:{apiname:['stats','smghits'], display: 'Sub machine guns', category: 'Finishing Hits'},
  606. shohits:{apiname:['stats','shohits'], display: 'Shotguns', category: 'Finishing Hits'},
  607. pishits:{apiname:['stats','pishits'], display: 'Pistols', category: 'Finishing Hits'},
  608. grehits:{apiname:['stats','grehits'], display: 'Temporary weapons', category: 'Finishing Hits'},
  609. piehits:{apiname:['stats','piehits'], display: 'Piercing weapons', category: 'Finishing Hits'},
  610. slahits:{apiname:['stats','slahits'], display: 'Slashing weapons', category: 'Finishing Hits'},
  611. axehits:{apiname:['stats','axehits'], display: 'Clubbed weapons', category: 'Finishing Hits'},
  612. chahits:{apiname:['stats','chahits'], display: 'Mechanical weapons', category: 'Finishing Hits'},
  613. mailssent:{apiname:['stats','mailssent'], display: 'Mail Sent', category: 'Communication'},
  614. friendmailssent:{apiname:['stats','friendmailssent'], display: 'Friend Mail Sent', category: 'Communication'},
  615. factionmailssent:{apiname:['stats','factionmailssent'], display: 'Faction Mail Sent', category: 'Communication'},
  616. companymailssent:{apiname:['stats','companymailssent'], display: 'Company Mail Sent', category: 'Communication'},
  617. spousemailssent:{apiname:['stats','spousemailssent'], display: 'Spouse Mail Sent', category: 'Communication'},
  618. classifiedadsplaced:{apiname:['stats','classifiedadsplaced'], display: 'Classified Newspaper Ads Placed', category: 'Communication'},
  619. personalsplaced:{apiname:['stats','personalsplaced'], display: 'Personal Placed', category: 'Communication'},
  620. bountiesplaced:{apiname:['stats','bountiesplaced'], display: 'Bounties Placed', category: 'Bounties'},
  621. totalbountyspent:{apiname:['stats','totalbountyspent'], display: 'Total Bounty Money Spent', category: 'Bounties', format: formatMoney},
  622. bountiescollected:{apiname:['stats','bountiescollected'], display: 'Bounties Collected', category: 'Bounties'},
  623. totalbountyreward:{apiname:['stats','totalbountyreward'], display: 'Total Bounty Money Gained', category: 'Bounties', format: formatMoney},
  624. bountiesreceived:{apiname:['stats','bountiesreceived'], display: 'Bounties Received', category: 'Bounties'},
  625. cityfinds:{apiname:['stats','cityfinds'], display: 'City Finds', category: 'Items'},
  626. itemsdumped:{apiname:['stats','itemsdumped'], display: 'Items Dumped', category: 'Items'},
  627. dumpsearches:{apiname:['stats','dumpsearches'], display: 'Dump Searches', category: 'Items'},
  628. dumpfinds:{apiname:['stats','dumpfinds'], display: 'Dump Finds', category: 'Items'},
  629. traveltimes:{apiname:['stats','traveltimes'], display: 'Travel Times', category: 'Travel'},
  630. itemsboughtabroad:{apiname:['stats','itemsboughtabroad'], display: 'Items Bought Abroad', category: 'Travel'},
  631. argtravel:{apiname:['stats','argtravel'], display: 'Argentina Traveled', category: 'Travel'},
  632. mextravel:{apiname:['stats','mextravel'], display: 'Mexico Traveled', category: 'Travel'},
  633. dubtravel:{apiname:['stats','dubtravel'], display: 'Dubai Traveled', category: 'Travel'},
  634. hawtravel:{apiname:['stats','hawtravel'], display: 'Hawaii Traveled', category: 'Travel'},
  635. japtravel:{apiname:['stats','japtravel'], display: 'Japan Traveled', category: 'Travel'},
  636. lontravel:{apiname:['stats','lontravel'], display: 'London Traveled', category: 'Travel'},
  637. soutravel:{apiname:['stats','soutravel'], display: 'South Africa Traveled', category: 'Travel'},
  638. switravel:{apiname:['stats','switravel'], display: 'Switzerland Traveled', category: 'Travel'},
  639. chitravel:{apiname:['stats','chitravel'], display: 'China Traveled', category: 'Travel'},
  640. cantravel:{apiname:['stats','cantravel'], display: 'Canada Traveled', category: 'Travel'},
  641. caytravel:{apiname:['stats','caytravel'], display: 'Cayman Islands Traveled', category: 'Travel'},
  642. drugsused:{apiname:['stats','drugsused'], display: 'Drug Used', category: 'Drugs'},
  643. overdosed:{apiname:['stats','overdosed'], display: 'Drug Overses', category: 'Drugs'},
  644. cantaken:{apiname:['stats','cantaken'], display: 'Canabis Taken', category: 'Drugs'},
  645. exttaken:{apiname:['stats','exttaken'], display: 'Ecstasy Taken', category: 'Drugs'},
  646. kettaken:{apiname:['stats','kettaken'], display: 'Ketamine Taken', category: 'Drugs'},
  647. lsdtaken:{apiname:['stats','lsdtaken'], display: 'LSD Taken', category: 'Drugs'},
  648. opitaken:{apiname:['stats','opitaken'], display: 'Opium Taken', category: 'Drugs'},
  649. shrtaken:{apiname:['stats','shrtaken'], display: 'Shrooms Taken', category: 'Drugs'},
  650. spetaken:{apiname:['stats','spetaken'], display: 'Speed Taken', category: 'Drugs'},
  651. pcptaken:{apiname:['stats','pcptaken'], display: 'PCP Taken', category: 'Drugs'},
  652. xantaken:{apiname:['stats','xantaken'], display: 'Xanax Taken', category: 'Drugs'},
  653. victaken:{apiname:['stats','victaken'], display: 'Vicodin Taken', category: 'Drugs'},
  654. networth:{apiname:['stats','networth'], display: 'Networth', format: formatMoney},
  655. logins:{apiname:['stats','logins'], display: 'Logins'},
  656. useractivity:{apiname:['stats','useractivity'], display: 'Time Played', format: formatSeconds},
  657. meritsbought:{apiname:['stats','meritsbought'], display: 'Merits Bought'},
  658. refills:{apiname:['stats','refills'], display: 'Refills'},
  659. trainsreceived:{apiname:['stats','trainsreceived'], display: 'Trains Received'},
  660. spydone:{apiname:['stats','spydone'], display: 'Spies Done'},
  661. statenhancersused:{apiname:['stats','statenhancersused'], display: 'Stat Enhancers Used'},
  662. virusescoded:{apiname:['stats','virusescoded'], display: 'Viruses Coded'},
  663. daysbeendonator:{apiname:['stats','daysbeendonator'], display: 'Days Been Donator'},
  664. missionscompleted:{apiname:['stats','missionscompleted'], display: 'Missions Completed', category: 'Missions'},
  665. contractscompleted:{apiname:['stats','contractscompleted'], display: 'Contracts Completed', category: 'Missions'},
  666. dukecontractscompleted:{apiname:['stats','dukecontractscompleted'], display: 'Duke Contracts Completed', category: 'Missions'},
  667. missioncreditsearned:{apiname:['stats','missioncreditsearned'], display: 'Mission Credits Earned', category: 'Missions'},
  668. sellingillegalproducts:{apiname:['crimes','selling_illegal_products'], display: 'Illegal Products Sold', category: 'Crimes'},
  669. theft:{apiname:['crimes','theft'], display: 'Theft', category: 'Crimes'},
  670. auto_theft:{apiname:['crimes','auto_theft'], display: 'Auto Theft', category: 'Crimes'},
  671. drug_deals:{apiname:['crimes','drug_deals'], display: 'Drug Deals', category: 'Crimes'},
  672. computer_crimes:{apiname:['crimes','computer_crimes'], display: 'Computer Crimes', category: 'Crimes'},
  673. murder:{apiname:['crimes','murder'], display: 'Murder', category: 'Crimes'},
  674. fraud_crimes:{apiname:['crimes','fraud_crimes'], display: 'Fraud Crimes', category: 'Crimes'},
  675. other:{apiname:['crimes','other'], display: 'Other Crimes', category: 'Crimes'},
  676. total:{apiname:['crimes','total'], display: 'Total Crimes', category: 'Crimes'},
  677. awards:{display: 'Awards', custom: function(id){return getAwards(id);}, tooltip: 'Your award count only updates when you visit your own profile.'}
  678. };
  679. }
  680.  
  681. function apiCall(url, cb){
  682. console.log(url);
  683. $.ajax({
  684. url: url,
  685. type: 'GET',
  686. success: function(data) {
  687. cb(data);
  688. },
  689. //async: false
  690. });
  691. }
  692.  
  693. function getAwards(id){
  694. var element = $(".profile-container.basic-info.bottom-round ul:nth-child(2) li:nth-child(8)");
  695. var content = element.first().contents().filter(function(){
  696. return this.nodeType == 3;
  697. });
  698. var value = parseInt(content.text().trim());
  699.  
  700. if(!id){
  701. if(data.me.awards)
  702. return data.me.awards;
  703. return -1;
  704. }
  705. if(id == data.me.id){
  706. data.me.awards = value;
  707. save();
  708. }
  709.  
  710. return value;
  711. }
  712.  
  713. function removeFirstAndLastLine(text){
  714. var lines = text.split('\n');
  715. lines.splice(0,1);
  716. lines.splice(-1,1);
  717. var newtext = lines.join('\n');
  718. }
  719.  
  720. function formatSeconds(s){
  721. var minutes = Math.floor(s/60)%60;
  722. var hours = Math.floor(s/(60*60))%24;
  723. var days = Math.floor(s/(60*60*24));
  724. var seconds = s%60;
  725.  
  726. return '{0}d {1}h {2}m {3}s'.format(days, hours, minutes, seconds);
  727. }
  728.  
  729. function formatNumber(n){
  730. return n.toString().replace(/(\d)(?=(\d\d\d)+(?!\d))/g, "$1,");
  731. }
  732.  
  733. function formatMoney(m){
  734. return '$'+formatNumber(m);
  735. }
  736.  
  737. // Taken from: http://stackoverflow.com/a/15724300/1832471
  738. function getCookieValue(name) {
  739. var nameEQ = name + "=";
  740. var ca = document.cookie.split(';');
  741. for(var i=0;i < ca.length;i++) {
  742. var c = ca[i];
  743. while (c.charAt(0)==' ') c = c.substring(1,c.length);
  744. if (c.indexOf(nameEQ) === 0) return c.substring(nameEQ.length,c.length);
  745. }
  746. return null;
  747. }
  748.  
  749. // Taken from: http://stackoverflow.com/a/901144/1832471
  750. function getParameterByName(name, url) {
  751. if (!url) {
  752. url = window.location.href;
  753. }
  754. name = name.replace(/[\[\]]/g, "\\$&");
  755. var regex = new RegExp("[?&]" + name + "(=([^&#]*)|&|#|$)"),
  756. results = regex.exec(url);
  757. if (!results) return null;
  758. if (!results[2]) return '';
  759. return decodeURIComponent(results[2].replace(/\+/g, " "));
  760. }
  761.  
  762.  
  763.  
  764.  
  765.  
  766.  
  767.  
  768.  
  769.  
  770.  
  771.  
  772.  
  773.  
  774.  
  775.  
  776.