WoTStatScript - ForumsExtendedStat

Adds stats and links for posts on official World of Tanks forums

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

  1. // ==UserScript==
  2. // @name WoTStatScript - ForumsExtendedStat
  3. // @version 0.9.13.0.2
  4. // @description Adds stats and links for posts on official World of Tanks forums
  5. // @author Orrie
  6. // @contributor seriych
  7. // @namespace http://forum.worldoftanks.eu/index.php?/topic/263423-
  8. // @icon http://dl.dropboxusercontent.com/u/12497046/wot/projects/statscript/img/icon.png
  9. // @include /^http:\/\/forum\.worldoftanks\.(eu|com|ru|asia|kr)\/\S+(\/topic\/|\?showtopic=)/
  10. // @include /^http:\/\/forum\.worldoftanks\.(eu|com|ru|asia|kr)\/index\.php\?app=members&module=messaging&section=view&do=showConversation&topicID=\d+/
  11. // @include /^http:\/\/forum\.worldoftanks\.(eu|com|ru|asia|kr)\/index\.php\?\/forum/\S+/
  12. // @grant GM_xmlhttpRequest
  13. // @license MIT License
  14. // ==/UserScript==
  15. (function() {
  16. // global vars
  17. var d = document;
  18.  
  19. // get server info and webpage
  20. var wg = {host:d.location.host, href:d.location.href, clan:{}};
  21. wg.srv = d.location.host.match(/\.(eu|ru|com|asia|kr)/)[1];
  22. wg.topic = /(\/topic\/|\?showtopic=|messaging\&)/.test(wg.href);
  23. wg.forum = /\/forum\//.test(wg.href);
  24. wg.login = !!d.getElementsByClassName('js-cm-login-link')[0];
  25.  
  26. // server, API and cluster settings
  27. var sc = {
  28. vers: "0.9.13.0.2",
  29. host: "http://greasyfork.org/scripts/660-wotstatscript-forumsextendedstat",
  30. top: {
  31. eu: "http://forum.worldoftanks.eu/index.php?showtopic=263423",
  32. na: "http://forum.worldoftanks.com/index.php?showtopic=404652"
  33. },
  34. api: {
  35. ru: "98ca7c4fb108175b67d6505b9c3f3ebd",
  36. eu: "a7595640a90bf2d19065f3f2683b171c",
  37. com: "bf5dba0efd444d75147b6222dd903fd2",
  38. asia: "95f8713eccd322e52dbf521dbd28b19c",
  39. kr: "ffea0f1c3c5f770db09357d94fe6abfb"
  40. },
  41. sym: ",",
  42. wn: "http://www.wnefficiency.net/exp/expected_tank_values_latest.json"
  43. };
  44. // script threadlink
  45. sc.link = "<div class='b-scriptlink'><a target='_blank' href="+sc.host+">Script</a> v"+sc.vers+" - <a target='_blank' href="+((wg.srv == "na") ? sc.top.na : sc.top.eu)+">Thread</a></div>";
  46.  
  47. // external site support - wotlabs, noobmeter, wot-news, wotstats, wotcs, wot-life, wotstats.com.au and wotreplays
  48. var srv = {
  49. s: "en", wl: false, nm: false, vb: false, ws: false, wr: false, ct: false, wlf: false
  50. };
  51. switch(wg.srv) {
  52. case ("eu"): // eu server
  53. srv.wl = srv.nm = srv.vb = srv.ws = srv.wr = srv.ct = srv.wlf = wg.srv;
  54. break;
  55. case ("ru"): // ru server
  56. srv.wl = srv.nm = srv.vb = srv.ws = srv.wr = srv.ct = srv.wlf = wg.srv;
  57. srv.s = "ru";
  58. sc.sym = " ";
  59. break;
  60. case ("com"): // na server
  61. srv.wl = srv.nm = srv.vb = srv.ws = srv.ct = "na"; srv.wr = "com";
  62. break;
  63. case ("asia"): // asia server
  64. srv.wl = srv.nm = srv.vb = srv.ws = "sea"; srv.ct = wg.srv; srv.wr = "com";
  65. break;
  66. case ("kr"): // korean server
  67. srv.wl = srv.nm = srv.vb = srv.ws = srv.ct = srv.ch = wg.srv; srv.wr = "com";
  68. break;
  69. default: break;
  70. }
  71.  
  72. // data uri
  73. var uri = {
  74. icon_arrow:"",
  75. q_mark: ""
  76. };
  77.  
  78. // fetch wnefficiency values - check if array exists in localStorage, otherwise fetch and reload page
  79. var statArr = [],
  80. wnExpValues = locStorage("wnExpValues", "", "get", "parse"),
  81. wnExpDate = locStorage("wnExpDate", "", "get", "parse")+12096e5 >= Date.now(), // true if timestamp is less than 2 weeks old, refresh list if false.
  82. wnExpVers = locStorage("wnExpVers", "", "get", "parse") || "";
  83. if (wnExpVers[0]==sc.vers && wnExpValues && wnExpDate) {
  84. statArr = wnExpValues.data;
  85. }
  86. else {
  87. reqHnd(sc.wn, wnHnd);
  88. }
  89.  
  90. // style contents
  91. var style = elem("style", "wotstatscript", "", "text/css"),
  92. styleText = [
  93. // script header rules
  94. "#common_menu .cm-menu__user > *:not(.cm-notifications):not(.js-cm-user-menu-dropdown) {display: inline-block}",
  95. "#common_menu .cm-parent-link:hover {cursor: inherit; color: #707273 !important;}",
  96. "#common_menu .b-scriptlink a {color: #E5B12E;}",
  97. "#common_menu .b-scriptlink a:hover {color: #FFBE4C; text-shadow: 0px 0px 7px rgba(255, 126, 0, 0.7);}",
  98. "#common_menu .cm-get-gold-link {margin: 0 12px;}",
  99. // settings menu rules
  100. "#common_menu .menu-settings {text-align: left;}",
  101. "#common_menu .menu-settings .cm-parent-link:hover {cursor: pointer;}",
  102. "#common_menu .menu-settings label {display: table; line-height: normal; cursor: pointer;}",
  103. "#common_menu .menu-settings .l-box {display: none;}",
  104. "#common_menu .menu-settings .b-checkbox {height: 16px; width: 16px; float: left; margin-right: 5px;}",
  105. "#common_menu .menu-settings .b-checkbox span {height: 16px; width: 16px;}",
  106. "#common_menu .menu-settings .b-combobox-label__checked {color: #DCDCDC;}",
  107. "#common_menu .menu-settings .settingItem .b-combobox-label:hover {color: #DCDCDC;}",
  108. "#common_menu .menu-settings .settingItem .b-combobox-label:hover .b-checkbox {background-position: 0px -34px; box-shadow: 0px 0px 10px 1px rgba(191, 166, 35, 0.15), 0px 0px 3px 1px rgba(191, 166, 35, 0.25);}",
  109. "#common_menu .menu-settings .settingItem .b-combobox-label:hover .b-checkbox.b-checkbox__checked {background-position: 0px -68px;}",
  110. "#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;}",
  111. "#common_menu .menu-settings textarea::-webkit-input-placeholder {color: #FFFFFF;}",
  112. "#common_menu .menu-settings textarea::-moz-placeholder {color: #FFFFFF;}",
  113. "#common_menu .menu-settings .b-settingLink {line-height: 26px;}",
  114. "#common_menu .menu-settings .b-settingLink a {cursor: pointer; color: #B1B2B3; text-shadow: 0px 1px 1px rgba(0, 0, 0, 0.5);}",
  115. "#common_menu .menu-settings .b-settingLink a:hover {color: #FFFFFF; text-shadow: 0px 1px 1px rgba(0, 0, 0, 0.75);}",
  116. // forum rules
  117. ".topic_list {table-layout: auto}",
  118. ".col_f_stats {width: 200px;}",
  119. ".col_f_preview {width: 10px !important;}",
  120. ".col_f_views {width: auto;}",
  121. ".col_f_post {width: auto;}",
  122. ".col_f_stats .i-xvm-lang {margin: 6px 0 0 0;}",
  123. ".t-row-stats {float: right; table-layout: fixed; width: 175px;}",
  124. ".realm_ru .t-row-stats {width: 200px;}",
  125. "table.ipb_table .t-row-stats td {border-bottom: 0; font-size: 11px; padding: 0;}",
  126. "#announcements .t-row-stats td {border-bottom: 0;}",
  127. ".s-no-stats {color: #A03737; display: table; font-size: 14px; margin: 0 auto;}",
  128. // thread rules
  129. ".author_info {min-height: 310px; padding: 0 10px 10px 14px;}",
  130. ".user_details > br {display: none;}",
  131. ".basic_info {margin: 0 0 4px;}",
  132. ".post_count.margin-bottom {margin: 0 0 5px;}",
  133. ".battles_count {display: none;}",
  134. ".i-xvm-lang {vertical-align: text-top;}",
  135. ".t-table-stats {margin: 5px 0; width: auto;}",
  136. ".t-table-stats td {padding: 0 5px 0 0; line-height: 16px;}",
  137. ".t-table-stats td:last-of-type {font-weight: bold;}",
  138. ".b-table-links {height: 25px; margin: 5px 0px 0px;}",
  139. ".t-table-links {background: #EDEDED; border: 1px solid #d5d5d5; border-radius: 3px; box-shadow: 0 2px 2px rgba(0, 0, 0, 0.1), 0 0 3px 3px rgba(255, 255, 255, 0.2); display: inline-block; overflow: hidden; padding: 3px 3px 0; position: absolute; text-overflow: ellipsis; white-space: nowrap; height: 25px; width: 90px; z-index: 1;}",
  140. ".t-table-links:hover {border-color: #7D7D7D; height: auto; width: auto;}",
  141. ".t-table-links tbody {display: table; width: 180px;}",
  142. ".t-table-links td {padding: 0 0 2px; font-weight: bold;}",
  143. ".t-table-links td a { background: url("+uri.icon_arrow+") no-repeat scroll 0 2px rgba(0, 0, 0, 0); padding: 0 0 0 10px; vertical-align: middle;}",
  144. ".t-table-links td a:hover {background: url("+uri.icon_arrow+") no-repeat scroll -244px 2px rgba(0, 0, 0, 0);}",
  145. ".t-table-links tr:last-of-type td {}",
  146. // korean realm widths
  147. ".realm_kr .t-table-links {width: 108px;}",
  148. ".realm_kr .t-table-links:hover {width: 215px;}",
  149. ".realm_kr .t-table-links tbody {width: 215px;}"
  150. ];
  151. style.textContent = styleText.join("");
  152. d.head.appendChild(style);
  153. // end style
  154.  
  155. // colour scale array
  156. var colArr = {
  157. // col wr bat wn8 wn7 eff nm pr
  158. sUni: [ "#5A3175", 65, 30000, 2900, 2050, 2050, 2000 ], // 99.99% super unicum
  159. uni: [ "#83579D", 60, 25000, 2450, 1850, 1800, 1950, 9930 ], // 99.90% unicum
  160. gr8: [ "#3972C6", 56, 21000, 2000, 1550, 1500, 1750, 8525 ], // 99.00% great
  161. vGud: [ "#4099BF", 54, 17000, 1600, 1350 ], // 95.00% very good
  162. good: [ "#4D7326", 52, 13000, 1200, 1100, 1200, 1450, 6340 ], // 82.00% good
  163. aAvg: [ "#849B24", 50, 10000, 900 ], // 63.00% above average
  164. avg: [ "#CCB800", 48, 7000, 650, 900, 900, 1250, 4185 ], // 40.00% average
  165. bAvg: [ "#CC7A00", 47, 3000, 450, 700, 600, 1150, 2020 ], // 20.00% below average
  166. bas: [ "#CD3333", 46, 1000, 300, 500 ], // 6.00% basic
  167. beg: [ "#930D0D", 0, 0, 0, 0, 0, 0, 0 ], // 0.00% beginner
  168. dft: [ "#6B6B6B" ], // default
  169. id: { "col": 0, "wr": 1, "bat": 2, "wn8": 3, "wn7": 4, "eff": 5, "nm": 6, "pr": 7 } // type identifier
  170. };
  171.  
  172. // localization
  173. var loc = [
  174. { en: "Client", ru: "Клиент"},
  175. { en: "Winrate:", ru: "Винрейт:"},
  176. { en: "Battles:", ru: "Боев:"},
  177. { en: "Rating:", ru: "Рейтинг:"},
  178. { en: "WN8:", ru: "WN8:"},
  179. { en: "Please Login for Stats", ru: "Please Login for Stats"},
  180. { en: "No Stats Found", ru: "No Stats Found"},
  181. { en: "Script Menu", ru: "Script Menu"},
  182. { en: "Refresh WN8 Table", ru: "Refresh WN8 Table"}
  183. //{ en: "f00_en", ru: "f00_ru"},
  184. ];
  185. // process localization
  186. for (var _l=0, l_len = loc.length; _l<l_len; _l++) {
  187. loc[_l] = loc[_l][srv.s];
  188. }
  189.  
  190. // script link
  191. var userLink_div = elem("div", "cm-parent-link", sc.link),
  192. userSet_div = elem("div", "menu-settings menu-top_item", "<span class='cm-parent-link js-cm-dropdown-link'>"+loc[7]+"<span class='cm-arrow'></span></span>"),
  193. userSet_list = elem("ul", "cm-sublist", "<li class='settingItem'><div class='b-settingLink'><a>"+loc[8]+" [v"+wnExpVers[1]+"]</a></div></li>"),
  194. navMenu = d.getElementById('common_menu'),
  195. navUser = navMenu.getElementsByClassName('cm-menu__user')[0];
  196. userSet_div.firstElementChild.addEventListener('click', function() {this.classList.toggle('cm-parent-link__opened'); this.nextSibling.classList.toggle('cm-sublist__opened');}, false);
  197. userSet_list.firstElementChild.firstElementChild.addEventListener('click', function() {localStorage.removeItem("wnExpValues"); location.reload();}, false);
  198. userSet_div.appendChild(userSet_list);
  199. if (navUser) {
  200. navUser.insertBefore(userLink_div, navUser.firstChild);
  201. navUser.insertBefore(userSet_div, navUser.firstChild);
  202. }
  203. else {
  204. var navLook = new MutationObserver(function() {
  205. var navUser = navMenu.getElementsByClassName('cm-menu__user')[0];
  206. navUser.insertBefore(userLink_div, navUser.firstChild);
  207. navUser.insertBefore(userSet_div, navUser.firstChild);
  208. navLook.disconnect();
  209. });
  210. navLook.observe(navMenu, {childList: true});
  211. }
  212.  
  213. // formula calculations and variables
  214. // create global post variable
  215. var postObj = {
  216. post: d.querySelectorAll(".post_wrap, .post_wrap__wg"),
  217. span: d.querySelectorAll("span.desc.lighter.blend_links"),
  218. cls: [],
  219. ids: [],
  220. num: []
  221. },
  222. s = {};
  223. switch(true) {
  224. case (wg.topic):
  225. postObj.cls = postObj.post; break;
  226. case (wg.forum):
  227. postObj.cls = postObj.span; break;
  228. default: break;
  229. }
  230.  
  231. // fetch userids and store all posts into one obj for later use
  232. if (wg.topic || (wg.forum && wg.login === false)) {
  233. for (var _t=0, _t_len = postObj.cls.length; _t<_t_len; _t++) {
  234. var id;
  235. switch(true) {
  236. case (wg.topic):
  237. id = postObj.post[_t].getElementsByClassName("ipsUserPhotoLink")[0].getAttribute('href').match(/\-(\d+)\//)[1]; break;
  238. case (wg.forum):
  239. var id_url = postObj.span[_t].getElementsByClassName("url")[0];
  240. if (id_url) {
  241. id = id_url.getAttribute('href').match(/\-(\d+)\//)[1];
  242. }
  243. break;
  244. default: break;
  245. }
  246. if (!isNaN(id)) {
  247. var users = postObj.ids.length,
  248. index = postObj.ids.indexOf(id);
  249. s[id] = {u:{},v:{frag:0,dmg:0,spot:0,def:0,win:0},wn8:""};
  250. if (index>-1) {
  251. postObj.num[index][postObj.num[index].length] = _t;
  252. }
  253. else {
  254. postObj.ids[users] = id;
  255. postObj.num[users] = [_t];
  256. }
  257. }
  258. }
  259. }
  260. else {
  261. navUser.insertBefore(elem("div", "cm-parent-link b-login-msg", "<div>"+loc[5]+"</div>"), navUser.firstChild);
  262. }
  263.  
  264. // request and retrieve statistics from API
  265. if (postObj.ids.length > 0) {
  266. sc.api.i = "http://api.worldoftanks."+wg.srv+"/wot/account/info/?application_id="+sc.api[wg.srv]+"&account_id="+postObj.ids.join(',');
  267. sc.api.v = "http://api.worldoftanks."+wg.srv+"/wot/account/tanks/?application_id="+sc.api[wg.srv]+"&account_id="+postObj.ids.join(',');
  268. reqHnd(sc.api.i, apiInfoHnd);
  269. }
  270. else {
  271. console.error("No post IDs found or not logged in");
  272. }
  273.  
  274. // processing information from player API
  275. function apiInfoHnd(resp) {
  276. var data = JSON.parse(resp).data;
  277. for (var a in data) {
  278. if (data.hasOwnProperty(a)) {
  279. var pData = data[a];
  280. if (pData !== null) {
  281. // store stats
  282. var pDataStats = pData.statistics.all;
  283. s[pData.account_id].u = {
  284. name: pData.nickname,
  285. id: pData.account_id,
  286. cid: pData.clan_id,
  287. bat: pDataStats.battles,
  288. win: (pDataStats.wins/pDataStats.battles)*100,
  289. dmg: pDataStats.damage_dealt/pDataStats.battles,
  290. frag: pDataStats.frags/pDataStats.battles,
  291. spot: pDataStats.spotted/pDataStats.battles,
  292. def: pDataStats.dropped_capture_points/pDataStats.battles,
  293. wgr: pData.global_rating,
  294. lng: pData.client_language
  295. };
  296. }
  297. else {
  298. // store stats
  299. s[parseFloat(a)].u = {
  300. bat: 0
  301. };
  302. }
  303. }
  304. }
  305. reqHnd(sc.api.v, apiVehHnd);
  306. }
  307.  
  308. // processing information from vehicle API and calculate WN8
  309. function apiVehHnd(resp) {
  310. var data = JSON.parse(resp).data;
  311. for (var p in data) {
  312. if (data.hasOwnProperty(p)) {
  313. var vData = data[p];
  314. if (vData !== null) {
  315. var rWin, rDmg, rFrag, rSpot, rDef, wn8 = 0;
  316. if (s[p].u.bat > 0) {
  317. for (var v in vData) {
  318. if (vData.hasOwnProperty(v)) {
  319. for (var _so=0, _so_len = statArr.length; _so<_so_len; _so++) {
  320. if (statArr[_so].IDNum == vData[v].tank_id) {
  321. var vehStat = statArr[_so],
  322. dataBattles = vData[v].statistics.battles;
  323. s[p].v.frag += vehStat.expFrag * dataBattles;
  324. s[p].v.dmg += vehStat.expDamage * dataBattles;
  325. s[p].v.spot += vehStat.expSpot * dataBattles;
  326. s[p].v.def += vehStat.expDef * dataBattles;
  327. s[p].v.win += vehStat.expWinRate * dataBattles;
  328. break;
  329. }
  330. }
  331. }
  332. }
  333. rWin = Math.max(((s[p].u.win /(s[p].v.win /s[p].u.bat)-0.71)/(1-0.71)),0);
  334. rDmg = Math.max(((s[p].u.dmg /(s[p].v.dmg /s[p].u.bat)-0.22)/(1-0.22)),0);
  335. rFrag = Math.max(Math.min(rDmg+0.2,((s[p].u.frag/(s[p].v.frag/s[p].u.bat)-0.12)/(1-0.12))),0);
  336. rSpot = Math.max(Math.min(rDmg+0.1,((s[p].u.spot/(s[p].v.spot/s[p].u.bat)-0.38)/(1-0.38))),0);
  337. rDef = Math.max(Math.min(rDmg+0.1,((s[p].u.def /(s[p].v.def /s[p].u.bat)-0.10)/(1-0.10))),0);
  338. wn8 = 980*rDmg + 210*rDmg*rFrag + 155*rFrag*rSpot + 75*rDef*rFrag + 145*Math.min(1.8,rWin);
  339. }
  340. // store wn8
  341. s[p].wn8 = colStat(wn8,"wn8",0);
  342. }
  343. }
  344. }
  345. statInsert();
  346. }
  347.  
  348. // insert stats and links to every post
  349. function statInsert() {
  350. for (var y in s) {
  351. if (s.hasOwnProperty(y)) {
  352. var iPost = postObj.ids.indexOf(y), flag;
  353. if (iPost >- 1 && s[y].u.id) {
  354. if (wg.topic) {
  355. var userStats = [
  356. "<td>"+loc[1]+"</td><td>"+colStat((s[y].u.bat > 0) ? s[y].u.win : 0,"wr",2,"%")+"</td>",
  357. "<td>"+loc[2]+"</td><td>"+colStat(s[y].u.bat,"bat",0,"")+"</td>",
  358. "<td>"+loc[3]+"</td><td>"+colStat(s[y].u.wgr,"pr",0,"")+"</td>",
  359. "<td>"+loc[4]+"</td><td>"+s[y].wn8+"</td>"
  360. ],
  361. userLinks = [
  362. [
  363. ["<a target='_blank' href='http://worldoftanks."+wg.srv+"/community/accounts/"+s[y].u.id+"-"+s[y].u.name+"/'>Player Profile</a>"],
  364. [srv.wl, "<a target='_blank' href='http://wotlabs.net/"+srv.wl+"/player/"+s[y].u.name+"'>WoTLabs</a>"]
  365. ],
  366. [
  367. [srv.vb, "<a target='_blank' href='http://www.vbaddict.net/player/"+s[y].u.name.toLowerCase()+"-"+srv.vb+"'>vBAddict</a>"],
  368. [srv.nm, "<a target='_blank' href='http://noobmeter.com/player/"+srv.nm+"/"+s[y].u.name+"'>Noobmeter</a>"]
  369. ],
  370. [
  371. [srv.ws, "<a target='_blank' href='http://wotstats.org/stats/"+srv.ws+"/"+s[y].u.name+"/'>WoTstats</a>"],
  372. [srv.ct, "<a target='_blank' href='http://clantools.us/servers/"+srv.ct+"/players?id="+s[y].u.id+"'>Clan Tools</a>"]
  373. ],
  374. [
  375. [srv.wr, "<a target='_blank' href='http://wotreplays."+srv.wr+"/player/"+s[y].u.name+"'>WoTReplays</a>"],
  376. (wg.srv=="ru") ? [srv.wr, "<a target='_blank' href='http://wots.com.ua/user/stats/"+s[y].u.name+"'>WoTS.com.ua</a>"] : [srv.wlf, "<a target='_blank' href='http://en.wot-life.com/"+srv.wlf+"/player/"+s[y].u.name+"/'>WoT-Life</a>"]
  377. ]
  378. ];
  379. for (var _i=0, _i_len = postObj.num[iPost].length; _i<_i_len; _i++) {
  380. var groupTitle = postObj.post[postObj.num[iPost][_i]].getElementsByClassName("group_title")[0],
  381. basicInfo = postObj.post[postObj.num[iPost][_i]].getElementsByClassName("basic_info")[0],
  382. infoClass = postObj.post[postObj.num[iPost][_i]].getElementsByClassName("author_info")[0],
  383. statTable = elem("table", "t-table-stats", ""),
  384. urlDiv = elem("div", "b-table-links", ""),
  385. urlTable = elem("table", "t-table-links", "<tbody></tbody>");
  386. flag = elem("img", "i-xvm-lang", "", "", "https://bytebucket.org/seriych/worldoftanksforumextendedstat.user.js/raw/tip/data/img/lang/"+s[y].u.lng+".png");
  387. for (var _sr=0, _sr_len = userStats.length; _sr<_sr_len; ++_sr) {
  388. statTable.appendChild(elem("tr", "", userStats[_sr]));
  389. }
  390. for (var _l=0, _l_len = userLinks.length; _l<_l_len; ++_l) {
  391. var uRow = elem("tr", "", "");
  392. for (var _lr=0, _lr_len = userLinks[_l].length; _lr<_lr_len; ++_lr) {
  393. uRow.appendChild((userLinks[_l][_lr][0] && userLinks[_l][_lr][1]) ? elem("td", "", userLinks[_l][_lr][1]) : elem("td", "", userLinks[_l][_lr][0]));
  394. }
  395. urlTable.firstElementChild.appendChild(uRow);
  396. }
  397. if (wg.srv !== "ru") {
  398. groupTitle.appendChild(flag);
  399. }
  400. flag.title = s[y].u.lng.toUpperCase()+" "+loc[0];
  401. flag.addEventListener('error', function(){this.src=uri.q_mark;});
  402. basicInfo.insertBefore(statTable, basicInfo.lastElementChild);
  403. urlDiv.appendChild(urlTable);
  404. infoClass.appendChild(urlDiv);
  405. }
  406. }
  407. else if (wg.forum) {
  408. for (var _r=0, _r_len = postObj.num[iPost].length; _r<_r_len; _r++) {
  409. var statCell, id_url = postObj.span[postObj.num[iPost][_r]].getElementsByClassName("url")[0],
  410. threadRow = postObj.span[postObj.num[iPost][_r]].parentNode.parentNode,
  411. threadRowClsList = threadRow.classList;
  412. if (threadRowClsList.contains("announcement")) {
  413. statCell = threadRow.insertCell(1);
  414. }
  415. else if (threadRowClsList.contains("__topic")) {
  416. statCell = threadRow.insertCell(3);
  417. }
  418. statCell.className = "col_f_stats";
  419. if (!id_url || s[y].u.bat === 0) {
  420. statCell.appendChild(elem("span", "s-no-stats", loc[6]));
  421. }
  422. else {
  423. var statsRow = [
  424. "<td>"+loc[1]+"</td><td>"+colStat((s[y].u.bat > 0) ? s[y].u.win : 0,"wr",2,"%")+"</td><td>"+loc[2]+"</td><td>"+colStat(s[y].u.bat,"bat",0,"")+"</td>",
  425. "<td>"+loc[3]+"</td><td>"+colStat(s[y].u.wgr,"pr",0,"")+"</td><td>"+loc[4]+"</td><td>"+s[y].wn8+"</td>",
  426. ],
  427. rowTable = elem("table", "t-row-stats", "");
  428. flag = elem("img", "i-xvm-lang", "", "", "https://bytebucket.org/seriych/worldoftanksforumextendedstat.user.js/raw/tip/data/img/lang/"+s[y].u.lng+".png");
  429. for (var _rr=0, _rr_len = statsRow.length; _rr<_rr_len; ++_rr) {
  430. rowTable.appendChild(elem("tr", "", statsRow[_rr]));
  431. }
  432. if (wg.srv !== "ru") {
  433. statCell.appendChild(flag);
  434. }
  435. flag.title = s[y].u.lng.toUpperCase()+" "+loc[0];
  436. flag.addEventListener('error', function(){this.src=uri.q_mark;});
  437. statCell.appendChild(rowTable);
  438. }
  439. }
  440. }
  441. }
  442. }
  443. }
  444. }
  445.  
  446. // helper functions
  447. // colouring
  448. function colStat(input, type, dec, sym) {
  449. var color = colArr.dft[0],
  450. output = input.toFixed(dec);
  451. if (sym) {
  452. output += sym;
  453. }
  454. if (input >= 1000) {
  455. output = input.toFixed(dec).toString().replace(/(\d)(?=(\d\d\d)+(?!\d))/g, "$1"+sc.sym);
  456. }
  457. for (var c in colArr) {
  458. if (colArr.hasOwnProperty(c)) {
  459. if (input >= colArr[c][colArr.id[type]]) {
  460. color = colArr[c][0]; break;
  461. }
  462. }
  463. }
  464. return "<font color='"+color+"'>"+output+"</font>";
  465. }
  466.  
  467. // quick creation of element
  468. function elem(tag, name, html, type, src) {
  469. var element = d.createElement(tag);
  470. if (name) {element.className = name;}
  471. if (html) {
  472. if (/</.test(html)) {
  473. element.innerHTML = html;
  474. }
  475. else {
  476. element.textContent = html;
  477. }
  478. }
  479. if (type) {element.type = type;}
  480. if (src) {element.src = src;}
  481. return element;
  482. }
  483.  
  484. // localStorage handler
  485. function locStorage(name, data, type, mode) {
  486. var storage;
  487. switch(type) {
  488. case ("set"):
  489. if (mode == "string") {
  490. data = JSON.stringify(data);
  491. }
  492. storage = localStorage.setItem(name, data);
  493. break;
  494. case ("get"):
  495. storage = localStorage.getItem(name);
  496. if (mode == "parse") {
  497. storage = JSON.parse(storage);
  498. }
  499. break;
  500. default: break;
  501. }
  502. return storage;
  503. }
  504. // end helper functions
  505.  
  506. // wnefficiency handler
  507. function wnHnd(resp) {
  508. locStorage("wnExpValues", resp, "set");
  509. locStorage("wnExpDate", Date.now(), "set");
  510. locStorage("wnExpVers", [sc.vers, JSON.parse(resp).header.version], "set", "string");
  511. location.reload();
  512. }
  513. // end wnefficiency handler
  514.  
  515. // request handler
  516. function reqHnd(url, handler, error) {
  517. GM_xmlhttpRequest({
  518. method: "GET",
  519. url: url,
  520. headers: {
  521. "Accept": "text/xml"
  522. },
  523. onload: function(resp) {
  524. if (resp.readyState == 4 && resp.status == 200 && resp.statusText == "OK") {
  525. handler(resp.responseText);
  526. }
  527. else {
  528. error(resp.responseText);
  529. }
  530. },
  531. onerror: function(resp) {
  532. error(resp.responseText);
  533. }
  534. });
  535. }
  536. }(window));