WoTStatScript - Clanpage

More info for World of Tanks clan page.

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

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