Torn Helper

Adds extra information to different pages all around Torn.

目前為 2017-01-15 提交的版本,檢視 最新版本

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