WoTStatScript - Clanpage

More info for World of Tanks clan page.

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