WoTStatScript - Clanpage

More info for World of Tanks clan page.

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

  1. // ==UserScript==
  2. // @name WoTStatScript - Clanpage
  3. // @version 0.9.13.0.5
  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. // @license MIT License
  11. // ==/UserScript==
  12. /*
  13. Changelogs:
  14. Profilepage: http://dl.dropboxusercontent.com/u/12497046/wot/projects/statscript/WoTStatScript-changelog.txt
  15. Clanpage: http://dl.dropboxusercontent.com/u/12497046/wot/projects/statscript/WoTStatScript-Clanpage-changelog.txt
  16. */
  17. (function() {
  18. // global vars
  19. var d = document, c = d.cookie;
  20.  
  21. // get server info and webpage
  22. var wg = {host:d.location.host, href:d.location.href, clan:{}};
  23. wg.srv = wg.host.match(/(eu|ru|na|com|asia|kr)/)[0].replace(/com/,"na");
  24. wg.m = (/players/i.test(wg.href) || /players\/wot/i.test(wg.href)) && !/wowp/i.test(wg.href);
  25.  
  26. // getting claninfo
  27. var emblemName = d.getElementsByClassName('page-header_emblem')[0],
  28. clanName = d.getElementsByClassName('clan_name')[0],
  29. sidebarName = d.getElementsByClassName('sidebar-clan_emblem')[0];
  30. wg.clan.id = wg.href.match(/\/(\d+)/)[1];
  31. wg.clan.name = (emblemName || clanName) ? ((clanName) ? clanName.firstElementChild.innerHTML.replace(/[\[\]]/g,"") : emblemName.alt) : sidebarName.alt;
  32. wg.p = new RegExp("\\["+wg.clan.name+"\\] \\|").test(d.title) && !/wowp/i.test(wg.href);
  33.  
  34. // script variables
  35. var sc = {
  36. vers: "0.9.13.0.5",
  37. host: "http://greasyfork.org/en/scripts/12137-wotstatscript-clans",
  38. user: "http://forum.wotlabs.net/index.php?/user/1618-orrie/",
  39. top: {
  40. eu: "http://forum.worldoftanks.eu/index.php?showtopic=263423",
  41. na: "http://forum.worldoftanks.com/index.php?showtopic=404652"
  42. },
  43. api: {
  44. ru: "98ca7c4fb108175b67d6505b9c3f3ebd",
  45. eu: "a7595640a90bf2d19065f3f2683b171c",
  46. na: "bf5dba0efd444d75147b6222dd903fd2",
  47. asia: "95f8713eccd322e52dbf521dbd28b19c",
  48. kr: "ffea0f1c3c5f770db09357d94fe6abfb"
  49. },
  50. wn: "http://www.wnefficiency.net/exp/expected_tank_values_latest.json",
  51. loc: c.match(/wgccfe_language=(\w+)/)[1],
  52. locSup: ["en", "ru", "cs", "de", "fr", "pl", "es", "tr"],
  53. date: Date.now()
  54. };
  55. // script threadlink
  56. sc.link = "<div class='b-scriptlink'><a target='_blank' href="+sc.host+">Script</a> v"+sc.vers+" - <a target='_blank' href="+((wg.srv == "na") ? sc.top.na : sc.top.eu)+">Thread</a></div>";
  57.  
  58. // api links without account id
  59. sc.api.i = "http://api.worldoftanks."+((wg.srv == "na") ? "com" : wg.srv)+"/wot/account/info/?application_id="+sc.api[wg.srv]+"&account_id=";
  60. sc.api.v = "http://api.worldoftanks."+((wg.srv == "na") ? "com" : wg.srv)+"/wot/account/tanks/?application_id="+sc.api[wg.srv]+"&account_id=";
  61.  
  62. // external site support
  63. var srv = {
  64. wl: false, // wotlabs
  65. nm: false, // noobmeter
  66. vb: false, // vbaddict
  67. ws: false, // wotstats
  68. cs: false, // wotcs
  69. wlf: false, // wot-life
  70. as: false, // away stats
  71. ct: false, // clan tools
  72. aos: false, // age of strife
  73. kttc: false, // kttc
  74. wots: false, // wots
  75. ch: false, // clan history
  76. wr: false // wotreplays
  77. };
  78.  
  79. // determine browser types
  80. var web = {
  81. gecko: typeof InstallTrigger !== 'undefined',
  82. opera: !!window.opera || /opera|opr/i.test(navigator.userAgent),
  83. chrome: !!window.chrome && !!window.chrome.webstore,
  84. safari: /constructor/i.test(window.HTMLElement)
  85. };
  86.  
  87. // fetch wnefficiency values - check if array exists in localStorage, otherwise fetch and reload page
  88. var statArr = [],
  89. wnExpValues = locStorage("wnExpValues", "", "get", "parse"),
  90. wnExpDate = locStorage("wnExpDate", "", "get", "parse")+12096e5 >= sc.date, // true if timestamp is less than 2 weeks old, refresh list if false.
  91. wnExpVers = locStorage("wnExpVers", "", "get", "parse") || "";
  92. if (wnExpVers[0]==sc.vers && wnExpValues && wnExpDate) {
  93. statArr = wnExpValues.data;
  94. }
  95. else {
  96. reqHnd(sc.wn, wnHnd, wnHnd_error);
  97. }
  98.  
  99. // fetch stored clanlist stats - check if array exists in localStorage, otherwise tag fetching to true
  100. var s = {clan:{},user:{}}, statFetch = false,
  101. ssValues = locStorage("statScriptValues_"+wg.clan.id, "", "get", "parse"),
  102. ssDate = locStorage("statScriptDate_"+wg.clan.id, "", "get", "parse")+6048e5 >= sc.date; // true if timestamp is less than 1 weeks old, refresh list if false.
  103. if (ssValues && ssDate) {
  104. s = ssValues;
  105. }
  106. else {
  107. statFetch = true;
  108. }
  109.  
  110. // inserting style into head
  111. var style = elem("style", "wotstatscript", "", "text/css");
  112. d.head.appendChild(style);
  113.  
  114. // colour scale array
  115. var colArr = {
  116. // col wr bat sr hr dmg wgr wn8 wn7 eff nm
  117. sUni: [ "#5A3175", 65, 30000, 50, 80, 300, 9900, 2900, 2050, 2050, 2000 ], // 99.99% super unicum
  118. uni: [ "#83579D", 60, 25000, 46, 75, 270, 9000, 2450, 1850, 1800, 1950 ], // 99.90% unicum
  119. gr8: [ "#3972C6", 56, 21000, 42, 70, 240, 8500, 2000, 1550, 1500, 1750 ], // 99.00% great
  120. vGud: [ "#4099BF", 54, 17000, 38, 65, 210, 6500, 1600, 1350 ], // 95.00% very good
  121. good: [ "#4D7326", 52, 13000, 34, 60, 180, 5000, 1200, 1100, 1200, 1450 ], // 82.00% good
  122. aAvg: [ "#849B24", 50, 10000, 30, 55, 150, 4000, 900 ], // 63.00% above average
  123. avg: [ "#CCB800", 48, 7000, 25, 50, 120, 3000, 650, 900, 900, 1250 ], // 40.00% average
  124. bAvg: [ "#CC7A00", 47, 3000, 20, 45, 90, 2000, 450, 700, 600, 1150 ], // 20.00% below average
  125. bas: [ "#CD3333", 46, 1000, 15, 40, 60, 1500, 300, 500 ], // 6.00% basic
  126. beg: [ "#930D0D", 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 ], // 0.00% beginner
  127. dft: [ "#6B6B6B" ], // default
  128. 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
  129. };
  130.  
  131. // localization
  132. // cz-czech - Crabtr33 and Ragnarocek
  133. // de-german - ArtiOpa, Crakker and multimill
  134. // fr-french - SuperPommeDeTerre
  135. // pl-polish - KeluMocy and pokapokami
  136. // es-spanish - Frodo45127
  137. // tr-turkish - Ufuko
  138. // ru-russian - dimon222
  139. var loc = [
  140. // thousands separator
  141. {en: ",", ru: " ", cs: " ", de: ".", fr: " ", pl: " ", es:" ", tr: "."},
  142. // clan page
  143. {en: "Clan Stats", ru: "Статистика клана", cs: "Stat. klanu", de: "Clanstatistiken", fr: "Statistiques du clan", pl: "Statystyki klanu", es: "Estadísticas del clan", tr: "Klan İstatistikleri"},
  144. {en: "Replays:", ru: "Реплеи:", cs: "Záznamy:", de: "Replays", fr: "Replays:", pl: "Powtórki:", es: "Repeticiones:", tr: "Replayler"},
  145. // memberlist
  146. {en: "Script Menu", ru: "Script Menu", cs: "Nastavení scriptu", de: "Script Menu", fr: "Script Menu", pl: "Script Menu", es:"Script Menu", tr: "Script Menu"},
  147. {en: "Load Stats Automatically", ru: "Load Stats Automatically", cs: "Nahrát stat. automaticky", de: "Load Stats Automatically", fr: "Load Stats Automatically", pl: "Load Stats Automatically", es:"Load Stats Automatically", tr: "Load Stats Automatically"},
  148. {en: "Use Whitelist", ru: "Use Whitelist", cs: "Použi whitelist", de: "Use Whitelist", fr: "Use Whitelist", pl: "Use Whitelist", es:"Use Whitelist", tr: "Use Whitelist"},
  149. {en: "Refresh WN8 Table", ru: "Refresh WN8 Table", cs: "Obnovit WN8 tabulku", de: "Refresh WN8 Table", fr: "Refresh WN8 Table", pl: "Refresh WN8 Table", es: "Refresh WN8 Table", tr: "Refresh WN8 Table"},
  150. {en: "Clean Script Database", ru: "Clean Script Database", cs: "Vyčisti db scriptu", de: "Clean Script Database", fr: "Clean Script Database", pl: "Clean Script Database", es: "Clean Script Database", tr: "Clean Script Database"},
  151. {en: "Average Winrate", ru: "Average Winrate", cs: "Průměrný winrate", de: "Average Winrate", fr: "Average Winrate", pl: "Average Winrate", es:"Average Winrate", tr: "Average Winrate"},
  152. {en: "Average WN8", ru: "Average WN8", cs: "Průměrné WN8", de: "Average WN8", fr: "Average WN8", pl: "Average WN8", es:"Average WN8", tr: "Average WN8"},
  153. {en: "Overall", ru: "Overall", cs: "Celkem", de: "Overall", fr: "Overall", pl: "Overall", es:"Overall", tr: "Overall"},
  154. {en: "Fetch WN8 for Clan", ru: "Fetch WN8 for Clan", cs: "Obnov WN8 pre klan", de: "Fetch WN8 for Clan", fr: "Fetch WN8 for Clan", pl: "Fetch WN8 for Clan", es:"Fetch WN8 for Clan", tr: "Fetch WN8 for Clan"},
  155. {en: "Fetching WN8 for Clan!", ru: "Fetching WN8 for Clan!", cs: "Obnovuju WN8 pro klan!", de: "Fetching WN8 for Clan!", fr: "Fetching WN8 for Clan!", pl: "Fetching WN8 for Clan!", es:"Fetching WN8 for Clan!", tr: "Fetching WN8 for Clan!"},
  156. {en: "WN8 Fetched for Clan!", ru: "WN8 Fetched for Clan!", cs: "WN8 obnoveno pro klan!", de: "WN8 Fetched for Clan!", fr: "WN8 Fetched for Clan!", pl: "WN8 Fetched for Clan!", es:"WN8 Fetched for Clan!", tr: "WN8 Fetched for Clan!"},
  157. {en: "Not Found", ru: "Not Found", cs: "Nenalezeno", de: "Not Found", fr: "Not Found", pl: "Not Found", es:"Not Found", tr: "Not Found"},
  158. {en: "New Members:", ru: "New Members:", cs: "Noví členové:", de: "New Members:", fr: "New Members:", pl: "New Members:", es:"New Members:", tr: "New Members:"},
  159. {en: "Banned Members:", ru: "Banned Members:", cs: "Noví členové:", de: "Banned Members:", fr: "Banned Members:", pl: "Banned Members:", es:"Banned Members:", tr: "Banned Members:"},
  160. {en: "Currently Unavailable", ru: "Currently Unavailable", cs: "Currently Unavailable", de: "Currently Unavailable", fr: "Currently Unavailable", pl: "Currently Unavailable", es:"Currently Unavailable", tr: "Currently Unavailable"},
  161. {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 Ended, but no login", pl: "Ban Ended, but no login", es:"Ban Ended, but no login", tr: "Ban Ended, but no login"}
  162. // {en: "", ru: "", cs: "", de: "", fr: "", pl: "", es:"", tr: ""}
  163. ];
  164.  
  165. // region settings for external sites
  166. switch(wg.srv) {
  167. case ("eu"): // eu server
  168. srv.wl = srv.nm = srv.vb = srv.ws = srv.cs = srv.wlf = srv.as = srv.ct = srv.kttc = srv.aos = srv.ch = srv.wr = wg.srv;
  169. break;
  170. case ("ru"): // ru server
  171. srv.wl = srv.nm = srv.vb = srv.ws = srv.cs = srv.ct = srv.kttc = srv.wots = srv.aos = srv.ch = srv.wr = wg.srv;
  172. break;
  173. case ("na"): // na server - american english
  174. srv.wl = srv.nm = srv.vb = srv.ws = srv.cs = srv.wlf = srv.ct = srv.kttc = srv.aos = srv.ch = wg.srv; srv.wr = "com";
  175. break;
  176. case ("asia"): // asia server
  177. srv.wl = srv.nm = srv.vb = srv.ws = srv.cs = "sea"; srv.ct = srv.kttc = srv.aos = srv.ch = wg.srv; srv.wr = "com";
  178. break;
  179. case ("kr"): // korean server
  180. srv.wl = srv.nm = srv.vb = srv.ws = srv.cs = srv.ct = srv.aos = srv.ch = wg.srv; srv.wr = "com";
  181. break;
  182. default: break;
  183. }
  184.  
  185. // set script language to english if an unsupported language is detected
  186. if (sc.locSup.indexOf(sc.loc) == -1) {
  187. sc.loc = "en";
  188. }
  189. // process localization
  190. for (var _l=0, l_len = loc.length; _l<l_len; _l++) {
  191. loc[_l] = loc[_l][sc.loc];
  192. }
  193.  
  194. // add language to body classname for language based styling
  195. d.body.classList.add("lang-"+sc.loc);
  196.  
  197. // variables for dropbox, css and data uri
  198. var css = {
  199. u: {
  200. cIcons: "",
  201. arrow: ""
  202. },
  203. i: {
  204. loader: "/clans/static/2.2.8/images/processing/loader.gif"
  205. }
  206. };
  207.  
  208. // style contents
  209. var styleClan = [
  210. // script header rules
  211. "#common_menu .cm-menu__user > *:not(.cm-notifications):not(.js-cm-user-menu-dropdown) {display: inline-block}",
  212. "#common_menu .cm-parent-link:hover {cursor: inherit; color: #707273 !important;}",
  213. "#common_menu .b-scriptlink a {color: #E5B12E;}",
  214. "#common_menu .b-scriptlink a:hover {color: #FFBE4C; text-shadow: 0px 0px 7px rgba(255, 126, 0, 0.7);}",
  215. ".profile__main {margin: 0; padding: 0;}",
  216. // fix width for header with low resolution
  217. "form.search {width: 25%;}",
  218. // loading text
  219. ".processing_loader span {display: table; margin: 0 auto;}",
  220. // links menu rules
  221. ".menu-clan_links {padding: 0;}",
  222. ".menu-clan_links.cm-parent-link__opened {border: 1px solid #313335;}",
  223. ".menu-clan_links .menu-top_link {cursor: pointer; padding: 0 8px 0 9px;}",
  224. ".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;}",
  225. ".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;}",
  226. ".menu-clan_links .cm-parent-link__opened .cm-arrow {opacity: 1; transform: rotate(180deg);}",
  227. ".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;}",
  228. ".menu-clan_links .cm-sublist__opened {display: block;}",
  229. ".menu-clan_links .clan-links td {padding: 0 10px;}",
  230. ".sl-icon {background: url('"+css.u.cIcons+"') no-repeat; display: inline-block; margin: -2px 8px 0px 0px; vertical-align: middle; height: 16px; width: 16px;}",
  231. ".sl-wl {background-position: 0px 0px;}",
  232. ".sl-nm {background-position: 0px -16px;}",
  233. ".sl-ct {background-position: 0px -32px;}",
  234. ".sl-cs {background-position: 0px -48px;}",
  235. ".sl-kttc {background-position: 0px -64px;}",
  236. ".sl-wlife {background-position: 0px -80px;}",
  237. ".sl-as {background-position: 0px -96px;}",
  238. ".sl-wr {background-position: 0px -112px;}",
  239. ".sl-vb {background-position: 0px -128px;}",
  240. // rating profile rules
  241. ".rating-profile {width: 70%; margin: 0px auto;}",
  242. // settings menu rules
  243. "#common_menu .menu-settings {text-align: left;}",
  244. "#common_menu .menu-settings .cm-parent-link:hover {cursor: pointer;}",
  245. "#common_menu .menu-settings .settingItem {margin: 6px 0px;}",
  246. "#common_menu .menu-settings label {display: table; line-height: normal; cursor: pointer;}",
  247. "#common_menu .menu-settings .l-box {display: none;}",
  248. "#common_menu .menu-settings .b-checkbox {height: 16px; width: 16px; float: left; margin-right: 5px;}",
  249. "#common_menu .menu-settings .b-checkbox span {height: 16px; width: 16px;}",
  250. "#common_menu .menu-settings .b-combobox-label__checked {color: #DCDCDC;}",
  251. "#common_menu .menu-settings .settingItem .b-combobox-label:hover {color: #DCDCDC;}",
  252. "#common_menu .menu-settings .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);}",
  253. "#common_menu .menu-settings .settingItem .b-combobox-label:hover .b-checkbox.b-checkbox__checked {background-position: 0px -68px;}",
  254. "#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;}",
  255. "#common_menu .menu-settings textarea::-webkit-input-placeholder {color: #FFFFFF;}",
  256. "#common_menu .menu-settings textarea::-moz-placeholder {color: #FFFFFF;}",
  257. "#common_menu .menu-settings .b-settingLink {line-height: 26px;}",
  258. "#common_menu .menu-settings .b-settingLink a {cursor: pointer; color: #B1B2B3; text-shadow: 0px 1px 1px rgba(0, 0, 0, 0.5);}",
  259. "#common_menu .menu-settings .b-settingLink a:hover {color: #FFFFFF; text-shadow: 0px 1px 1px rgba(0, 0, 0, 0.75);}",
  260. // memberpage rules
  261. ".page-header {padding: 30px 0 30px 0}",
  262. ".page-header_meminfo {display: table; margin: 0px auto; position: absolute; top: 3px; right: 0px; left: 0px; text-align: center;}",
  263. ".page-header_meminfo span {margin: 0 5px;}",
  264. ".page-header_ban {color: #E5B12E;}",
  265. ".page-header_mem {color: #E5B12E;}",
  266. ".js-page-header-view .page-header_mem {margin-left: 25px;}",
  267. // button fetch rules
  268. ".b-button-stats {border: 1px solid rgba(255, 255, 255, 0.2); border-radius: 2px; position: absolute; right: 0; top: 9px;}",
  269. ".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;}",
  270. ".b-button-stats a:hover {background: rgba(229, 177, 46, 0.25);}",
  271. // rating players rules
  272. ".rating-players {height: 200px;}",
  273. ".rating-players tbody {width: 95%; display: table; margin: 0px auto;}",
  274. ".rating-players_item__data {padding-top: 25px; width: 14%;}",
  275. ".rating-players_item__average {padding-top: 10px; width: 16%;}",
  276. ".rating-players_stats {font-size: 40px;}",
  277. // membertable rules
  278. "#js-playerslist-table {margin-top: -40px;}",
  279. ".rating-table_wn a {cursor: pointer;}",
  280. ".player_time {position: absolute; right: 0; top: 18px;}"
  281. ];
  282. style.textContent = styleClan.join("");
  283. // end style
  284.  
  285. // add animated loading icon for progress indication
  286. var layoutHolder = d.getElementsByClassName('layout_holder')[0],
  287. loadGif = elem("div", "processing js-hidden", "<div class='processing_loader'><img src='"+css.i.loader+"' alt='Processing...'><span id='js-wn8-status'></span></div>");
  288. layoutHolder.appendChild(loadGif);
  289.  
  290. // script link and settings
  291. var clanHead_div = elem("div", "cm-parent-link", sc.link),
  292. clanSet_div = elem("div", "menu-settings menu-top_item", "<span class='cm-parent-link js-cm-dropdown-link'>"+loc[3]+"<span class='cm-arrow'></span></span>"),
  293. clanSet_list = elem("ul", "cm-sublist", ""),
  294. enableOnPageload = locStorage("statScript_onPageload", "", "get", "parse"),
  295. enableWhitelist = locStorage("statScript_whitelist", "", "get", "parse"),
  296. enableWhiteList_list = locStorage("statScript_whitelist_list", "", "get"),
  297. whiteListArray = (enableWhiteList_list) ? enableWhiteList_list.split(",") : "",
  298. clanSet_list_items = [
  299. settingsHnd("onPageload", loc[4], enableOnPageload, false),
  300. settingsHnd("whitelist", loc[5], enableWhitelist, false, enableWhiteList_list),
  301. settingsHnd("wnRefresh", loc[6]+" [v"+wnExpVers[1]+"]"),
  302. settingsHnd("cleanStorage", loc[7]),
  303. ];
  304. linksHnd(clanSet_list, clanSet_list_items);
  305. clanSet_div.firstElementChild.addEventListener('click', function() {this.classList.toggle('cm-parent-link__opened'); this.nextSibling.classList.toggle('cm-sublist__opened');}, false);
  306. clanSet_div.appendChild(clanSet_list);
  307. // add script info and settings if user menu exists, else wait
  308. var navMenu = d.getElementById('common_menu'),
  309. navUser = navMenu.getElementsByClassName('cm-menu__user')[0];
  310. if (navUser) {
  311. navUser.insertBefore(clanHead_div, navUser.firstChild);
  312. navUser.insertBefore(clanSet_div, navUser.firstChild);
  313. }
  314. else {
  315. var setLook = new MutationObserver(function() {
  316. navUser = navMenu.getElementsByClassName('cm-menu__user')[0];
  317. navUser.insertBefore(clanHead_div, navUser.firstChild);
  318. navUser.insertBefore(clanSet_div, navUser.firstChild);
  319. setLook.disconnect();
  320. });
  321. setLook.observe(navMenu, {childList: true});
  322. }
  323.  
  324. // clan statistic links
  325. var menu_class = d.getElementsByClassName('menu-top')[0],
  326. clanMenu_div = elem("div", "menu-clan_links menu-top_item", "<span class='menu-top_link'>"+loc[1]+"<span class='cm-arrow'></span></span>"),
  327. clanMenu_list = elem("ul", "clan-links cm-sublist", ""),
  328. clanMenu_list_items = [
  329. [srv.wl, "<a target='_blank' href='http://wotlabs.net/"+srv.wl+"/clan/"+wg.clan.name+"'><span class='sl-icon sl-wl'></span>WoTLabs</a>"],
  330. [srv.nm, "<a target='_blank' href='http://noobmeter.com/clan/"+srv.nm+"/"+wg.clan.name+"/"+wg.clan.id+"'><span class='sl-icon sl-nm'></span>Noobmeter</a>"],
  331. [srv.vb, "<a target='_blank' href='http://vbaddict.net/clan/worldoftanks."+srv.vb+"/"+wg.clan.id+"/clan-"+wg.clan.name.toLowerCase()+"'><span class='sl-icon sl-vb'></span>vBAddict</a>"],
  332. [srv.ct, "<a target='_blank' href='http://clantools.us/servers/"+srv.ct+"/clans?id="+wg.clan.id+"'><span class='sl-icon sl-ct'></span>Clan Tools</a>"],
  333. [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>"],
  334. [srv.kttc, "<a target='_blank' href='http://"+((wg.srv=="ru") ? "" : srv.kttc+".")+"kttc.ru/clan/"+wg.clan.id+"/'><span class='sl-icon sl-kttc'></span>KTTC</a>"],
  335. [srv.wlf, "<a target='_blank' href='http://en.wot-life.com/"+srv.wlf+"/clan/"+wg.clan.name+"-"+wg.clan.id+"/'><span class='sl-icon sl-wlife'></span>WoT-Life</a>"],
  336. [srv.as, "<a target='_blank' href='http://stats.teamaway.net/clan/"+wg.clan.id+"/'><span class='sl-icon sl-as'></span>AWAY Stats</a>"],
  337. [srv.wr, "<a target='_blank' href='http://wotreplays."+srv.wr+"/clan/"+wg.clan.name+"'><span class='sl-icon sl-wr'></span>WoTReplays</a>"]
  338. ];
  339. linksHnd(clanMenu_list, clanMenu_list_items);
  340. clanMenu_div.firstElementChild.addEventListener('click', function() {this.classList.toggle('cm-parent-link__opened'); this.nextSibling.classList.toggle('cm-sublist__opened');}, false);
  341. clanMenu_div.appendChild(clanMenu_list);
  342. menu_class.appendChild(clanMenu_div);
  343.  
  344. // add clan total stats if they exist
  345. if (wg.p && s.clan) {
  346. var clanProfileValue = d.getElementsByClassName('rating-profile_item');
  347. if (s.clan.win) {
  348. 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'>"+colStat(s.clan.win,"wr",2,"%")+"</span><span class='rating-profile_key'>"+loc[8]+"</span>";
  349. }
  350. if (s.clan.wn8) {
  351. 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'>"+colStat(s.clan.wn8,"wn8",0)+"</span><span class='rating-profile_key'>"+loc[9]+"</span>";
  352. }
  353. }
  354.  
  355. // check if on memberlist page
  356. if (wg.m) {
  357. // formula calculations and variables
  358. var memObj = {
  359. cls: d.getElementsByClassName('rating-table_body')[0],
  360. ids: [],
  361. bans: {ids:[],f:false}
  362. };
  363.  
  364. // add manual stat fetching button
  365. var timeFrame_class = d.getElementsByClassName('timeframe')[0],
  366. refreshBtn_div = elem("div", "b-button-stats", "<a>"+loc[11]+"</a>");
  367. refreshBtn_div.addEventListener('click', function() {tableFetch();}, false);
  368. timeFrame_class.appendChild(refreshBtn_div);
  369.  
  370. // prepare stat fetcher, store stats in localStorage and reload page
  371. var ratLook = new MutationObserver(function() {
  372. tableFetch();
  373. ratLook.disconnect();
  374. });
  375.  
  376. // fetch stats automatically if enabled or check whitelist for whitelisted clan
  377. if (statFetch && (enableOnPageload || (enableWhitelist && whiteListArray.indexOf(wg.clan.id) > -1))) {
  378. ratLook.observe(memObj.cls, {childList: true});
  379. }
  380. else {
  381. // no stats fetching, check if stats already exist and add if they do
  382. var clanPlayersValue = d.getElementsByClassName('rating-players')[0].rows[0],
  383. pageHeader = d.getElementsByClassName('page-header')[0],
  384. ratHeadStatus = false;
  385. // add clan total stats if they exist
  386. clanPlayersValue.cells[1].getElementsByClassName('rating-players_key')[0].textContent = loc[8];
  387. if (s.clan) {
  388. if (s.clan.win) {
  389. var clanWinCell = clanPlayersValue.insertCell(2);
  390. clanWinCell.className = "rating-players_item rating-players_item__data";
  391. clanWinCell.innerHTML = "<i class='rating-players_icon i i__rating-common i__wot-victories'></i><span class='rating-players_value rating-players_stats'>"+colStat(s.clan.win,"wr",2,"%")+"</span><span class='rating-players_key'>"+loc[10]+" "+loc[8]+"</span>";
  392. }
  393. if (s.clan.wn8) {
  394. var clanWn8Cell = clanPlayersValue.insertCell(4);
  395. clanWn8Cell.className = "rating-players_item rating-players_item__data";
  396. clanWn8Cell.innerHTML = "<i class='rating-players_icon i i__rating-common i__wot-experience'></i><span class='rating-players_value rating-players_stats'>"+colStat(s.clan.wn8,"wn8",0)+"</span><span class='rating-players_key'>"+loc[10]+" "+loc[9]+"</span>";
  397. }
  398. }
  399. // add container for member counters
  400. var memInfo_div = elem("div", "page-header_meminfo", "");
  401. pageHeader.appendChild(memInfo_div);
  402. // wait for table to be filled before adding wn8
  403. var ratInsert = new MutationObserver(function(muto) {
  404. if (muto[0].previousSibling === null) {
  405. var newMem = 0, banMem = d.getElementsByClassName('rating-table_tr__banned').length;
  406. // add a counter for amount of banned people in clan
  407. if (banMem > 0) {
  408. var banMem_span = d.getElementsByClassName('page-header_ban')[0];
  409. if (!banMem_span) {
  410. banMem_span = elem("span", "page-header_ban", loc[16]+" "+banMem);
  411. memInfo_div.appendChild(banMem_span);
  412. }
  413. else {
  414. banMem_span.textContent = loc[16]+" "+banMem;
  415. }
  416. }
  417. // table header for wn8
  418. if (ratHeadStatus === false && Object.keys(s.clan).length !== 0) {
  419. var headName = d.getElementsByClassName('rating-table_th__player')[0],
  420. wnHead = elem("td", "rating-table_th rating-table_th__key rating-table_wn", "<a class='table-sorter js-table-sorter js-sort-wn'><i class='table-sorter_icon table-sorter_icon__media i i__table-params i__wot-aeb'></i><span class='table-sorter_key'>WN8</span></a>");
  421. headName.parentNode.insertBefore(wnHead, headName.nextSibling);
  422. ratHeadStatus = true;
  423. }
  424. // add wn8 for each member and colorize stats
  425. var userCheck = Object.keys(s.user).length !== 0;
  426. for (var _rt=0, _rt_len = memObj.cls.rows.length; _rt<_rt_len; _rt++) {
  427. var row = memObj.cls.rows[_rt];
  428. if (!row.classList.contains("rating-table_tr__card")) {
  429. var id = row.getAttribute('data-account_id'),
  430. memName = row.getElementsByClassName('rating-table_td__player')[0],
  431. memWGR = row.getElementsByClassName('rating-table_td__value')[0],
  432. memWins = row.getElementsByClassName('rating-table_td__value')[2];
  433. if (userCheck) {
  434. var wnRow = elem("td", "rating-table_td rating-table_td__value", "");
  435. memName.parentNode.insertBefore(wnRow, memName.nextSibling);
  436. if (s.user[id]) {
  437. wnRow.innerHTML = colStat(s.user[id].wn8,"wn8",0);
  438. }
  439. else {
  440. wnRow.innerHTML = loc[14];
  441. newMem ++;
  442. }
  443. }
  444. if (memWGR.innerHTML !== "0" && memWins.innerHTML !== "0.00%") {
  445. memWGR.innerHTML = colStat(filter(memWGR.innerHTML,1),"wgr",0);
  446. memWins.innerHTML = colStat(filter(memWins.innerHTML.replace(/[,]/g,"."),3),"wr",2,"%");
  447. }
  448. }
  449. }
  450. // add a counter for new people in the clan, compared to store stats
  451. if (newMem > 0) {
  452. var newMem_span = d.getElementsByClassName('page-header_mem')[0];
  453. if (!newMem_span) {
  454. newMem_span = elem("span", "page-header_mem", loc[15]+" "+newMem);
  455. memInfo_div.appendChild(newMem_span);
  456. }
  457. else {
  458. newMem_span.textContent = loc[15]+" "+newMem;
  459. }
  460. }
  461. // check for length on bans
  462. if (memObj.bans.f) {
  463. apiBanHnd();
  464. }
  465. else {
  466. memObj.bans.cls = d.getElementsByClassName('js-banned');
  467. for (var _bm=0, _bm_len = memObj.bans.cls.length; _bm<_bm_len; _bm++) {
  468. var bannedId = memObj.bans.cls[_bm].dataset.account_id;
  469. memObj.bans.ids.push(bannedId);
  470. }
  471. if (memObj.bans.ids.length > 0) {
  472. sc.api.b = sc.api.i+memObj.bans.ids.join(',')+"&fields=ban_time";
  473. reqHnd(sc.api.b , apiBanHnd);
  474. }
  475. }
  476. }
  477. });
  478. ratInsert.observe(memObj.cls, {childList: true});
  479. }
  480. }
  481.  
  482. function tableFetch() {
  483. // show animated loading gear
  484. loadGif.classList.remove("js-hidden");
  485. // find required info from table player rows
  486. for (var _rt=0, _rt_len = memObj.cls.rows.length; _rt<_rt_len; _rt++) {
  487. var id = memObj.cls.rows[_rt].getAttribute('data-account_id'),
  488. name = memObj.cls.rows[_rt].getElementsByClassName('player_name')[0].innerHTML.match(/[\w\_]+/)[0];
  489. if (!isNaN(id)) {
  490. memObj.ids.push(id);
  491. s.user[id] = {u:{name:name,id:id}, v:{frag:0,dmg:0,spot:0,def:0,win:0}, wn8:""};
  492. }
  493. }
  494. s.clan = {wn8:0, win:0, mem: _rt_len};
  495. // request and retrieve statistics from API
  496. if (s.clan.mem > 0) {
  497. reqHnd(sc.api.i+memObj.ids.join(','), apiInfoHnd);
  498. }
  499. }
  500.  
  501. // processing information from player API
  502. function apiInfoHnd(resp) {
  503. var data = JSON.parse(resp).data;
  504. for (var a in data) {
  505. if (data.hasOwnProperty(a)) {
  506. var pData = data[a];
  507. if (pData !== null) {
  508. // store stats
  509. var pDataStats = pData.statistics.all;
  510. s.user[pData.account_id].u = {
  511. name: pData.nickname,
  512. id: pData.account_id,
  513. cid: pData.clan_id,
  514. bat: pDataStats.battles,
  515. win: (pDataStats.wins/pDataStats.battles)*100,
  516. dmg: pDataStats.damage_dealt/pDataStats.battles,
  517. frag: pDataStats.frags/pDataStats.battles,
  518. spot: pDataStats.spotted/pDataStats.battles,
  519. def: pDataStats.dropped_capture_points/pDataStats.battles,
  520. wgr: pData.global_rating,
  521. lng: pData.client_language
  522. };
  523. s.clan.win += (!isNaN(s.user[pData.account_id].u.win)) ? s.user[pData.account_id].u.win : 0;
  524. }
  525. }
  526. }
  527. d.getElementById('js-wn8-status').textContent = "50%";
  528. reqHnd(sc.api.v+memObj.ids.join(','), apiVehHnd);
  529. }
  530.  
  531. // processing information from vehicle API and calculate WN8
  532. function apiVehHnd(resp) {
  533. var data = JSON.parse(resp).data;
  534. for (var p in data) {
  535. if (data.hasOwnProperty(p)) {
  536. var vData = data[p];
  537. if (vData !== null) {
  538. var rWin, rDmg, rFrag, rSpot, rDef, wn8 = 0;
  539. if (s.user[p].u.bat > 0) {
  540. for (var v in vData) {
  541. if (vData.hasOwnProperty(v)) {
  542. // go through each vehicle to get expected stats
  543. for (var _so=0, _so_len = statArr.length; _so<_so_len; _so++) {
  544. if (statArr[_so].IDNum == vData[v].tank_id) {
  545. var vehStat = statArr[_so],
  546. dataBattles = vData[v].statistics.battles;
  547. s.user[p].v.frag += vehStat.expFrag * dataBattles;
  548. s.user[p].v.dmg += vehStat.expDamage * dataBattles;
  549. s.user[p].v.spot += vehStat.expSpot * dataBattles;
  550. s.user[p].v.def += vehStat.expDef * dataBattles;
  551. s.user[p].v.win += vehStat.expWinRate * dataBattles;
  552. break;
  553. }
  554. }
  555. }
  556. }
  557. // start calculating wn8
  558. rWin = Math.max(((s.user[p].u.win /(s.user[p].v.win /s.user[p].u.bat)-0.71)/(1-0.71)),0);
  559. rDmg = Math.max(((s.user[p].u.dmg /(s.user[p].v.dmg /s.user[p].u.bat)-0.22)/(1-0.22)),0);
  560. rFrag = Math.max(Math.min(rDmg+0.2,((s.user[p].u.frag/(s.user[p].v.frag/s.user[p].u.bat)-0.12)/(1-0.12))),0);
  561. rSpot = Math.max(Math.min(rDmg+0.1,((s.user[p].u.spot/(s.user[p].v.spot/s.user[p].u.bat)-0.38)/(1-0.38))),0);
  562. rDef = Math.max(Math.min(rDmg+0.1,((s.user[p].u.def /(s.user[p].v.def /s.user[p].u.bat)-0.10)/(1-0.10))),0);
  563. wn8 = 980*rDmg + 210*rDmg*rFrag + 155*rFrag*rSpot + 75*rDef*rFrag + 145*Math.min(1.8,rWin);
  564. }
  565. // store wn8 and add to clan total
  566. s.user[p].wn8 = wn8;
  567. s.clan.wn8 += wn8;
  568. }
  569. }
  570. }
  571. // calculate average wn8 / winrate and store everything in localStorage, then reload page
  572. s.clan.wn8 = s.clan.wn8/s.clan.mem;
  573. s.clan.win = s.clan.win/s.clan.mem;
  574. locStorage("statScriptValues_"+wg.clan.id, s, "set", "string");
  575. locStorage("statScriptDate_"+wg.clan.id, sc.date, "set");
  576. d.getElementById('js-wn8-status').textContent = "100%";
  577. location.reload();
  578. }
  579.  
  580. // processing information from banned API
  581. function apiBanHnd(resp) {
  582. var data;
  583. if (!memObj.bans.api) {
  584. data = JSON.parse(resp).data;
  585. memObj.bans.api = data;
  586. memObj.bans.f = true;
  587. }
  588. else {
  589. data = memObj.bans.api;
  590. }
  591. for (var a in data) {
  592. if (data.hasOwnProperty(a)) {
  593. var bData = data[a];
  594. var memClass = "js-tooltip-id_js-playerslist-account-banned-"+a,
  595. memCell = d.getElementsByClassName(memClass)[0].parentNode;
  596. if (bData.ban_time !== null) {
  597. var banTime = (bData.ban_time > 0) ? new Date(bData.ban_time*1000).toUTCString() : loc[17];
  598. memCell.appendChild(elem("p", "player_rank player_time", banTime));
  599. }
  600. else {
  601. memCell.appendChild(elem("p", "player_rank player_time", loc[18]));
  602. }
  603. }
  604. }
  605. }
  606.  
  607. // helper functions
  608. // filter
  609. function filter(input, type) {
  610. var inputStr = input.toString();
  611. switch(type) {
  612. case (1): // input string into number
  613. return parseFloat(inputStr.replace(/[^\d]/g,""));
  614. case (2): // output number with locale symbol
  615. return inputStr.replace(/(\d)(?=(\d\d\d)+(?!\d))/g, "$1"+loc[0]);
  616. case (3): // input string into number - exclude dots
  617. return parseFloat(inputStr.replace(/[^\d\.]/g,""));
  618. default:
  619. console.error("Error filtering: ", input);
  620. return input;
  621. }
  622. }
  623.  
  624. // colouring
  625. function colStat(input, type, dec, sym) {
  626. var color = colArr.dft[0],
  627. output = input.toFixed(dec);
  628. if (input >= 1000) {
  629. output = filter(input.toFixed(dec),2);
  630. }
  631. for (var c in colArr) {
  632. if (colArr.hasOwnProperty(c)) {
  633. if (input >= colArr[c][colArr.id[type]]) {
  634. color = colArr[c][0]; break;
  635. }
  636. }
  637. }
  638. if (loc[0] !== "," && dec !== 0) {
  639. output = output.replace(/\.(\d+)*$/g,",$1");
  640. }
  641. if (sym) {
  642. output += sym;
  643. }
  644. return "<font color='"+color+"'>"+output+"</font>";
  645. }
  646.  
  647. // quick creation of element
  648. function elem(tag, name, html, type) {
  649. var element = d.createElement(tag);
  650. if (name) {
  651. element.className = name;
  652. }
  653. if (html) {
  654. if (/</.test(html)) {
  655. element.innerHTML = html;
  656. }
  657. else {
  658. element.textContent = html;
  659. }
  660. }
  661. if (type) {
  662. element.type = type;
  663. }
  664. return element;
  665. }
  666.  
  667. // settings handler
  668. function settingsHnd(name, text, state, dftState, wlist) {
  669. var listItem = elem("li", "settingItem", ""),
  670. listItems = d.createDocumentFragment();
  671. if (name == "wnRefresh") {
  672. var refreshBtn = elem("div", "b-settingLink", "<a>"+text+"</a>");
  673. refreshBtn.addEventListener('click', function() {localStorage.removeItem("wnExpValues"); location.reload();}, false);
  674. listItems.appendChild(refreshBtn);
  675. }
  676. else if (name == "cleanStorage") {
  677. var cleanBtn = elem("div", "b-settingLink", "<a>"+text+"</a>");
  678. cleanBtn.addEventListener('click', function() {localStorage.clear(); location.reload();}, false);
  679. listItems.appendChild(cleanBtn);
  680. }
  681. else {
  682. var optCheckDiv = elem("div", "b-checkbox", "<span class='b-checkbox_checker'></span>"),
  683. optLabel = elem("label", "b-combobox-label", text),
  684. optCheck = elem("input", "l-box", "", "checkbox");
  685. optLabel.htmlFor = name;
  686. optCheck.name = name;
  687. optCheck.id = name;
  688. if (state) {
  689. optCheckDiv.classList.add("b-checkbox__checked");
  690. optLabel.classList.add("b-combobox-label__checked");
  691. }
  692. optCheck.checked = (state !== undefined) ? state : dftState;
  693. optCheck.addEventListener('click', function() {
  694. locStorage('statScript_' + this.name, this.checked, "set");
  695. d[this.name] = this.checked;
  696. this.parentNode.classList.toggle('b-checkbox__checked');
  697. this.parentNode.parentNode.classList.toggle('b-combobox-label__checked');
  698. return this.checked;
  699. }, false);
  700. d[optCheck.name] = optCheck.checked;
  701. optCheckDiv.insertBefore(optCheck, optCheckDiv.firstChild);
  702. optLabel.appendChild(optCheckDiv);
  703. listItems.appendChild(optLabel);
  704. if (name == "whitelist") {
  705. var optText = elem("textarea", "l-textarea", "");
  706. optText.placeholder = "Add clanID seperated by comma without spaces: 500004502,500010805";
  707. if (wlist) {
  708. optText.value = wlist;
  709. }
  710. optText.addEventListener('input', function() {
  711. locStorage('statScript_whitelist_list', optText.value.split(","), "set");
  712. }, false);
  713. listItems.appendChild(optText);
  714. }
  715. }
  716. listItem.appendChild(listItems);
  717. return listItem;
  718. }
  719.  
  720. // links handler
  721. function linksHnd(parent, links) {
  722. var uRows = d.createDocumentFragment();
  723. for (var _l=0, _l_len = links.length; _l<_l_len; ++_l) {
  724. if (links[_l] instanceof HTMLElement) {
  725. uRows.appendChild(links[_l]);
  726. }
  727. else {
  728. uRows.appendChild((links[_l][0] && links[_l][1]) ? elem("li", "", links[_l][1]) : elem("li", "statname", links[_l][0]));
  729. }
  730. }
  731. parent.appendChild(uRows);
  732. }
  733.  
  734. // localStorage handler
  735. function locStorage(name, data, type, mode) {
  736. var storage;
  737. switch(type) {
  738. case ("set"):
  739. if (mode == "string") {
  740. data = JSON.stringify(data);
  741. }
  742. storage = localStorage.setItem(name, data);
  743. break;
  744. case ("get"):
  745. storage = localStorage.getItem(name);
  746. if (mode == "parse") {
  747. storage = JSON.parse(storage);
  748. }
  749. break;
  750. default: break;
  751. }
  752. return storage;
  753. }
  754. // end helper functions
  755.  
  756. // wnefficiency handler
  757. function wnHnd(resp) {
  758. locStorage("wnExpValues", resp, "set");
  759. locStorage("wnExpDate", sc.date, "set");
  760. locStorage("wnExpVers", [sc.vers, JSON.parse(resp).header.version], "set", "string");
  761. location.reload();
  762. }
  763. function wnHnd_error(error) {
  764. console.error("Error accessing WNEfficiency.net", error);
  765. }
  766. // end wnefficiency handler
  767.  
  768. // retrieval function
  769. function reqHnd(url, handler, error) {
  770. GM_xmlhttpRequest({
  771. method: "GET",
  772. url: url,
  773. headers: {
  774. "Accept": "text/xml"
  775. },
  776. onload: function(resp) {
  777. if (resp.readyState == 4 && resp.status == 200 && resp.statusText == "OK") {
  778. handler(resp.responseText);
  779. }
  780. else {
  781. error(resp.responseText);
  782. }
  783. },
  784. onerror: function(resp) {
  785. error(resp.responseText);
  786. }
  787. });
  788. }
  789. }(window));