WoTStatScript - Clanpage

More info for World of Tanks clan page.

当前为 2016-08-16 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name WoTStatScript - Clanpage
  3. // @version 0.9.15.2.3
  4. // @description More info for World of Tanks clan page.
  5. // @author Orrie
  6. // @namespace http://forum.worldoftanks.eu/index.php?/topic/263423-
  7. // @icon http://dl.dropboxusercontent.com/u/12497046/wot/projects/statscript/img/icon.png
  8. // @include http*://*.wargaming.net/clans/*/*
  9. // @grant GM_xmlhttpRequest
  10. // @connect www.wnefficiency.net
  11. // @connect api.worldoftanks.eu
  12. // @connect api.worldoftanks.ru
  13. // @connect api.worldoftanks.com
  14. // @connect api.worldoftanks.asia
  15. // @connect api.worldoftanks.kr
  16. // @connect eu.wargaming.net
  17. // @require https://gf.qytechs.cn/scripts/18946-tablesort/code/Tablesort.js?version=120660
  18. // @license MIT License
  19. // ==/UserScript==
  20. (function() {
  21. // global vars
  22. var d = document, c = d.cookie;
  23.  
  24. // new page handler making sure it has loaded
  25. var pageLook = new MutationObserver(function() {
  26. pageLook.disconnect();
  27. // get server info and webpage
  28. var wg = {host:d.location.host, href:d.location.href, clan:{}};
  29. wg.srv = wg.host.match(/(eu|ru|na|asia|kr)/)[0];
  30. wg.m = (/players/i.test(wg.href) || /players\/wot/i.test(wg.href)) && !/wowp/i.test(wg.href);
  31. wg.sb = /clan_battles/i.test(wg.href);
  32.  
  33. // getting claninfo
  34. wg.clan.id = wg.href.match(/\/(\d+)/)[1];
  35. wg.p = /clan \|/i.test(d.title) && !/wowp/i.test(wg.href);
  36.  
  37. // script variables
  38. var sc = {
  39. vers: ((GM_info) ? GM_info.script.version : ""),
  40. host: "http://gf.qytechs.cn/en/scripts/12137-wotstatscript-clans",
  41. user: {
  42. wl: "http://forum.wotlabs.net/index.php?/user/1618-orrie/",
  43. wot: "http://worldoftanks.eu/community/accounts/505838943-Orrie/"
  44. },
  45. top: {
  46. eu: "http://forum.worldoftanks.eu/index.php?showtopic=263423",
  47. na: "http://forum.worldoftanks.com/index.php?showtopic=404652"
  48. },
  49. cred: { // translators
  50. cs: "<tr><td><a class='b-orange-arrow' href='http://worldoftanks.eu/community/accounts/500744969/'>Crabtr33</a></td><td><a class='b-orange-arrow' href='http://worldoftanks.eu/community/accounts/508323506/'>Ragnarocek</a></td></tr><tr><td><a class='b-orange-arrow' href='http://worldoftanks.eu/community/accounts/508904714/'>jViks</a></td></tr>" ,
  51. de: "<tr><td><a class='b-orange-arrow' href='http://worldoftanks.eu/community/accounts/504873051/'>ArtiOpa</a></td><td><a class='b-orange-arrow' href='http://worldoftanks.eu/community/accounts/501118529/'>Crakker</a></td></tr><tr><td><a class='b-orange-arrow' href='http://worldoftanks.eu/community/accounts/501072645/'>multimill</a></td><td><a class='b-orange-arrow' href='http://worldoftanks.eu/community/accounts/500373105/'>coolathlon</a></td></tr>",
  52. fr: "<tr><td><a class='b-orange-arrow' href='http://worldoftanks.eu/community/accounts/506641783/'>SuperPommeDeTerre</a></td></tr>",
  53. pl: "<tr><td><a class='b-orange-arrow' href='http://worldoftanks.eu/community/accounts/501801562/'>KeluMocy</a></td><td><a class='b-orange-arrow' href='http://worldoftanks.eu/community/accounts/504412736/'>pokapokami</a></td></tr>",
  54. es: "<tr><td><a class='b-orange-arrow' href='http://worldoftanks.eu/community/accounts/512759883/'>Frodo45127</a></td></tr>",
  55. tr: "<tr><td><a class='b-orange-arrow' href='http://worldoftanks.eu/community/accounts/500400806/'>Ufuko</a></td></tr>",
  56. ru: "<tr><td><a class='b-orange-arrow' href='http://worldoftanks.ru/community/accounts/291063/'>Bananium</a></td><td>dimon222</td></tr>"
  57. },
  58. api: {
  59. wg_key: "a7595640a90bf2d19065f3f2683b171c"
  60. },
  61. srv: {
  62. wl: false, // wotlabs
  63. nm: false, // noobmeter
  64. vb: false, // vbaddict
  65. ws: false, // wotstats
  66. cs: false, // wotcs
  67. wlf: false, // wot-life
  68. ct: false, // clan tools
  69. kttc: false, // kttc
  70. wots: false, // wots
  71. ch: false, // clan history
  72. wr: false, // wotreplays
  73. we: false // wot event stats
  74. },
  75. wn: "http://www.wnefficiency.net/exp/expected_tank_values_latest.json",
  76. loc: {
  77. cur: (c.match(/django_language=(\w+)/)) ? c.match(/django_language=(\w+)/)[1] : "en",
  78. sup: ["en", "ru", "cs", "de", "fr", "pl", "es", "tr"],
  79. miss: 0
  80. },
  81. date: {
  82. raw: new Date(),
  83. now: Date.now(),
  84. format: {ru: "ru-RU", eu: "en-GB", na: "en-US", asia: "en-AU", kr: "ko-KR"}
  85. },
  86. col: { // colour scale array
  87. // col wr bat sr hr dmg wgr wn8 wn7 eff nm
  88. sUni: [ "#5A3175", 65, 30000, 50, 80, 300, 9900, 2900, 2050, 2050, 2000 ], // 99.99% super unicum
  89. uni: [ "#83579D", 60, 25000, 46, 75, 270, 9000, 2450, 1850, 1800, 1950 ], // 99.90% unicum
  90. gr8: [ "#3972C6", 56, 21000, 42, 70, 240, 8500, 2000, 1550, 1500, 1750 ], // 99.00% great
  91. vGud: [ "#4099BF", 54, 17000, 38, 65, 210, 6500, 1600, 1350 ], // 95.00% very good
  92. good: [ "#4D7326", 52, 13000, 34, 60, 180, 5000, 1200, 1100, 1200, 1450 ], // 82.00% good
  93. aAvg: [ "#849B24", 50, 10000, 30, 55, 150, 4000, 900 ], // 63.00% above average
  94. avg: [ "#CCB800", 48, 7000, 25, 50, 120, 3000, 650, 900, 900, 1250 ], // 40.00% average
  95. bAvg: [ "#CC7A00", 47, 3000, 20, 45, 90, 2000, 450, 700, 600, 1150 ], // 20.00% below average
  96. bas: [ "#CD3333", 46, 1000, 15, 40, 60, 1500, 300, 500 ], // 6.00% basic
  97. beg: [ "#930D0D", 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 ], // 0.00% beginner
  98. dft: [ "#6B6B6B" ], // default
  99. id: { "col": 0, "wr": 1, "bat": 2, "sr": 3, "hr": 4, "dmg": 5, "wgr": 6, "wn8": 7, "wn7": 8, "eff": 9, "nm": 10 } // type identifier
  100. },
  101. web: {
  102. gecko: typeof InstallTrigger !== 'undefined',
  103. opera: !!window.opera || /opera|opr/i.test(navigator.userAgent),
  104. chrome: !!window.chrome && !!window.chrome.webstore,
  105. safari: /constructor/i.test(window.HTMLElement)
  106. },
  107. debug: true
  108. },
  109. // battle scheduler variables
  110. bs = {
  111. cw: {
  112. status: false,
  113. event: false,
  114. fame: 400,
  115. tier: "Ɵ",
  116. elo: "Ɵ",
  117. bats: "Ɵ",
  118. current: 0
  119. },
  120. dyn: {
  121. conc: [],
  122. check: 0,
  123. gold: 0
  124. },
  125. clan: {
  126. id: wg.clan.id,
  127. tag: "Ɵ",
  128. emblem: "Ɵ",
  129. color: "Ɵ"
  130. },
  131. time: {
  132. h: sc.date.raw.getHours(),
  133. m: sc.date.raw.getMinutes(),
  134. o: ((sc.date.raw.getTimezoneOffset() > 0) ? -Math.abs(sc.date.raw.getTimezoneOffset()) : Math.abs(sc.date.raw.getTimezoneOffset()))/60
  135. },
  136. table: {
  137. eu: [17, 18, 19, 20, 21, 22, 23],
  138. na: [23, 0, 1, 2, 3, 4, 5, 6],
  139. ru: [9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23],
  140. asia: [8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18],
  141. c: "",
  142. s: []
  143. }
  144. };
  145. bs.api = {
  146. clan: "https://"+wg.srv+".wargaming.net/globalmap/game_api/clan/"+bs.clan.id+"/battles",
  147. divs: "https://"+wg.srv+".wargaming.net/globalmap/game_api/wot/clan_tactical_data",
  148. map: "https://"+wg.srv+".wargaming.net/globalmap/game_api/map_fill_info?aliases=",
  149. tourney: "https://"+wg.srv+".wargaming.net/globalmap/game_api/tournament_info?alias=",
  150. prov: "https://"+wg.srv+".wargaming.net/globalmap/game_api/province_info?alias=",
  151. event: "https://api.worldoftanks."+((wg.srv == "na") ? "com" : wg.srv)+"/wot/globalmap/events/?application_id="+sc.api.wg_key+"&limit=1",
  152. provs: "https://api.worldoftanks."+((wg.srv == "na") ? "com" : wg.srv)+"/wot/globalmap/clanprovinces/?application_id="+sc.api.wg_key+"&clan_id="+bs.clan.id
  153. };
  154. bs.time.r = ((bs.time.m >= 15 && bs.time.m <= 45) ? [bs.time.h,"30"] : ((bs.time.m <= 15) ? [bs.time.h,"00"] : [(bs.time.h+1),"00"]));
  155. bs.time.t = bs.time.r[0]+"_"+bs.time.r[1];
  156.  
  157. // script functions
  158. var sf = {
  159. tableFetch: function () {
  160. // show animated loading gear
  161. loadGif.classList.remove("js-hidden");
  162. // find required info from table player rows
  163. for (var _rt=0, _rt_len = memObj.cls.children.length; _rt<_rt_len; _rt++) {
  164. var row = memObj.cls.children[_rt];
  165. if (!row.classList.contains("tbl-rating_tr__card")) {
  166. var id = row.getAttribute('data-account_id'),
  167. name = row.getElementsByClassName('player_name')[0].innerHTML,
  168. role = row.getElementsByClassName('player_rank')[0].innerHTML;
  169. if (!isNaN(id)) {
  170. memObj.ids.push(id);
  171. ss.user[id] = {u:{name:name,id:id,role:role}, v:{frag:0,dmg:0,spot:0,def:0,win:0}, wn8:""};
  172. }
  173. }
  174. }
  175. ss.clan = {name:(wg.clan.name) ? (wg.clan.name) : "???", wn8:0, win:0, mem: _rt_len};
  176. // request and retrieve statistics from API
  177. if (ss.clan.mem > 0) {
  178. sf.request("infoData", sc.api.i+memObj.ids.join(','), sf.apiInfoHnd);
  179. }
  180. },
  181. apiInfoHnd: function (resp) { // processing information from player API
  182. var data = resp.data;
  183. for (var a in data) {
  184. if (data.hasOwnProperty(a)) {
  185. var pData = data[a];
  186. if (pData !== null) {
  187. // store stats
  188. var pDataStats = pData.statistics.all;
  189. ss.user[pData.account_id].u = {
  190. name: pData.nickname,
  191. id: pData.account_id,
  192. role: ss.user[pData.account_id].u.role,
  193. cid: pData.clan_id,
  194. bat: pDataStats.battles,
  195. win: (pDataStats.wins/pDataStats.battles)*100,
  196. dmg: pDataStats.damage_dealt/pDataStats.battles,
  197. frag: pDataStats.frags/pDataStats.battles,
  198. spot: pDataStats.spotted/pDataStats.battles,
  199. def: pDataStats.dropped_capture_points/pDataStats.battles,
  200. wgr: pData.global_rating,
  201. lng: pData.client_language
  202. };
  203. ss.clan.win += (!isNaN(ss.user[pData.account_id].u.win)) ? ss.user[pData.account_id].u.win : 0;
  204. }
  205. }
  206. }
  207. d.getElementById('js-wn8-status').textContent = "50%";
  208. sf.request("vehData", sc.api.v+memObj.ids.join(','), sf.apiVehHnd);
  209. },
  210. apiVehHnd: function (resp) { // processing information from vehicle API and calculate WN8
  211. var data = resp.data;
  212. for (var p in data) {
  213. if (data.hasOwnProperty(p)) {
  214. var vData = data[p];
  215. if (vData !== null) {
  216. var rWin, rDmg, rFrag, rSpot, rDef, wn8 = 0, battles = 0,
  217. userStat = ss.user[p];
  218. if (userStat.u.bat > 0) {
  219. for (var v in vData) {
  220. if (vData.hasOwnProperty(v)) {
  221. // go through each vehicle to get expected stats
  222. for (var _so=0, _so_len = wn.stat.length; _so<_so_len; _so++) {
  223. if (wn.stat[_so].IDNum == vData[v].tank_id) {
  224. var vehStat = wn.stat[_so],
  225. dataBattles = vData[v].statistics.battles;
  226. userStat.v.frag += vehStat.expFrag * dataBattles;
  227. userStat.v.dmg += vehStat.expDamage * dataBattles;
  228. userStat.v.spot += vehStat.expSpot * dataBattles;
  229. userStat.v.def += vehStat.expDef * dataBattles;
  230. userStat.v.win += vehStat.expWinRate * dataBattles;
  231. battles += dataBattles;
  232. break;
  233. }
  234. }
  235. }
  236. }
  237. // start calculating wn8
  238. rWin = Math.max(((userStat.u.win /(userStat.v.win /battles)-0.71)/(1-0.71)),0);
  239. rDmg = Math.max(((userStat.u.dmg /(userStat.v.dmg /battles)-0.22)/(1-0.22)),0);
  240. rFrag = Math.max(Math.min(rDmg+0.2,((userStat.u.frag/(userStat.v.frag/battles)-0.12)/(1-0.12))),0);
  241. rSpot = Math.max(Math.min(rDmg+0.1,((userStat.u.spot/(userStat.v.spot/battles)-0.38)/(1-0.38))),0);
  242. rDef = Math.max(Math.min(rDmg+0.1,((userStat.u.def /(userStat.v.def /battles)-0.10)/(1-0.10))),0);
  243. wn8 = 980*rDmg + 210*rDmg*rFrag + 155*rFrag*rSpot + 75*rDef*rFrag + 145*Math.min(1.8,rWin);
  244. }
  245. // store wn8 and add to clan total
  246. userStat.wn8 = wn8;
  247. ss.clan.wn8 += wn8;
  248. }
  249. }
  250. }
  251. // calculate average wn8 / winrate and store everything in localStorage, then reload page
  252. ss.clan.wn8 = ss.clan.wn8/ss.clan.mem;
  253. ss.clan.win = ss.clan.win/ss.clan.mem;
  254. sf.storage("statScriptValues_"+wg.clan.id, {clan: ss.clan, user: ss.user}, "set", "string");
  255. sf.storage("statScriptDate_"+wg.clan.id, sc.date.now, "set");
  256. d.getElementById('js-wn8-status').textContent = "100%";
  257. location.reload();
  258. },
  259. apiBanHnd: function (resp) { // processing information from banned API
  260. var data;
  261. if (!memObj.bans.api) {
  262. data = resp.data;
  263. memObj.bans.api = data;
  264. memObj.bans.f = true;
  265. }
  266. else {
  267. data = memObj.bans.api;
  268. }
  269. for (var a in data) {
  270. if (data.hasOwnProperty(a)) {
  271. var bData = data[a];
  272. var memClass = "js-tooltip-id_js-playerslist-account-name-tooltip-"+a,
  273. memCell = d.getElementsByClassName(memClass)[0].parentNode;
  274. if (bData.ban_time !== null) {
  275. var banTime = (bData.ban_time > 0) ? new Date(bData.ban_time*1000).toLocaleString(sc.date.format[wg.srv]) : loc[18];
  276. memCell.appendChild(sf.elem("p", "player_time", banTime));
  277. }
  278. else {
  279. memCell.appendChild(sf.elem("p", "player_time", loc[19]));
  280. }
  281. }
  282. }
  283. },
  284. clan: function () { // clan links
  285. // look for potential elements with clan name
  286. var emblems = [
  287. d.getElementsByClassName('page-header_emblem')[0], // first method
  288. d.getElementsByClassName('clan_name')[0], // second method
  289. d.getElementsByClassName('js-clan-name')[0] // third method - your own clan
  290. ];
  291. wg.clan.name = (emblems[0] || emblems[1]) ? ((emblems[1]) ? emblems[1].firstElementChild.innerHTML.replace(/[\[\]]/g,"") : emblems[0].alt) : (emblems[2]) ? emblems[2].innerHTML : false;
  292. // create the stat site menu
  293. var menu_class = d.getElementsByClassName('menu-top')[0],
  294. clanMenu_div = sf.elem("div", "menu-clan_links menu-top_item", "<span class='menu-top_link'>"+loc[1]+"<span class='cm-arrow'></span></span>"),
  295. clanMenu_list = sf.elem("ul", "clan-links cm-sublist", ""),
  296. clanMenu_list_items = [
  297. [sc.srv.wl, "<a target='_blank' href='http://wotlabs.net/"+sc.srv.wl+"/clan/"+wg.clan.name+"'><span class='sl-icon sl-wl'></span>WoTLabs</a>"],
  298. [sc.srv.nm, "<a target='_blank' href='http://noobmeter.com/clan/"+sc.srv.nm+"/"+wg.clan.name+"/"+wg.clan.id+"'><span class='sl-icon sl-nm'></span>Noobmeter</a>"],
  299. [sc.srv.vb, "<a target='_blank' href='http://vbaddict.net/clan/worldoftanks."+sc.srv.vb+"/"+wg.clan.id+"/clan-"+wg.clan.name.toLowerCase()+"'><span class='sl-icon sl-vb'></span>vBAddict</a>"],
  300. [sc.srv.ct, "<a target='_blank' href='http://clantools.us/servers/"+sc.srv.ct+"/clans?id="+wg.clan.id+"'><span class='sl-icon sl-ct'></span>Clan Tools</a>"],
  301. [sc.srv.cs, "<a target='_blank' href='http://wotcs.com/clan.php?wid="+wg.clan.id+"'><span class='sl-icon sl-cs'></span>WoT-CS</a>"],
  302. [sc.srv.kttc, "<a target='_blank' href='http://"+((wg.srv=="ru") ? "" : sc.srv.kttc+".")+"kttc.ru/clan/"+wg.clan.id+"/'><span class='sl-icon sl-kttc'></span>KTTC</a>"],
  303. [sc.srv.wlf, "<a target='_blank' href='http://en.wot-life.com/"+sc.srv.wlf+"/clan/"+wg.clan.name+"-"+wg.clan.id+"/'><span class='sl-icon sl-wlife'></span>WoT-Life</a>"],
  304. [sc.srv.we, "<a target='_blank' href='http://wotevent.guildity.com/clans/"+wg.clan.name+"/'><span class='sl-icon sl-we'></span>WoT Event Stats</a>"],
  305. [sc.srv.wr, "<a target='_blank' href='http://wotreplays."+sc.srv.wr+"/clan/"+wg.clan.name+"'><span class='sl-icon sl-wr'></span>WoTReplays</a>"]
  306. ];
  307. sf.links(clanMenu_list, clanMenu_list_items);
  308. clanMenu_div.firstElementChild.addEventListener('click', function() {this.classList.toggle('cm-parent-link__opened'); this.nextSibling.classList.toggle('cm-sublist__opened');}, false);
  309. clanMenu_div.appendChild(clanMenu_list);
  310. menu_class.appendChild(clanMenu_div);
  311. },
  312. clanInsert: function (mode) { // overall clan stat insertion
  313. switch(mode) {
  314. case ("main"):
  315. var clanProfileValue = d.getElementsByClassName('rating-profile_item');
  316. if (ss.clan.win) {
  317. clanProfileValue[1].innerHTML = "<i class='rating-profile_icon i i__rating-common i__wot-victories'></i><span class='rating-profile_value rating-players_stats js-format-number'>"+sf.color(ss.clan.win,"wr",2,"%")+"</span><span class='rating-profile_key'>"+loc[8]+"</span>";
  318. }
  319. if (ss.clan.wn8) {
  320. clanProfileValue[3].innerHTML = "<i class='rating-profile_icon i i__rating-common i__wot-experience'></i><span class='rating-profile_value rating-players_stats js-format-number'>"+sf.color(ss.clan.wn8,"wn8",0)+"</span><span class='rating-profile_key'>"+loc[9]+"</span>";
  321. }
  322. break;
  323. case ("list"):
  324. var clanPlayersValue = d.getElementsByClassName('rating-players')[0].rows[0];
  325. clanPlayersValue.cells[1].getElementsByClassName('rating-players_key')[0].textContent = loc[8];
  326. if (statsInsertionStatus === false && ss.clan) {
  327. if (ss.clan.win) {
  328. var clanWinCell = clanPlayersValue.insertCell(2);
  329. clanWinCell.className = "rating-players_item rating-players_item__data";
  330. clanWinCell.innerHTML = "<i class='rating-players_icon i i__rating-common i__wot-victories'></i><span class='rating-players_value rating-players_stats'>"+sf.color(ss.clan.win,"wr",2,"%")+"</span><span class='rating-players_key'>"+loc[10]+"</span>";
  331. }
  332. if (ss.clan.wn8) {
  333. var clanWn8Cell = clanPlayersValue.insertCell(4);
  334. clanWn8Cell.className = "rating-players_item rating-players_item__data";
  335. clanWn8Cell.innerHTML = "<i class='rating-players_icon i i__rating-common i__wot-experience'></i><span class='rating-players_value rating-players_stats'>"+sf.color(ss.clan.wn8,"wn8",0)+"</span><span class='rating-players_key'>"+loc[11]+"</span>";
  336. }
  337. }
  338. break;
  339. default:
  340. break;
  341. }
  342. },
  343. ratInsert: function () { //memberlist stat insertion
  344. // add container for member counters
  345. var pageHeader = d.getElementsByClassName('page-header')[0],
  346. memInfo_div = sf.elem("div", "page-header_meminfo", "");
  347. pageHeader.appendChild(memInfo_div);
  348. var newMem = 0, banMem = d.getElementsByClassName('tbl-rating_tr__state-banned').length;
  349. // add a counter for amount of banned people in clan
  350. if (banMem > 0) {
  351. var banMem_span = d.getElementsByClassName('page-header_ban')[0];
  352. if (!banMem_span) {
  353. banMem_span = sf.elem("span", "page-header_ban", loc[17]+" "+banMem);
  354. memInfo_div.appendChild(banMem_span);
  355. }
  356. else {
  357. banMem_span.textContent = loc[17]+" "+banMem;
  358. }
  359. }
  360. // table header for wn8
  361. if (headerInsertionStatus === false && Object.keys(ss.clan).length !== 0) {
  362. var headName = d.getElementsByClassName('tbl-rating_th__basis')[0],
  363. wnHead = sf.elem("div", "tbl-rating_th tbl-rating_th__wn tbl-rating_th__right", "<a href='#' data-tooltip-content='WN8' class='sorter js-table-sorter js-simple-tooltip js-sort-field_wn8 js-tooltip'><i class='sorter_icon sorter_icon__mq-hidden i i__table-params i__wot-aeb'></i><span class='sorter_caption'>WN8</span></a>");
  364. headName.parentNode.insertBefore(wnHead, headName.nextSibling);
  365. headerInsertionStatus = true;
  366. }
  367. // add wn8 for each member and colorize stats
  368. var userCheck = Object.keys(ss.user).length !== 0;
  369. for (var _rt=0, _rt_len = memObj.cls.children.length; _rt<_rt_len; _rt++) {
  370. var row = memObj.cls.children[_rt];
  371. if (!row.classList.contains("tbl-rating_tr__card")) {
  372. var id = row.getAttribute('data-account_id'),
  373. memName = row.getElementsByClassName('tbl-rating_td__basis')[0],
  374. memWGR = row.getElementsByClassName('tbl-rating_value')[0],
  375. memWins = row.getElementsByClassName('tbl-rating_value')[2];
  376. if (userCheck) {
  377. var wnRow = sf.elem("div", "tbl-rating_td tbl-rating_td__right", "");
  378. memName.parentNode.insertBefore(wnRow, memName.nextSibling);
  379. if (ss.user[id]) {
  380. wnRow.innerHTML = "<span class='tbl-rating_value'>"+sf.color(ss.user[id].wn8,"wn8",0)+"</span>";
  381. }
  382. else {
  383. wnRow.innerHTML = "<span class='tbl-rating_value'>"+loc[15]+"</span>";
  384. newMem ++;
  385. }
  386. }
  387. if (memWGR.innerHTML !== "0" && memWGR.innerHTML !== "—") {
  388. memWGR.innerHTML = sf.color(sf.format(memWGR.innerHTML,1),"wgr",0);
  389. }
  390. if (memWins.innerHTML !== "0.00%" && memWins.innerHTML !== "—") {
  391. memWins.innerHTML = sf.color(sf.format(memWins.innerHTML.replace(/[,]/g,"."),3),"wr",2,"%");
  392. }
  393. }
  394. }
  395. // add a counter for new people in the clan, compared to store stats
  396. if (newMem > 0) {
  397. var newMem_span = d.getElementsByClassName('page-header_mem')[0];
  398. if (!newMem_span) {
  399. newMem_span = sf.elem("span", "page-header_mem", loc[16]+" "+newMem);
  400. memInfo_div.appendChild(newMem_span);
  401. }
  402. else {
  403. newMem_span.textContent = loc[16]+" "+newMem;
  404. }
  405. }
  406. // check for length on bans
  407. if (memObj.bans.f) {
  408. sf.apiBanHnd();
  409. }
  410. else {
  411. memObj.bans.cls = d.getElementsByClassName('js-banned');
  412. for (var _bm=0, _bm_len = memObj.bans.cls.length; _bm<_bm_len; _bm++) {
  413. var bannedId = memObj.bans.cls[_bm].dataset.account_id;
  414. memObj.bans.ids.push(bannedId);
  415. }
  416. if (memObj.bans.ids.length > 0) {
  417. sc.api.b = sc.api.i+memObj.bans.ids.join(',')+"&fields=ban_time";
  418. sf.request("banData", sc.api.b , sf.apiBanHnd);
  419. }
  420. }
  421. },
  422. format: function (input, type) { // input and output formatting
  423. var inputStr = input.toString();
  424. switch(type) {
  425. case (1): // input string into number
  426. return parseFloat(inputStr.replace(/[^\d]/g,""));
  427. case (2): // output number with locale symbol
  428. return inputStr.replace(/(\d)(?=(\d\d\d)+(?!\d))/g, "$1"+loc[0]);
  429. case (3): // input string into number - exclude dots
  430. return parseFloat(inputStr.replace(/[^\d\.]/g,""));
  431. default:
  432. console.error("Error filtering: ", input);
  433. return input;
  434. }
  435. },
  436. color: function (input, type, dec, sym) { // color formatting
  437. var color = sc.col.dft[0],
  438. output = input.toFixed(dec);
  439. if (input >= 1000) {
  440. output = sf.format(input.toFixed(dec),2);
  441. }
  442. for (var c in sc.col) {
  443. if (sc.col.hasOwnProperty(c)) {
  444. if (input >= sc.col[c][sc.col.id[type]]) {
  445. color = sc.col[c][0]; break;
  446. }
  447. }
  448. }
  449. if (loc[0] !== "," && dec !== 0) {
  450. output = output.replace(/\.(\d+)*$/g,",$1");
  451. }
  452. if (sym) {
  453. output += sym;
  454. }
  455. return "<font color='"+color+"'>"+output+"</font>";
  456. },
  457. elem: function (tag, name, html, type, src) { // element creation
  458. var element = d.createElement(tag);
  459. if (name) {
  460. element.className = name;
  461. }
  462. if (html) {
  463. if (/</.test(html)) {
  464. element.innerHTML = html;
  465. }
  466. else {
  467. element.textContent = html;
  468. }
  469. }
  470. if (type) {
  471. element.type = type;
  472. }
  473. if (src) {element.src = src;}
  474. return element;
  475. },
  476. settings: function (name, text, state, dftState, wlist) { // script menu handler
  477. var listItem = sf.elem("li", "b-settingItem "+name, ""),
  478. listItems = d.createDocumentFragment();
  479. if (name == "wnRefresh") {
  480. var refreshBtn = sf.elem("div", "b-settingParent", "<a>"+text+"</a>");
  481. listItem.classList.add("settingSeperator");
  482. refreshBtn.addEventListener('click', function() {localStorage.removeItem("wnExpValues"); location.reload();}, false);
  483. listItems.appendChild(refreshBtn);
  484. }
  485. else if (name == "cleanStorage") {
  486. var cleanBtn = sf.elem("div", "b-settingParent", "<a>"+text+"</a>");
  487. cleanBtn.addEventListener('click', function() {localStorage.clear(); location.reload();}, false);
  488. listItems.appendChild(cleanBtn);
  489. }
  490. else {
  491. var optCheckDiv = sf.elem("div", "b-checkbox", "<span class='b-checkbox_checker'></span>"),
  492. optLabel = sf.elem("label", "b-combobox-label", text),
  493. optCheck = sf.elem("input", "l-box", "", "checkbox");
  494. optLabel.htmlFor = name;
  495. optCheck.name = name;
  496. optCheck.id = name;
  497. if (state) {
  498. optCheckDiv.classList.add("b-checkbox__checked");
  499. optLabel.classList.add("b-combobox-label__checked");
  500. }
  501. optCheck.checked = (state !== undefined) ? state : dftState;
  502. optCheck.addEventListener('click', function() {
  503. sf.storage('statScript_' + this.name, this.checked, "set");
  504. d[this.name] = this.checked;
  505. this.parentNode.classList.toggle('b-checkbox__checked');
  506. this.parentNode.parentNode.classList.toggle('b-combobox-label__checked');
  507. return this.checked;
  508. }, false);
  509. d[optCheck.name] = optCheck.checked;
  510. optCheckDiv.insertBefore(optCheck, optCheckDiv.firstChild);
  511. optLabel.appendChild(optCheckDiv);
  512. listItems.appendChild(optLabel);
  513. if (name == "whitelist") {
  514. var optText = sf.elem("textarea", "l-textarea", "");
  515. optText.placeholder = "Add clanID seperated by comma without spaces: 500004502,500010805";
  516. if (wlist) {
  517. optText.value = wlist;
  518. }
  519. optText.addEventListener('input', function() {
  520. sf.storage('statScript_whitelist_list', optText.value.split(","), "set");
  521. }, false);
  522. listItems.appendChild(optText);
  523. }
  524. }
  525. listItem.appendChild(listItems);
  526. return listItem;
  527. },
  528. links: function (parent, links) { // links handler
  529. var uRows = d.createDocumentFragment();
  530. for (var _l=0, _l_len = links.length; _l<_l_len; ++_l) {
  531. if (links[_l] instanceof HTMLElement) {
  532. uRows.appendChild(links[_l]);
  533. }
  534. else {
  535. uRows.appendChild((links[_l][0] && links[_l][1]) ? sf.elem("li", "", links[_l][1]) : sf.elem("li", "statname", links[_l][0]));
  536. }
  537. }
  538. parent.appendChild(uRows);
  539. },
  540. storage: function (name, data, type, mode) { // localStorage handler
  541. var storage;
  542. switch(type) {
  543. case ("set"):
  544. if (mode == "string") {
  545. data = JSON.stringify(data);
  546. }
  547. storage = localStorage.setItem(name, data);
  548. break;
  549. case ("get"):
  550. storage = localStorage.getItem(name);
  551. if (mode == "parse") {
  552. storage = JSON.parse(storage);
  553. }
  554. break;
  555. default: break;
  556. }
  557. return storage;
  558. },
  559. wn: function (resp) { // wnefficiency handler
  560. sf.storage("wnExpValues", resp, "set", "string");
  561. sf.storage("wnExpDate", sc.date.now, "set");
  562. sf.storage("wnExpVers", [sc.vers, resp.header.version], "set", "string");
  563. location.reload();
  564. },
  565. // battle scheduler functions
  566. handlerEvent: function(data) { // event checker
  567. // check if active event exists
  568. var event = data.data[0];
  569. if (event.status == "ACTIVE" || !bs.cw.status) {
  570. if (event.status == "ACTIVE") {
  571. bs.cw.event = true;
  572. }
  573. sf.request("mainData", bs.api.clan, sf.handlerMain);
  574. }
  575. else {
  576. // empty table
  577. table.lastElementChild.innerHTML = "";
  578. table.lastElementChild.appendChild(sf.elem("tr", "t-cwText", "<td colspan='8'>"+loc[39]+"</td>"));
  579. clearInterval(updateInterval);
  580. }
  581. // insert update timestamp
  582. d.getElementById('js-batttleUpdate').textContent = new Date().toLocaleTimeString("en-GB");
  583. },
  584. handlerMain: function(data) { // data handler
  585. var battleProvinces = [], battle,
  586. battleFragment = d.createDocumentFragment();
  587. // store data
  588. bs.clan.tag = data.clan.tag;
  589. bs.clan.emblem = data.clan.emblem_url;
  590. bs.clan.color = data.clan.color;
  591. bs.cw.bats = data.clan.appointed_battles_count;
  592. bs.cw.current = data.battles.length;
  593. bs.cw.elo = {
  594. 6: data.clan.elo_rating_6,
  595. 8: data.clan.elo_rating_8,
  596. 10: data.clan.elo_rating_10
  597. };
  598. style.textContent += ".t-clantag {color: "+bs.clan.color+";}";
  599. // go through battles and planned battles
  600. for (var _b=0, _b_len = data.battles.length; _b<_b_len; _b++) {
  601. battle = data.battles[_b];
  602. if (battleProvinces.indexOf(battle.province_id) == -1) {
  603. battleProvinces.push(battle.province_id);
  604. battleFragment.appendChild(sf.elem("tr", "battle "+battle.province_id, "<td><a class='link link__external' target='_blank' href='https://"+wg.srv+".wargaming.net/globalmap/#province/"+battle.province_id+"'>"+battle.province_name+"<i class='link_icon i i__external-links i__regular'></i></a></td><td>"+sf.mapFix(battle.arena_name)+"</td><td></td><td></td><td></td><td></td><td></td><td id='"+battle.enemy.id+"'><a class='link link__external' target='_blank' href='http://"+wg.srv+".wargaming.net/clans/"+battle.enemy.id+"/globalmap'><span style='color: "+battle.enemy.color+";'>["+battle.enemy.tag+"]</span> <img src='"+battle.enemy.emblem_url+"'><i class='link_icon i i__external-links i__regular'></i></a></td><td class='t-battle'>Ɵ</td><td class='t-battle t-border'>Ɵ</td>"+bs.table.c));
  605. }
  606. }
  607. for (var _bp=0, _bp_len = data.planned_battles.length; _bp<_bp_len; _bp++) {
  608. battle = data.planned_battles[_bp];
  609. if (battleProvinces.indexOf(battle.province_id) == -1) {
  610. battleProvinces.push(battle.province_id);
  611. battleFragment.appendChild(sf.elem("tr", "battle planned "+battle.province_id, "<td><a class='link link__external' target='_blank' href='https://"+wg.srv+".wargaming.net/globalmap/#province/"+battle.province_id+"'>"+battle.province_name+"<i class='link_icon i i__external-links i__regular'></i></a></td><td>"+sf.mapFix(battle.arena_name)+"</td><td></td><td>"+((bs.cw.event) ? "" : battle.province_revenue)+"</td><td></td><td></td><td></td><td>"+loc[40]+"</td><td class='t-battle'>Ɵ</td><td class='t-battle t-border'>Ɵ</td>"+bs.table.c));
  612. }
  613. }
  614. // show foes and battle count if clan has any battles and remove loading indicator
  615. if (bs.cw.bats > 0) {
  616. style.textContent += ".t-battle {display: table-cell !important;}";
  617. table.lastElementChild.innerHTML = ""; // empty table
  618. }
  619. table.lastElementChild.appendChild(battleFragment);
  620. // insert battle count
  621. d.getElementById('js-battles').textContent = bs.cw.bats;
  622. // send request for detailed battle information
  623. sf.request("batsData", bs.api.map+battleProvinces.join("&aliases="), sf.handlerBats);
  624. },
  625. handlerBats: function(data) { // battles handler
  626. for (var _bd=0, _bd_len = data.data.length; _bd<_bd_len; _bd++) {
  627. var battle = data.data[_bd],
  628. battleRow = d.getElementsByClassName(battle.alias)[0],
  629. enemyID = battleRow.children[7].id,
  630. battleType = ((battle.owner_clan_id == bs.clan.id) ? loc[43] : ((battle.owner_clan_id == enemyID) ? loc[44] : loc[45])),
  631. primeTime = [sf.time(parseFloat(battle.primetime.match(/\d+/g)[0])), battle.primetime.match(/\d+/g)[1], parseFloat(battle.primetime.match(/\d+/g)[0])];
  632. // modify cells cells
  633. battleRow.children[2].textContent = primeTime[0]+":"+primeTime[1];
  634. battleRow.children[2].dataset.sort = primeTime[2];
  635. battleRow.children[3].innerHTML = (bs.cw.event) ? battle.display_fame_points : "<span class='gold'>"+battle.revenue+"<i class='i i__currencies i__gold'></i></span>";
  636. battleRow.children[5].textContent = battle.owner_clan_rating;
  637. battleRow.children[6].textContent = battleType;
  638. // get correct battle count and schedule
  639. sf.request("tourneyData", bs.api.tourney+battle.alias+"&round=1", sf.handlerTourney);
  640. }
  641. // send request for clan provinces
  642. sf.request("provData", bs.api.provs+"&round=1", sf.handlerProv);
  643. // refresh table
  644. sortTable.refresh();
  645. },
  646. handlerTourney: function(data) { // tournament handler
  647. var battleRow = d.getElementsByClassName(data.province_id)[0],
  648. primeTime = [sf.time(parseFloat(data.start_time.match(/\d+/g)[0])), data.start_time.match(/\d+/g)[1], parseFloat(data.start_time.match(/\d+/g)[0])],
  649. ownerClan = (data.owner) ? (data.owner.id == bs.clan.id) || false : false,
  650. cellOwnerTime = false,
  651. provFame = bs.cw.fame*((data.owner) ? (Math.floor(data.owner.occupy/24)+1) : 1),
  652. attackers = [data.pretenders, 0];
  653. bs.dyn.check ++;
  654. // check attackers
  655. if (data.is_superfinal) {
  656. attackers = 1;
  657. }
  658. else if (attackers[0]) {
  659. attackers = attackers[0].length;
  660. }
  661. else {
  662. attackers = attackers[1];
  663. for (var _bc=0, _bc_len = data.battles.length; _bc<_bc_len; _bc++) {
  664. attackers += ((data.battles[_bc].is_fake) ? 1 : 2);
  665. }
  666. }
  667. // find how many battles
  668. var battles = (attackers !== 0) ? Math.ceil(Math.log2(attackers))+1 : 0;
  669. // modify cells
  670. battleRow.children[1].textContent = sf.mapFix(data.arena_name);
  671. battleRow.children[2].textContent = primeTime[0]+":"+primeTime[1];
  672. battleRow.children[3].innerHTML = (bs.cw.event) ? provFame : "<span class='gold'>"+data.province_revenue+"<i class='i i__currencies i__gold'></i></span>";
  673. battleRow.children[4].innerHTML = (data.owner) ? "<a class='link link__external' target='_blank' href='http://"+wg.srv+".wargaming.net/clans/"+data.owner.id+"/globalmap'><span style='color: "+data.owner.color+";'>["+data.owner.tag+"]</span> <img src='"+data.owner.emblem_url+"'><i class='link_icon i i__external-links i__regular'></i></a>" : loc[47];
  674. if (ownerClan) {
  675. bs.dyn.gold += data.province_revenue;
  676. d.getElementById('js-gold').textContent = sf.format(bs.dyn.gold,2); // insert gold count
  677. if (bs.cw.tier !== "Ɵ") {
  678. battleRow.children[5].textContent = data.owner["elo_rating_"+bs.cw.tier];
  679. }
  680. }
  681. // only continue if there are any attackers
  682. if (attackers) {
  683. var emptyCells = ((primeTime[2]-bs.table[wg.srv][0])*2)+9,
  684. lastBattle = battles+emptyCells;
  685. battleRow.children[8].textContent = attackers;
  686. battleRow.children[9].textContent = battles;
  687. for (var _e=10; _e<battleRow.childElementCount; _e++) {
  688. var cell = battleRow.children[_e];
  689. if (_e > emptyCells && _e <= lastBattle) {
  690. var timeClass = "."+cell.classList.item(1),
  691. timePrevClass = "."+cell.previousElementSibling.classList.item(1);
  692. if (bs.table.s.indexOf(timeClass) == -1 || bs.table.s.indexOf(timePrevClass) == -1) {
  693. bs.table.s.push(timePrevClass, timeClass);
  694. if (_e == lastBattle) {
  695. bs.table.s.push(timePrevClass+" + th", timePrevClass+" + td", timeClass+" + th", timeClass+" + td");
  696. }
  697. }
  698. if (ownerClan && _e !== lastBattle) {
  699. cell.className += " t-noFight";
  700. }
  701. else {
  702. cell.className += " t-fight";
  703. if (bs.dyn.conc[_e]) {
  704. bs.dyn.conc[_e] ++;
  705. }
  706. else {
  707. bs.dyn.conc[_e] = 1;
  708. }
  709. }
  710. if (_e == lastBattle) {
  711. if (ownerClan) {
  712. cell.className += " js-last";
  713. }
  714. if (!data.owner) {
  715. cell.className += " t-noOwner";
  716. }
  717. cell.innerHTML = "♖";
  718. cellOwnerTime = [parseFloat(cell.classList.item(1).match(/\d+/g)[0]), parseFloat(cell.classList.item(1).match(/\d+/g)[1])];
  719. }
  720. else {
  721. cell.innerHTML = "&#9876;";
  722. }
  723. }
  724. }
  725. if (bs.dyn.check == bs.cw.bats) {
  726. d.getElementById('js-battlesConc').textContent = bs.dyn.conc.sort(function(a,b){return b-a;})[0];
  727. style.textContent += bs.table.s.join(", ")+" {display: table-cell !important;}";
  728. }
  729. // check if battle is planned or not started and change state to ongoing
  730. if (battleRow.children[6].innerHTML !== loc[45] && !/\[/i.test(battleRow.children[7].textContent) && new Date().getHours() >= primeTime[0]-1 && new Date().getHours() < cellOwnerTime[0]) {
  731. switch(battleRow.children[6].innerHTML) {
  732. case (loc[42]):
  733. battleRow.children[6].textContent = loc[43];
  734. battleRow.children[7].textContent = loc[41];
  735. break;
  736. case (loc[43]):
  737. battleRow.children[7].textContent = loc[41];
  738. break;
  739. default:
  740. break;
  741. }
  742. }
  743. // check if no opponent - free round
  744. var lastBattleRound = data.battles[data.battles.length-1];
  745. if (battleRow.children[7].innerHTML == loc[40] && lastBattleRound && lastBattleRound.is_fake && lastBattleRound.first_competitor.id == bs.clan.id) {
  746. battleRow.children[7].textContent = loc[46];
  747. battleRow.children[7].classList.add("t-bold");
  748. }
  749. }
  750. else {
  751. battleRow.children[8].textContent = "ERROR";
  752. battleRow.children[8].classList.add("t-error");
  753. }
  754. // refresh table
  755. battleRow.children[2].dataset.sort = primeTime[2]+"."+((battleRow.children[6].innerHTML == loc[45]) ? 0+""+battles : 1+""+cellOwnerTime[0]+""+cellOwnerTime[1]);
  756. sortTable.refresh();
  757. },
  758. handlerProv: function(data) { // provinces handler
  759. var provs = data.data[bs.clan.id],
  760. provFragment = d.createDocumentFragment();
  761. if (table.rows[1] && table.rows[1].classList.contains("t-cwText")) {
  762. table.lastElementChild.innerHTML = ""; // empty table
  763. }
  764. if (provs) {
  765. for (var _p=0, _p_len = provs.length; _p<_p_len; _p++) {
  766. var prov = provs[_p],
  767. battleRow = d.getElementsByClassName(prov.province_id)[0],
  768. primeTime = [sf.time(parseFloat(prov.prime_time.match(/\d+/g)[0])), prov.prime_time.match(/\d+/g)[1]],
  769. provFame = 400*(Math.floor(prov.turns_owned/24)+1);
  770. if (!battleRow) {
  771. provFragment.appendChild(sf.elem("tr", "province "+prov.province_id, "<td><a class='link link__external' target='_blank' href='https://"+wg.srv+".wargaming.net/globalmap/#province/"+prov.province_id+"'>"+prov.province_name+"<i class='link_icon i i__external-links i__regular'></i></a></td><td>"+sf.mapFix(prov.arena_name)+"</td><td data-sort='"+(primeTime[0]+10)+"'>"+primeTime[0]+":"+primeTime[1]+"</td><td>"+((bs.cw.event) ? provFame : "<span class='gold'>"+prov.daily_revenue+"<i class='i i__currencies i__gold'></i></span>")+"</td><td><a class='link link__external' target='_blank' href='http://"+wg.srv+".wargaming.net/clans/"+bs.clan.id+"/globalmap'><span class='t-clantag'>["+bs.clan.tag+"]</span> <img src='"+bs.clan.emblem+"'><i class='link_icon i i__external-links i__regular'></i></a></td><td>"+bs.cw.elo[prov.max_vehicle_level]+"</td><td>"+loc[43]+"</td><td>"+loc[48]+"</td><td class='t-battle' data-sort='99'>Ɵ</td><td class='t-battle t-border' data-sort='99'>Ɵ</td>"+bs.table.c));
  772. bs.dyn.gold += prov.daily_revenue;
  773. bs.cw.tier = prov.max_vehicle_level;
  774. }
  775. }
  776. // insert gold count
  777. d.getElementById('js-gold').textContent = sf.format(bs.dyn.gold,2);
  778. // send request for divisions
  779. sf.request("divsData", bs.api.divs, sf.handlerDivs);
  780. }
  781. else if (bs.cw.bats === 0) {
  782. provFragment.appendChild(sf.elem("tr", "t-cwText", "<td colspan='8'>"+loc[49]+"</td>"));
  783. }
  784. table.lastElementChild.appendChild(provFragment);
  785. // refresh table
  786. sortTable.refresh();
  787. },
  788. handlerDivs: function(data) { // divisions handler
  789. if (data.data[0].division.clan_id == bs.clan.id) {
  790. for (var _p=0, _p_len = data.data.length; _p<_p_len; _p++) {
  791. var div = data.data[_p],
  792. battleRow = d.getElementsByClassName(div.alias)[0];
  793. if (!div.division && battleRow.classList.contains('province')) {
  794. var defBattle = battleRow.getElementsByClassName("js-last")[0];
  795. battleRow.children[7].textContent = loc[50];
  796. battleRow.children[7].classList.add("t-bold");
  797. if (defBattle) {
  798. defBattle.classList.remove("t-fight");
  799. defBattle.classList.add("t-noFight");
  800. }
  801. }
  802. else {
  803. // sometimes future defenses wont show up in planned battles
  804. if (div.attackers.length > 0 && battleRow.classList.contains('province')) {
  805. table.lastElementChild.appendChild(sf.elem("tr", "planned "+div.alias, "<td><a class='link link__external' target='_blank' href='https://"+wg.srv+".wargaming.net/globalmap/#province/"+div.alias+"'>"+div.name+"<i class='link_icon i i__external-links i__regular'></i></a></td><td></td><td></td><td>"+battleRow.children[3].innerHTML+"</td><td></td><td></td><td></td><td>"+loc[42]+"</td>"));
  806. battleRow.parentNode.removeChild(battleRow);
  807. sf.request("tourneyData", bs.api.tourney+div.alias+"&round=1", sf.handlerTourney);
  808. // refresh table
  809. sortTable.refresh();
  810. }
  811. }
  812. }
  813. }
  814. else {
  815. d.getElementById('js-error').textContent = " • "+loc[51];
  816. }
  817. },
  818. handlerError: function(name, data) { // error handler
  819. console.error("errorData", name, data);
  820. switch(name) {
  821. case ("mainData"):
  822. table.lastElementChild.appendChild(sf.elem("tr", "t-cwText", "<td colspan='8'>"+loc[52]+"</td>"));
  823. break;
  824. case ("divsData"):
  825. d.getElementById('js-error').textContent = " • "+loc[51];
  826. break;
  827. default: break;
  828. }
  829. },
  830. time: function (hour, min, type) { // time converter
  831. var time = hour+bs.time.o;
  832. if (time >= 24) {
  833. time -= 24;
  834. }
  835. else if (time <= 0) {
  836. time += 24;
  837. }
  838. if (type == "s") {
  839. time = "t-"+time+"_"+min+((time === 0 && min == "00") ? " t-24_00" : "");
  840. }
  841. return time;
  842. },
  843. timer: function () { // timestamp handler
  844. var dateNow = new Date(),
  845. time = {
  846. h: sf.time(bs.table[wg.srv][0]-1)-dateNow.getHours(),
  847. m: 60-dateNow.getMinutes()-1,
  848. s: 60-dateNow.getSeconds()-1
  849. };
  850. var timeSpan = d.getElementById('js-timePrime');
  851. if (!bs.cw.event && bs.cw.status) {
  852. timeSpan.textContent = loc[53];
  853. timeSpan.classList.add("t-bold");
  854. clearInterval(timeInterval);
  855. }
  856. else if (time.h >= 0 && (time.s > 0 || time.m > 0)) {
  857. timeSpan.textContent = ((time.h > 0) ? time.h+" "+loc[55]+", " : "")+((time.m > 0) ? time.m+" "+loc[56]+", " : "")+time.s+" "+loc[57];
  858. }
  859. else if (time.h < 0) {
  860. if (bs.cw.bats == "Ɵ") {
  861. if (bs.cw.bats === 0) {
  862. timeSpan.textContent = loc[54];
  863. timeSpan.classList.add("t-bold");
  864. clearInterval(timeInterval);
  865. }
  866. }
  867. else {
  868. timeSpan.classList.add("h-shadow");
  869. timeSpan.textContent = loc[59];
  870. if (bs.cw.current > 0) {
  871. d.getElementById('js-provStatus').textContent = loc[61];
  872. }
  873. clearInterval(timeInterval);
  874. }
  875. }
  876. },
  877. mapFix: function(name) { // map name fixer
  878. var fixedNames = {
  879. "114_czech/name": "Pilsen"
  880. };
  881. return (fixedNames[name]) ? fixedNames[name] : name;
  882. },
  883. updater: function () { // updater handler
  884. var dateNow = new Date(),
  885. newDate = [dateNow.getHours(), dateNow.getMinutes()],
  886. newTime = ((newDate[1] >= 15 && newDate[1] <= 45) ? [newDate[0],"30"] : ((newDate[1] <= 15) ? [newDate[0],"00"] : [(newDate[0]+1),"00"]));
  887. if (bs.time.r[0] !== newTime[0] || bs.time.r[1] !== newTime[1]) {
  888. bs.time.r = newTime;
  889. bs.dyn = {conc:[],check:0,gold:0};
  890. bs.table.s = [];
  891. sf.request("mainData", bs.api.clan, sf.handlerMain);
  892. // insert update timestamp
  893. d.getElementById('js-batttleUpdate').textContent = new Date().toLocaleTimeString("en-GB");
  894. }
  895. },
  896. request: function (name, api, handler) { // request handler
  897. GM_xmlhttpRequest({
  898. method: "GET",
  899. url: api,
  900. headers: {
  901. "Accept": "application/json"
  902. },
  903. onload: function(resp) {
  904. var data = JSON.parse(resp.responseText);
  905. if (resp.status == 200) {
  906. if (sc.debug) {console.info(name, data);}
  907. handler(data);
  908. }
  909. else {
  910. sf.handlerError(name, resp);
  911. }
  912. },
  913. onerror: function(resp) {
  914. console.error("Error:", name, api, resp);
  915. }
  916. });
  917. }
  918. };
  919.  
  920. // api links without account id
  921. sc.api.i = "http://api.worldoftanks."+((wg.srv == "na") ? "com" : wg.srv)+"/wot/account/info/?application_id="+sc.api.wg_key+"&account_id=";
  922. sc.api.v = "http://api.worldoftanks."+((wg.srv == "na") ? "com" : wg.srv)+"/wot/account/tanks/?application_id="+sc.api.wg_key+"&account_id=";
  923.  
  924. // fetch wnefficiency values - check if array exists in localStorage, otherwise fetch and reload page
  925. var wn = {
  926. stat: [],
  927. values: sf.storage("wnExpValues", "", "get", "parse"),
  928. date: sf.storage("wnExpDate", "", "get", "parse")+12096e5 >= sc.date.now, // true if timestamp is less than 2 weeks old, refresh list if false.
  929. vers: sf.storage("wnExpVers", "", "get", "parse") || ""
  930. };
  931. if (wn.vers[0]==sc.vers && wn.values && wn.date) {
  932. wn.stat = wn.values.data;
  933. }
  934. else {
  935. sf.request("wnData", sc.wn, sf.wn);
  936. }
  937.  
  938. // fetch stored clanlist stats - check if array exists in localStorage, otherwise tag fetching to true
  939. var ss = {
  940. val: sf.storage("statScriptValues_"+wg.clan.id, "", "get", "parse"),
  941. date: sf.storage("statScriptDate_"+wg.clan.id, "", "get", "parse")+6048e5 >= sc.date.now, // true if timestamp is less than 1 weeks old, refresh list if false.
  942. clan: {},
  943. user: {},
  944. statFetch: false
  945. };
  946. if (ss.val && ss.date) {
  947. ss.clan = ss.val.clan;
  948. ss.user = ss.val.user;
  949. }
  950. else {
  951. ss.statFetch = true;
  952. }
  953.  
  954. // inserting style into head
  955. var style = sf.elem("style", "wotstatscript", "", "text/css");
  956. d.head.appendChild(style);
  957.  
  958. // localization
  959. // cz-czech - Crabtr33 and Ragnarocek
  960. // de-german - ArtiOpa, Crakker and multimill
  961. // fr-french - SuperPommeDeTerre
  962. // pl-polish - KeluMocy and pokapokami
  963. // es-spanish - Frodo45127
  964. // tr-turkish - Ufuko
  965. // ru-russian - dimon222
  966. var loc = [
  967. { i:0, en: ",", ru: " ", cs: " ", de: ".", fr: " ", pl: " ", es:" ", tr: "."}, // thousands separator
  968. { i:1, en: "Clan Stats", ru: "Статистика клана", cs: "Stat. klanu", de: "Clanstatistiken", fr: "Statistiques du clan", pl: "Statystyki klanu", es: "Estadísticas del clan", tr: "Klan İstatistikleri"},
  969. { i:2, en: "Replays:", ru: "Реплеи:", cs: "Záznamy:", de: "Replays", fr: "Replays:", pl: "Powtórki:", es: "Repeticiones:", tr: "Replayler"},
  970. { i:3, en: "Script Menu", ru: "Меню скрипта", cs: "Nastavení scriptu", de: "Script-Menü", fr: "Menu du script", pl: "Script Menu", es:"Script Menu", tr: "Script Menu"},
  971. { i:4, en: "Load Stats Automatically", ru: "Load Stats Automatically", cs: "Nahrát stat. automaticky", de: "Load Stats Automatically", fr: "Charger les statistiques automatiquement", pl: "Load Stats Automatically", es:"Load Stats Automatically", tr: "Load Stats Automatically"},
  972. { i:5, en: "Use Whitelist", ru: "Use Whitelist", cs: "Použi whitelist", de: "Use Whitelist", fr: "Utiliser la liste blanche", pl: "Use Whitelist", es:"Use Whitelist", tr: "Use Whitelist"},
  973. { i:6, en: "Refresh WN8 Table", ru: "Обновить таблицу WN8", cs: "Obnov WN8 Tabulku", de: "WN8-Tabelle neu laden", fr: "Rafraîchir la table WN8", pl: "Refresh WN8 Table", es: "Refresh WN8 Table", tr: "Refresh WN8 Table"},
  974. { i:7, en: "Clean Script Database", ru: "Clean Script Database", cs: "Vyčisti db scriptu", de: "Clean Script Database", fr: "Nettoyer la base de données du script", pl: "Clean Script Database", es: "Clean Script Database", tr: "Clean Script Database"},
  975. { i:8, en: "Average Winrate", ru: "Average Winrate", cs: "Průměrný winrate", de: "Average Winrate", fr: "Ratio de victoire moyen", pl: "Average Winrate", es:"Average Winrate", tr: "Average Winrate"},
  976. { i:9, en: "Average WN8", ru: "Average WN8", cs: "Průměrné WN8", de: "Average WN8", fr: "Average WN8", pl: "WN8 moyen", es:"Average WN8", tr: "Average WN8"},
  977. { i:10, en: "Overall Average Winrate", ru: "Overall Average Winrate", cs: "Průměrný winrate", de: "Overall Average Winrate", fr: "Overall Average Winrate", pl: "Overall Average Winrate", es:"Overall Average Winrate", tr: "Overall Average Winrate"},
  978. { i:11, en: "Overall Average WN8", ru: "Overall Average WN8", cs: "Overall Average WN8", de: "Overall Average WN8", fr: "Overall Average WN8", pl: "Overall Average WN8", es:"Overall Average WN8", tr: "Overall Average WN8"},
  979. { i:12, en: "Fetch WN8 for Clan", ru: "Fetch WN8 for Clan", cs: "Obnov WN8 pre klan", de: "Fetch WN8 for Clan", fr: "Calculer le WN8 pour le clan", pl: "Fetch WN8 for Clan", es:"Fetch WN8 for Clan", tr: "Fetch WN8 for Clan"},
  980. { i:13, en: "Fetching WN8 for Clan!", ru: "Fetching WN8 for Clan!", cs: "Obnovuju WN8 pro klan!", de: "Fetching WN8 for Clan!", fr: "Walcul du WN8 pour le clan !", pl: "Fetching WN8 for Clan!", es:"Fetching WN8 for Clan!", tr: "Fetching WN8 for Clan!"},
  981. { i:14, en: "WN8 Fetched for Clan!", ru: "WN8 Fetched for Clan!", cs: "WN8 obnoveno pro klan!", de: "WN8 Fetched for Clan!", fr: "WN8 calculé pour le clan !", pl: "WN8 Fetched for Clan!", es:"WN8 Fetched for Clan!", tr: "WN8 Fetched for Clan!"},
  982. { i:15, en: "Not Found", ru: "Not Found", cs: "Nenalezeno", de: "Not Found", fr: "Non trouvé", pl: "Not Found", es:"Not Found", tr: "Not Found"},
  983. { i:16, en: "New Members:", ru: "New Members:", cs: "Noví členové:", de: "New Members:", fr: "Nouveaux membres :", pl: "New Members:", es:"New Members:", tr: "New Members:"},
  984. { i:17, en: "Banned Members:", ru: "Banned Members:", cs: "Noví členové:", de: "Banned Members:", fr: "Membres bannis:", pl: "Banned Members:", es:"Banned Members:", tr: "Banned Members:"},
  985. { i:18, en: "Currently Unavailable", ru: "Currently Unavailable", cs: "Currently Unavailable", de: "Currently Unavailable", fr: "Indisponible actuellement", pl: "Currently Unavailable", es:"Currently Unavailable", tr: "Currently Unavailable"},
  986. { i:19, en: "Ban ended, but no login", ru: "Ban ended, but no login", cs: "Ban ended, but no login", de: "Ban ended, but no login", fr: "Ban terminé, mais aucune connexion", pl: "Ban ended, but no login", es:"Ban ended, but no login", tr: "Ban ended, but no login"},
  987. { i:20, en: "Script Author:", ru: "Автор скрипта:", cs: "Autor skriptu:", de: "Script-Autor:", fr: "Auteur du script:", pl: "Script Author:", es:"Script Author:", tr: "Script Author:"},
  988. { i:21, en: "Contributors", ru: "Contributors", cs: "Kontributoři", de: "Contributors", fr: "Contributeurs", pl: "Contributors", es:"Contributors", tr: "Contributors"},
  989. { i:22, en: "Battle Schedule", ru: "Battle Schedule", cs: "Battle Schedule", de: "Battle Schedule", fr: "Battle Schedule", pl: "Battle Schedule", es:"Battle Schedule", tr: "Battle Schedule"},
  990. { i:23, en: "Clan Wars Countdown:", ru: "Clan Wars Countdown:", cs: "Clan Wars Countdown:", de: "Clan Wars Countdown:", fr: "Clan Wars Countdown:", pl: "Clan Wars Countdown:", es:"Clan Wars Countdown:", tr: "Clan Wars Countdown:"},
  991. { i:24, en: "Total Battles:", ru: "Total Battles:", cs: "Total Battles:", de: "Total Battles:", fr: "Total Battles:", pl: "Total Battles:", es:"Total Battles:", tr: "Total Battles:"},
  992. { i:25, en: "Potential Gold Income:", ru: "Potential Gold Income:", cs: "Potential Gold Income:", de: "Potential Gold Income:", fr: "Potential Gold Income:", pl: "Potential Gold Income:", es:"Potential Gold Income:", tr: "Potential Gold Income:"},
  993. { i:26, en: "Province", ru: "Провинция", cs: "Province", de: "Province", fr: "Province", pl: "Province", es:"Province", tr: "Province"},
  994. { i:27, en: "Map", ru: "Игровая карта", cs: "Map", de: "Map", fr: "Map", pl: "Map", es:"Map", tr: "Map"},
  995. { i:28, en: "Timezone", ru: "Timezone", cs: "Timezone", de: "Timezone", fr: "Timezone", pl: "Timezone", es:"Timezone", tr: "Timezone"},
  996. { i:29, en: "Fame", ru: "Fame", cs: "Fame", de: "Fame", fr: "Fame", pl: "Fame", es:"Fame", tr: "Fame"},
  997. { i:30, en: "Gold", ru: "Gold", cs: "Gold", de: "Gold", fr: "Gold", pl: "Gold", es:"Gold", tr: "Gold"},
  998. { i:31, en: "Owner", ru: "Owner", cs: "Owner", de: "Owner", fr: "Owner", pl: "Owner", es:"Owner", tr: "Owner"},
  999. { i:32, en: "ELO", ru: "ELO", cs: "ELO", de: "ELO", fr: "ELO", pl: "ELO", es:"ELO", tr: "ELO"},
  1000. { i:33, en: "Type", ru: "Type", cs: "Type", de: "Type", fr: "Type", pl: "Type", es:"Type", tr: "Type"},
  1001. { i:34, en: "Status", ru: "Status", cs: "Status", de: "Status", fr: "Status", pl: "Status", es:"Status", tr: "Status"},
  1002. { i:35, en: "Attackers", ru: "Attackers", cs: "Attackers", de: "Attackers", fr: "Attackers", pl: "Attackers", es:"Attackers", tr: "Attackers"},
  1003. { i:36, en: "Turns", ru: "Turns", cs: "Turns", de: "Turns", fr: "Turns", pl: "Turns", es:"Turns", tr: "Turns"},
  1004. { i:37, en: "Last Updated:", ru: "Last Updated:", cs: "Last Updated:", de: "Last Updated:", fr: "Last Updated:", pl: "Last Updated:", es:"Last Updated:", tr: "Last Updated:"},
  1005. { i:38, en: "Updating...", ru: "Updating...", cs: "Updating...", de: "Updating...", fr: "Updating...", pl: "Updating...", es:"Updating...", tr: "Updating..."},
  1006. { i:39, en: "See you next time.", ru: "See you next time.", cs: "See you next time.", de: "See you next time.", fr: "See you next time.", pl: "See you next time.", es:"See you next time.", tr: "See you next time."},
  1007. { i:40, en: "Not Started", ru: "Not Started", cs: "Not Started", de: "Not Started", fr: "Not Started", pl: "Not Started", es:"Not Started", tr: "Not Started"},
  1008. { i:41, en: "Ongoing", ru: "Ongoing", cs: "Ongoing", de: "Ongoing", fr: "Ongoing", pl: "Ongoing", es:"Ongoing", tr: "Ongoing"},
  1009. { i:42, en: "Planned", ru: "Planned", cs: "Planned", de: "Planned", fr: "Planned", pl: "Planned", es:"Planned", tr: "Planned"},
  1010. { i:43, en: "Defense", ru: "Defense", cs: "Defense", de: "Defense", fr: "Defense", pl: "Defense", es:"Defense", tr: "Defense"},
  1011. { i:44, en: "Owner", ru: "Owner", cs: "Owner", de: "Owner", fr: "Owner", pl: "Owner", es:"Owner", tr: "Owner"},
  1012. { i:45, en: "Attack", ru: "Attack", cs: "Attack", de: "Attack", fr: "Attack", pl: "Attack", es:"Attack", tr: "Attack"},
  1013. { i:46, en: "Free Round", ru: "Free Round", cs: "Free Round", de: "Free Round", fr: "Free Round", pl: "Free Round", es:"Free Round", tr: "Free Round"},
  1014. { i:47, en: "No Owner", ru: "No Owner", cs: "No Owner", de: "No Owner", fr: "No Owner", pl: "No Owner", es:"No Owner", tr: "No Owner"},
  1015. { i:48, en: "No Attacks", ru: "No Attacks", cs: "No Attacks", de: "No Attacks", fr: "No Attacks", pl: "No Attacks", es:"No Attacks", tr: "No Attacks"},
  1016. { i:49, en: "No Battles", ru: "No Battles", cs: "No Battles", de: "No Battles", fr: "No Battles", pl: "No Battles", es:"No Battles", tr: "No Battles"},
  1017. { i:50, en: "No Division", ru: "No Division", cs: "No Division", de: "No Division", fr: "No Division", pl: "No Division", es:"No Division", tr: "No Division"},
  1018. { i:51, en: "Division Data not Available!", ru: "Division Data not Available!", cs: "Division Data not Available!", de: "Division Data not Available!", fr: "Division Data not Available!", pl: "Division Data not Available!", es:"Division Data not Available!", tr: "Division Data not Available!"},
  1019. { i:52, en: "Clan ID Error", ru: "Clan ID Error", cs: "Clan ID Error", de: "Clan ID Error", fr: "Clan ID Error", pl: "Clan ID Error", es:"Clan ID Error", tr: "Clan ID Error"},
  1020. { i:53, en: "No Event Campaign", ru: "No Event Campaign", cs: "No Event Campaign", de: "No Event Campaign", fr: "No Event Campaign", pl: "No Event Campaign", es:"No Event Campaign", tr: "No Event Campaign"},
  1021. { i:54, en: "No Planned Battles", ru: "No Planned Battles", cs: "No Planned Battles", de: "No Planned Battles", fr: "No Planned Battles", pl: "No Planned Battles", es:"No Planned Battles", tr: "No Planned Battles"},
  1022. { i:55, en: "Hours", ru: "Hours", cs: "Hours", de: "Hours", fr: "Hours", pl: "Hours", es:"Hours", tr: "Hours"},
  1023. { i:56, en: "Mins", ru: "Mins", cs: "Mins", de: "Mins", fr: "Mins", pl: "Mins", es:"Mins", tr: "Mins"},
  1024. { i:57, en: "Secs", ru: "Secs", cs: "Secs", de: "Secs", fr: "Secs", pl: "Secs", es:"Secs", tr: "Secs"},
  1025. { i:58, en: "Event Only Schedule", ru: "Event Only Schedule", cs: "Event Only Schedule", de: "Event Only Schedule", fr: "Event Only Schedule", pl: "Event Only Schedule", es:"Event Only Schedule", tr: "Event Only Schedule"},
  1026. { i:59, en: "Currently Running", ru: "Currently Running", cs: "Currently Running", de: "Currently Running", fr: "Currently Running", pl: "Currently Running", es:"Currently Running", tr: "Currently Running"},
  1027. { i:60, en: "Concurrent Battles:", ru: "Concurrent Battles:", cs: "Concurrent Battles:", de: "Concurrent Battles:", fr: "Concurrent Battles:", pl: "Concurrent Battles:", es:"Concurrent Battles:", tr: "Concurrent Battles:"},
  1028. { i:61, en: "Next Opponent", ru: "Next Opponent", cs: "Next Opponent", de: "Next Opponent", fr: "Next Opponent", pl: "Next Opponent", es:"Next Opponent", tr: "Next Opponent"}
  1029. // {en: "", ru: "", cs: "", de: "", fr: "", pl: "", es:"", tr: ""}
  1030. ];
  1031.  
  1032. // region settings for external sites
  1033. switch(wg.srv) {
  1034. case ("eu"): // eu server
  1035. sc.srv.wl = sc.srv.nm = sc.srv.vb = sc.srv.ws = sc.srv.cs = sc.srv.wlf = sc.srv.ct = sc.srv.kttc = sc.srv.ch = sc.srv.wr = sc.srv.we = wg.srv;
  1036. break;
  1037. case ("ru"): // ru server
  1038. sc.srv.wl = sc.srv.nm = sc.srv.vb = sc.srv.ws = sc.srv.cs = sc.srv.ct = sc.srv.kttc = sc.srv.wots = sc.srv.ch = sc.srv.wr = wg.srv;
  1039. break;
  1040. case ("na"): // na server - american english
  1041. sc.srv.wl = sc.srv.nm = sc.srv.vb = sc.srv.ws = sc.srv.cs = sc.srv.wlf = sc.srv.ct = sc.srv.kttc = sc.srv.ch = wg.srv; sc.srv.wr = "com";
  1042. break;
  1043. case ("asia"): // asia server
  1044. sc.srv.wl = sc.srv.nm = sc.srv.vb = sc.srv.ws = sc.srv.cs = "sea"; sc.srv.ct = sc.srv.kttc = sc.srv.ch = wg.srv; sc.srv.wr = "com";
  1045. break;
  1046. case ("kr"): // korean server
  1047. sc.srv.wl = sc.srv.nm = sc.srv.vb = sc.srv.ws = sc.srv.cs = sc.srv.ct = sc.srv.ch = wg.srv; sc.srv.wr = "com";
  1048. break;
  1049. default: break;
  1050. }
  1051.  
  1052. // set script language to english if an unsupported language is detected
  1053. if (sc.loc.sup.indexOf(sc.loc.cur) == -1) {
  1054. sc.loc.cur = "en";
  1055. }
  1056.  
  1057. // process localization
  1058. for (var _l=0, l_len = loc.length; _l<l_len; _l++) {
  1059. var langLoc = loc[_l][sc.loc.cur];
  1060. if (sc.loc.cur !== "en" && langLoc == loc[_l].en && !loc[_l].f) {
  1061. sc.loc.miss ++;
  1062. console.info("Missing translation at line "+(_l+825)+" - en:\""+loc[_l].en+"\"", sc.loc.cur+":\""+loc[_l][sc.loc.cur]+"\"");
  1063. }
  1064. loc[_l] = langLoc;
  1065. }
  1066.  
  1067. // add language to body classname for language based styling
  1068. d.body.classList.add("lang-"+sc.loc.cur);
  1069.  
  1070. // variables for dropbox, css and data uri
  1071. var css = {
  1072. u: {
  1073. cIcons: "",
  1074. arrow: ""
  1075. },
  1076. i: {
  1077. arrow: "http://static-ptl-eu.gcdn.co/static/wot/common/css/scss/content/links/img/orange_arrow.png"
  1078. }
  1079. };
  1080.  
  1081. // style contents
  1082. var styleClan = [
  1083. // fix width for header with low resolution
  1084. ".search-form__place-header {width: 30%;}",
  1085. ".search-form__place-header.search-form__state-on {width: 410px;}",
  1086. // loading text
  1087. ".processing.wn8-loader span {margin: 25px 0px 0px -20px; text-align: center; width: 45px; position: absolute; top: 50%; left: 50%;}",
  1088. // links menu rules
  1089. ".menu-clan_links {padding: 0;}",
  1090. ".menu-clan_links.cm-parent-link__opened {border: 1px solid #313335;}",
  1091. ".menu-clan_links .menu-top_link {cursor: pointer; padding: 0 8px 0 9px;}",
  1092. ".menu-clan_links .menu-top_link.cm-parent-link__opened {background: #0E0E0E; border-left: 1px solid #313335; border-right: 1px solid #313335; margin-left: -1px;}",
  1093. ".menu-clan_links .cm-arrow {background-image: url('"+css.u.arrow+"'); display: inline-block; margin-left: 5px; opacity: 0.5; vertical-align: middle; transition: opacity 0.2s ease 0s; height: 4px; width: 7px;}",
  1094. ".menu-clan_links .cm-parent-link__opened .cm-arrow {opacity: 1; transform: rotate(180deg);}",
  1095. ".menu-clan_links .clan-links {background: rgba(14, 14, 14, 0.99); border: 1px solid #313335; display: none; box-shadow: 0px 0px 25px rgba(0, 0, 0, 0.4); margin-left: -1px; padding: 14px 16px; position: absolute;}",
  1096. ".menu-clan_links .cm-sublist__opened {display: block;}",
  1097. ".menu-clan_links .clan-links td {padding: 0 10px;}",
  1098. ".menu-clan_links .clan-links a {color: #E5E5E5; font-family: Arial,'Helvetica CY',Helvetica,sans-serif; font-size: 14px;}",
  1099. ".sl-icon {background: url('"+css.u.cIcons+"') no-repeat; display: inline-block; margin: -2px 8px 0px 0px; vertical-align: middle; height: 16px; width: 16px;}",
  1100. ".sl-wl {background-position: 0px 0px;}",
  1101. ".sl-nm {background-position: 0px -16px;}",
  1102. ".sl-ct {background-position: 0px -32px;}",
  1103. ".sl-cs {background-position: 0px -48px;}",
  1104. ".sl-kttc {background-position: 0px -64px;}",
  1105. ".sl-wlife {background-position: 0px -80px;}",
  1106. ".sl-as {background-position: 0px -96px;}",
  1107. ".sl-wr {background-position: 0px -112px;}",
  1108. ".sl-vb {background-position: 0px -128px;}",
  1109. ".sl-we {background-position: 0px -144px;}",
  1110. // rating profile rules
  1111. ".rating-profile {width: 70%; margin: 0px auto;}",
  1112. // settings menu rules
  1113. "#common_menu .menu-settings {color: #7C7E80; display: inline-block;}",
  1114. "#common_menu .menu-settings .cm-user-menu-link {margin: 0 10px 0 0;}",
  1115. "#common_menu .menu-settings .cm-user-menu-link_cutted-text {max-width: unset;}",
  1116. "#common_menu .menu-settings .cm-user-menu {min-width: 200px; padding: 15px;}",
  1117. "#common_menu .menu-settings .cm-parent-link:hover {cursor: pointer;}",
  1118. "#common_menu .menu-settings .b-settingItem {margin: 6px 0px; text-align: center;}",
  1119. "#common_menu .menu-settings label {display: table; line-height: normal; cursor: pointer; margin: 0 auto;}",
  1120. "#common_menu .menu-settings .l-box {display: none;}",
  1121. "#common_menu .menu-settings .b-checkbox {background-position: -4px -4px; height: 16px; width: 16px; float: left; margin-right: 5px;}",
  1122. "#common_menu .menu-settings .b-checkbox.b-checkbox__checked {background-position: -28px -4px;}",
  1123. "#common_menu .menu-settings .b-checkbox .b-checkbox_checker {background-position: -76px -4px; height: 16px; width: 16px;}",
  1124. "#common_menu .menu-settings .b-combobox-label__checked {color: #DCDCDC;}",
  1125. "#common_menu .menu-settings .b-settingItem .b-combobox-label:hover {color: #DCDCDC;}",
  1126. "#common_menu .menu-settings .b-settingItem .b-combobox-label:hover .b-checkbox {box-shadow: 0px 0px 10px 1px rgba(191, 166, 35, 0.15), 0px 0px 3px 1px rgba(191, 166, 35, 0.25);}",
  1127. "#common_menu .menu-settings textarea.l-textarea {background: rgba(255, 255, 255, 0.1); border: 1px solid rgba(255, 255, 255, 0.2); border-radius: 2px; color: #FFFFFF; line-height: normal; padding: 5px; min-height: 50px; margin: 5px 0 5px 0; min-width: 175px;}",
  1128. "#common_menu .menu-settings textarea::-webkit-input-placeholder {color: #FFFFFF;}",
  1129. "#common_menu .menu-settings textarea::-moz-placeholder {color: #FFFFFF;}",
  1130. "#common_menu .menu-settings .b-settingParent {line-height: 26px;}",
  1131. "#common_menu .menu-settings .b-settingParent a {cursor: pointer; color: #B1B2B3; text-shadow: 0px 1px 1px rgba(0, 0, 0, 0.5);}",
  1132. "#common_menu .menu-settings .b-settingParent a:hover {color: #FFFFFF; text-shadow: 0px 1px 1px rgba(0, 0, 0, 0.75); text-decoration: underline;}",
  1133. "#common_menu .menu-settings .settingCredits {margin: 2px 0px;}",
  1134. "#common_menu .menu-settings .settingCredits h1 {color: #B1B2B3;}",
  1135. "#common_menu .menu-settings .settingCredits table {font-size: 12px; margin: 0 auto; width: unset;}",
  1136. "#common_menu .menu-settings .settingCredits table td {padding: 0 5px;}",
  1137. "#common_menu .menu-settings .settingCredits p {font-size: 12px; padding: 2px 0;}",
  1138. "#common_menu .menu-settings .settingCredits .b-orange-arrow {background: url('"+css.i.arrow+"') 100% 0 no-repeat; color: #F25322; line-height: 14px; padding-right: 9px;}",
  1139. "#common_menu .menu-settings .settingCredits .b-orange-arrow:hover {color: #FF7432;}",
  1140. "#common_menu .menu-settings .settingCredits.settingSeperator {border-top: 1px dashed #212123; margin-top: 6px; padding-top: 12px;}",
  1141. "#common_menu .menu-settings .settingCredits.settingSeperator.b-wnRefresh {margin-top: 11px; padding-top: 6px;}",
  1142. "#common_menu .menu-settings .settingCredits.settingLinks a {margin: 0 5px;}",
  1143. // memberpage rules
  1144. ".page-header {padding: 30px 0 30px 75px}",
  1145. ".page-header_meminfo {display: table; margin: 0px auto; position: absolute; top: 3px; right: 0px; left: 0px; text-align: center;}",
  1146. ".page-header_meminfo span {margin: 0 5px;}",
  1147. ".page-header_ban {color: #E5B12E;}",
  1148. ".page-header_mem {color: #E5B12E;}",
  1149. ".js-page-header-view .page-header_mem {margin-left: 25px;}",
  1150. // button fetch rules
  1151. ".b-button-stats {border: 1px solid rgba(255, 255, 255, 0.2); border-radius: 2px; position: absolute; right: 0; top: 9px;}",
  1152. ".b-button-stats a {background: rgba(255, 255, 255, 0.1); color: #FFFFFF; cursor: pointer; font-size: 17px; line-height: 45px; display: block; padding: 0px 15px; transition: all 0.2s ease 0s;}",
  1153. ".b-button-stats a:hover {background: rgba(229, 177, 46, 0.25);}",
  1154. // rating players rules
  1155. ".rating-players {height: 200px;}",
  1156. ".rating-players tbody {width: 95%; display: table; margin: 0px auto;}",
  1157. ".rating-players_item__data {padding-top: 25px; width: 14%;}",
  1158. ".rating-players_item__average {padding-top: 10px; width: 16%;}",
  1159. ".rating-players_stats {font-size: 40px;}",
  1160. // membertable rules
  1161. ".js-expander-link-view {display: table; position: absolute; top: 333px;}",
  1162. "#js-playerslist-table {margin-top: 7px;}",
  1163. ".tbl-rating_th.tbl-rating_th__right a.js-sort-field_days {display: table; margin: 0 auto;}",
  1164. ".tbl-rating_th.tbl-rating_th__right a.js-sort-field_days span {line-height: 44px;}",
  1165. ".tbl-rating_td.tbl-rating_td__right span {display: table; margin: 0 auto;}",
  1166. ".player_time {position: absolute; right: 0; top: 18px;}"
  1167. ];
  1168. style.textContent = styleClan.join("");
  1169. // end style
  1170.  
  1171. // add animated loading icon for progress indication
  1172. var loadGif = sf.elem("div", "processing wn8-loader js-hidden", "<span id='js-wn8-status'></span>");
  1173. d.body.appendChild(loadGif);
  1174.  
  1175. // load and store settings
  1176. sc.set = {
  1177. onPageload: sf.storage("statScript_onPageload", "", "get", "parse"),
  1178. useWhitelist: sf.storage("statScript_whitelist", "", "get", "parse"),
  1179. listWhitelist: sf.storage("statScript_whitelist_list", "", "get"),
  1180. eventOnly: sf.storage("statScript_eventOnly", "", "get", "parse")
  1181. };
  1182. bs.cw.status = sc.set.eventOnly;
  1183.  
  1184. // script link and settings
  1185. var clanSet_div = sf.elem("div", "menu-settings menu-top_item", "<a class='cm-user-menu-link' href='#' onClick='return false;'><span class='cm-user-menu-link_cutted-text'>"+loc[3]+"</span><span class='cm-arrow'></span></span>"),
  1186. clanSet_list = sf.elem("ul", "cm-user-menu", ""),
  1187. clanSet_list_locItem = sf.elem("li", "b-settingItem settingCredits settingSeperator", ""),
  1188. whitelistArray = (sc.set.listWhitelist) ? sc.set.listWhitelist.split(",") : "",
  1189. clanSet_list_items = [
  1190. sf.settings("onPageload", loc[4], sc.set.onPageload, false),
  1191. sf.settings("whitelist", loc[5], sc.set.useWhitelist, false, sc.set.listWhitelist),
  1192. sf.settings("eventOnly", loc[58], sc.set.eventOnly, false),
  1193. sf.settings("wnRefresh", loc[6]+" [v"+wn.vers[1]+"]"),
  1194. sf.settings("cleanStorage", loc[7]),
  1195. sf.elem("li", "b-settingItem settingCredits settingSeperator", "<p>Version: "+sc.vers+"</p>"),
  1196. sf.elem("li", "b-settingItem settingCredits", "<p>"+loc[20]+" <a class='b-orange-arrow' href='"+sc.user.wot+"'>Orrie</a></p>"+((sc.cred[sc.loc.cur]) ? "<p>"+loc[21]+" ("+sc.loc.cur.toUpperCase()+"):</p><table>"+sc.cred[sc.loc.cur]+"</table>" : "")),
  1197. sf.elem("li", "b-settingItem settingCredits settingLinks", "<p><a class='b-orange-arrow' href='"+sc.host+"'>Greasy Fork镜像</a><a class='b-orange-arrow' href='"+((wg.srv == "na") ? sc.top.na : sc.top.eu)+"'>Support Thread</a></p>")
  1198. ];
  1199. if (sc.loc.sup.indexOf(sc.loc.cur) == -1) {
  1200. clanSet_list_locItem.innerHTML = "<h1>Script Translation</h1><p>Unsupported language detected!</p><p>If you want to contribute with translation, please contact <a class='b-orange-arrow' href='"+sc.user.wl+"'>Orrie</a></p>";
  1201. clanSet_list_items.push(clanSet_list_locItem);
  1202. }
  1203. else if (sc.loc.miss > 0) {
  1204. clanSet_list_locItem.innerHTML = "<h1>Script Translation</h1><p>Currently "+sc.loc.miss+" out of "+_l+" strings not translated in your language!</p><p>If you want to contribute, open the browser console, translate the strings and send them to <a class='b-orange-arrow' href='"+sc.user.wl+"'>Orrie</a></p>";
  1205. clanSet_list_items.push(clanSet_list_locItem);
  1206. }
  1207. sf.links(clanSet_list, clanSet_list_items);
  1208. clanSet_div.firstElementChild.addEventListener('click', function() {this.classList.toggle('cm-user-menu-link__opened'); this.nextSibling.classList.toggle('cm-user-menu__opened');}, false);
  1209. clanSet_div.appendChild(clanSet_list);
  1210. // add script info and settings if user menu exists, else wait
  1211. var navMenu = d.getElementById('common_menu'),
  1212. navUser = (navMenu) ? d.getElementsByClassName('cm-menu__user')[0] : false;
  1213. if (navUser) {
  1214. navUser.appendChild(clanSet_div);
  1215. }
  1216. else {
  1217. var setLook = new MutationObserver(function() {
  1218. navUser = d.getElementsByClassName('cm-menu__user')[0];
  1219. navUser.appendChild(clanSet_div);
  1220. setLook.disconnect();
  1221. });
  1222. setLook.observe(d.body, {childList: true});
  1223. }
  1224.  
  1225. // clan statistic links - observe html change for clan name
  1226. var emblems = [
  1227. d.getElementsByClassName('page-header_emblem')[0], // first method
  1228. d.getElementsByClassName('clan_name')[0], // second method
  1229. d.getElementsByClassName('js-clan-name')[0] // third method - your own clan
  1230. ];
  1231. if (emblems[0] || emblems[1]) {
  1232. sf.clan();
  1233. }
  1234. else {
  1235. var clanInfo = d.getElementById('js-general-info-block'),
  1236. nameLook = new MutationObserver(function() {
  1237. sf.clan();
  1238. nameLook.disconnect();
  1239. });
  1240. if (clanInfo) {
  1241. nameLook.observe(clanInfo, {childList: true});
  1242. }
  1243. else if (emblems[2]) {
  1244. sf.clan();
  1245. }
  1246. }
  1247.  
  1248. // add clan total stats if they exist
  1249. if (wg.p && ss.clan) {
  1250. var clanRating = d.getElementById('js-rating-block'),
  1251. clanProfileValue = d.getElementsByClassName('rating-profile_item');
  1252. if (clanProfileValue.length === 0) {
  1253. var ratingLook = new MutationObserver(function() {
  1254. sf.clanInsert("main");
  1255. ratingLook.disconnect();
  1256. });
  1257. ratingLook.observe(clanRating, {childList: true});
  1258. }
  1259. else {
  1260. sf.clanInsert("main");
  1261. }
  1262. }
  1263.  
  1264. // check if on memberlist page
  1265. if (wg.m) {
  1266. // formula calculations and variables
  1267. var memObj = {
  1268. cls: d.getElementsByClassName('tbl-rating_body')[0],
  1269. ids: [],
  1270. bans: {ids:[],f:false}
  1271. };
  1272.  
  1273. // add manual stat fetching button
  1274. var filter_class = d.getElementsByClassName('filter')[0],
  1275. refreshBtn_div = sf.elem("div", "b-button-stats", "<a>"+loc[12]+"</a>");
  1276. refreshBtn_div.addEventListener('click', function() {sf.tableFetch();}, false);
  1277. filter_class.appendChild(refreshBtn_div);
  1278.  
  1279. // prepare stat fetcher, store stats in localStorage and reload page
  1280. var ratLook = new MutationObserver(function() {
  1281. sf.tableFetch();
  1282. ratLook.disconnect();
  1283. });
  1284.  
  1285. // fetch stats automatically if enabled or check whitelist for whitelisted clan
  1286. if (ss.statFetch && (sc.set.onPageload || (sc.set.useWhitelist && whitelistArray.indexOf(wg.clan.id) > -1))) {
  1287. ratLook.observe(memObj.cls, {childList: true});
  1288. }
  1289. else {
  1290. // no stats fetching, check if stats already exist and add if they do
  1291. var statsInsertionStatus = false,
  1292. headerInsertionStatus = false;
  1293. // add clan total stats if they exist
  1294. var clanStatsPanel = d.getElementsByClassName('js-clan-statistics-container')[0],
  1295. statsLook = new MutationObserver(function() {
  1296. sf.clanInsert("list");
  1297. });
  1298. if (sc.web.chrome) {
  1299. sf.clanInsert("list");
  1300. }
  1301. else {
  1302. statsLook.observe(clanStatsPanel, {childList: true});
  1303. }
  1304. // wait for table to be filled before adding wn8
  1305. if (memObj.cls.childElementCount === 0) {
  1306. var ratInsert = new MutationObserver(function(muto) {
  1307. if (muto[0].previousSibling === null) {
  1308. sf.ratInsert();
  1309. }
  1310. });
  1311. ratInsert.observe(memObj.cls, {childList: true});
  1312. }
  1313. else {
  1314. sf.ratInsert();
  1315. }
  1316. }
  1317. }
  1318. else if (wg.sb) { // check if on globalmap page for battle scheduler
  1319. // inserting style into head
  1320. var styleSch = [
  1321. "h3 {text-align: center;}",
  1322. ".b-battles {font-size: 12px; margin: 0px 0 60px; width: 100%;}",
  1323. ".b-battles .h-battles {font-size: 15px; position: relative;}",
  1324. ".b-battles .h-battles .h-battles-info {border-bottom: 1px solid #000; box-shadow: inset 0 -1px rgba(255,255,255,.05); text-align: center; padding: 10px 0;}",
  1325. ".b-battles .h-battles .h-battles-info img {max-height: 16px; vertical-align: bottom;}",
  1326. ".b-battles .h-battles .h-battles-info .h-shadow {font-weight: bold; text-shadow: 0px 0px 1px rgba(27,27,28, 1), 0px 0px 2px rgba(27,27,28, 1);}",
  1327. ".b-battles .h-battles .h-battles-infotable {margin: 21px 10px; min-width: 150px; position: absolute; top: 0px;}",
  1328. ".b-battles .h-battles .h-battles-infotable td {padding: 0 2px;}",
  1329. ".b-battles .h-battles .h-battles-infotable td.gold {padding-right: 16px;}",
  1330. ".b-battles .b-battles-holder {background-color: rgba(0, 0, 0, 0.75);}",
  1331. ".b-battles .b-battles-holder .t-battles {border-spacing: 0; box-shadow: inset -1px 0 rgba(255,255,255,.05); text-align: center; width: 100%;}",
  1332. ".b-battles .b-battles-holder .t-battles thead tr {}",
  1333. ".b-battles .b-battles-holder .t-battles tbody tr:nth-child(even) td {background-color: rgba(80, 60, 60, 0.1);}",
  1334. ".b-battles .b-battles-holder .t-battles tbody tr:nth-child(odd) td {background-color: rgba(123, 123, 123, 0.1);}",
  1335. ".b-battles .b-battles-holder .t-battles tbody tr:hover {background-color: rgba(100, 100, 100, 0.20);}",
  1336. ".b-battles .b-battles-holder .t-battles thead tr th.t-"+bs.time.t+", .b-battles .b-battles-holder .t-battles tbody tr td.t-"+bs.time.t+" {background-color: rgba(254,252,223, 0.15); border-left: 1px solid #808080; border-right: 1px solid #808080;}",
  1337. ".b-battles .b-battles-holder .t-battles thead tr th.t-"+bs.time.t+" + th, .b-battles .b-battles-holder .t-battles tbody tr td.t-"+bs.time.t+" + td {background-color: rgba(224,223,218, 0.1); border-right: 1px solid #808080;}",
  1338. ".b-battles .b-battles-holder .t-battles tr .t-border {border-right: 2px solid rgba(194, 173, 173, 0.1);}",
  1339. ".b-battles .b-battles-holder .t-battles tr th {line-height: 35px; border-top: 1px solid rgba(255,255,255,.1); box-shadow: inset 1px -1px rgba(255,255,255,.05); position: relative;}",
  1340. ".b-battles .b-battles-holder .t-battles tr th .sorter_caption {margin: 0; line-height: unset;}",
  1341. ".b-battles .b-battles-holder .t-battles tr th .sorter::after {margin-top: "+((sc.web.chrome) ? "-1" : "-2")+"px;}",
  1342. ".b-battles .b-battles-holder .t-battles tr th:hover {color: #FFFFFF;}",
  1343. ".b-battles .b-battles-holder .t-battles tr th:hover .sorter::after {opacity: 1;}",
  1344. ".b-battles .b-battles-holder .t-battles tr th.sort-up, .b-battles .b-battles-holder .t-battles tr th.sort-down {color: #DADADB;}",
  1345. ".b-battles .b-battles-holder .t-battles tr th.sort-up .sorter::after {background-position-y: -10px; margin-top: "+((sc.web.chrome) ? "-3" : "-4")+"px; opacity: 1;}",
  1346. ".b-battles .b-battles-holder .t-battles tr th.sort-down .sorter::after {background-position-y: -5px; opacity: 1;}",
  1347. ".b-battles .b-battles-holder .t-battles tr td {line-height: 27px; border-top: 1px solid rgba(255,255,255,.1); border-bottom: 1px solid #000; box-shadow: inset 1px -1px rgba(255,255,255,.05); padding: 0 2px;}",
  1348. ".b-battles .b-battles-holder .t-battles tr td:first-of-type {max-width: 125px; width: 125px; overflow: hidden; padding: 0 5px; text-overflow: ellipsis; white-space: nowrap;}",
  1349. ".b-battles .b-battles-holder .t-battles tr td.t-title {font-weight: bold;}",
  1350. ".b-battles .b-battles-holder .t-battles tr td.t-good {color: #4D7326;}",
  1351. ".b-battles .b-battles-holder .t-battles tr td.t-bad {color: #930D0D;}",
  1352. ".b-battles .b-battles-holder .t-battles tr td.t-plan {color: #FFE400;}",
  1353. ".b-battles .b-battles-holder .t-battles tr td.t-fight {color: #4D7326; font-size: 15px; font-weight: bold;}",
  1354. ".b-battles .b-battles-holder .t-battles tr td.t-noFight {color: #808080; font-size: 14px;}",
  1355. ".b-battles .b-battles-holder .t-battles tr td.t-fight.t-noOwner {color: #808080;}",
  1356. ".b-battles .b-battles-holder .t-battles tr td.t-error {color: #CD2911;}",
  1357. ".b-battles .b-battles-holder .t-battles tr.t-cwText td {font-size: 26px; line-height: 54px;}",
  1358. ".b-battles .b-battles-holder .t-battles img {height: 16px; margin-bottom: 5px; vertical-align: bottom;}",
  1359. ".b-battles .f-battles {border-top: 1px solid #000; box-shadow: inset 0 1px rgba(255,255,255,.05); font-size: 15px; padding: 10px 0; text-align: center;}",
  1360. ".b-battles .f-battles img {max-height: 16px; vertical-align: bottom;}",
  1361. ".b-battles .t-bold {font-weight: bold;}",
  1362. ".t-battle {display: none;}",
  1363. ".t-time {display: none;}",
  1364. ".b-display-none {display: none;}",
  1365. ".b-display-block {display: block}"
  1366. ];
  1367. style.textContent += styleSch.join("");
  1368.  
  1369. // prepare static html and table reference for further use
  1370. var layout_holder = d.getElementsByClassName("layout_holder")[0],
  1371. battlesPanel = sf.elem("div", "b-battles", "<div class='h-battles'><h3>"+loc[22]+"</h3><div class='h-battles-info'>"+loc[23]+" <span id='js-timePrime'>Ɵ</span></div><table class='h-battles-infotable'><tr><td>"+loc[24]+"</td><td id='js-battles'>0</td><td>"+loc[60]+"</td><td id='js-battlesConc'>0</td></tr><tr><td>"+loc[25]+"</td><td class='gold'><span id='js-gold'>0</span><i class='i i__currencies i__gold'></i></td></tr></table></div><div class='b-battles-holder'><table class='t-battles sortable'><thead><tr><th><a href='#' class='sorter js-table-sorter js-sort-field_localized_front_name'><span class='sorter_caption'>"+loc[26]+"</span></a></th><th><a href='#' class='sorter js-table-sorter js-sort-field_localized_front_name'><span class='sorter_caption'>"+loc[27]+"</span></a></th><th id='js-sort' class='sort-default' data-sort-order='desc'><a href='#' class='sorter js-table-sorter js-sort-field_localized_front_name'><span class='sorter_caption'>"+loc[28]+"</span></a></th><th><a href='#' class='sorter js-table-sorter js-sort-field_localized_front_name'><span class='sorter_caption'>"+((bs.cw.event) ? loc[29] : loc[30])+"</span></a></th><th><a href='#' class='sorter js-table-sorter js-sort-field_localized_front_name'><span class='sorter_caption'>"+loc[31]+"</span></a></th><th><a href='#' class='sorter js-table-sorter js-sort-field_localized_front_name'><span class='sorter_caption'>"+loc[32]+"</span></a></th><th><a href='#' class='sorter js-table-sorter js-sort-field_localized_front_name'><span class='sorter_caption'>"+loc[33]+"</span></a></th><th><a href='#' class='sorter js-table-sorter js-sort-field_localized_front_name'><span id='js-provStatus' class='sorter_caption'>"+loc[34]+"</span></a></th><th class='t-battle'><a href='#' class='sorter js-table-sorter js-sort-field_localized_front_name'><span class='sorter_caption'>"+loc[35]+"</span></a></th><th class='t-battle t-border'><a href='#' class='sorter js-table-sorter js-sort-field_localized_front_name'><span class='sorter_caption'>"+loc[36]+"</span></a></th></tr></thead><tbody></tbody></table></div><div class='f-battles'>"+loc[37]+" <span id='js-batttleUpdate'>Ɵ</span> [UTC"+((bs.time.o >= 0) ? "+" : "")+bs.time.o+"]<span id='js-error'></span></div>"),
  1372. table = battlesPanel.children[1].firstElementChild,
  1373. layoutLook = new MutationObserver(function() {
  1374. layout_holder.insertBefore(battlesPanel, layout_holder.children[1]);
  1375. layoutLook.disconnect();
  1376. });
  1377. if (layout_holder.childElementCount > 0) {
  1378. layout_holder.insertBefore(battlesPanel, layout_holder.children[1]);
  1379. }
  1380. else {
  1381. layoutLook.observe(layout_holder, {childList: true});
  1382. }
  1383.  
  1384. // time cells for header and body rows
  1385. var timeCells = bs.table[wg.srv],
  1386. timeFragment = d.createDocumentFragment();
  1387. for (var _tc=0, _tc_len = timeCells.length; _tc<_tc_len; _tc++) {
  1388. var t = timeCells[_tc],
  1389. times = [sf.time(t,"00","s"), sf.time(t)+":00", sf.time(t,"30","s"), sf.time(t)+":30"];
  1390. timeFragment.appendChild(sf.elem("th", "t-time "+times[0], "<a href='#' class='sorter js-table-sorter js-sort-field_localized_front_name'><span class='sorter_caption'>"+times[1]+"</span></a>"));
  1391. bs.table.c += "<td class='t-time "+times[0]+"'></td>";
  1392. if (_tc !== _tc_len-1) {
  1393. timeFragment.appendChild(sf.elem("th", "t-time "+times[2], "<a href='#' class='sorter js-table-sorter js-sort-field_localized_front_name'><span class='sorter_caption'>"+times[3]+"</span></a>"));
  1394. bs.table.c += "<td class='t-time "+times[2]+"'></td>";
  1395. }
  1396. }
  1397. table.firstElementChild.firstElementChild.appendChild(timeFragment);
  1398.  
  1399. // add intervals for time and round updater
  1400. var timeInterval = setInterval(sf.timer,1000), // 1 second
  1401. updateInterval = setInterval(sf.updater,120000); // 2 minutes
  1402.  
  1403. // activate tablesort function
  1404. var sortTable = false;
  1405. if (Tablesort) {
  1406. // Numeric sort
  1407. Tablesort.extend('number', function(item) {
  1408. return item.match(/^-?(\d)*-?([,\.]){0,1}-?(\d)+([E,e][\-+][\d]+)?%?$/); // Number
  1409. }, function(a, b) {
  1410. a = parseFloat(a);
  1411. b = parseFloat(b);
  1412.  
  1413. a = isNaN(a) ? 0 : a;
  1414. b = isNaN(b) ? 0 : b;
  1415. return a - b;
  1416. });
  1417. sortTable = new Tablesort(table);
  1418. }
  1419. else {
  1420. window.alert("Error activating tablesort, please refresh - if this shit continues, poke Orrie");
  1421. }
  1422.  
  1423. // insert update status
  1424. table.lastElementChild.appendChild(sf.elem("tr", "t-cwText", "<td colspan='8'>"+loc[38]+"</td>"));
  1425.  
  1426. // send request to wargaming api to see if an event is running
  1427. sf.request("eventData", bs.api.event, sf.handlerEvent);
  1428. }
  1429. });
  1430. pageLook.observe(d.body, {childList: true});
  1431. }(window));

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址