WoTStatScript - Clanpage

More info for World of Tanks clan page.

目前為 2015-11-05 提交的版本,檢視 最新版本

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