// ==UserScript==
// @name WoWsStatInfo
// @author Vov_chiK
// @description Расширенная статистика и функционал на сайте World of Warships.
// @copyright 2015+, Vov_chiK
// @license GPL version 3 or any later version; http://www.gnu.org/copyleft/gpl.html
// @namespace http://forum.walkure.pro/
// @version 0.5.2.27
// @creator Vov_chiK
// @include http://worldofwarships.ru/ru/community/accounts/*
// @include http://forum.worldofwarships.ru/index.php?/topic/*
// @include http://forum.worldofwarships.ru/index.php?/user/*
// @include http://ru.wargaming.net/clans/*/players*
// @include http://worldofwarships.eu/*/community/accounts/*
// @include http://forum.worldofwarships.eu/index.php?/topic/*
// @include http://forum.worldofwarships.eu/index.php?/user/*
// @include http://eu.wargaming.net/clans/*/players*
// @include http://worldofwarships.com/*/community/accounts/*
// @include http://forum.worldofwarships.com/index.php?/topic/*
// @include http://forum.worldofwarships.com/index.php?/user/*
// @include http://na.wargaming.net/clans/*/players*
// @include http://worldofwarships.asia/*/community/accounts/*
// @include http://forum.worldofwarships.asia/index.php?/topic/*
// @include http://forum.worldofwarships.asia/index.php?/user/*
// @include http://asia.wargaming.net/clans/*/players*
// @match http://worldofwarships.ru/ru/community/accounts/*
// @match http://forum.worldofwarships.ru/index.php?/topic/*
// @match http://forum.worldofwarships.ru/index.php?/user/*
// @match http://ru.wargaming.net/clans/*/players*
// @match http://worldofwarships.eu/*/community/accounts/*
// @match http://forum.worldofwarships.eu/index.php?/topic/*
// @match http://forum.worldofwarships.eu/index.php?/user/*
// @match http://eu.wargaming.net/clans/*/players*
// @match http://worldofwarships.com/*/community/accounts/*
// @match http://forum.worldofwarships.com/index.php?/topic/*
// @match http://forum.worldofwarships.com/index.php?/user/*
// @match http://na.wargaming.net/clans/*/players*
// @match http://worldofwarships.asia/*/community/accounts/*
// @match http://forum.worldofwarships.asia/index.php?/topic/*
// @match http://forum.worldofwarships.asia/index.php?/user/*
// @match http://asia.wargaming.net/clans/*/players*
// @grant GM_xmlhttpRequest
// @downloadURL none
// ==/UserScript==
(function(window){
	/* ===== Main function ===== */
	function WoWsStatInfo(){
		var VersionWoWsStatInfo = '0.5.2.27';
		
		var WoWsStatInfoLinkLoc = [];
		WoWsStatInfoLinkLoc['ru'] = 'http://forum.worldofwarships.ru/index.php?/topic/19158-';
		WoWsStatInfoLinkLoc['asia'] = 'http://forum.worldofwarships.asia/index.php?/topic/8950-';
		WoWsStatInfoLinkLoc['na'] = 'http://forum.worldofwarships.com/index.php?/topic/47436-';
		WoWsStatInfoLinkLoc['eu'] = 'http://forum.worldofwarships.eu/index.php?/topic/14650-';
		
		
		var WoWsStatInfoLinkNameLoc = [];
		WoWsStatInfoLinkNameLoc['ru'] = '['+VersionWoWsStatInfo.substring(0, 5)+'] [WoWsStatInfo] Расширенная статистика на оф. сайте.';
		WoWsStatInfoLinkNameLoc['asia'] = '['+VersionWoWsStatInfo.substring(0, 5)+'] [WoWsStatInfo] Extended stat-info for official WoWS profile.';
		WoWsStatInfoLinkNameLoc['na'] = '['+VersionWoWsStatInfo.substring(0, 5)+'] [WoWsStatInfo] Extended stat-info for official WoWS profile.';
		WoWsStatInfoLinkNameLoc['eu'] = '['+VersionWoWsStatInfo.substring(0, 5)+'] [WoWsStatInfo] Extended stat-info for official WoWS profile.';
		
		var lang = 'ru';
		var realm = 'ru';
		var realm_host = 'ru';
		if(window.location.host.indexOf(".wargaming.net") > -1){
			lang = getCookie('wgccfe_language');
			realm = window.location.host.split('.')[0];
			realm_host = realm; if(realm == 'na'){realm_host = 'com';}
		}else if(window.location.host.indexOf("worldofwarships") > -1){
			lang = getCookie('hllang');
			var host_split = window.location.host.split('.'); realm = host_split[host_split.length - 1]; if(realm == 'com'){realm = 'na';}
			realm_host = realm; if(realm == 'na'){realm_host = 'com';}
		}
		if(lang == null){
			if(realm == 'ru'){
				lang = 'ru';
			}else{
				lang = 'en';
			}
		}
		if(window.location.host.indexOf("cm-ru.wargaming.net") > -1){return;}
		
		var localizationText = getlocalizationText(lang);
		var application_id = getApplicationId(realm);
		
		var WoWsStatInfoLink = WoWsStatInfoLinkLoc[realm];
		var WoWsStatInfoLinkName = WoWsStatInfoLinkNameLoc[realm];
		
		var WGAPI = 'http://api.worldoftanks.'+realm_host+'/wgn/';
		var WOWSAPI = 'http://api.worldofwarships.'+realm_host+'/wows/';
		var WoWsStatInfoHref = 'http://vzhabin.ru/US_WoWsStatInfo/';
	
		var Process = 0;
		var MaxProcess = 3;
		
		var SettingsWCI = null;
		{ /* ===== Default Settings UserScript ===== */
			var defaultSettingsWoWsStatInfo = ''+
				'{' +
					'"role_i18n": {"table": "1", "check": 1},' + // col 1
					'"clan_days": {"table": "1", "check": 1},' +
					'"info.last_battle_time": {"table": "1", "check": 1},' +
					'"info.logout_at": {"table": "1", "check": 0},' +
					'"null_1": {"table": "spacerow", "check": 0},' + // spacerow
					'"info.statistics.pvp.battles": {"table": "1", "check": 1},' +
					'"info.statistics.pvp.wins": {"table": "1", "check": 0},' +
					'"info.statistics.pvp.losses": {"table": "1", "check": 0},' +
					'"info.statistics.pvp.draws": {"table": "1", "check": 0},' +
					'"info.statistics.pvp.survived_battles": {"table": "1", "check": 0},' +
					'"info.statistics.pvp.survived_wins": {"table": "1", "check": 0},' +
					'"null_2": {"table": "spacerow", "check": 0},' + // spacerow
					'"info.statistics.pvp.kill_dead": {"table": "1", "check": 1},' +
					'"info.statistics.pvp.xp": {"table": "1", "check": 0},' +
					'"info.statistics.pvp.damage_dealt": {"table": "1", "check": 0},' +
					'"null_3": {"table": "spacerow", "check": 0},' + // spacerow
					'"info.statistics.pvp.frags": {"table": "1", "check": 0},' +
					'"info.statistics.pvp.planes_killed": {"table": "1", "check": 0},' +
					'"info.statistics.pvp.capture_points": {"table": "1", "check": 0},' +
					'"info.statistics.pvp.dropped_capture_points": {"table": "1", "check": 0},' +
					'"null_4": {"table": "spacerow", "check": 0},' + // spacerow
					'"info.statistics.pvp.avg_xp": {"table": "2", "check": 1},' + // col 2
					'"info.statistics.pvp.avg_damage_dealt": {"table": "2", "check": 1},' + 
					'"info.statistics.pvp.avg_frags": {"table": "2", "check": 0},' + 
					'"info.statistics.pvp.avg_planes_killed": {"table": "2", "check": 0},' + 
					'"info.statistics.pvp.avg_capture_points": {"table": "2", "check": 0},' + 
					'"info.statistics.pvp.avg_dropped_capture_points": {"table": "2", "check": 0},' + 
					'"null_5": {"table": "spacerow", "check": 0},' + // spacerow
					'"info.statistics.pvp.max_xp": {"table": "2", "check": 0},' +
					'"info.statistics.pvp.max_damage_dealt": {"table": "2", "check": 0},' +  
					'"info.statistics.pvp.max_frags_battle": {"table": "2", "check": 0},' +  
					'"info.statistics.pvp.max_planes_killed": {"table": "2", "check": 0},' +  
					'"null_6": {"table": "spacerow", "check": 0},' + // spacerow
					'"info.statistics.pvp.wins_percents": {"table": "2", "check": 1},' +
					'"info.statistics.pvp.survived_battles_percents": {"table": "2", "check": 0},' +
					'"null_7": {"table": "spacerow", "check": 0},' + // spacerow
					'"info.statistics.pvp.wr": {"table": "3", "check": 1},' + // col 3
					'"info.statistics.pvp.wtr": {"table": "3", "check": 1},' +
					'"null_8": {"table": "spacerow", "check": 0},' + // spacerow
					'"info.ships_x_level": {"table": "3", "check": 1},' +
					'"view_table": {' +
						'"table": "position", ' +
						'"check": 0, ' +
						'"sort": "' +
							'role_i18n,' +
							'clan_days,' +
							'info.last_battle_time,' +
							'info.ships_x_level,' +
							'info.statistics.pvp.battles,' +
							'info.statistics.pvp.wins_percents,' +
							'info.statistics.pvp.avg_xp,' +
							'info.statistics.pvp.avg_damage_dealt,' +
							'info.statistics.pvp.kill_dead,' +
							'info.statistics.pvp.wr,' +
							'info.statistics.pvp.wtr' +
						'"' +
					'}' +
				'}' +
			'';
		}
		
		var MembersArray = [];
		var StatPvPMemberArray = [];
		var Encyclopedia = null;
		
		var typeStat = ["pvp", "pve", "pvp_solo", "pvp_div", "pvp_div2", "pvp_div3"];
		var typeShip = ["Battleship", "AirCarrier", "Cruiser", "Destroyer"];
		
		var color = new Array();
		color['very_bad'] = '#FE0E00'; // очень плохо, хуже чем у 85%
		color['bad'] = '#FE7903'; // плохо, хуже чем у 50%
		color['normal'] = '#F8F400'; // средне, лучше чем у 50%
		color['good'] = '#60FF00';  // хорошо, лучше чем у 75%
		color['very_good'] = '#02C9B3'; // очень хорошо, лучше чем у 95%
		color['unique'] = '#D042F3'; // уникально, лучше чем у 99%
		
		var colorStat = null;
		var ExpShips = null;
		var ExpWTR = null;
		
		/* ===== Style and Script UserScript ===== */
		{
			var StyleWoWsStatInfo = '' +
				'div.div-link-block{font-size:13px; color: #fff; text-align: right; padding-top: 10px; padding-bottom: 10px;}' +
				'span.link-block:hover{border-bottom: 1px dotted #fff; cursor: pointer;}' +
				'span.link-block div.icon-link-block{background: url("http://'+realm+'.wargaming.net/clans/static/0.1.0.1/images/table-sorter/table-sorter_arrow_sprite.png") no-repeat 0 0; width: 10px; height: 10px; margin: -15px 100%;}' +
				'span.hide-block div.icon-link-block{background-position: 100% -26px;}' +
				'span.show-block div.icon-link-block{background-position: 100% -16px;}' +
				'div#userscript-block{border-radius: 2px; background-color: rgba(255, 255, 255, 0.03); border: 1px solid rgba(255, 255, 255, 0.1); padding: 10px; margin: 10px 0; line-height: 20px;}' +
				'div.hide-block{display: none;}' +
				'div.wowsstatinfo-stat{text-align: center; margin-top: 10px; color: white; font-size: 16px;}' +
				'span.name-stat{color: #ffcc33;}' +
				'table.account-table td{white-space: nowrap;}' +
				'li.account-tab div._title{padding: 0 10px;}' +
				'div.chart_div{text-align: center; float: left; margin-right: 40px;}' +
				'.userscript-placeholder {box-sizing: border-box;width: 550px;height: 330px;padding: 5px 15px 15px 40px;margin: 15px auto 30px auto;border: 1px solid #ddd;background: #fff;background: linear-gradient(#f6f6f6 0, #fff 50px);background: -o-linear-gradient(#f6f6f6 0, #fff 50px);background: -ms-linear-gradient(#f6f6f6 0, #fff 50px);background: -moz-linear-gradient(#f6f6f6 0, #fff 50px);background: -webkit-linear-gradient(#f6f6f6 0, #fff 50px);box-shadow: 0 3px 10px rgba(0,0,0,0.15);-o-box-shadow: 0 3px 10px rgba(0,0,0,0.1);-ms-box-shadow: 0 3px 10px rgba(0,0,0,0.1);-moz-box-shadow: 0 3px 10px rgba(0,0,0,0.1);-webkit-box-shadow: 0 3px 10px rgba(0,0,0,0.1);}' +
				'.chart-placeholder {width: 500px;height: 300px;font-size: 14px;line-height: 1.2em;}' +
				'.legend table {border-spacing: 5px;}' +
				'.flot-y-axis {margin-left: -40px; margin-top: -12px;}' +
				'.flot-x-axis {margin-left: -15px;}' +
			'';
			var StyleWoWsStatInfoAdd = document.createElement("style");
			StyleWoWsStatInfoAdd.textContent = StyleWoWsStatInfo.toString();
			document.head.appendChild(StyleWoWsStatInfoAdd);
			
			var ScriptChart = document.createElement("script");
			ScriptChart.setAttribute("src", "http://www.flotcharts.org/javascript/jquery.flot.min.js");
			document.head.appendChild(ScriptChart);
		}
		
		/* ===== Message UserScript ===== */
		if(window.location.host != 'forum.worldofwarships.'+realm_host){
			var message = document.createElement("div");
			message.setAttribute("id", "message-wowsstatinfo");
			message.setAttribute("class", "wsi-ui-dialog wsi-ui-widget wsi-ui-widget-content wsi-ui-corner-all wsi-ui-front");
			message.setAttribute("style", "display: none; z-index:9999; left: 50%; margin-left: 0px; top: 0px;");
			message.innerHTML = '' +
				'' +
				'
' +
				'' +
					'' +
					'' +
				'
' +
			'';
			document.body.appendChild(message);
			
			var messagebg = document.createElement("div");
			messagebg.setAttribute("id", "userscript-message-bg");
			messagebg.setAttribute("style", "display: none; z-index:9998; background: url('http://"+realm+".wargaming.net/clans/static/0.1.0.1/images/plugins/jquery-ui/widget_overlay-pattern.png'); position: fixed; top: 0; left: 0; width: 100%; height: 100%;");
			document.body.appendChild(messagebg);
		}	
		
		getBrowser();
		
		function getBrowser(){
			var nVer = navigator.appVersion;
			var nAgt = navigator.userAgent;
			var browserName = navigator.appName;
			var fullVersion = ''+parseFloat(navigator.appVersion); 
			var majorVersion = parseInt(navigator.appVersion, 10);
			var nameOffset, verOffset, ix;
			// In Opera 15+, the true version is after "OPR/" 
			if((verOffset = nAgt.indexOf("OPR/")) != -1){
				browserName = "Opera";
				fullVersion = nAgt.substring(verOffset + 4);
			}
			
			// In older Opera, the true version is after "Opera" or after "Version"
			else if((verOffset = nAgt.indexOf("Opera")) != -1){
				browserName = "Opera";
				fullVersion = nAgt.substring(verOffset + 6);
				if((verOffset = nAgt.indexOf("Version")) != -1) 
					fullVersion = nAgt.substring(verOffset + 8);
			}
			
			// In MSIE, the true version is after "MSIE" in userAgent
			else if((verOffset = nAgt.indexOf("MSIE")) != -1){
				browserName = "Microsoft Internet Explorer";
				fullVersion = nAgt.substring(verOffset + 5);
			}
			// In Chrome, the true version is after "Chrome" 
			else if((verOffset = nAgt.indexOf("Chrome")) != -1) {
				browserName = "Chrome";
				fullVersion = nAgt.substring(verOffset + 7);
			}
			
			// In Safari, the true version is after "Safari" or after "Version" 
			else if((verOffset = nAgt.indexOf("Safari")) != -1){
				browserName = "Safari";
				fullVersion = nAgt.substring(verOffset + 7);
				if((verOffset = nAgt.indexOf("Version")) != -1) 
					fullVersion = nAgt.substring(verOffset + 8);
			}
			
			// In Firefox, the true version is after "Firefox" 
			else if((verOffset = nAgt.indexOf("Firefox")) != -1){
				browserName = "Firefox";
				fullVersion = nAgt.substring(verOffset + 8);
			}
			
			// In most other browsers, "name/version" is at the end of userAgent 
			else if((nameOffset = nAgt.lastIndexOf(' ') + 1) < (verOffset = nAgt.lastIndexOf('/'))){
				browserName = nAgt.substring(nameOffset, verOffset);
				fullVersion = nAgt.substring(verOffset + 1);
				if(browserName.toLowerCase() == browserName.toUpperCase()){
					browserName = navigator.appName;
				}
			}
			
			// trim the fullVersion string at semicolon/space if present
			if((ix = fullVersion.indexOf(";")) != -1)
				fullVersion = fullVersion.substring(0, ix);
			if((ix = fullVersion.indexOf(" ")) != -1)
				fullVersion = fullVersion.substring(0, ix);
			majorVersion = parseInt(''+fullVersion, 10);
			if(isNaN(majorVersion)){
				fullVersion  = ''+parseFloat(navigator.appVersion); 
				majorVersion = parseInt(navigator.appVersion, 10);
			}
			
			var url = window.location.href;
			
			console.log(''
				+'----------------------- WoWsStatInfo '+VersionWoWsStatInfo+' -----------------------\n'
				+'| Lang                = '+lang+'\n'
				+'| URL                 = '+url+'\n'
				+'| Browser name        = '+browserName+'\n'
				+'| Full version        = '+fullVersion+'\n'
				+'| Major version       = '+majorVersion+'\n'
				+'| navigator.appName   = '+navigator.appName+'\n'
				+'| navigator.userAgent = '+navigator.userAgent+'\n'
				+'---------------------------------------------------------------------\n'
			);			
			
			var navigatorInfo = [];
			navigatorInfo['browserName'] = browserName;
			navigatorInfo['fullVersion'] = fullVersion;
			navigatorInfo['majorVersion'] = majorVersion;
			navigatorInfo['appName'] = navigator.appName;
			navigatorInfo['userAgent'] = navigator.userAgent;
			
			return navigatorInfo;
		}
		
		/* ===== Check load page ===== */
		if(window.location.href.indexOf("accounts") > -1 && window.location.href.split('/').length >= 8 && window.location.href.split('/')[6].match(/[0-9]+/) != null){
			checkJson();
			
			getStatSaveMember();
			
			lang = window.location.href.split('/')[3].match(/[a-z\s-]+/); if(lang == 'zh-tw'){lang = 'zh-tw';}
			localizationText = getlocalizationText(lang);
			getJson(WoWsStatInfoHref+'version.php?random='+Math.floor(Math.random()*100000001), doneLastVersion, errorLastVersion);
			var account_id = window.location.href.split('/')[6].match(/[0-9]+/);
			MemberProfilePage();
		}else if(window.location.host == 'forum.worldofwarships.'+realm_host && window.location.href.indexOf("/user/") > -1){
			ForumUserPage();
		}else if(window.location.host == 'forum.worldofwarships.'+realm_host && window.location.href.indexOf("/topic/") > -1){
			ForumTopicPage();
		}else if(window.location.href.indexOf("clans") > -1 && window.location.href.split('/')[4].match(/[0-9]+/) != null 
			&& window.location.href.indexOf("players") > -1 && window.location.href.split('/').length >= 6
		){
			checkJson();
			SettingsWCI = getSettingsClanPage();
			getJson(WoWsStatInfoHref+'version.php?random='+Math.floor(Math.random()*100000001), doneLastVersion, errorLastVersion);
			var ClanId = window.location.href.split('/')[4].match(/[0-9]+/);
			ClanPage();
		}
		
		function doneLastVersion(url, response){
			var data = response;
			if(VersionWoWsStatInfo != data['version']){
				onShowMessage(
					localizationText['Box'],
					localizationText['NewVersion']+' WoWsStatInfo '+data['version']+'' +
						// ''+localizationText['profile-wows']+': ' +
						// ''+nickname.innerHTML+'  ' +
				// '';
				
				ipsList_data.innerHTML += '' +
					// profile +
					'' +
						''+localizationText['profile-clan']+': ' +
						' ' +
				'';
				
				var language = lang; if(language == 'zh-tw'){language = 'zh-cn';}else if(language == 'ja' || language == 'es-mx' || language == 'pt-br'){language = 'en';}
				getJson(WGAPI+'clans/membersinfo/?application_id='+application_id+'&language='+language+'&account_id='+user_id, doneForumClanInfo, errorForumClanInfo);
			}
		}
		function ForumTopicPage(){
			var ForumTopicMembers = [];
			var basic_info = document.getElementsByClassName('basic_info');
			for(var i = 0; i < basic_info.length; i++){
				var ipsUserPhotoLink = basic_info[i].getElementsByClassName('ipsUserPhotoLink')[0];
				if(undefined === ipsUserPhotoLink){continue;}
				if(ipsUserPhotoLink.id == null){continue;}
				if(ipsUserPhotoLink.id.indexOf('anonymous_element') > -1){
					var linkParse = ipsUserPhotoLink.href.split('/');
					var accountParse = linkParse[5].split('-');
					var account_id = accountParse[accountParse.length - 1];
					if(ForumTopicMembers['member_'+account_id] === undefined){
						ForumTopicMembers['member_'+account_id] = account_id;
						
						var language = lang; if(language == 'zh-tw'){language = 'zh-cn';}else if(language == 'ja' || language == 'es-mx' || language == 'pt-br'){language = 'en';}
						getJson(WGAPI+'clans/membersinfo/?application_id='+application_id+'&language='+language+'&account_id='+account_id, doneForumClanInfo, errorForumClanInfo);
					}
					basic_info[i].innerHTML += '' +
						'' +
							' ' +
					'';
				}
			}
		}
		function ClanPage(){
			if(colorStat == null || ExpShips == null || ExpWTR == null){
				console.log('colorStat == null || ExpShips == null || ExpWTR == null');
				setTimeout(function(){ClanPage();}, 1000);
				return;
			}
		
			var view_block_history = getLocalStorage('clan-member-history', false);
			if(view_block_history == null){view_block_history = 'hide';}
			
			var page_header = document.getElementsByClassName("page-header")[0];
			page_header.outerHTML += '' +
				'' +
					'
' +
						'' +
							'' +
								'' +
									''+localizationText['get-settings-button']+' ' +
								' ' +
							' ' +
						' ' +
					'
' +
					getUserScriptDeveloperBlock() + 
					'
' +
						'
' +
							localizationText['block-link-clan-member-history'] +
							'
'+
						' ' +
					'
' +
					'
' +
						'
' +
					'
' +
					'
' +
						'
' +
							'' +						
								'' +
									'' +
										'' +
											'' +
												''+localizationText['statistic-clan-button-0']+' ' +
											' ' +
										' ' +
									' ' +
									'' +
										localizationText['statistic-clan-load-text']+' 0%'+localizationText['statistic-load-text-lost']+' 0 '+localizationText['statistic-load-text-min']+' 0 '+localizationText['statistic-load-text-sec'] +
									' '+
								' ' +
								'' +
									' ' +							
							' ' +
						'
' +
					'
' +
				'
' +
						'' +
							' ' +
						'['+clansInfo['clan']['tag']+'] ' +
					' ' +
				'';
				jQ('.member_'+account_id).html(html);
			}else{
				jQ('.member_'+account_id).html('');
			}
		}
		function errorForumClanInfo(url){
			jQ('.member_'+account_id).html('');
		}
		
		/* ===== MemberProfilePage function ===== */
		function doneClanInfo(url, response){
			if(response.status && response.status == "error"){
				errorClanInfo();
				return;
			}
			
			var vars = getUrlVars(url);
			var account_id = vars['account_id'];
			
			if(response['data'][account_id] == null){errorClanInfo(); return;}
			
			MembersArray[0]['clan'] = response['data'][account_id]['clan'];
			MembersArray[0]['clan']['account_id'] = response['data'][account_id]['account_id'];
			MembersArray[0]['clan']['role_i18n'] = response['data'][account_id]['role_i18n'];
			MembersArray[0]['clan']['joined_at'] = response['data'][account_id]['joined_at'];
			MembersArray[0]['clan']['role'] = response['data'][account_id]['role'];
			MembersArray[0]['clan']['account_name'] = response['data'][account_id]['account_name'];		
			
			viewMemberClan();
		}
		function errorClanInfo(url){
			MembersArray[0]['clan'] = null;
			
			viewMemberClan();
		}
		function viewMemberClan(){
			var wowsstatinfo_profile_clan = document.getElementById('wowsstatinfo-profile-clan');
		
			if(MembersArray[0]['clan'] != null){
				var day = dateDiffInDays(MembersArray[0]['clan']['joined_at'] * 1000, new Date().getTime());
				
				var icon = MembersArray[0]['clan']['emblems']['x32']['portal'];
				if(icon === undefined){
					for(var key in MembersArray[0]['clan']['emblems']['x32']){
						if(key != 'wot'){
							icon = MembersArray[0]['clan']['emblems']['x32'][key];
						}
					}					
				}
				
				wowsstatinfo_profile_clan.innerHTML = '' +
					'' +
				'';
			}else{
				wowsstatinfo_profile_clan.innerHTML = '';
			}			
		}
		function viewMainPageProfile(){
			if(Encyclopedia == null){console.log('Encyclopedia == null'); setTimeout(function(){viewMainPageProfile();}, 1000);return;}
			
			if(MembersArray[0]['info']['hidden_profile']){
				console.log(MembersArray[0]['info']['account_id']+' hidden profile!!!');
				return;
			}
			
			if(!calcStat(0)){
				console.log('Error calcStat '+MembersArray[0]['info']['account_id']);
				return;
			}
			
			var tabContainer = null;
			var tab_container = document.getElementsByClassName('tab-container');
			for(var tc = 0; tc < tab_container.length; tc++){
				if(tab_container[tc].getAttribute('js-tab-cont-id') != 'pvp'){continue;}
				tabContainer = tab_container[tc];
			}
			
			if(tabContainer != null){
				var cm_user_menu_link_cutted_text = document.getElementsByClassName('cm-user-menu-link_cutted-text')[0];
				var login_name = null; if(cm_user_menu_link_cutted_text != null){login_name = cm_user_menu_link_cutted_text.textContent;}
				
				var userbar = '';
				if(login_name == MembersArray[0]['info']['nickname']){
					userbar += ''+localizationText['generator-userbar']+' ';
					onSaveStatMember();
				}
				userbar += '' +
					'' +
							'' +
								'' +
									'' +
										'' +
											'' +
												'' +
													''+localizationText['stat-table-4']+' ' +
												' ' +
											' ' +
										' ' +
										'' +
											'' +
												'' +
													''+localizationText['battles_days']+' ' +
												' ' +
												'' +
													''+valueFormat((MembersArray[0]['info']['statistics']['pvp']['battles_days']).toFixed(0))+' '+
												' ' +
											' ' +
											'' +
												'' +
													''+localizationText['number-ships-x']+' ' +
												' ' +
												'' +
													''+MembersArray[0]['info']['ships_x_level']+' '+
												' ' +
											' ' +	
											'' +
												'' +
													''+localizationText['max_ship_level']+' ' +
												' ' +
												'' +
													''+
														MembersArray[0]['info']['statistics']['pvp']['max_ship_level']+
													' ' +
												' ' +
											' ' +
											'' +
												'' +
													''+localizationText['avg_battles_level']+' ' +
												' ' +
												'' +
													''+valueFormat((MembersArray[0]['info']['statistics']['pvp']['avg_battles_level']).toFixed(1))+' '+
												' ' +
											' ' +
											'' +
												'' +
													''+
														''+
															localizationText['wr']+
														' ' +
													' ' +
												' ' +
												'' +
													''+
														valueFormat((MembersArray[0]['info']['statistics']['pvp']['wr']).toFixed(2)) + 
													' '+
												' ' +
											' ' +
											'' +
												'' +
													'' +
														''+
															localizationText['wtr'] +
														' ' +
													' ' +
												' ' +
												'' +
													''+
														valueFormat((MembersArray[0]['info']['statistics']['pvp']['wtr']).toFixed(2)) + 
													' '+
												' ' +
											' ' +
										' ' +
									'
' +
								' '+
								''+
								' '+
								''+
									userbar +
								' ' +
							' ' +
						'
'
					'';
					addStatHover(document.getElementById('info-stat-pvp-wr'), 'wr');
					addStatHover(document.getElementById('info-stat-pvp-wtr'), 'wtr');
					
					var img = new Image();
					img.onload = function(){
						var userbar_img = document.getElementById('userbar-img');
						if(userbar_img != null){
							userbar_img.src = WoWsStatInfoHref+'userbar/'+MembersArray[0]['info']['nickname']+'.png'+'?'+Math.floor(Math.random()*100000001);
							
							var userbar_link = document.getElementById('userbar-link');
							userbar_link.textContent = '[url='+WoWsStatInfoHref+'?realm_search='+realm+'&nickname='+MembersArray[0]['info']['nickname']+'][img]'+WoWsStatInfoHref+'userbar/'+MembersArray[0]['info']['nickname']+'.png[/img][/url]';
						}
					}
					img.src = WoWsStatInfoHref+'userbar/'+MembersArray[0]['info']['nickname']+'.png'+'?'+Math.floor(Math.random()*100000001);
					
					jQ('#generator-userbar').click(function(){
						getJson(WoWsStatInfoHref+'bg/bg.php?'+Math.floor(Math.random()*100000001), doneUserBarBG, errorUserBarBG);
					});
					
					var _values = tabContainer.getElementsByClassName('_values')[0];
					var main_stat = _values.getElementsByTagName('div');
					main_stat[0].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['battles'], 'battles', 'main');
					addStatHover(main_stat[0], 'battles');
					
					main_stat[1].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['wins_percents'], 'wins_percents', 'main');
					addStatHover(main_stat[1], 'wins_percents');
					
					main_stat[2].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['avg_xp'], 'avg_xp', 'main');
					addStatHover(main_stat[2], 'avg_xp');
					
					
					main_stat[3].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['avg_damage_dealt'], 'avg_damage_dealt', 'main');
					addStatHover(main_stat[3], 'avg_damage_dealt');
					
					main_stat[4].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['kill_dead'], 'kill_dead', 'main');
					addStatHover(main_stat[4], 'kill_dead');
					
					var account_battle_stats = tabContainer.getElementsByClassName('account-battle-stats')[0];
					if(account_battle_stats != null){
						var account_table = account_battle_stats.getElementsByClassName('account-table');
						
						//Общие результаты
						account_table[0].rows[1].cells[1].getElementsByTagName('span')[0].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['battles'], 'battles', 'main');
						addStatHover(account_table[0].rows[1], 'battles');
						
						//Средние показатели за бой
						account_table[1].rows[1].cells[1].getElementsByTagName('span')[0].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['avg_xp'], 'avg_xp', 'main');
						addStatHover(account_table[1].rows[1], 'avg_xp');
						
						account_table[1].rows[2].cells[1].getElementsByTagName('span')[0].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['avg_damage_dealt'], 'avg_damage_dealt', 'main');
						addStatHover(account_table[1].rows[2], 'avg_damage_dealt');
						
						account_table[1].rows[3].cells[1].getElementsByTagName('span')[0].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['avg_frags'], 'avg_frags', 'main');
						addStatHover(account_table[1].rows[3], 'avg_frags');
						
						account_table[1].rows[4].cells[1].getElementsByTagName('span')[0].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['avg_planes_killed'], 'avg_planes_killed', 'main');
						addStatHover(account_table[1].rows[4], 'avg_planes_killed');
						
						//account_table[1].rows[5].cells[1].getElementsByTagName('span')[0].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['avg_capture_points'], 'avg_capture_points', 'main');
						//account_table[1].rows[6].cells[1].getElementsByTagName('span')[0].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['avg_dropped_capture_points'], 'avg_dropped_capture_points', 'main');
						
						// Рекордные показатели
						account_table[2].rows[1].cells[1].getElementsByTagName('span')[0].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['max_xp'], 'max_xp', 'main');
						addStatHover(account_table[2].rows[1], 'max_xp');
						
						account_table[2].rows[2].cells[1].getElementsByTagName('span')[0].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['max_damage_dealt'], 'max_damage_dealt', 'main');
						addStatHover(account_table[2].rows[2], 'max_damage_dealt');
						
						account_table[2].rows[3].cells[1].getElementsByTagName('span')[0].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['max_frags_battle'], 'max_frags_battle', 'main');
						addStatHover(account_table[2].rows[3], 'max_frags_battle');
						
						account_table[2].rows[4].cells[1].getElementsByTagName('span')[0].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['max_planes_killed'], 'max_planes_killed', 'main');
						addStatHover(account_table[2].rows[4], 'max_planes_killed');
						
						//Дополнительно
						account_table[0].rows[2].cells[1].getElementsByTagName('small')[0].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['wins_percents'], 'wins_percents', 'main');
						addStatHover(account_table[0].rows[2], 'wins_percents');
						
						if(account_table[0].rows[3].cells[1].getElementsByClassName('small-survived_battles_percents')[0] == undefined){
							account_table[0].rows[3].cells[1].innerHTML += '('+valueFormat((MembersArray[0]['info']['statistics']['pvp']['survived_battles_percents']).toFixed(2))+'%) ';
							account_table[0].rows[3].cells[1].getElementsByTagName('small')[0].style.color = findColorASC(MembersArray[0]['info']['statistics']['pvp']['survived_battles_percents'], 'survived_battles_percents', 'main');
							addStatHover(account_table[0].rows[3], 'survived_battles_percents');
						}					
						if(account_table[2].rows[1].cells[0].getElementsByClassName('small-max_xp_ship')[0] == undefined 
							&& Encyclopedia[MembersArray[0]['info']['statistics']['pvp']['max_xp_ship_id']] != null && Encyclopedia[MembersArray[0]['info']['statistics']['pvp']['max_xp_ship_id']] !== undefined){
							account_table[2].rows[1].cells[0].innerHTML += '('+Encyclopedia[MembersArray[0]['info']['statistics']['pvp']['max_xp_ship_id']]['name']+') ';
						}
						if(account_table[2].rows[2].cells[0].getElementsByClassName('small-max_damage_dealt_ship')[0] == undefined 
							&& Encyclopedia[MembersArray[0]['info']['statistics']['pvp']['max_damage_dealt_ship_id']] != null && Encyclopedia[MembersArray[0]['info']['statistics']['pvp']['max_damage_dealt_ship_id']] !== undefined){
							account_table[2].rows[2].cells[0].innerHTML += '('+Encyclopedia[MembersArray[0]['info']['statistics']['pvp']['max_damage_dealt_ship_id']]['name']+') ';
						}
						if(account_table[2].rows[3].cells[0].getElementsByClassName('small-max_frags_ship')[0] == undefined 
							&& Encyclopedia[MembersArray[0]['info']['statistics']['pvp']['max_frags_ship_id']] != null && Encyclopedia[MembersArray[0]['info']['statistics']['pvp']['max_frags_ship_id']] !== undefined){
							account_table[2].rows[3].cells[0].innerHTML += '('+Encyclopedia[MembersArray[0]['info']['statistics']['pvp']['max_frags_ship_id']]['name']+') ';
						}
						if(account_table[2].rows[4].cells[0].getElementsByClassName('small-max_planes_killed_ship')[0] == undefined 
							&& Encyclopedia[MembersArray[0]['info']['statistics']['pvp']['max_planes_killed_ship_id']] != null && Encyclopedia[MembersArray[0]['info']['statistics']['pvp']['max_planes_killed_ship_id']] !== undefined){
							account_table[2].rows[4].cells[0].innerHTML += '('+Encyclopedia[MembersArray[0]['info']['statistics']['pvp']['max_planes_killed_ship_id']]['name']+') ';
						}
						
					}
				}
				
				var ships_detail_stats = tabContainer.getElementsByClassName('ships-detail-stats')[0];
				if(ships_detail_stats != null){
					for(var i = 0; i < ships_detail_stats.rows.length; i++){
						var row = ships_detail_stats.rows[i];
						
						if(i == 0 && row.cells.length < 6){
							var th = document.createElement('th');
							th.setAttribute('class', '_value');
							th.innerHTML = ''+localizationText['wr']+' ';
							row.appendChild(th);
							
							th = document.createElement('th');
							th.setAttribute('class', '_value');
							th.innerHTML = ''+localizationText['wtr']+' ';
							row.appendChild(th);
							
							continue;
						}
						
						if(row.getAttribute('class') == null){
							var _icon = row.getElementsByClassName('_icon')[0];
							var div_icon = _icon.getElementsByTagName('div')[0];
							var ship_class = div_icon.getAttribute('class').split('-')[1];
							
							if(row.cells.length < 6){
								for(var t = 0; t < typeShip.length; t++){
									var type = typeShip[t].toLowerCase();
									if(ship_class == type){
										ship_class = typeShip[t];
									}
								}
							
								var td = document.createElement('td');
								td.innerHTML = ''+valueFormat((MembersArray[0]['info']['statistics']['pvp']['wr_'+ship_class]).toFixed(0))+' ';
								row.appendChild(td);
								
								td = document.createElement('td');
								td.innerHTML = ''+valueFormat((MembersArray[0]['info']['statistics']['pvp']['wtr_'+ship_class]).toFixed(0))+' ';
								row.appendChild(td);
							}
							
							var battles = htmlParseMemberStatistic(row.cells[2]);
							if(battles > 0){
								var wins = htmlParseMemberStatistic(row.cells[3]);
								var avg_xp = htmlParseMemberStatistic(row.cells[4]);
								var wins_percents = (wins/battles)*100; if(isNaN(wins_percents)){wins_percents = 0;}
								
								row.cells[3].setAttribute('style', 'white-space: nowrap;');
								row.cells[3].innerHTML = valueFormat(wins)+' ('+valueFormat((wins_percents).toFixed(0))+'%) ';							
								
								row.cells[4].setAttribute('style', 'white-space: nowrap;');
								row.cells[4].innerHTML = ' '+valueFormat(avg_xp)+' ';
							}
							
							continue;
						}
						
						if(row.getAttribute('class').indexOf('_expandable') > -1){
							var ship_id = row.getAttribute('js-has-extension').split('-')[0];
							
							if(row.cells.length < 5){
								var name_text = row.cells[0].getElementsByClassName('_text')[0];
								var is_premium = Encyclopedia[ship_id]['is_premium'];
								if(is_premium){
									name_text.setAttribute('style', 'color: #ffab34;');
								}
							
								var td = document.createElement('td');
								td.setAttribute('id', 'wr-'+ship_id);
								td.innerHTML = '0 ';
								row.appendChild(td);
								
								td = document.createElement('td');
								td.setAttribute('id', 'wtr-'+ship_id);
								td.innerHTML = '0 ';
								row.appendChild(td);
							}
							
							var battles = htmlParseMemberStatistic(row.cells[1]);
							if(battles > 0){
								var wins = htmlParseMemberStatistic(row.cells[2]);
								var avg_xp = htmlParseMemberStatistic(row.cells[3]);
								var wins_percents = (wins/battles)*100; if(isNaN(wins_percents)){wins_percents = 0;}
								
								row.cells[2].setAttribute('style', 'white-space: nowrap;');
								row.cells[2].innerHTML = valueFormat(wins)+' ('+valueFormat((wins_percents).toFixed(0))+'%) ';							
								
								row.cells[3].setAttribute('style', 'white-space: nowrap;');
								row.cells[3].innerHTML = ' '+valueFormat(avg_xp)+' ';
							}
							
							continue;
						}
						
						if(row.getAttribute('class').indexOf('_ship-entry-stat') > -1){
							row.cells[0].setAttribute('colspan', '7');
							
							continue;
						}
					}
					
					for(var shipI = 0; shipI < MembersArray[0]['ships'].length; shipI++){
						var ship_id = MembersArray[0]['ships'][shipI]['ship_id'];
						var wr_cell = document.getElementById('wr-'+ship_id);
						if(wr_cell != null){
							wr_cell.setAttribute('style', 'white-space: nowrap;');
							wr_cell.innerHTML = ''+valueFormat((MembersArray[0]['ships'][shipI]['pvp']['wr']).toFixed(0))+' ';
						}
						
						var wtr_cell = document.getElementById('wtr-'+ship_id);
						if(wtr_cell != null){
							wtr_cell.setAttribute('style', 'white-space: nowrap;');
							wtr_cell.innerHTML = ''+valueFormat((MembersArray[0]['ships'][shipI]['pvp']['wtr']).toFixed(0))+' ';
						}
					}
				}
			
				var achieves_block = tabContainer.getElementsByClassName('achieves-block')[0];
				if(achieves_block != null){
					var achieve_item = tabContainer.getElementsByClassName('achieve-item');
					for(var i = 0; i < achieve_item.length; i++){
						var item = achieve_item[i];
						var js_tooltip_show = item.getAttribute('js-tooltip-show');
						var _counter = item.getElementsByClassName('_counter')[0];
						if(_counter != null && item.getElementsByClassName('_counter').length == 1){
							_counter.setAttribute('style', 'left: 70%; background-color: #F7882E; min-width: 3em;');
							
							var battlesAchievement = 0;
							if(MembersArray[0]['achievements']['battle'][js_tooltip_show+'_battle'] === undefined){
								battlesAchievement = 'Error';
							}else{
								battlesAchievement = MembersArray[0]['achievements']['battle'][js_tooltip_show+'_battle'];
							}
							
							item.innerHTML += ''+battlesAchievement+'
';
						}
					}
					
					var achieve_counter = tabContainer.getElementsByClassName('achieve-counter')[0];
					if(achieve_counter == null){
						achieves_block.outerHTML += '' +
							'' +
						'';
					}
				}
				
				var typeStatAdd = ["pvp_div", "pvp_solo"];
				for(var i = 0; i < typeStatAdd.length; i++){
					var type = typeStatAdd[i];
					
					var account_tab = tabContainer.getElementsByClassName('account-tab-'+type)[0];
					if(account_tab == null){
						var account_tab_detail_stats = tabContainer.getElementsByClassName('account-tab-detail-stats')[0];
						if(account_tab_detail_stats != null){
							account_tab_detail_stats.outerHTML += '' +
								'' +
									'
' +
										'
' +
										'
' +
											'
' +
												'
' +
												'
' +
												'
' +
												'
' +
												'
' +
											'
' +
											'
' +
												'
'+localizationText['title_battles']+'
' +
												'
'+localizationText['title_wins_percents']+'
' +
												'
'+localizationText['title_avg_xp']+'
' +
												'
'+localizationText['title_avg_damage_dealt']+'
' +
												'
'+localizationText['title_kill_dead']+'
' +
											'
' +
											'
' +
												'
'+
													valueFormat((MembersArray[0]['info']['statistics'][type]['battles']).toFixed(0))+
												'
' +
												'
'+
													valueFormat((MembersArray[0]['info']['statistics'][type]['wins_percents']).toFixed(2))+
												'%
' +
												'
'+
													valueFormat((MembersArray[0]['info']['statistics'][type]['avg_xp']).toFixed(0))+
												'
' +
												'
'+
													valueFormat((MembersArray[0]['info']['statistics'][type]['avg_damage_dealt']).toFixed(0))+
												'
' +
												'
'+
													valueFormat((MembersArray[0]['info']['statistics'][type]['kill_dead']).toFixed(2))+
												'
' +
											'
' +
										'
' +
									'
' +
									'
' +
									'
' +
										getHTMLStat(MembersArray[0]['info']['statistics'][type], 'main')+
										'
' +
											'
' +
												'
' +
													'' +
													'' +
														'' +
															''+localizationText['stat-table-4']+' ' +
														' ' +
													' ' +
													' ' +
													'' +
														'' +
															'' +
																''+localizationText['battles_days']+' ' +
															' ' +
															'' +
																''+
																	(MembersArray[0]['info']['statistics'][type]['battles_days']).toFixed(0)+
																' ' +
															' ' +
														' ' +
														'' +
															'' +
																''+localizationText['max_ship_level']+' ' +
															' ' +
															'' +
																''+
																	MembersArray[0]['info']['statistics'][type]['max_ship_level']+
																' ' +
															' ' +
														' ' +
														'' +
															'' +
																''+localizationText['avg_battles_level']+' ' +
															' ' +
															'' +
																''+
																	valueFormat((MembersArray[0]['info']['statistics'][type]['avg_battles_level']).toFixed(1))+
																' ' +
															' ' +
														' ' +
														'' +
															'' +
																'' +
																	''+
																		localizationText['wr'] +
																	' ' +
																' ' +
															' ' +
															'' +
																''+
																	valueFormat((MembersArray[0]['info']['statistics'][type]['wr']).toFixed(2)) + 
																' '+
															' ' +
														' ' +
														'' +
															'' +
																'' +
																	'' +
																		localizationText['wtr'] +
																	' ' +
																' ' +
															' ' +
															'' +
																''+
																	valueFormat((MembersArray[0]['info']['statistics'][type]['wtr']).toFixed(2)) + 
																' '+
															' ' +
														' ' +
													' ' +
												'
' +
											'
' +
										'
' +
									'
' +
									'
' +
									'
' +
										'
'+localizationText['ships_stat']+' ' +
										'
'+
											''+
												''+
													''+
														'
'+
													' '+
													''+
														localizationText['title_ships']+
													' '+
													''+
														'
'+
														''+localizationText['battles']+' '+
													' '+
													''+
														'
'+
														''+localizationText['wins']+' '+
													' '+
													''+
														'
'+
														''+localizationText['title_avg_xp']+' '+
													' '+
													''+
														''+localizationText['wr']+' '+
													' '+
													''+
														''+localizationText['wtr']+' '+
													' '+
												' '+
											' '+
											getHTMLShipStat(MembersArray[0], type)+
										'
'+
									'
' +
								'
'+
									''+localizationText[type]+'
'+
									''+
								' '+
							'');
							jQ(tabContainer).find('div.account-tabs-mobile ul').append(''+
								''+localizationText[type]+' ' +
							'');
							
							jQ('.ships-class-'+type).click(function(){
								if(jQ(this).attr('open-active') == 'true'){
									jQ(this).removeClass('_active-btn');
									jQ('.'+jQ(this).attr('open-block')).removeClass('_active');
									jQ(this).attr('open-active', 'false');
								}else{
									jQ(this).addClass('_active-btn');
									jQ('.'+jQ(this).attr('open-block')).addClass('_active');
									jQ(this).attr('open-active', 'true');
								}								
							});
							
							jQ('.ship-stat-'+type).click(function(){
								if(jQ(this).attr('open-active') == 'true'){
									jQ(this).removeClass('_active-btn');
									jQ('.'+jQ(this).attr('open-block')).removeClass('_active');
									jQ(this).attr('open-active', 'false');
								}else{
									jQ(this).addClass('_active-btn');
									jQ('.'+jQ(this).attr('open-block')).addClass('_active');
									jQ(this).attr('open-active', 'true');
								}								
							});
						}
					}
				}				
			}
		}
		function getHTMLStat(StatArray, type_stat){
			var html = '';
			
			var battles_percents = '';
			var max_xp_ship = '';
			var max_damage_dealt_ship = '';
			var max_frags_ship = '';
			var max_planes_killed_ship = '';
			if(type_stat == 'main'){
				battles_percents = '('+valueFormat((StatArray['battles_percents']).toFixed(2))+'%) ';
				max_xp_ship = ' ('+Encyclopedia[''+StatArray['max_xp_ship_id']+'']['name']+') ';
				max_damage_dealt_ship = ' ('+Encyclopedia[''+StatArray['max_damage_dealt_ship_id']+'']['name']+') ';
				max_frags_ship = ' ('+Encyclopedia[''+StatArray['max_frags_ship_id']+'']['name']+') ';
				max_planes_killed_ship = ' ('+Encyclopedia[''+StatArray['max_planes_killed_ship_id']+'']['name']+') ';
			}
			
			html = ''+
				'' +
					'
' +
						'
' +
							'' +
								'' +
									'' +
										''+localizationText['stat-table-1']+' ' +
									' ' +
								' ' +
							' ' +
							'' +
								'' +
									'' +
										''+localizationText['battles']+' ' +
									' ' +
									'' +
										''+
											valueFormat((StatArray['battles']).toFixed(0))+
										' ' +
										battles_percents+
									' ' +
								' ' +
								'' +
									'' +
										''+localizationText['wins']+' ' +
									' ' +
									'' +
										''+
											valueFormat((StatArray['wins']).toFixed(0))+
										' ' +
										'('+
											valueFormat((StatArray['wins_percents']).toFixed(2))+
										'%) '+
									' ' +
								' ' +
								'' +
									'' +
										''+localizationText['survived_battles']+' ' +
									' ' +
									'' +
										''+
											valueFormat((StatArray['survived_battles']).toFixed(0))+
										' ' +
										'('+
											valueFormat((StatArray['survived_battles_percents']).toFixed(2))+
										'%) '+
									' ' +
								' ' +
								'' +
									'' +
										''+localizationText['damage_dealt']+' ' +
									' ' +
									'' +
										''+valueFormat((StatArray['damage_dealt']).toFixed(0))+' ' +
									' ' +
								' ' +
								'' +
									'' +
										''+localizationText['frags']+' ' +
									' ' +
									'' +
										''+valueFormat((StatArray['frags']).toFixed(0))+' ' +
									' ' +
								' ' +
								'' +
									'' +
										''+localizationText['planes_killed']+' ' +
									' ' +
									'' +
										''+valueFormat((StatArray['planes_killed']).toFixed(0))+' ' +
									' ' +
								' ' +
								'' +
									'' +
										''+localizationText['capture_points']+' ' +
									' ' +
									'' +
										''+valueFormat((StatArray['capture_points']).toFixed(0))+' ' +
									' ' +
								' ' +
								'' +
									'' +
										''+localizationText['dropped_capture_points']+' ' +
									' ' +
									'' +
										''+valueFormat((StatArray['dropped_capture_points']).toFixed(0))+' ' +
									' ' +
								' ' +
							' ' +
						'
' +
					'
' +
					'
' +
						'
' +
							'' +
							'' +
								'' +
									''+localizationText['stat-table-2']+' ' +
								' ' +
							' ' +
							' ' +
							'' +
								'' +
									'' +
										''+localizationText['avg_xp']+' ' +
									' ' +
									'' +
										''+
											valueFormat((StatArray['avg_xp']).toFixed(2))+
										' ' +
									' ' +
								' ' +
								'' +
									'' +
										''+localizationText['avg_damage_dealt']+' ' +
									' ' +
									'' +
										''+
											valueFormat((StatArray['avg_damage_dealt']).toFixed(2))+
										' ' +
									' ' +
								' ' +
								'' +
									'' +
										''+localizationText['avg_frags']+' ' +
									' ' +
									'' +
										''+
											valueFormat((StatArray['avg_frags']).toFixed(2))+
										' ' +
									' ' +
								' ' +
								'' +
									'' +
										''+localizationText['avg_planes_killed']+' ' +
									' ' +
									'' +
										''+
											valueFormat((StatArray['avg_planes_killed']).toFixed(2))+
										' ' +
									' ' +
								' ' +
								'' +
									'' +
										''+localizationText['avg_capture_points']+' ' +
									' ' +
									'' +
										//''
										''
											+valueFormat((StatArray['avg_capture_points']).toFixed(2))+
										' ' +
									'  ' +
								' ' +
								'' +
									'' +
										''+localizationText['avg_dropped_capture_points']+' ' +
									' ' +
									'' +
										//''+
										''+
											valueFormat((StatArray['avg_dropped_capture_points']).toFixed(0))+
										' ' +
									'  ' +
								' ' +
							' ' +
						'
' +
					'
' +
					'
' +
						'
' +
							'' +
							'' +
								'' +
									''+localizationText['stat-table-3']+' ' +
								' ' +
							' ' +
							' ' +
							'' +
								'' +
									'' +
										''+localizationText['max_xp']+' ' +
										max_xp_ship +
									' ' +
									'' +
										''+
											valueFormat((StatArray['max_xp']).toFixed(0))+
										' ' +
									' ' +
								' ' +
								'' +
									'' +
										''+localizationText['max_damage_dealt']+' ' +
										max_damage_dealt_ship +
									' ' +
									'' +
										''+
											valueFormat((StatArray['max_damage_dealt']).toFixed(0))+
										' ' +
									' ' +
								' ' +
								'' +
									'' +
										''+localizationText['max_frags_battle']+' ' +
										max_frags_ship+
									' ' +
									'' +
										''+
											valueFormat((StatArray['max_frags_battle']).toFixed(0))+
										' ' +
									' ' +
								' ' +
								'' +
									'' +
										''+localizationText['max_planes_killed']+' ' +
										max_planes_killed_ship+
									' ' +
									'' +
										''+
											valueFormat((StatArray['max_planes_killed']).toFixed(0))+
										' ' +
									' ' +
								' ' +
							' ' +
						'
' +
					'
' +	
				'
' +
							'' +
								'
' +
								''+ship_lvl+' ' +
								''+ship_name+' ' +
							' ' +
							''+valueFormat((Ship[type]['battles']).toFixed(0))+' ' +
							'' +
								Ship[type]['wins']+' ' +
								'' +
									'('+valueFormat((Ship[type]['wins_percents']).toFixed(0))+'%)' +
								' ' +
							' ' +
							'' +
								'' +
									valueFormat((Ship[type]['avg_xp']).toFixed(0)) +
								' ' +
							' ' +
							'' +
								'' +
									valueFormat((Ship[type]['wr']).toFixed(0)) +
								' ' +
							' ' +
							'' +
								'' +
									valueFormat((Ship[type]['wtr']).toFixed(0)) +
								' ' +
							' ' +
						' ' +
						'' +
							'' +
								getHTMLStat(Ship[type], 'ship') +
							' ' +
						' '+
					'';
				}
				
			}
			
			var html = '';
			
			for(var tS = 0; tS < typeShip.length; tS++){
				var wins_percents = (StatClassArray[typeShip[tS]]['wins']/StatClassArray[typeShip[tS]]['battles'])*100; if(isNaN(wins_percents)){wins_percents = 0;}
				var avg_xp = StatClassArray[typeShip[tS]]['xp']/StatClassArray[typeShip[tS]]['battles']; if(isNaN(avg_xp)){avg_xp = 0;}
			
				html += '' +
					'' +
						'' +
							'
' +
								localizationText[typeShip[tS].toLowerCase()]+' ('+StatClassArray[typeShip[tS]]['count']+')' +
							' ' +
							''+valueFormat((StatClassArray[typeShip[tS]]['battles']).toFixed(0))+' ' +
							'' +
								valueFormat((StatClassArray[typeShip[tS]]['wins']).toFixed(0))+' ' +
								'' +
									'('+valueFormat((wins_percents).toFixed(0))+'%)' +
								' ' +
							' ' +
							' ' +
								'' +
									valueFormat((avg_xp).toFixed(0)) +
								' ' +
							' ' +
							'' +
								'' +
									valueFormat((StatArray['info']['statistics'][type]['wr_'+typeShip[tS]]).toFixed(0)) +
								' ' +
							' ' +
							'' +
								'' +
									valueFormat((StatArray['info']['statistics'][type]['wtr_'+typeShip[tS]]).toFixed(0)) +
								' ' +
							' ' +
						' ' +
					' ' +
					'' +
						htmlArray[typeShip[tS]] +
					' ' +
				'';
			}
			
			return html;
		}
		function GeneratorUserBar(userbarbg){
			var jsonString = 'json='+JSON.stringify(MembersArray[0])+'&type=userbar&userbarbg='+userbarbg+'&lang='+lang;
			
			var xmlhttp;
			try{
				xmlhttp = new ActiveXObject('Msxml2.XMLHTTP');
			}catch(e){
				try{
					xmlhttp = new ActiveXObject('Microsoft.XMLHTTP');
				}catch(E){
					xmlhttp = false;
				}
			}
			if(!xmlhttp && typeof XMLHttpRequest != 'undefined'){
				xmlhttp = new XMLHttpRequest();
			}
			xmlhttp.open('POST', ''+WoWsStatInfoHref+'userbar.php?random='+Math.floor(Math.random()*100000001), true);
			xmlhttp.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');
			xmlhttp.onreadystatechange = function(){
				if(xmlhttp.readyState == 4){
					if(xmlhttp.status == 200){
						var userbar_img = document.getElementById('userbar-img');
						userbar_img.src = xmlhttp.responseText+'?'+Math.floor(Math.random()*100000001);
						
						var userbar_link = document.getElementById('userbar-link');
						userbar_link.textContent = '[url='+WoWsStatInfoHref+'?realm_search='+realm+'&nickname='+MembersArray[0]['info']['nickname']+'][img]'+xmlhttp.responseText+'[/img][/url]';
					}
				}
			};
			xmlhttp.send(jsonString);
		}
		var UserBarBGData = null;
		var CountUserBar = [];
		function doneUserBarBG(url, response){
			UserBarBGData = response;
			
			CountUserBar['all'] = 0;
			CountUserBar['clan'] = 0;
			CountUserBar['noclassification'] = 0;
			
			CountUserBar['battleship'] = [];
			CountUserBar['battleship']['count'] = 0;
			CountUserBar['battleship']['japan'] = 0;
			CountUserBar['battleship']['ussr'] = 0;
			CountUserBar['battleship']['germany'] = 0;
			CountUserBar['battleship']['uk'] = 0;
			CountUserBar['battleship']['usa'] = 0;
			
			CountUserBar['aircarrier'] = [];
			CountUserBar['aircarrier']['count'] = 0;
			CountUserBar['aircarrier']['japan'] = 0;
			CountUserBar['aircarrier']['ussr'] = 0;
			CountUserBar['aircarrier']['germany'] = 0;
			CountUserBar['aircarrier']['uk'] = 0;
			CountUserBar['aircarrier']['usa'] = 0;
			
			CountUserBar['cruiser'] = [];
			CountUserBar['cruiser']['count'] = 0;
			CountUserBar['cruiser']['japan'] = 0;
			CountUserBar['cruiser']['ussr'] = 0;
			CountUserBar['cruiser']['germany'] = 0;
			CountUserBar['cruiser']['uk'] = 0;
			CountUserBar['cruiser']['usa'] = 0;
			
			CountUserBar['destroyer'] = [];
			CountUserBar['destroyer']['count'] = 0;
			CountUserBar['destroyer']['japan'] = 0;
			CountUserBar['destroyer']['ussr'] = 0;
			CountUserBar['destroyer']['germany'] = 0;
			CountUserBar['destroyer']['uk'] = 0;
			CountUserBar['destroyer']['usa'] = 0;
			
			for(var i = 0; i < UserBarBGData.length; i++){
				var img = UserBarBGData[i].split('_');
				if(img.length > 1){
					for(var i_id = 1; i_id < img.length; i_id++){
						if(MembersArray[0]['clan'] == null){break;}
						if(img[i_id] == MembersArray[0]['clan']['clan_id']){
							CountUserBar['clan']++;
							CountUserBar['all']++;
						}
					}
				}else{
					CountUserBar['all']++;
					var shipsBG = UserBarBGData[i].split('-');
					if(shipsBG.length == 2){
						CountUserBar['noclassification']++;
					}else if(shipsBG.length == 3){
						var type = shipsBG[1];
						var nation = shipsBG[2];
						CountUserBar[type]['count']++;
						CountUserBar[type][nation]++;
					}
				}
			}
			
			var html = '' +
				'' +
					localizationText['userbar-filters']+' ' +
					'' +
						''+localizationText['filters-all']+' ('+CountUserBar['all']+') ' +
						''+localizationText['filters-clan']+' ('+CountUserBar['clan']+') ' +
						''+localizationText['filters-noclassification']+' ('+CountUserBar['noclassification']+') ' +
						''+localizationText['filters-battleship']+' ('+CountUserBar['battleship']['count']+') ' +
						''+localizationText['filters-aircarrier']+' ('+CountUserBar['aircarrier']['count']+') ' +
						''+localizationText['filters-cruiser']+' ('+CountUserBar['cruiser']['count']+') ' +
						''+localizationText['filters-destroyer']+' ('+CountUserBar['destroyer']['count']+') ' +
					' ' +
					'' +
						''+localizationText['filters-japan']+' ' +
						''+localizationText['filters-ussr']+' ' +
						''+localizationText['filters-germany']+' ' +
						''+localizationText['filters-uk']+' ' +
						''+localizationText['filters-usa']+' ' +
					' ' +
				'
' +
			'';
			
			var check = true;
			html += '';
			for(var i = 0; i < UserBarBGData.length; i++){				
				var imgbgview = false;
				var img = UserBarBGData[i].split('_');
				if(img.length > 1){
					for(var i_id = 1; i_id < img.length; i_id++){
						if(MembersArray[0]['clan'] == null){break;}
						if(img[i_id] == MembersArray[0]['clan']['clan_id']){
							imgbgview = true;
						}
					}
				}else{
					imgbgview = true;
				}
				
				if(imgbgview){
					var checked = ''; if(check){checked = 'checked="checked"'; check = false;}
					html += '
 '+UserBarBGData[i]+'
';
					html += '
';
				}
			}
			html += '
' +
					localizationText['userbar-your-background'] +
				' ' +
			'';
			
			onShowMessage(
				localizationText['userbar-bg'],
				html,
				function(){
					var userbarbg = 'userbar';
					
					var radios = document.getElementsByName('userbar-bg');
					for(var i = 0; i < radios.length; i++){
						if(radios[i].checked){
							userbarbg = radios[i].value;
							break;
						}
					}
				
					GeneratorUserBar(userbarbg); 
					
					onCloseMessage();
				},
				localizationText['Ok'],
				true
			);
			jQ('#userbar-your-background').click(function(){
				onCloseMessage();
				
				var html = '' +
					'' + 
						'
'+localizationText['img-max-size']+', '+localizationText['img-max-px']+', '+localizationText['img-format']+'
' +
						'
' +
						'
' +
					'
' +
					'
 userbar
' +
					'
' +
				'
';
						
							var Keys = Object.keys(StatPvPMemberArray);
							var IndexLast = Keys.length - 1;
							var IndexOld = Keys.length - 2;
							
							var battles = StatPvPMemberArray[Keys[IndexLast]]['battles'] - StatPvPMemberArray[Keys[IndexOld]]['battles'];
							var wins_percents = StatPvPMemberArray[Keys[IndexLast]]['wins_percents'] - StatPvPMemberArray[Keys[IndexOld]]['wins_percents'];
							var avg_xp = StatPvPMemberArray[Keys[IndexLast]]['avg_xp'] - StatPvPMemberArray[Keys[IndexOld]]['avg_xp'];
							var avg_damage_dealt = StatPvPMemberArray[Keys[IndexLast]]['avg_damage_dealt'] - StatPvPMemberArray[Keys[IndexOld]]['avg_damage_dealt'];
							var kill_dead = StatPvPMemberArray[Keys[IndexLast]]['kill_dead'] - StatPvPMemberArray[Keys[IndexOld]]['kill_dead'];
							var avg_frags = StatPvPMemberArray[Keys[IndexLast]]['avg_frags'] - StatPvPMemberArray[Keys[IndexOld]]['avg_frags'];
							var avg_planes_killed = StatPvPMemberArray[Keys[IndexLast]]['avg_planes_killed'] - StatPvPMemberArray[Keys[IndexOld]]['avg_planes_killed'];
							var avg_capture_points = StatPvPMemberArray[Keys[IndexLast]]['avg_capture_points'] - StatPvPMemberArray[Keys[IndexOld]]['avg_capture_points'];
							var avg_dropped_capture_points = StatPvPMemberArray[Keys[IndexLast]]['avg_dropped_capture_points'] - StatPvPMemberArray[Keys[IndexOld]]['avg_dropped_capture_points'];
							var avg_battles_level = StatPvPMemberArray[Keys[IndexLast]]['avg_battles_level'] - StatPvPMemberArray[Keys[IndexOld]]['avg_battles_level'];
							var wr = StatPvPMemberArray[Keys[IndexLast]]['wr'] - StatPvPMemberArray[Keys[IndexOld]]['wr'];
							var wtr = StatPvPMemberArray[Keys[IndexLast]]['wtr'] - StatPvPMemberArray[Keys[IndexOld]]['wtr'];
							
							var _values = tabContainer.getElementsByClassName('_values')[0];
							var main_stat = _values.getElementsByTagName('div');
							main_stat[0].innerHTML += getHTMLDif(battles, 0);
							main_stat[1].innerHTML += getHTMLDif(wins_percents, 2);
							main_stat[2].innerHTML += getHTMLDif(avg_xp, 2);
							main_stat[3].innerHTML += getHTMLDif(avg_damage_dealt, 0);
							main_stat[4].innerHTML += getHTMLDif(kill_dead, 2);
							
							if(account_battle_stats != null){
								var account_table = account_battle_stats.getElementsByClassName('account-table');
								
								account_table[1].rows[1].cells[1].innerHTML  += getHTMLDif(avg_xp, 2);
								account_table[1].rows[2].cells[1].innerHTML  += getHTMLDif(avg_damage_dealt, 2);
								account_table[1].rows[3].cells[1].innerHTML  += getHTMLDif(avg_frags, 2);
								account_table[1].rows[4].cells[1].innerHTML  += getHTMLDif(avg_planes_killed, 2);
								//account_table[1].rows[5].cells[1].innerHTML  += getHTMLDif(avg_capture_points, 2);
								//account_table[1].rows[6].cells[1].innerHTML  += getHTMLDif(avg_dropped_capture_points, 2);
								
								addStatHover(account_table[0].rows[1], 'battles');
								addStatHover(account_table[0].rows[2], 'wins_percents');
								addStatHover(account_table[0].rows[3], 'survived_battles_percents');
								
								addStatHover(account_table[1].rows[1], 'avg_xp');
								addStatHover(account_table[1].rows[2], 'avg_damage_dealt');
								addStatHover(account_table[1].rows[3], 'avg_frags');
								addStatHover(account_table[1].rows[4], 'avg_planes_killed');
								
								addStatHover(account_table[2].rows[1], 'max_xp');
								addStatHover(account_table[2].rows[2], 'max_damage_dealt');
								addStatHover(account_table[2].rows[3], 'max_frags_battle');
								addStatHover(account_table[2].rows[4], 'max_planes_killed');
							}
							
							var main_page_script_block = document.getElementById('main-page-script-block');
							if(main_page_script_block != null){
								var account_table = main_page_script_block.getElementsByClassName('account-table');
								
								account_table[0].rows[4].cells[1].innerHTML  += getHTMLDif(avg_battles_level, 2);
								account_table[0].rows[5].cells[1].innerHTML  += getHTMLDif(wr, 2);
								account_table[0].rows[6].cells[1].innerHTML  += getHTMLDif(wtr, 2);
							}
						}
					}
					
					var account_tab = tabContainer.getElementsByClassName('account-tab-charts')[0];
					if(account_tab == null){					
						var account_tab_detail_stats = tabContainer.getElementsByClassName('account-tab-detail-stats')[0];
						if(account_tab_detail_stats != null){
							var date = [];
							var value = [];
							var html_chart = '';
							var chart_value = ['wins_percents', 'avg_xp', 'avg_damage_dealt', 'wr', 'kill_dead', 'avg_battles_level'];
							
							for(var key in chart_value){
								var title = chart_value[key];
								html_chart += '' +
									'' +
										'
'+localizationText['title_'+title]+' ' +
										'
' +
									'
' +
									'
' +
										html_chart + 
									'
' +
								'
'+
									''+localizationText['charts']+'
'+
									''+
								' '+
							'');
							jQ(tabContainer).find('div.account-tabs-mobile ul').append(''+
								''+localizationText['charts']+' ' +
							'');
						}
					}
				}
			}
		}
		function updateStatPvPMemberArray(response){
			if(response == null){
				StatPvPMemberArray = [];
			}else{
				StatPvPMemberArray = jQ.parseJSON(response);
			}
		}
		function viewChart(title, date, value){
			$.plot(
				'#chart_'+title, 
				[value], 
				{
					series: {lines: {show: true}, points: {show: true}},
					xaxis: {ticks: date},
					grid: {hoverable: true, clickable: true},
					colors: ['#00c0c0']
				}
			);
			
			$('
').css({
				position: "absolute",
				display: "none",
				border: "1px solid #fdd",
				padding: "2px",
				"background-color": "#00c0c0",
				opacity: 0.80,
				zIndex: 999,
				color: "black"
			}).appendTo("body");
			$('#chart_'+title).bind("plothover", function (event, pos, item) {
				if(item){
					var y = item.datapoint[1].toFixed(2);
					$('#tooltip-chart-'+title).html(y).css({top: item.pageY - 35, left: item.pageX - 15}).fadeIn(200);
				}else{
					$('#tooltip-chart-'+title).hide();
				}
			});
		}
		function getHTMLDif(value, fixed){
			var text = valueFormat(value.toFixed(fixed));
			var color = '#fff';
			if(value > 0){
				text = '+'+text;
				color = 'green';
			}else if(value < 0){
				color = 'red';
			}
		
			return ' '+text+' 
').css({
					width: "200px",
					position: "absolute",
					display: "none",
					textAlign: "center",
					"background-color": "#066",
					padding: "0px", 
					top: "0", 
					left: "0"
				}).appendTo("body");
			}
		}
		function hoverStatIn(){
			var type_stat_hover = $(this).attr('type-stat');
			
			var bodyRect = document.body.getBoundingClientRect(), 
			elemRect = this.getBoundingClientRect(), 
			offsetTop = elemRect.top - bodyRect.top, 
			offsetLeft = elemRect.left - bodyRect.left;
			
			var offsetWidth = this.offsetWidth;
			var offsetHeight = this.offsetHeight;
			
			var paddingLeft = Number($(this).css('padding-left').replace(/[^0-9\.]+/g,""));
			var paddingTop = Number($(this).css('padding-top').replace(/[^0-9\.]+/g,""));
			var paddingRight = Number($(this).css('padding-bottom').replace(/[^0-9\.]+/g,""));
			var paddingBottom = Number($(this).css('padding-bottom').replace(/[^0-9\.]+/g,""));
			
			var statValue = MembersArray[0]['info']['statistics']['pvp'][type_stat_hover];
			
			var plusValue = 0.01;
			var tofixedNum = 2; 
			if(type_stat_hover == 'battles' || type_stat_hover.indexOf("max_") > -1){
				tofixedNum = 0;
				plusValue = 1;
			}
			
			var color1 = color['very_bad'];
			var color2 = color['very_bad'];
			var color3 = color['very_bad'];
			
			var value1 = parseFloat(0);
			var value3 = parseFloat(0);
			
			if(parseFloat(statValue) <= parseFloat(colorStat[type_stat_hover][5])){
				color1 = color['very_good'];
				color2 = color['unique'];
				color3 = color['unique'];
				
				value1 = (parseFloat(colorStat[type_stat_hover][4])).toFixed(tofixedNum);
				value3 = '∞';
			}
			if(parseFloat(statValue) <= parseFloat(colorStat[type_stat_hover][4])){
				color1 = color['good'];
				color2 = color['very_good'];
				color3 = color['unique'];
				
				value1 = (parseFloat(colorStat[type_stat_hover][3])).toFixed(tofixedNum);
				value3 = (parseFloat(colorStat[type_stat_hover][4]) + plusValue).toFixed(tofixedNum);
			}
			if(parseFloat(statValue) <= parseFloat(colorStat[type_stat_hover][3])){
				color1 = color['normal'];
				color2 = color['good'];
				color3 = color['very_good'];
				
				value1 = (parseFloat(colorStat[type_stat_hover][2])).toFixed(tofixedNum);
				value3 = (parseFloat(colorStat[type_stat_hover][3]) + plusValue).toFixed(tofixedNum);
			}
			if(parseFloat(statValue) <= parseFloat(colorStat[type_stat_hover][2])){
				color1 = color['bad'];
				color2 = color['normal'];
				color3 = color['good'];
				
				value1 = (parseFloat(colorStat[type_stat_hover][1])).toFixed(tofixedNum);
				value3 = (parseFloat(colorStat[type_stat_hover][2]) + plusValue).toFixed(tofixedNum);
			}
			if(parseFloat(statValue) <= parseFloat(colorStat[type_stat_hover][1])){
				color1 = color['very_bad'];
				color2 = color['bad'];
				color3 = color['normal'];
				
				value1 = (parseFloat(colorStat[type_stat_hover][0])).toFixed(tofixedNum);
				value3 = (parseFloat(colorStat[type_stat_hover][1]) + plusValue).toFixed(tofixedNum);
			}
			if(parseFloat(statValue) <= parseFloat(colorStat[type_stat_hover][0])){
				color1 = color['very_bad'];
				color2 = color['very_bad'];
				color3 = color['bad'];
				
				value1 = (parseFloat(0)).toFixed(tofixedNum);
				value3 = (parseFloat(colorStat[type_stat_hover][0]) + plusValue).toFixed(tofixedNum);
			}
			
			var next_percent_wins_html = '';
			if(type_stat_hover == 'wins_percents'){
				var next_percent_losses = Math.floor(100 - MembersArray[0]['info']['statistics']['pvp']['wins_percents']);
				var next_percent_wins = 100 - next_percent_losses;
				var next_losses_rate = next_percent_losses / 100;
				var next_battles = Math.ceil(
					(
						(
							MembersArray[0]['info']['statistics']['pvp']['battles'] - MembersArray[0]['info']['statistics']['pvp']['wins']
						) / next_losses_rate
					) - MembersArray[0]['info']['statistics']['pvp']['battles']
				);
				
				next_percent_wins_html = ''+next_battles+' '+localizationText['to']+' '+next_percent_wins+'%
';
			}
			
			
			$('#tooltip-stat-'+type_stat_hover).html('' +
				''+localizationText['info.statistics.pvp.'+type_stat_hover]+'
' +
				'' +
				''+value1+' ' +
				' ⇐ '+(statValue).toFixed(tofixedNum)+'  ⇒ ' +
				''+value3+' ' +
				'
' +
				next_percent_wins_html +
			'').css({
				width: offsetWidth+"px", //(offsetWidth - paddingRight - paddingLeft)
				display: "block", 
				top: offsetTop + offsetHeight - paddingTop - paddingBottom, 
				left: offsetLeft //+ paddingLeft
			}).fadeIn(200);
		}
		function hoverStatOut(){
			var type_stat_hover = $(this).attr('type-stat');
			$('#tooltip-stat-'+type_stat_hover).hide();
		}	
		
		/* ===== ClanPage function ===== */
		function getSettingsClanPage(){
			var SettingsWoWsStatInfo = getLocalStorage('SettingsWoWsStatInfo', false);
			if(SettingsWoWsStatInfo == null){
				setLocalStorage('SettingsWoWsStatInfo', defaultSettingsWoWsStatInfo, false);
				return jQ.parseJSON(defaultSettingsWoWsStatInfo);
			}else{
				return jQ.parseJSON(SettingsWoWsStatInfo);
			}
		}
		function runSettingsClanPage(){
			var page_header = document.getElementsByClassName("page-header")[0];
			countColumn = (page_header.offsetWidth / 100).toFixed(0);
			
			var html = '';
			html += '';
			html += ''+localizationText['table-setting-caption']+'  (0 /'+countColumn+')';
			html += '';
			
			var countCheck = 0;
			var tableNum = null;
			for(attr in SettingsWCI){
				if(tableNum != SettingsWCI[attr]['table'] && 'spacerow' != SettingsWCI[attr]['table'] && 'position' != SettingsWCI[attr]['table']){
					if(tableNum != null){
						html += '  
';
						html += '';
						html += '';						
					}
					
					html += '';
			html += ' ';
			html += '';
			html += '';
			
			html += '' +
				'' +
					'' +
						'' +
							'' +
								''+localizationText['set-settings-default']+' ' +
							' ' +
						' ' +
					' ' +
				'
' +
			'';
			
			onShowMessage(
				localizationText['get-settings-button'], 
				html, 
				function(){onOkSettingsClanPage(); onCloseMessage();}, 
				localizationText['Ok'], 
				true
			);
			
			checkSettingsClanPage();
			
			jQ('.table-settings-row-input').click(function(){checkSettingsClanPage();});
			jQ(".table-setting-structure-up").click(function(){
				var table_setting_structure = document.getElementsByClassName("table-setting-structure")[0];
				if(table_setting_structure.selectedIndex == 0 || table_setting_structure.selectedIndex == -1){return;}
				var structure_table_selected_value = table_setting_structure.options[table_setting_structure.selectedIndex].value;
				var structure_table_selected_text = table_setting_structure.options[table_setting_structure.selectedIndex].text;
				for(i = 0; i < table_setting_structure.options.length; i++){
					if(table_setting_structure.options[i+1].value == structure_table_selected_value){
						table_setting_structure.options[i+1].value = table_setting_structure.options[i].value;
						table_setting_structure.options[i+1].text = table_setting_structure.options[i].text;
						table_setting_structure.options[i].value = structure_table_selected_value;
						table_setting_structure.options[i].text = structure_table_selected_text;
						table_setting_structure.selectedIndex = i;
						break;
					}
				}
			});
			jQ(".table-setting-structure-down").click(function(){
				var table_setting_structure = document.getElementsByClassName("table-setting-structure")[0];
				if(table_setting_structure.selectedIndex == (table_setting_structure.options.length - 1) || table_setting_structure.selectedIndex == -1){return;}
				var structure_table_selected_value = table_setting_structure.options[table_setting_structure.selectedIndex].value;
				var structure_table_selected_text = table_setting_structure.options[table_setting_structure.selectedIndex].text;
				for(i = 0; i < table_setting_structure.options.length; i++){
					if(table_setting_structure.options[i].value == structure_table_selected_value){
						table_setting_structure.options[i].value = table_setting_structure.options[i+1].value;
						table_setting_structure.options[i].text = table_setting_structure.options[i+1].text;
						table_setting_structure.options[i+1].value = structure_table_selected_value;
						table_setting_structure.options[i+1].text = structure_table_selected_text;
						table_setting_structure.selectedIndex = i + 1;
						break;
					}
				}
			});
			jQ('#set-settings-default').click(function(){
				window.localStorage.removeItem('SettingsWoWsStatInfo');
				setTimeout(function(){window.location.reload();}, 2000);
			});
		}
		function checkSettingsClanPage(){
			var countCheck = 0;
			var table_settings_row_input = document.getElementsByClassName("table-settings-row-input");
			var table_setting_structure = document.getElementsByClassName("table-setting-structure")[0];
			for(var i = 0; i < table_settings_row_input.length; i++){
				if(table_settings_row_input[i].checked){
					countCheck++;
					
					for(j = 0; j < table_setting_structure.options.length; j++){
						if(table_settings_row_input[i].name == table_setting_structure.options[j].value){
							break;
						}
						if(j == table_setting_structure.options.length - 1){
							var option = document.createElement("option");
							option.text = localizationText[table_settings_row_input[i].name];
							option.value = table_settings_row_input[i].name;
							table_setting_structure.add(option);							
						}
					}
					if(table_setting_structure.options.length == 0){
						var option = document.createElement("option");
						option.text = localizationText[table_settings_row_input[i].name];
						option.value = table_settings_row_input[i].name;
						table_setting_structure.add(option);
					}
				}else{
					for(j = 0; j < table_setting_structure.options.length; j++){
						if(table_settings_row_input[i].name == table_setting_structure.options[j].value){
							table_setting_structure.remove(j);
							break;
						}
					}
				}
			}
			
			jQ('#table-setting-count-check').html(countCheck);
		
			if(countCheck < countColumn){
				jQ('.table-settings-row-input').attr('disabled', false);
			}else{
				jQ('.table-settings-row-input').attr('disabled', true);
			}
			
			for(var i = 0; i < table_settings_row_input.length; i++){
				if(table_settings_row_input[i].checked){
					table_settings_row_input[i].disabled = false;
				}
			}
		}
		function onOkSettingsClanPage(){
			var getSettings = '';
			var table_settings_row_input = document.getElementsByClassName("table-settings-row-input");
			for(var i = 0; i < table_settings_row_input.length; i++){
				var input = table_settings_row_input[i];
				var attr = input.getAttribute('name');
				var table = input.getAttribute('tablenum');
				var check = 0; if(input.checked){check = 1;}
				
				if(table == 'position'){
					var sort = '';
					var table_setting_structure = document.getElementsByClassName("table-setting-structure")[0];
					for(j = 0; j < table_setting_structure.options.length; j++){
						sort += table_setting_structure.options[j].value+',';
					}
					sort = sort.substring(sort.length - 1, 0);
					getSettings += '"'+attr+'": {"table": "'+table+'", "check": '+check+', "sort": "'+sort+'"}';
				}else{
					getSettings += '"'+attr+'": {"table": "'+table+'", "check": '+check+'},';
				}
			}
			var saveSettingsClanPage = '{'+getSettings+'}';
			setLocalStorage('SettingsWoWsStatInfo', saveSettingsClanPage, false);
			
			SettingsWCI = getSettingsClanPage();
			
			if(loadMemberClan){
				var scrollTop = window.pageYOffset || document.documentElement.scrollTop || document.body.scrollTop;
			
				var table_clan_statistic = document.getElementById('table-clan-statistic');
				table_clan_statistic.parentNode.removeChild(table_clan_statistic);
				
				var statistic_clan_button_text = document.getElementById('statistic-clan-button-text');
				statistic_clan_button_text.innerHTML = localizationText['statistic-clan-button-1'];
				
				if(clickRunCount % 2 == 1){
					clickRunCount++;
				}
				
				viewMainPageClan();
				
				window.pageYOffset = scrollTop;
				document.documentElement.scrollTop = scrollTop;
				document.body.scrollTop = scrollTop;
			}
		}
		function getClanMembersList(){
			var table = document.getElementsByClassName("tbl-rating_skeleton")[0];
			
			if(table.rows.length <= 1){
				setTimeout(function(){getClanMembersList();}, 1000);
				return;
			}
			
			for(var i = 1; i < table.rows.length; i++){
				var row = table.rows[i];
				
				var account_id = row.getAttribute('data-account_id');
				if(account_id != null){
					var index = MembersArray.length;
					MembersArray[index] = [];
					MembersArray[index]['account_id'] = account_id;
					
					var player_name = row.getElementsByClassName("player_name")[0];
					var account_name = player_name.innerHTML.trim();
					MembersArray[index]['account_name'] = account_name;
					
					var tbl_rating_epaulettes = row.getElementsByClassName('tbl-rating_epaulettes')[0];
					var epaulettesSplit = tbl_rating_epaulettes.getAttribute('class').split(' ');
					for(eS = 0; eS < epaulettesSplit.length; eS++){
						if(epaulettesSplit[eS].indexOf('js-tooltip-id_js-role_') > -1){
							var role = epaulettesSplit[eS].replace('js-tooltip-id_js-role_', '');
							role = role.replace('-tooltip', '');
							
							MembersArray[index]['role'] = role;
							MembersArray[index]['role_i18n'] = getRoleText(role);
							MembersArray[index]['role_sort_num'] = getRoleSortNum(role);
							
							break;
						}
					}
					
					var colNum = (row.cells.length - 2) - 1;
					var clan_days = row.cells[colNum].getElementsByClassName('tbl-rating_value')[0].innerHTML.trim();
					MembersArray[index]['clan_days'] = clan_days;
				}
			}
			
			if(MembersArray.length > 0){
				viewClanMemberHistory();
			}
		}
		function viewClanMemberHistory(){
			var oldClanMemberSave = getLocalStorage('ClanMemberSave', true);
			
			var oldClanChangeSave = getLocalStorage('ClanChangeSave', true);
			if(oldClanChangeSave == null){oldClanChangeSave = '';}
			
			var stringJSON = '';
			for(var i = 0; i < MembersArray.length; i++){
				stringJSON += '"'+MembersArray[i]['account_id']+'":{"account_id":'+MembersArray[i]['account_id']+',"account_name":"'+MembersArray[i]['account_name']+'","role":"'+MembersArray[i]['role']+'"},';
			}
			stringJSON = stringJSON.substring(stringJSON.length - 1, 0);
			var newClanMemberSave = '{"members": {'+stringJSON+'}}';
			
			var newClanChangeSave = '';
			
			if(oldClanMemberSave != null){
				var oldClanMemberArr = jQ.parseJSON(oldClanMemberSave);
				var newClanMemberArr = jQ.parseJSON(newClanMemberSave);
				
				for(id in oldClanMemberArr['members']){
					if(newClanMemberArr['members'][id] === undefined){
						oldClanChangeSave += '1:'+oldClanMemberArr['members'][id]['account_id']+':'+oldClanMemberArr['members'][id]['account_name']+':null:null;';
					}else{
						if(newClanMemberArr['members'][id]['role'] != oldClanMemberArr['members'][id]['role']){
							oldClanChangeSave += '3:'+oldClanMemberArr['members'][id]['account_id']+':'+oldClanMemberArr['members'][id]['account_name']+':'+oldClanMemberArr['members'][id]['role']+':'+newClanMemberArr['members'][id]['role']+';';
						}
						if(newClanMemberArr['members'][id]['account_name'] != oldClanMemberArr['members'][id]['account_name']){
							oldClanChangeSave += '4:'+oldClanMemberArr['members'][id]['account_id']+':'+oldClanMemberArr['members'][id]['account_name']+':'+oldClanMemberArr['members'][id]['account_name']+':'+newClanMemberArr['members'][id]['account_name']+';';
						}
					}
				}
				
				for(id in newClanMemberArr['members']){
					if(oldClanMemberArr['members'][id] === undefined){
						oldClanChangeSave += '2:'+newClanMemberArr['members'][id]['account_id']+':'+newClanMemberArr['members'][id]['account_name']+':null:null;';
						if(newClanMemberArr['members'][id]['role'] != 'recruit'){
							oldClanChangeSave += '3:'+newClanMemberArr['members'][id]['account_id']+':'+newClanMemberArr['members'][id]['account_name']+':recruit:'+newClanMemberArr['members'][id]['role']+';';
						}
					}
				}
			}
			
			var html = '';
			var clan_member_history = document.getElementsByClassName("clan-member-history")[0];
			
			if(oldClanChangeSave != ''){
				var arrOldChangeClan = oldClanChangeSave.split(';');
				if(arrOldChangeClan.length > 50){
					var deleteCountLine = arrOldChangeClan.length - 50;
					for(var i = 0; i < deleteCountLine; i++){
						arrOldChangeClan.splice(0, 1);
					}
					for(var i = 0; i < arrOldChangeClan.length; i++){
						if(arrOldChangeClan[i] == ''){continue;}
						newClanChangeSave += arrOldChangeClan[i]+';';
					}
				}else{
					newClanChangeSave = oldClanChangeSave;
				}
				
				var arrNewChangeClan = newClanChangeSave.split(';');
				for(var i = (arrNewChangeClan.length - 1); i >= 0; i--){
					var arrChange = arrNewChangeClan[i].split(':');
					var type = arrChange[0];
					var mId = arrChange[1];
					var mName = arrChange[2];
					var mOld = arrChange[3];
					var mNew = arrChange[4];
					
					if(type == '1'){
						var text = localizationText['member-history-leave'];
						text = text.replace('%NAME%',''+mName+' ');
						html +=  ''+				
							''+
								' '+text+' '+
							' '+
						' ';						
					}else if(type == '2'){
						var text = localizationText['member-history-join'];
						text = text.replace('%NAME%',''+mName+' ');					
						html +=  ''+				
							''+
								' '+text+' '+
							' '+
						' ';						
					}else if(type == '3'){
						var text = localizationText['member-history-rerole'];
						text = text.replace('%NAME%',''+mName+' ');					
						text = text.replace('%OLDROLE%',getRoleText(mOld));					
						text = text.replace('%NEWROLE%',getRoleText(mNew));					
						html +=  ''+				
							''+
								text+
							' '+
						' ';					
					}else if(type == '4'){
						var text = localizationText['member-history-rename'];
						text = text.replace('%OLDNAME%',mOld);						
						text = text.replace('%NEWNAME%',''+mNew+' ');						
						html +=  ''+				
							''+
								text+
							' '+
						' ';					
					}
				}
			}else{
				html += '' +
					'' +
						'' +
							localizationText['member-history-notchange'] +
						' ' +
					' ' +
				'';
			}
			
			html += ' 
' +
					'' +
						'' +
							'' +
								''+localizationText['member-history-clear']+' ' +
							' ' +
						' ' +
					' ' +
				'
' +
			'';
			clan_member_history.innerHTML = html;
			
			jQ('#run-history-clear').click(function(){
				setLocalStorage('ClanChangeSave', '', true);
				viewClanMemberHistory();
			});
			
			setLocalStorage('ClanMemberSave', newClanMemberSave, true);
			setLocalStorage('ClanChangeSave', newClanChangeSave, true);
		}		
		var clickRunCount = 0;
		var loadMemberClan = false;
		var loadTimeStart = 0;
		var countMembers = 0;
		var loadOnePercent = 0;
		var loadMemberCount = 0;
		function runGetStatisticClan(){
			if(clickRunCount % 2 == 0){
				var statistic_clan_button_text = document.getElementById('statistic-clan-button-text');
				statistic_clan_button_text.innerHTML = localizationText['statistic-clan-button-1'];
				
				if(!loadMemberClan){
					var get_clan_statistic_block = document.getElementById('get-clan-statistic-block');
					get_clan_statistic_block.style.display = 'none';
					
					var statistic_clan_load_icon = document.getElementById('statistic-clan-load-icon');
					statistic_clan_load_icon.style.display = 'block';
					
					var statistic_clan_load_text = document.getElementById('statistic-clan-load-text');
					statistic_clan_load_text.style.display = 'block';
					statistic_clan_load_text.innerHTML = localizationText['statistic-clan-load-text']+' 0%'+localizationText['statistic-load-text-lost']+' 0 '+localizationText['statistic-load-text-min']+' 0 '+localizationText['statistic-load-text-sec'];
					
					if(Encyclopedia == null){
						getJson(WOWSAPI+'encyclopedia/ships/?application_id='+application_id+'&fields=name,images,tier,nation,is_premium,images,type', doneEncyclopedia, errorEncyclopedia);
					}
					
					loadTimeStart = new Date().getTime();
					countMembers = MembersArray.length;
					loadOnePercent = 100 / countMembers;
					for(var i = 0; i < MembersArray.length; i++){
						getJson(WOWSAPI+'account/info/?application_id='+application_id+'&extra=statistics.pve,statistics.pvp_solo,statistics.pvp_div2,statistics.pvp_div3&account_id='+MembersArray[i]['account_id']+'&index='+i+'&type=clan', doneAccountInfo, errorAccountInfo);
					}
				}else{
					jQ('.timeframe').hide();
					jQ('#js-clan-statistics').hide();
					jQ('#js-playerslist-table').hide();
					jQ('#js-players-management').hide();
			
					jQ('#table-clan-statistic').show();
				}
			}else if(clickRunCount % 2 == 1){
				var statistic_clan_button_text = document.getElementById('statistic-clan-button-text');
				statistic_clan_button_text.innerHTML = localizationText['statistic-clan-button-0'];
				
				jQ('.timeframe').show();
				jQ('#js-clan-statistics').show();
				jQ('#js-playerslist-table').show();
				jQ('#js-players-management').show();
				
				jQ('#table-clan-statistic').hide();
			}
			
			clickRunCount++;
		}
		function viewMainPageClan(){
			if(loadMemberCount != countMembers){
				var loadPercent = (loadMemberCount * loadOnePercent).toFixed(0);
				
				var loadTime = (((new Date().getTime() - loadTimeStart) / loadMemberCount) * countMembers) + loadTimeStart;
				var lostTime = loadTime - new Date().getTime();
				var ss = (lostTime / 1000 % 60).toFixed(0);
				var mm = parseInt((lostTime / (60 * 1000) % 60), 10);
				
				var statistic_clan_load_text = document.getElementById('statistic-clan-load-text');
				statistic_clan_load_text.innerHTML = localizationText['statistic-clan-load-text']+' '+loadPercent+'%'+
				localizationText['statistic-load-text-lost']+
				' '+mm+' '+localizationText['statistic-load-text-min']+' '+ss+' '+localizationText['statistic-load-text-sec'];
				
				return;
			}
			
			loadMemberClan = true;
			
			var statistic_clan_load_icon = document.getElementById('statistic-clan-load-icon');
			statistic_clan_load_icon.style.display = 'none';
			
			var statistic_clan_load_text = document.getElementById('statistic-clan-load-text');
			statistic_clan_load_text.style.display = 'none';
			
			var get_clan_statistic_block = document.getElementById('get-clan-statistic-block');
			get_clan_statistic_block.style.display = 'block';
			
			jQ('.timeframe').hide();
			jQ('#js-clan-statistics').hide();
			jQ('#js-playerslist-table').hide();
			jQ('#js-players-management').hide();
			
			var page_header = document.getElementsByClassName("page-header")[0];
			var tableSizeWidth = page_header.offsetWidth;
			
			var table = document.createElement('table');
			table.setAttribute('id', 'table-clan-statistic');
			table.setAttribute('style', 'border-collapse: separate; font-size: 13px; border-bottom: 1px solid rgba(255, 255, 255, 0.1); margin-top: 20px; margin-left: -40px; width: '+tableSizeWidth+'px;');
			
			var thead = document.createElement('thead');
			var tr_head = document.createElement('tr');
			var th = document.createElement('th');
			th.setAttribute('class', 'table-default_heading');
			th.innerHTML = '' +
				localizationText['account_name'] +
				' '; 
			tr_head.appendChild(th);
			
			var column = SettingsWCI['view_table']['sort'].split(',');
			for(var i = 0; i < column.length; i++){          
				var th = document.createElement('th');
				th.setAttribute('class', 'table-default_heading');
				th.innerHTML = '' +
					localizationText[column[i]] +
					' ';
				tr_head.appendChild(th);
			}
			thead.appendChild(tr_head);
			table.appendChild(thead);
			
			var tbody = document.createElement('tbody');
			for(var index = 0; index < MembersArray.length; index++){			
				if(!calcStat(index)){
					console.log('Error calcStat '+MembersArray[index]['account_id']);
				}
			
				var tr = document.createElement('tr');
				var td = document.createElement('td');
				td.setAttribute('class', 'table-default_cell');
				td.innerHTML = (index + 1)+'. ' +
					MembersArray[index]['account_name'] + 
				' ';
				tr.appendChild(td);
				
				for(var i = 0; i < column.length; i++){
					var td = document.createElement('td');
					td.setAttribute('class', 'table-default_cell');
					td.innerHTML = getStatisticClanMember(index, column[i], td);
					tr.appendChild(td);
				}
				tbody.appendChild(tr);
			}
			table.appendChild(tbody);
			
			var ClanPageElement = document.getElementsByClassName('rating')[0];
			ClanPageElement.appendChild(table);
			
			jQ('.table-clan-statistic-sorter').click(function(){sortStatisticClan(this);});
		}
		function getStatisticClanMember(index, attr, td){
			var layer = attr.split('\.');
			var value = null;
			for(var i = 0; i < layer.length; i++){
				if(value == null){
					value = MembersArray[index][layer[i]];
				}else{
					value = value[layer[i]];
				}
			}
			
			if(attr.indexOf("last_battle_time") > -1 || attr.indexOf("logout_at") > -1){
				return TimeToDate(value);
			}else if(attr.indexOf("role_i18n") > -1){
				return value;
			}
			
			if(attr.indexOf(".avg_") > -1 || attr.indexOf("_percents") > -1 || attr.indexOf(".kill_dead") > -1 || attr.indexOf(".wr") > -1 || attr.indexOf(".wtr") > -1){
				value = (value).toFixed(2);
			}
			
			var value_html = '';
			if(attr.indexOf(".avg_") > -1 || attr.indexOf("_percents") > -1 || (attr.indexOf(".max_") > -1 && attr.indexOf(".max_xp") == -1) 
				|| attr.indexOf(".battles") > -1 || attr.indexOf(".kill_dead") > -1 || attr.indexOf(".wr") > -1 || attr.indexOf(".wtr") > -1){
				value_html = ''+valueFormat(value)+' ';
				td.setAttribute('style', 'white-space: nowrap;');
			}else{
				value_html = valueFormat(value);
				td.setAttribute('style', 'white-space: nowrap;');
			}
			
			return value_html;
		}
		function sortStatisticClan(elementSort){
			var scrollTop = window.pageYOffset || document.documentElement.scrollTop || document.body.scrollTop;
			
			var attr = elementSort.getAttribute('id');
			var sort = elementSort.getAttribute('sort');
			var table_clan_statistic = document.getElementById('table-clan-statistic');
			table_clan_statistic.parentNode.removeChild(table_clan_statistic);
			
			if(sort === 'desc' || sort === 'none'){
				MembersArray.sort(ASC(attr));
			}else{
				MembersArray.sort(DESC(attr));
			}
			
			viewMainPageClan();
			
			window.pageYOffset = scrollTop;
			document.documentElement.scrollTop = scrollTop;
			document.body.scrollTop = scrollTop;
			
			var table_clan_statistic_sorter = document.getElementById(attr);
			if(sort === 'desc' || sort === 'none'){
				table_clan_statistic_sorter.setAttribute('sort', 'asc');
				table_clan_statistic_sorter.setAttribute('class', 'table-clan-statistic-sorter table-sorter table-sorter__up');
			}else{
				table_clan_statistic_sorter.setAttribute('sort', 'desc');
				table_clan_statistic_sorter.setAttribute('class', 'table-clan-statistic-sorter table-sorter table-sorter__down');
			}
		}
		
		/* ===== UserScript function ===== */
		function checkJson(){
			var date = new Date();
			var d = date.getDate();
			if(d < 10){d = '0'+d;}
			var m = date.getMonth(); m++;
			if(m < 10){m = '0'+m;}
			var y = date.getFullYear();
			var numJson = y+''+m+''+d;
			
			if(getLocalStorage('numJson', false) < numJson){
				setLocalStorage('numJson', numJson, false);
				getJson(WoWsStatInfoHref+'get/color.php?'+Math.floor(Math.random()*100000001), doneColorStat, errorColorStat);
				getJson(WoWsStatInfoHref+'get/expships.php?'+Math.floor(Math.random()*100000001), doneExpShips, errorExpShips);
				getJson(WoWsStatInfoHref+'get/expwtr.php?'+Math.floor(Math.random()*100000001), doneExpWTR, errorExpWTR);
			}else{
				getIndexedDB('ColorStat', updateColorStat, updateColorStat);
				getIndexedDB('ExpShips', updateExpShips, updateExpShips);
				getIndexedDB('ExpWTR', updateExpWTR, updateExpWTR);
			}
		}
		function doneColorStat(url, response){
			setIndexedDB('ColorStat', response, updateColorStat, updateColorStat);
		}
		function errorColorStat(url){}
		function doneExpShips(url, response){
			setIndexedDB('ExpShips', response, updateExpShips, updateExpShips);
		}
		function doneExpWTR(url, response){
			setIndexedDB('ExpWTR', response, updateExpWTR, updateExpWTR);
		}
		function updateColorStat(response){
			if(response == null){
				colorStat = jQ.parseJSON('{"wtr":["789.20","983.98","1125.80","1360.69","1548.68","99999.00"],"max_frags_battle":[4,5,6,7,8,99],"avg_planes_killed":["0.19","0.86","1.69","3.59","5.88","99.00"],"max_damage_dealt":[78397,106983,128431,163532,194270,9999999],"wr":["553.98","855.26","1093.56","1524.67","1904.51","99999.00"],"kill_dead":["0.60","0.94","1.25","1.92","2.66","99.00"],"avg_capture_points":["0.05","0.33","0.64","1.33","2.03","99.00"],"survived_battles_percents":["16.53","25.86","33.09","44.92","53.61","100.00"],"max_xp":[1658,2266,2850,3887,4670,99999],"avg_damage_dealt":["14866.40","21346.71","26575.10","36498.53","46017.30","999999.00"],"avg_dropped_capture_points":["0.29","2.13","4.10","7.28","9.91","99.00"],"max_planes_killed":[6,21,31,49,62,999],"avg_xp":["465.20","618.98","764.13","1090.82","1382.56","99999.00"],"avg_frags":["0.48","0.69","0.85","1.12","1.35","99.00"],"wins_percents":["44.73","48.64","51.40","55.94","59.96","100.00"],"battles":[243,415,686,1396,2179,99999]}');
			}else{
				colorStat = response;
			}
		}
		function updateExpShips(response){
			if(response == null){
				ExpShips = jQ.parseJSON('{"4292818736":{"expDamage":44381.43,"expFrags":0.73,"expPlanesKilled":2.04},"4293834544":{"expDamage":7892.27,"expFrags":0.62,"expPlanesKilled":0.01},"4287542992":{"expDamage":27407.18,"expFrags":0.74,"expPlanesKilled":1.76},"4188976592":{"expDamage":8576.18,"expFrags":0.66,"expPlanesKilled":0.01},"4288558800":{"expDamage":22197.05,"expFrags":0.67,"expPlanesKilled":0.45},"4289607376":{"expDamage":19683.82,"expFrags":0.63,"expPlanesKilled":0.53},"4277057520":{"expDamage":34068.77,"expFrags":0.63,"expPlanesKilled":4.13},"4180588336":{"expDamage":45910.69,"expFrags":0.82,"expPlanesKilled":1.66},"4276041424":{"expDamage":81883.32,"expFrags":0.95,"expPlanesKilled":5.05},"4180555216":{"expDamage":27119.9,"expFrags":0.9,"expPlanesKilled":0.33},"4264441840":{"expDamage":14992.99,"expFrags":0.49,"expPlanesKilled":0.27},"4266538992":{"expDamage":12175.31,"expFrags":0.56,"expPlanesKilled":0.02},"4293867504":{"expDamage":17220.69,"expFrags":0.62,"expPlanesKilled":0.03},"4183734064":{"expDamage":27119.78,"expFrags":0.8,"expPlanesKilled":0.68},"4290655952":{"expDamage":25809.42,"expFrags":0.93,"expPlanesKilled":0.12},"4291737040":{"expDamage":30029.34,"expFrags":0.91,"expPlanesKilled":1.25},"4277122768":{"expDamage":85354.31,"expFrags":1.2,"expPlanesKilled":21.0},"4286527184":{"expDamage":25561.71,"expFrags":0.77,"expPlanesKilled":0.17},"4259231440":{"expDamage":67179.42,"expFrags":1.0,"expPlanesKilled":2.67},"3768497968":{"expDamage":16331.25,"expFrags":0.95,"expPlanesKilled":0.01},"4293866960":{"expDamage":35088.17,"expFrags":1.06,"expPlanesKilled":0.04},"4293834736":{"expDamage":9009.93,"expFrags":0.59,"expPlanesKilled":0.01},"4289640144":{"expDamage":19815.55,"expFrags":0.57,"expPlanesKilled":0.35},"4267620048":{"expDamage":28338.06,"expFrags":1.06,"expPlanesKilled":0.97},"4286461936":{"expDamage":26535.73,"expFrags":0.84,"expPlanesKilled":0.3},"4288657392":{"expDamage":30896.72,"expFrags":0.6,"expPlanesKilled":12.07},"4280170480":{"expDamage":18827.3,"expFrags":0.73,"expPlanesKilled":0.39},"4256085712":{"expDamage":10922.49,"expFrags":0.8,"expPlanesKilled":0.01},"4259264496":{"expDamage":37141.92,"expFrags":0.87,"expPlanesKilled":0.84},"4268635856":{"expDamage":36638.37,"expFrags":1.43,"expPlanesKilled":0.01},"4182652368":{"expDamage":33643.02,"expFrags":0.8,"expPlanesKilled":0.6},"4248745968":{"expDamage":38728.96,"expFrags":1.29,"expPlanesKilled":0.24},"4258182864":{"expDamage":12852.49,"expFrags":0.58,"expPlanesKilled":0.01},"4280170192":{"expDamage":16574.53,"expFrags":1.04,"expPlanesKilled":0.01},"4185831216":{"expDamage":14764.96,"expFrags":0.51,"expPlanesKilled":0.04},"3554621136":{"expDamage":44472.51,"expFrags":1.02,"expPlanesKilled":1.45},"4291737584":{"expDamage":12708.43,"expFrags":0.72,"expPlanesKilled":0.01},"4286527472":{"expDamage":42856.08,"expFrags":0.8,"expPlanesKilled":1.51},"4292818896":{"expDamage":34310.54,"expFrags":0.79,"expPlanesKilled":1.89},"4282267344":{"expDamage":52355.34,"expFrags":0.87,"expPlanesKilled":0.36},"4280203248":{"expDamage":31678.18,"expFrags":0.64,"expPlanesKilled":3.12},"4181636912":{"expDamage":35338.8,"expFrags":0.73,"expPlanesKilled":1.63},"4293801680":{"expDamage":16574.53,"expFrags":1.04,"expPlanesKilled":0.01},"4247697392":{"expDamage":38728.96,"expFrags":1.29,"expPlanesKilled":0.24},"4285445840":{"expDamage":40190.77,"expFrags":0.88,"expPlanesKilled":1.89},"4279154384":{"expDamage":16733.58,"expFrags":0.66,"expPlanesKilled":0.04},"4283381456":{"expDamage":13551.31,"expFrags":0.78,"expPlanesKilled":0.01},"4293867216":{"expDamage":18258.15,"expFrags":0.71,"expPlanesKilled":0.06},"4290754544":{"expDamage":20433.88,"expFrags":0.44,"expPlanesKilled":10.19},"4286494416":{"expDamage":39360.63,"expFrags":1.0,"expPlanesKilled":2.26},"4282365648":{"expDamage":53436.14,"expFrags":0.91,"expPlanesKilled":15.37},"4288624624":{"expDamage":27267.71,"expFrags":0.68,"expPlanesKilled":0.57},"4291704528":{"expDamage":20835.45,"expFrags":0.81,"expPlanesKilled":0.09},"4282300400":{"expDamage":25381.43,"expFrags":0.6,"expPlanesKilled":2.12},"4183700944":{"expDamage":18152.88,"expFrags":0.67,"expPlanesKilled":0.17},"4272830448":{"expDamage":33689.36,"expFrags":0.9,"expPlanesKilled":0.57},"4269684432":{"expDamage":14803.92,"expFrags":0.67,"expPlanesKilled":0.02},"4290689008":{"expDamage":19356.58,"expFrags":0.74,"expPlanesKilled":0.01},"4293834192":{"expDamage":20521.1,"expFrags":0.81,"expPlanesKilled":0.02},"4292786160":{"expDamage":10000.76,"expFrags":0.45,"expPlanesKilled":0.01},"4284430032":{"expDamage":44777.07,"expFrags":0.81,"expPlanesKilled":1.09},"4288559088":{"expDamage":19957.26,"expFrags":0.72,"expPlanesKilled":0.32},"4279220208":{"expDamage":104791.58,"expFrags":1.51,"expPlanesKilled":24.74},"4288624336":{"expDamage":41083.36,"expFrags":0.93,"expPlanesKilled":0.85},"4282365936":{"expDamage":47308.47,"expFrags":0.75,"expPlanesKilled":27.01},"4289607664":{"expDamage":17929.55,"expFrags":0.68,"expPlanesKilled":0.45},"4281284304":{"expDamage":55626.33,"expFrags":0.93,"expPlanesKilled":2.77},"4285511376":{"expDamage":40394.81,"expFrags":0.84,"expPlanesKilled":9.8},"3553572560":{"expDamage":44472.51,"expFrags":1.02,"expPlanesKilled":1.45},"4293801424":{"expDamage":22581.8,"expFrags":0.82,"expPlanesKilled":0.34},"4274927600":{"expDamage":17863.3,"expFrags":0.74,"expPlanesKilled":0.03},"4288657104":{"expDamage":40183.56,"expFrags":0.91,"expPlanesKilled":7.3},"4292753392":{"expDamage":9508.99,"expFrags":0.56,"expPlanesKilled":0.01},"4273911792":{"expDamage":53723.44,"expFrags":0.88,"expPlanesKilled":3.49},"4276041712":{"expDamage":60091.8,"expFrags":0.87,"expPlanesKilled":4.71},"4179506640":{"expDamage":40629.69,"expFrags":0.81,"expPlanesKilled":1.0},"4284397008":{"expDamage":19949.31,"expFrags":0.95,"expPlanesKilled":0.01},"4282333168":{"expDamage":52216.96,"expFrags":0.85,"expPlanesKilled":4.43},"4186846672":{"expDamage":11102.51,"expFrags":0.58,"expPlanesKilled":0.01},"4292753104":{"expDamage":11726.37,"expFrags":0.68,"expPlanesKilled":0.01},"4277024464":{"expDamage":36638.37,"expFrags":1.43,"expPlanesKilled":0.01},"4269717488":{"expDamage":17935.15,"expFrags":0.56,"expPlanesKilled":0.13},"4292851408":{"expDamage":34585.0,"expFrags":0.83,"expPlanesKilled":6.16},"4186879792":{"expDamage":15100.18,"expFrags":0.67,"expPlanesKilled":0.01},"4184782640":{"expDamage":27766.33,"expFrags":0.87,"expPlanesKilled":0.25},"4292785616":{"expDamage":19949.31,"expFrags":0.95,"expPlanesKilled":0.01},"4283414224":{"expDamage":49607.57,"expFrags":0.95,"expPlanesKilled":14.8},"4281284592":{"expDamage":27190.76,"expFrags":0.79,"expPlanesKilled":0.01},"3555669712":{"expDamage":44472.51,"expFrags":1.02,"expPlanesKilled":1.45},"4267587280":{"expDamage":36638.37,"expFrags":1.43,"expPlanesKilled":0.01},"4284463088":{"expDamage":44282.2,"expFrags":0.8,"expPlanesKilled":16.78},"4288591856":{"expDamage":22777.69,"expFrags":0.68,"expPlanesKilled":2.75},"4277090288":{"expDamage":72559.28,"expFrags":0.92,"expPlanesKilled":6.02},"4287543280":{"expDamage":30122.78,"expFrags":0.87,"expPlanesKilled":2.84},"3769513264":{"expDamage":21617.8,"expFrags":0.73,"expPlanesKilled":0.19},"4290721776":{"expDamage":26461.37,"expFrags":0.75,"expPlanesKilled":0.45},"4255037136":{"expDamage":28131.73,"expFrags":0.55,"expPlanesKilled":1.72},"3555636944":{"expDamage":39360.63,"expFrags":1.0,"expPlanesKilled":2.26},"4179539760":{"expDamage":63527.57,"expFrags":0.99,"expPlanesKilled":2.5},"4185798096":{"expDamage":13712.59,"expFrags":0.58,"expPlanesKilled":0.07},"4287510224":{"expDamage":26909.03,"expFrags":0.7,"expPlanesKilled":0.35},"4291770064":{"expDamage":20597.6,"expFrags":0.55,"expPlanesKilled":0.49},"3552523984":{"expDamage":44472.51,"expFrags":1.02,"expPlanesKilled":1.45},"4281219056":{"expDamage":42846.63,"expFrags":0.85,"expPlanesKilled":0.83},"4281317360":{"expDamage":79876.74,"expFrags":1.28,"expPlanesKilled":21.92},"4187895248":{"expDamage":8276.49,"expFrags":0.58,"expPlanesKilled":0.01},"3762206160":{"expDamage":30798.65,"expFrags":0.71,"expPlanesKilled":3.47},"4292851696":{"expDamage":22020.05,"expFrags":0.44,"expPlanesKilled":12.71},"4184749520":{"expDamage":17748.09,"expFrags":0.7,"expPlanesKilled":0.17},"4282300112":{"expDamage":41896.23,"expFrags":0.74,"expPlanesKilled":2.61},"4289640432":{"expDamage":26846.88,"expFrags":0.82,"expPlanesKilled":1.04},"4182685488":{"expDamage":26828.93,"expFrags":0.66,"expPlanesKilled":1.05},"4290688720":{"expDamage":19138.12,"expFrags":0.67,"expPlanesKilled":1.65},"4279219920":{"expDamage":73950.93,"expFrags":1.21,"expPlanesKilled":20.0},"4272895696":{"expDamage":54736.52,"expFrags":0.78,"expPlanesKilled":3.09},"4181603792":{"expDamage":24902.28,"expFrags":0.74,"expPlanesKilled":0.41},"4292785968":{"expDamage":13731.52,"expFrags":0.87,"expPlanesKilled":0.01},"4284364496":{"expDamage":36019.61,"expFrags":0.74,"expPlanesKilled":0.29},"4287575760":{"expDamage":44472.51,"expFrags":1.02,"expPlanesKilled":1.45},"4281251536":{"expDamage":23428.4,"expFrags":0.77,"expPlanesKilled":0.27}}');
			}else{
				ExpShips = response;
			}
		}
		function updateExpWTR(response){
			if(response == null){
				ExpWTR = jQ.parseJSON('{"coefficients":{"wins_weight":0.25,"damage_weight":0.5,"frags_weight":0.25,"capture_weight":0,"dropped_capture_weight":0,"ship_frags_importance_weight":10,"nominal_rating":1000},"expected":{"4184782640":{"ship_id":4184782640,"wins":0.51122170686722,"damage_dealt":29010.787109375,"frags":0.86259603500366,"capture_points":0.056900948286057,"dropped_capture_points":0.26138630509377,"planes_killed":0.27677825093269},"4185831216":{"ship_id":4185831216,"wins":0.47541585564613,"damage_dealt":14922.84765625,"frags":0.48721393942833,"capture_points":0.14922149479389,"dropped_capture_points":0.45502883195877,"planes_killed":0.034130837768316},"4293801680":{"ship_id":4293801680,"wins":0.54129618406296,"damage_dealt":14387.209960938,"frags":0.94595927000046,"capture_points":0.20993874967098,"dropped_capture_points":0.33782848715782,"planes_killed":0},"4286527472":{"ship_id":4286527472,"wins":0.47352772951126,"damage_dealt":41785.0234375,"frags":0.71913027763367,"capture_points":0.24547809362411,"dropped_capture_points":1.111743927002,"planes_killed":1.8227601051331},"4287542992":{"ship_id":4287542992,"wins":0.48403823375702,"damage_dealt":27601.314453125,"frags":0.69836097955704,"capture_points":0.47521156072617,"dropped_capture_points":2.2203254699707,"planes_killed":1.8591247797012},"4282300112":{"ship_id":4282300112,"wins":0.48449957370758,"damage_dealt":43626.83203125,"frags":0.70653027296066,"capture_points":0.36395245790482,"dropped_capture_points":2.004123210907,"planes_killed":2.5518198013306},"4293801424":{"ship_id":4293801424,"wins":0.52637714147568,"damage_dealt":27949.580078125,"frags":0.98774564266205,"capture_points":1.0703727006912,"dropped_capture_points":3.5271558761597,"planes_killed":0.35358089208603},"4272895696":{"ship_id":4272895696,"wins":0.47099414467812,"damage_dealt":53934.96875,"frags":0.68613040447235,"capture_points":0.19086363911629,"dropped_capture_points":1.1639505624771,"planes_killed":3.3138031959534},"4283414224":{"ship_id":4283414224,"wins":0.50300073623657,"damage_dealt":50991.78125,"frags":0.95326614379883,"capture_points":0.18270418047905,"dropped_capture_points":3.0336711406708,"planes_killed":16.524854660034},"4277090288":{"ship_id":4277090288,"wins":0.48147720098495,"damage_dealt":74156.7890625,"frags":0.85653483867645,"capture_points":0.31039190292358,"dropped_capture_points":1.6609683036804,"planes_killed":6.0576696395874},"4282300400":{"ship_id":4282300400,"wins":0.46984308958054,"damage_dealt":26931.9921875,"frags":0.59478557109833,"capture_points":0.42359617352486,"dropped_capture_points":2.0665423870087,"planes_killed":2.278792142868},"4268635856":{"ship_id":4268635856,"wins":0.62282401323318,"damage_dealt":31894.259765625,"frags":1.1750483512878,"capture_points":0,"dropped_capture_points":0.01160541549325,"planes_killed":0},"3762206160":{"ship_id":3762206160,"wins":0.5086122751236,"damage_dealt":33298.26953125,"frags":0.66069889068604,"capture_points":0.00036819276283495,"dropped_capture_points":0.00069036142667755,"planes_killed":3.9224264621735},"4287575760":{"ship_id":4287575760,"wins":0.48555356264114,"damage_dealt":32007.111328125,"frags":0.71060824394226,"capture_points":0.45665016770363,"dropped_capture_points":2.0577249526978,"planes_killed":1.2519056797028},"4292753392":{"ship_id":4292753392,"wins":0.46952879428864,"damage_dealt":9180.6474609375,"frags":0.5257385969162,"capture_points":0.42997989058495,"dropped_capture_points":2.7973961830139,"planes_killed":0.00033417690428905},"4291770064":{"ship_id":4291770064,"wins":0.46288484334946,"damage_dealt":20910.033203125,"frags":0.5083144903183,"capture_points":0.61298954486847,"dropped_capture_points":2.6720056533813,"planes_killed":0.63197612762451},"4281284304":{"ship_id":4281284304,"wins":0.50753146409988,"damage_dealt":57384.82421875,"frags":0.87232065200806,"capture_points":0.2268174290657,"dropped_capture_points":0.95550882816315,"planes_killed":3.1812968254089},"4292851408":{"ship_id":4292851408,"wins":0.5024933218956,"damage_dealt":37074.6953125,"frags":0.86261510848999,"capture_points":0.24805398285389,"dropped_capture_points":1.5132809877396,"planes_killed":6.6838707923889},"3555636944":{"ship_id":3555636944,"wins":0.57035320997238,"damage_dealt":41847.2421875,"frags":0.93676894903183,"capture_points":0,"dropped_capture_points":0,"planes_killed":2.0414590835571},"4264441840":{"ship_id":4264441840,"wins":0.45099514722824,"damage_dealt":16430.048828125,"frags":0.50936496257782,"capture_points":0.80799180269241,"dropped_capture_points":2.55335521698,"planes_killed":0.2141999900341},"4287510224":{"ship_id":4287510224,"wins":0.48084256052971,"damage_dealt":27977.234375,"frags":0.62787091732025,"capture_points":0.53109759092331,"dropped_capture_points":0.6688591837883,"planes_killed":0.4331274330616},"4290754544":{"ship_id":4290754544,"wins":0.46277719736099,"damage_dealt":23377.509765625,"frags":0.47787865996361,"capture_points":0.087762981653214,"dropped_capture_points":0.81610888242722,"planes_killed":11.153073310852},"4293834192":{"ship_id":4293834192,"wins":0.51636666059494,"damage_dealt":23010.470703125,"frags":0.88609629869461,"capture_points":0.8088019490242,"dropped_capture_points":5.3639154434204,"planes_killed":0.01646813005209},"4286527184":{"ship_id":4286527184,"wins":0.49807155132294,"damage_dealt":25288.439453125,"frags":0.71628284454346,"capture_points":0.5487242937088,"dropped_capture_points":2.6521990299225,"planes_killed":0.1598627269268},"4279154384":{"ship_id":4279154384,"wins":0.47584357857704,"damage_dealt":16365.153320312,"frags":0.61520731449127,"capture_points":0.60201609134674,"dropped_capture_points":4.4364356994629,"planes_killed":0.035235047340393},"4292785968":{"ship_id":4292785968,"wins":0.52037113904953,"damage_dealt":14674.416992188,"frags":0.84994626045227,"capture_points":0.18533332645893,"dropped_capture_points":0.63516169786453,"planes_killed":6.2376857385971e-5},"4280170192":{"ship_id":4280170192,"wins":0.58620691299438,"damage_dealt":15345.862304688,"frags":0.87931036949158,"capture_points":1.3678160905838,"dropped_capture_points":4.9195404052734,"planes_killed":0},"4279219920":{"ship_id":4279219920,"wins":0.49208694696426,"damage_dealt":79023.796875,"frags":1.237321972847,"capture_points":0.16529600322247,"dropped_capture_points":3.9133138656616,"planes_killed":22.65545463562},"4182685488":{"ship_id":4182685488,"wins":0.493243932724,"damage_dealt":27271.353515625,"frags":0.63743996620178,"capture_points":0.0060345153324306,"dropped_capture_points":0.014768753200769,"planes_killed":1.0250433683395},"4255037136":{"ship_id":4255037136,"wins":0.46467468142509,"damage_dealt":31155.099609375,"frags":0.57337117195129,"capture_points":0.34689751267433,"dropped_capture_points":2.0904424190521,"planes_killed":1.6390644311905},"4181636912":{"ship_id":4181636912,"wins":0.50579386949539,"damage_dealt":36072.90234375,"frags":0.68282055854797,"capture_points":0.012051488272846,"dropped_capture_points":0.051826748996973,"planes_killed":1.4525104761124},"4183700944":{"ship_id":4183700944,"wins":0.51602953672409,"damage_dealt":20301.447265625,"frags":0.71119356155396,"capture_points":0.01843904517591,"dropped_capture_points":0.013811847195029,"planes_killed":0.19603800773621},"4180588336":{"ship_id":4180588336,"wins":0.50769966840744,"damage_dealt":45930.7578125,"frags":0.75055509805679,"capture_points":0.030917989090085,"dropped_capture_points":0.14783334732056,"planes_killed":1.4655365943909},"4273911792":{"ship_id":4273911792,"wins":0.48156395554543,"damage_dealt":53287.31640625,"frags":0.81086224317551,"capture_points":0.38113671541214,"dropped_capture_points":2.2883222103119,"planes_killed":4.8028225898743},"4293834544":{"ship_id":4293834544,"wins":0.49174481630325,"damage_dealt":8885.615234375,"frags":0.70324790477753,"capture_points":0.080272294580936,"dropped_capture_points":0.21217910945415,"planes_killed":0.0036505574826151},"4293867504":{"ship_id":4293867504,"wins":0.47524207830429,"damage_dealt":17662.810546875,"frags":0.59320122003555,"capture_points":0.88193660974503,"dropped_capture_points":4.5931978225708,"planes_killed":0.019640145823359},"3769513264":{"ship_id":3769513264,"wins":0.52074003219604,"damage_dealt":23681.662109375,"frags":0.72903996706009,"capture_points":0.0071594417095184,"dropped_capture_points":0.013677516952157,"planes_killed":0.14705194532871},"4277024464":{"ship_id":4277024464,"wins":0.54966104030609,"damage_dealt":29922.33984375,"frags":1.0423370599747,"capture_points":0.14278312027454,"dropped_capture_points":0.093596316874027,"planes_killed":0.0023008289281279},"4290688720":{"ship_id":4290688720,"wins":0.4907740354538,"damage_dealt":21443.482421875,"frags":0.73005795478821,"capture_points":0.75107079744339,"dropped_capture_points":4.7987771034241,"planes_killed":2.154886007309},"4292753104":{"ship_id":4292753104,"wins":0.51494568586349,"damage_dealt":13503.28125,"frags":0.76045894622803,"capture_points":0.69377225637436,"dropped_capture_points":2.3867702484131,"planes_killed":0.0025299144908786},"4288657392":{"ship_id":4288657392,"wins":0.47228679060936,"damage_dealt":32027.70703125,"frags":0.5947328209877,"capture_points":0.21192662417889,"dropped_capture_points":1.8065805435181,"planes_killed":12.462384223938},"4292786160":{"ship_id":4292786160,"wins":0.46855843067169,"damage_dealt":9711.1904296875,"frags":0.41686633229256,"capture_points":0.61553996801376,"dropped_capture_points":5.966287612915,"planes_killed":0.0002320865605725},"4284430032":{"ship_id":4284430032,"wins":0.49145954847336,"damage_dealt":47097.96875,"frags":0.78241211175919,"capture_points":0.24562656879425,"dropped_capture_points":1.0172488689423,"planes_killed":1.2799953222275},"4286494416":{"ship_id":4286494416,"wins":0.50471705198288,"damage_dealt":35171.00390625,"frags":0.77655363082886,"capture_points":0.38664507865906,"dropped_capture_points":1.8615604639053,"planes_killed":2.0774765014648},"4179539760":{"ship_id":4179539760,"wins":0.49690985679626,"damage_dealt":60654.109375,"frags":0.84266096353531,"capture_points":0.0163519307971,"dropped_capture_points":0.14261803030968,"planes_killed":2.3118026256561},"4291704528":{"ship_id":4291704528,"wins":0.49236604571342,"damage_dealt":21284.619140625,"frags":0.79264974594116,"capture_points":0.82577568292618,"dropped_capture_points":1.28300344944,"planes_killed":0.081362120807171},"4276041712":{"ship_id":4276041712,"wins":0.49023562669754,"damage_dealt":60521.5,"frags":0.81944251060486,"capture_points":0.38277718424797,"dropped_capture_points":1.6925464868546,"planes_killed":5.2104725837708},"4279220208":{"ship_id":4279220208,"wins":0.56659376621246,"damage_dealt":122938.859375,"frags":1.7224745750427,"capture_points":0.19586570560932,"dropped_capture_points":2.1930289268494,"planes_killed":22.241962432861},"4289607664":{"ship_id":4289607664,"wins":0.47798517346382,"damage_dealt":17561.3046875,"frags":0.62715280056,"capture_points":0.44180572032928,"dropped_capture_points":1.1330152750015,"planes_killed":0.42930275201797},"4274927600":{"ship_id":4274927600,"wins":0.48380756378174,"damage_dealt":17806.7890625,"frags":0.71286630630493,"capture_points":0.55572903156281,"dropped_capture_points":1.749541759491,"planes_killed":0.015720445662737},"4288624336":{"ship_id":4288624336,"wins":0.49999806284904,"damage_dealt":43621.703125,"frags":0.88851475715637,"capture_points":0.3166811466217,"dropped_capture_points":1.4821939468384,"planes_killed":0.9398313164711},"4182652368":{"ship_id":4182652368,"wins":0.53374767303467,"damage_dealt":37103.78515625,"frags":0.84917390346527,"capture_points":0.085267201066017,"dropped_capture_points":0.20361058413982,"planes_killed":0.72545832395554},"4180555216":{"ship_id":4180555216,"wins":0.55949187278748,"damage_dealt":31422.1015625,"frags":0.96666514873505,"capture_points":0.018552985042334,"dropped_capture_points":0.027523621916771,"planes_killed":0.41133809089661},"4276041424":{"ship_id":4276041424,"wins":0.52375638484955,"damage_dealt":86309.8828125,"frags":0.86914587020874,"capture_points":0.2110887914896,"dropped_capture_points":1.1167989969254,"planes_killed":5.0761866569519},"3768497968":{"ship_id":3768497968,"wins":0.53151720762253,"damage_dealt":16703.125,"frags":0.89476084709167,"capture_points":0,"dropped_capture_points":0,"planes_killed":7.6689400884788e-5},"4292818896":{"ship_id":4292818896,"wins":0.49132812023163,"damage_dealt":38581.73046875,"frags":0.80932784080505,"capture_points":0.51086294651031,"dropped_capture_points":2.1963210105896,"planes_killed":1.7055611610413},"4291737584":{"ship_id":4291737584,"wins":0.51265150308609,"damage_dealt":13162.381835938,"frags":0.74851536750793,"capture_points":1.0853387117386,"dropped_capture_points":7.9745111465454,"planes_killed":0.00054676347645},"4256085712":{"ship_id":4256085712,"wins":0.51617801189423,"damage_dealt":11983.815429688,"frags":0.88629257678986,"capture_points":0.96862757205963,"dropped_capture_points":7.711003780365,"planes_killed":0.009332112967968},"4288559088":{"ship_id":4288559088,"wins":0.48756995797157,"damage_dealt":19911.185546875,"frags":0.66164863109589,"capture_points":0.38923868536949,"dropped_capture_points":0.78198432922363,"planes_killed":0.32376030087471},"4284397008":{"ship_id":4284397008,"wins":0.59825325012207,"damage_dealt":23505.876953125,"frags":1.1091703176498,"capture_points":1.22707426548,"dropped_capture_points":13.84716129303,"planes_killed":0},"3555669712":{"ship_id":3555669712,"wins":0.57258361577988,"damage_dealt":48332.26953125,"frags":1.0680457353592,"capture_points":0,"dropped_capture_points":0,"planes_killed":1.500967502594},"4292851696":{"ship_id":4292851696,"wins":0.46156921982765,"damage_dealt":21037.59375,"frags":0.40915170311928,"capture_points":0.076929539442062,"dropped_capture_points":1.5164833068848,"planes_killed":13.995998382568},"4184749520":{"ship_id":4184749520,"wins":0.50903660058975,"damage_dealt":20120.134765625,"frags":0.75349593162537,"capture_points":0.06822320073843,"dropped_capture_points":0.13170683383942,"planes_killed":0.17915205657482},"4282365936":{"ship_id":4282365936,"wins":0.47331801056862,"damage_dealt":46967.5078125,"frags":0.71830648183823,"capture_points":0.10427645593882,"dropped_capture_points":1.2506062984467,"planes_killed":27.577516555786},"4285445840":{"ship_id":4285445840,"wins":0.51204824447632,"damage_dealt":42242.2734375,"frags":0.84760069847107,"capture_points":0.37564527988434,"dropped_capture_points":2.0849678516388,"planes_killed":2.0687417984009},"4282365648":{"ship_id":4282365648,"wins":0.48905548453331,"damage_dealt":59900.9921875,"frags":1.0054889917374,"capture_points":0.17719039320946,"dropped_capture_points":3.4725511074066,"planes_killed":15.908694267273},"4293867312":{"ship_id":4293867312,"wins":0.33333334326744,"damage_dealt":47393.109375,"frags":1.111111164093,"capture_points":0,"dropped_capture_points":13.777777671814,"planes_killed":0.55555558204651},"4287543280":{"ship_id":4287543280,"wins":0.49591055512428,"damage_dealt":30844.71484375,"frags":0.84339910745621,"capture_points":0.5514662861824,"dropped_capture_points":3.1818056106567,"planes_killed":2.906042098999},"4289607376":{"ship_id":4289607376,"wins":0.4749498963356,"damage_dealt":20338.333984375,"frags":0.57600152492523,"capture_points":0.81533247232437,"dropped_capture_points":0.70073652267456,"planes_killed":0.56654810905457},"4283381456":{"ship_id":4283381456,"wins":0.51551496982574,"damage_dealt":12964.6875,"frags":0.71449911594391,"capture_points":0.98751544952393,"dropped_capture_points":3.3637158870697,"planes_killed":0},"4290655952":{"ship_id":4290655952,"wins":0.49981626868248,"damage_dealt":26826.720703125,"frags":0.90305662155151,"capture_points":0.95347154140472,"dropped_capture_points":1.1497843265533,"planes_killed":0.11331869661808},"4186846672":{"ship_id":4186846672,"wins":0.49038279056549,"damage_dealt":12225.55078125,"frags":0.62605762481689,"capture_points":0.1398436576128,"dropped_capture_points":0.26155284047127,"planes_killed":0.0012192274443805},"4280203248":{"ship_id":4280203248,"wins":0.4759079515934,"damage_dealt":31191.642578125,"frags":0.59089314937592,"capture_points":0.46612244844437,"dropped_capture_points":2.1409313678741,"planes_killed":3.3640120029449},"4247697392":{"ship_id":4247697392,"wins":0.62326872348785,"damage_dealt":42531.140625,"frags":1.3490304946899,"capture_points":0.88365650177002,"dropped_capture_points":6.9501385688782,"planes_killed":0.25484764575958},"4188976592":{"ship_id":4188976592,"wins":0.51687461137772,"damage_dealt":10430.244140625,"frags":0.81734579801559,"capture_points":0.095328889787197,"dropped_capture_points":0.2455480247736,"planes_killed":0.00084729748778045},"4269717488":{"ship_id":4269717488,"wins":0.4744436442852,"damage_dealt":19657.0234375,"frags":0.58517348766327,"capture_points":0.51520311832428,"dropped_capture_points":4.1102132797241,"planes_killed":0.15040400624275},"4285511376":{"ship_id":4285511376,"wins":0.50200510025024,"damage_dealt":42934.97265625,"frags":0.8718563914299,"capture_points":0.1599772721529,"dropped_capture_points":3.0300908088684,"planes_killed":9.7274026870728},"4290721776":{"ship_id":4290721776,"wins":0.48683840036392,"damage_dealt":27189.314453125,"frags":0.71860271692276,"capture_points":0.74586910009384,"dropped_capture_points":3.6231265068054,"planes_killed":0.51523298025131},"4277122768":{"ship_id":4277122768,"wins":0.52040976285934,"damage_dealt":97933.34375,"frags":1.4317626953125,"capture_points":0.26329359412193,"dropped_capture_points":4.6207022666931,"planes_killed":25.154556274414},"4280202960":{"ship_id":4280202960,"wins":0.40000000596046,"damage_dealt":25757.029296875,"frags":0.54285717010498,"capture_points":0,"dropped_capture_points":2.2571427822113,"planes_killed":0.65714287757874},"4292785616":{"ship_id":4292785616,"wins":0.53021615743637,"damage_dealt":16832.44921875,"frags":0.82588958740234,"capture_points":0.47461375594139,"dropped_capture_points":2.9290931224823,"planes_killed":0},"4284463088":{"ship_id":4284463088,"wins":0.46770951151848,"damage_dealt":42765.28515625,"frags":0.7275470495224,"capture_points":0.092292763292789,"dropped_capture_points":1.7809463739395,"planes_killed":17.947797775269},"4282267344":{"ship_id":4282267344,"wins":0.49789351224899,"damage_dealt":54761.546875,"frags":0.82376223802567,"capture_points":0.64457195997238,"dropped_capture_points":0.71425980329514,"planes_killed":0.40821501612663},"4277057520":{"ship_id":4277057520,"wins":0.4621470272541,"damage_dealt":33403.171875,"frags":0.58620339632034,"capture_points":0.51035594940186,"dropped_capture_points":2.5576140880585,"planes_killed":4.8932147026062},"4293867216":{"ship_id":4293867216,"wins":0.48672068119049,"damage_dealt":17365.701171875,"frags":0.64515465497971,"capture_points":0.99146997928619,"dropped_capture_points":4.0451827049255,"planes_killed":0.055872082710266},"4282333168":{"ship_id":4282333168,"wins":0.48037457466125,"damage_dealt":51151.23828125,"frags":0.77452915906906,"capture_points":0.31005719304085,"dropped_capture_points":1.2688862085342,"planes_killed":4.4019017219543},"4286461936":{"ship_id":4286461936,"wins":0.50703746080399,"damage_dealt":26263.830078125,"frags":0.75541549921036,"capture_points":0.41792750358582,"dropped_capture_points":0.79157638549805,"planes_killed":0.30637675523758},"4186879792":{"ship_id":4186879792,"wins":0.4904902279377,"damage_dealt":15572.170898438,"frags":0.65610313415527,"capture_points":0.23056210577488,"dropped_capture_points":0.70989382266998,"planes_killed":0},"4267620048":{"ship_id":4267620048,"wins":0.56769609451294,"damage_dealt":32817.11328125,"frags":1.1899466514587,"capture_points":0.97768026590347,"dropped_capture_points":5.0320897102356,"planes_killed":1.1803220510483},"4281284592":{"ship_id":4281284592,"wins":0.51938199996948,"damage_dealt":32038.33203125,"frags":0.90314656496048,"capture_points":0.94859611988068,"dropped_capture_points":4.8209056854248,"planes_killed":0},"4289640144":{"ship_id":4289640144,"wins":0.46131098270416,"damage_dealt":20093.380859375,"frags":0.55474579334259,"capture_points":0.47441563010216,"dropped_capture_points":2.3597092628479,"planes_killed":0.37398678064346},"4280170480":{"ship_id":4280170480,"wins":0.47194939851761,"damage_dealt":18749.478515625,"frags":0.69936400651932,"capture_points":0.4646797478199,"dropped_capture_points":1.4791560173035,"planes_killed":0.40327444672585},"4284364496":{"ship_id":4284364496,"wins":0.48605459928513,"damage_dealt":38620.859375,"frags":0.68300211429596,"capture_points":0.68985867500305,"dropped_capture_points":0.77824079990387,"planes_killed":0.34508037567139},"4185798096":{"ship_id":4185798096,"wins":0.49030685424805,"damage_dealt":15445.366210938,"frags":0.62646013498306,"capture_points":0.10897959023714,"dropped_capture_points":0.21636413037777,"planes_killed":0.067896746098995},"4281219056":{"ship_id":4281219056,"wins":0.50688892602921,"damage_dealt":46298.609375,"frags":0.84025180339813,"capture_points":0.57574254274368,"dropped_capture_points":0.9680113196373,"planes_killed":0.97797679901123},"4187895248":{"ship_id":4187895248,"wins":0.49087762832642,"damage_dealt":9034.1240234375,"frags":0.60579288005829,"capture_points":0.080421641469002,"dropped_capture_points":0.20948664844036,"planes_killed":0.00037944468203932},"4181603792":{"ship_id":4181603792,"wins":0.52121806144714,"damage_dealt":28608.521484375,"frags":0.7731249332428,"capture_points":0.016832668334246,"dropped_capture_points":0.044563118368387,"planes_killed":0.47679218649864},"4258182864":{"ship_id":4258182864,"wins":0.49692192673683,"damage_dealt":12993.342773438,"frags":0.56344938278198,"capture_points":0.67929142713547,"dropped_capture_points":5.9911632537842,"planes_killed":7.1776434197091e-5},"4272830448":{"ship_id":4272830448,"wins":0.51577967405319,"damage_dealt":36368.69140625,"frags":0.8646759390831,"capture_points":0.58087933063507,"dropped_capture_points":1.0310314893723,"planes_killed":0.67544251680374},"4288657104":{"ship_id":4288657104,"wins":0.51160848140717,"damage_dealt":42630.8203125,"frags":0.9451676607132,"capture_points":0.21757699549198,"dropped_capture_points":2.4586248397827,"planes_killed":7.4511098861694},"4288591856":{"ship_id":4288591856,"wins":0.45406526327133,"damage_dealt":22931.974609375,"frags":0.63379961252213,"capture_points":0.51421225070953,"dropped_capture_points":2.7220969200134,"planes_killed":2.6097500324249},"4288558800":{"ship_id":4288558800,"wins":0.4821991622448,"damage_dealt":21729.326171875,"frags":0.59099590778351,"capture_points":0.62569862604141,"dropped_capture_points":0.83074468374252,"planes_killed":0.48786357045174},"4266538992":{"ship_id":4266538992,"wins":0.46971708536148,"damage_dealt":11930.59375,"frags":0.53524053096771,"capture_points":0.4190699160099,"dropped_capture_points":1.9530692100525,"planes_killed":0.0099149942398071},"4183734064":{"ship_id":4183734064,"wins":0.49475267529488,"damage_dealt":27303.80078125,"frags":0.74871665239334,"capture_points":0.0082841766998172,"dropped_capture_points":0.018642133101821,"planes_killed":0.65558570623398},"4281317360":{"ship_id":4281317360,"wins":0.5076670050621,"damage_dealt":91473.1015625,"frags":1.3835434913635,"capture_points":0.12333177775145,"dropped_capture_points":1.6780213117599,"planes_killed":20.631332397461},"4292818736":{"ship_id":4292818736,"wins":0.49458310008049,"damage_dealt":49375.0546875,"frags":0.72911477088928,"capture_points":0.13192163407803,"dropped_capture_points":0.52681165933609,"planes_killed":1.8101972341537},"4289640432":{"ship_id":4289640432,"wins":0.4901687502861,"damage_dealt":27815.310546875,"frags":0.82094019651413,"capture_points":0.57474219799042,"dropped_capture_points":4.1309590339661,"planes_killed":1.1518355607986},"4259264496":{"ship_id":4259264496,"wins":0.48876741528511,"damage_dealt":39468.45703125,"frags":0.83219093084335,"capture_points":0.3554590344429,"dropped_capture_points":1.6597150564194,"planes_killed":1.0688138008118},"4288624624":{"ship_id":4288624624,"wins":0.47002255916595,"damage_dealt":29196.048828125,"frags":0.66977375745773,"capture_points":0.48569974303246,"dropped_capture_points":2.3473603725433,"planes_killed":0.72276109457016},"4281251536":{"ship_id":4281251536,"wins":0.49260646104813,"damage_dealt":22843.80859375,"frags":0.72023290395737,"capture_points":0.56403440237045,"dropped_capture_points":3.9352245330811,"planes_killed":0.29223653674126},"4269684432":{"ship_id":4269684432,"wins":0.48727235198021,"damage_dealt":14679.861328125,"frags":0.64838409423828,"capture_points":0.66714358329773,"dropped_capture_points":1.7604387998581,"planes_killed":0.018125910311937},"4290689008":{"ship_id":4290689008,"wins":0.50464558601379,"damage_dealt":20325.57421875,"frags":0.75058627128601,"capture_points":0.875792324543,"dropped_capture_points":6.6569604873657,"planes_killed":0.004384093452245},"3554621136":{"ship_id":3554621136,"wins":1,"damage_dealt":15548,"frags":1,"capture_points":0,"dropped_capture_points":0,"planes_killed":4},"4179506640":{"ship_id":4179506640,"wins":0.53261315822601,"damage_dealt":48924.546875,"frags":0.83661127090454,"capture_points":0.080373391509056,"dropped_capture_points":0.17830094695091,"planes_killed":0.98696649074554},"4259231440":{"ship_id":4259231440,"wins":0.51405346393585,"damage_dealt":69867.1171875,"frags":0.92596477270126,"capture_points":0.27620103955269,"dropped_capture_points":1.8662161827087,"planes_killed":2.713700056076},"4248745968":{"ship_id":4248745968,"wins":0.56045067310333,"damage_dealt":36988.48046875,"frags":1.121866106987,"capture_points":0.34049141407013,"dropped_capture_points":0.82540702819824,"planes_killed":0.18764664232731},"4293866960":{"ship_id":4293866960,"wins":0.59622633457184,"damage_dealt":40642.046875,"frags":1.1839680671692,"capture_points":0.18553453683853,"dropped_capture_points":0.39293265342712,"planes_killed":0.031857497990131},"4291737040":{"ship_id":4291737040,"wins":0.5352366566658,"damage_dealt":34966.93359375,"frags":1.0280692577362,"capture_points":0.53413772583008,"dropped_capture_points":3.632979631424,"planes_killed":1.3394631147385},"4293834736":{"ship_id":4293834736,"wins":0.47978937625885,"damage_dealt":9667.21484375,"frags":0.63977980613708,"capture_points":0.6472572684288,"dropped_capture_points":5.7667627334595,"planes_killed":0.0055624567903578},"4267587280":{"ship_id":4267587280,"wins":0.54329973459244,"damage_dealt":25633.357421875,"frags":0.90539002418518,"capture_points":0,"dropped_capture_points":0,"planes_killed":0.0030576402787119}}}');
			}else{
				ExpWTR = response;
			}
		}
		function errorExpShips(url){}
		function errorExpWTR(url){}
		function doneAccountInfo(url, response){
			if(response.status && response.status == 'error'){
				errorAccountInfo(url);
				return;
			}
			
			var vars = getUrlVars(url);
			var account_id = vars['account_id'];
			var index = vars['index'];
			var type = vars['type'];
			
			MembersArray[index]['info'] = response['data'][account_id];
			
			getJson(WOWSAPI+'ships/stats/?application_id='+application_id+'&extra=pve,pvp_solo,pvp_div2,pvp_div3&account_id='+account_id+'&index='+index+'&type='+type, doneShipsStats, errorShipsStats);
		}
		function errorAccountInfo(url){
			var vars = getUrlVars(url);
			var account_id = vars['account_id'];
			var index = vars['index'];
			var type = vars['type'];
			
			console.log('Error AccountInfo '+account_id);
			
			if(type == 'profile'){
				onShowMessage(
					localizationText['Box'],
					localizationText['ErrorAPI'],
					onCloseMessage,
					localizationText['Ok'],
					false
				);
			}else if(type == 'clan'){
				getJson(WOWSAPI+'ships/stats/?application_id='+application_id+'&extra=pve,pvp_solo,pvp_div2,pvp_div3&account_id='+account_id+'&index='+index+'&type='+type, doneShipsStats, errorShipsStats);
			}
		}
		function doneShipsStats(url, response){
			if(response.status && response.status == 'error'){
				errorShipsStats(url);
				return;
			}
			
			var vars = getUrlVars(url);
			var account_id = vars['account_id'];
			var index = vars['index'];
			var type = vars['type'];
			
			MembersArray[index]['ships'] = response['data'][account_id];
			
			getJson(WOWSAPI+'account/achievements/?application_id='+application_id+'&account_id='+account_id+'&index='+index+'&type='+type, doneAchievements, errorAchievements);
		}
		function errorShipsStats(url){
			var vars = getUrlVars(url);
			var account_id = vars['account_id'];
			var index = vars['index'];
			var type = vars['type'];
			
			console.log('Error ShipsStats '+account_id);
			
			if(type == 'profile'){
				onShowMessage(
					localizationText['Box'],
					localizationText['ErrorAPI'],
					onCloseMessage,
					localizationText['Ok'],
					false
				);
			}else if(type == 'clan'){
				getJson(WOWSAPI+'account/achievements/?application_id='+application_id+'&account_id='+account_id+'&index='+index+'&type='+type, doneAchievements, errorAchievements);
			}
		}
		function doneAchievements(url, response){
			if(response.status && response.status == 'error'){
				errorAchievements(url);
				return;
			}
			
			var vars = getUrlVars(url);
			var account_id = vars['account_id'];
			var index = vars['index'];
			var type = vars['type'];
			
			MembersArray[index]['achievements'] = response['data'][account_id];
			
			if(type == 'profile'){
				viewMainPageProfile();
			}else if(type == 'clan'){
				loadMemberCount++;
				viewMainPageClan();
			}
		}
		function errorAchievements(url){
			var vars = getUrlVars(url);
			var account_id = vars['account_id'];
			var index = vars['index'];
			var type = vars['type'];
			
			console.log('Error Achievements '+account_id);
			
			if(type == 'profile'){
				onShowMessage(
					localizationText['Box'],
					localizationText['ErrorAPI'],
					onCloseMessage,
					localizationText['Ok'],
					false
				);
			}else if(type == 'clan'){
				loadMemberCount++;
				viewMainPageClan();
			}
		}
		function calcStat(index){
			if(MembersArray[index]['info'] == null || MembersArray[index]['ships'] == null){
				MembersArray[index]['info'] = [];
				MembersArray[index]['info']['last_battle_time'] = 0;
				MembersArray[index]['info']['logout_at'] = 0;
				MembersArray[index]['info']['ships_x_level'] = 0;
				
				MembersArray[index]['info']['statistics'] = [];
				MembersArray[index]['info']['statistics']['pvp'] = [];
				MembersArray[index]['info']['statistics']['pvp']['battles'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['wins'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['losses'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['draws'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['survived_battles'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['survived_wins'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['kill_dead'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['xp'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['damage_dealt'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['frags'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['planes_killed'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['capture_points'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['dropped_capture_points'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['avg_xp'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['avg_damage_dealt'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['avg_frags'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['avg_planes_killed'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['avg_capture_points'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['avg_dropped_capture_points'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['max_xp'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['max_damage_dealt'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['max_frags_battle'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['max_planes_killed'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['wins_percents'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['survived_battles_percents'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['wr'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['wtr'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['avg_battles_level'] = 0;
				MembersArray[index]['info']['statistics']['pvp']['max_ship_level'] = 0;
				
				return false;
			}
			
			for(var t = 0; t < typeStat.length; t++){
				var type = typeStat[t];
				
				if(type == 'pvp_div' && MembersArray[index]['info']['statistics']['pvp_div'] == undefined){
					MembersArray[index]['info']['statistics']['pvp_div'] = [];
					
					if(MembersArray[index]['info']['statistics']['pvp_div2']['max_xp'] > MembersArray[index]['info']['statistics']['pvp_div3']['max_xp']){
						MembersArray[index]['info']['statistics']['pvp_div']['max_xp'] = MembersArray[index]['info']['statistics']['pvp_div2']['max_xp'];
						MembersArray[index]['info']['statistics']['pvp_div']['max_xp_ship_id'] = MembersArray[index]['info']['statistics']['pvp_div2']['max_xp_ship_id'];
					}else{
						MembersArray[index]['info']['statistics']['pvp_div']['max_xp'] = MembersArray[index]['info']['statistics']['pvp_div3']['max_xp'];
						MembersArray[index]['info']['statistics']['pvp_div']['max_xp_ship_id'] = MembersArray[index]['info']['statistics']['pvp_div3']['max_xp_ship_id'];
					}
					
					if(MembersArray[index]['info']['statistics']['pvp_div2']['max_frags_battle'] > MembersArray[index]['info']['statistics']['pvp_div3']['max_frags_battle']){
						MembersArray[index]['info']['statistics']['pvp_div']['max_frags_battle'] = MembersArray[index]['info']['statistics']['pvp_div2']['max_frags_battle'];
						MembersArray[index]['info']['statistics']['pvp_div']['max_frags_ship_id'] = MembersArray[index]['info']['statistics']['pvp_div2']['max_frags_ship_id'];
					}else{
						MembersArray[index]['info']['statistics']['pvp_div']['max_frags_battle'] = MembersArray[index]['info']['statistics']['pvp_div3']['max_frags_battle'];
						MembersArray[index]['info']['statistics']['pvp_div']['max_frags_ship_id'] = MembersArray[index]['info']['statistics']['pvp_div3']['max_frags_ship_id'];
					}
					
					if(MembersArray[index]['info']['statistics']['pvp_div2']['max_planes_killed'] > MembersArray[index]['info']['statistics']['pvp_div3']['max_planes_killed']){
						MembersArray[index]['info']['statistics']['pvp_div']['max_planes_killed'] = MembersArray[index]['info']['statistics']['pvp_div2']['max_planes_killed'];
						MembersArray[index]['info']['statistics']['pvp_div']['max_planes_killed_ship_id'] = MembersArray[index]['info']['statistics']['pvp_div2']['max_planes_killed_ship_id'];
					}else{
						MembersArray[index]['info']['statistics']['pvp_div']['max_planes_killed'] = MembersArray[index]['info']['statistics']['pvp_div3']['max_planes_killed'];
						MembersArray[index]['info']['statistics']['pvp_div']['max_planes_killed_ship_id'] = MembersArray[index]['info']['statistics']['pvp_div3']['max_planes_killed_ship_id'];
					}
					
					if(MembersArray[index]['info']['statistics']['pvp_div2']['max_damage_dealt'] > MembersArray[index]['info']['statistics']['pvp_div3']['max_damage_dealt']){
						MembersArray[index]['info']['statistics']['pvp_div']['max_damage_dealt'] = MembersArray[index]['info']['statistics']['pvp_div2']['max_damage_dealt'];
						MembersArray[index]['info']['statistics']['pvp_div']['max_damage_dealt_ship_id'] = MembersArray[index]['info']['statistics']['pvp_div2']['max_damage_dealt_ship_id'];
					}else{
						MembersArray[index]['info']['statistics']['pvp_div']['max_damage_dealt'] = MembersArray[index]['info']['statistics']['pvp_div3']['max_damage_dealt'];
						MembersArray[index]['info']['statistics']['pvp_div']['max_damage_dealt_ship_id'] = MembersArray[index]['info']['statistics']['pvp_div3']['max_damage_dealt_ship_id'];
					}
					
					MembersArray[index]['info']['statistics']['pvp_div']['xp'] = MembersArray[index]['info']['statistics']['pvp_div2']['xp'] + MembersArray[index]['info']['statistics']['pvp_div3']['xp'];
					MembersArray[index]['info']['statistics']['pvp_div']['survived_battles'] = MembersArray[index]['info']['statistics']['pvp_div2']['survived_battles'] + MembersArray[index]['info']['statistics']['pvp_div3']['survived_battles'];
					MembersArray[index]['info']['statistics']['pvp_div']['dropped_capture_points'] = MembersArray[index]['info']['statistics']['pvp_div2']['dropped_capture_points'] + MembersArray[index]['info']['statistics']['pvp_div3']['dropped_capture_points'];
					MembersArray[index]['info']['statistics']['pvp_div']['draws'] = MembersArray[index]['info']['statistics']['pvp_div2']['draws'] + MembersArray[index]['info']['statistics']['pvp_div3']['draws'];
					MembersArray[index]['info']['statistics']['pvp_div']['wins'] = MembersArray[index]['info']['statistics']['pvp_div2']['wins'] + MembersArray[index]['info']['statistics']['pvp_div3']['wins'];
					MembersArray[index]['info']['statistics']['pvp_div']['damage_dealt'] = MembersArray[index]['info']['statistics']['pvp_div2']['damage_dealt'] + MembersArray[index]['info']['statistics']['pvp_div3']['damage_dealt'];
					MembersArray[index]['info']['statistics']['pvp_div']['losses'] = MembersArray[index]['info']['statistics']['pvp_div2']['losses'] + MembersArray[index]['info']['statistics']['pvp_div3']['losses'];
					MembersArray[index]['info']['statistics']['pvp_div']['frags'] = MembersArray[index]['info']['statistics']['pvp_div2']['frags'] + MembersArray[index]['info']['statistics']['pvp_div3']['frags'];
					MembersArray[index]['info']['statistics']['pvp_div']['capture_points'] = MembersArray[index]['info']['statistics']['pvp_div2']['capture_points'] + MembersArray[index]['info']['statistics']['pvp_div3']['capture_points'];
					MembersArray[index]['info']['statistics']['pvp_div']['survived_wins'] = MembersArray[index]['info']['statistics']['pvp_div2']['survived_wins'] + MembersArray[index]['info']['statistics']['pvp_div3']['survived_wins'];
					MembersArray[index]['info']['statistics']['pvp_div']['battles'] = MembersArray[index]['info']['statistics']['pvp_div2']['battles'] + MembersArray[index]['info']['statistics']['pvp_div3']['battles'];
					MembersArray[index]['info']['statistics']['pvp_div']['planes_killed'] = MembersArray[index]['info']['statistics']['pvp_div2']['planes_killed'] + MembersArray[index]['info']['statistics']['pvp_div3']['planes_killed'];
				}
				
				var Statistics = MembersArray[index]['info']['statistics'][type];
				
				var timestamp = Math.round(+new Date()/1000);
				var created_at = MembersArray[index]['info']['created_at'];
				var days = (timestamp - created_at)/60/60/24;
				var battles_days = Statistics['battles'] / days;
				MembersArray[index]['info']['statistics'][type]['battles_days'] = battles_days;
				
				MembersArray[index]['info']['statistics'][type]['avg_xp'] = Statistics['xp'] / Statistics['battles'];
				if(isNaN(MembersArray[index]['info']['statistics'][type]['avg_xp'])){MembersArray[index]['info']['statistics'][type]['avg_xp'] = 0;}
				
				MembersArray[index]['info']['statistics'][type]['avg_damage_dealt'] = Statistics['damage_dealt'] / Statistics['battles'];
				if(isNaN(MembersArray[index]['info']['statistics'][type]['avg_damage_dealt'])){MembersArray[index]['info']['statistics'][type]['avg_damage_dealt'] = 0;}
				
				MembersArray[index]['info']['statistics'][type]['avg_frags'] = Statistics['frags'] / Statistics['battles'];
				if(isNaN(MembersArray[index]['info']['statistics'][type]['avg_frags'])){MembersArray[index]['info']['statistics'][type]['avg_frags'] = 0;}
				
				MembersArray[index]['info']['statistics'][type]['avg_planes_killed'] = Statistics['planes_killed'] / Statistics['battles'];
				if(isNaN(MembersArray[index]['info']['statistics'][type]['avg_planes_killed'])){MembersArray[index]['info']['statistics'][type]['avg_planes_killed'] = 0;}
				
				MembersArray[index]['info']['statistics'][type]['avg_capture_points'] = Statistics['capture_points'] / Statistics['battles'];
				if(isNaN(MembersArray[index]['info']['statistics'][type]['avg_capture_points'])){MembersArray[index]['info']['statistics'][type]['avg_capture_points'] = 0;}
				
				MembersArray[index]['info']['statistics'][type]['avg_dropped_capture_points'] = Statistics['dropped_capture_points'] / Statistics['battles'];
				if(isNaN(MembersArray[index]['info']['statistics'][type]['avg_dropped_capture_points'])){MembersArray[index]['info']['statistics'][type]['avg_dropped_capture_points'] = 0;}
				
				MembersArray[index]['info']['statistics'][type]['wins_percents'] = (Statistics['wins']/Statistics['battles'])*100;
				if(isNaN(MembersArray[index]['info']['statistics'][type]['wins_percents'])){MembersArray[index]['info']['statistics'][type]['wins_percents'] = 0;}
				
				MembersArray[index]['info']['statistics'][type]['survived_battles_percents'] = (Statistics['survived_battles']/Statistics['battles'])*100;
				if(isNaN(MembersArray[index]['info']['statistics'][type]['survived_battles_percents'])){MembersArray[index]['info']['statistics'][type]['survived_battles_percents'] = 0;}
				
				if(Statistics['battles'] == Statistics['survived_battles']){
					MembersArray[index]['info']['statistics'][type]['kill_dead'] = Statistics['frags']/Statistics['battles'];
				}else{
					MembersArray[index]['info']['statistics'][type]['kill_dead'] = Statistics['frags']/(Statistics['battles']-Statistics['survived_battles']);
				}
				if(isNaN(MembersArray[index]['info']['statistics'][type]['kill_dead'])){MembersArray[index]['info']['statistics'][type]['kill_dead'] = 0;}
				
				if(type != 'pvp' && type != 'pve'){
					var type_start = type.split('_')[0];
					MembersArray[index]['info']['statistics'][type]['battles_percents'] = (Statistics['battles']/MembersArray[index]['info']['statistics'][type_start]['battles'])*100;
					if(isNaN(MembersArray[index]['info']['statistics'][type]['battles_percents'])){MembersArray[index]['info']['statistics'][type]['battles_percents'] = 0;}
				}
			}
			
			MembersArray[index]['info']['ships_x_level'] = 0;
			MembersArray[index]['info']['statistics']['pvp_div']['avg_battles_level'] = 0;
			MembersArray[index]['info']['statistics']['pvp_div']['max_ship_level'] = 0;
			
			for(var t = 0; t < typeStat.length; t++){
				var type = typeStat[t];
				
				MembersArray[index]['info']['statistics'][type]['avg_battles_level'] = 0;
				MembersArray[index]['info']['statistics'][type]['max_ship_level'] = 0;
				MembersArray[index]['info']['statistics'][type]['wr'] = 0;
				MembersArray[index]['info']['statistics'][type]['wtr'] = 0;
				
				var StatShips = [];
				
				StatShips['damage_dealt'] = 0;
				StatShips['frags'] = 0;
				StatShips['planes_killed'] = 0;
				StatShips['expDamage'] = 0;
				StatShips['expFrags'] = 0;
				StatShips['expPlanesKilled'] = 0;
				
				StatShips['actual.wins'] = 0;
				StatShips['actual.damage_dealt'] = 0;
				StatShips['actual.frags'] = 0;
				StatShips['actual.planes_killed'] = 0;
				StatShips['actual.capture_points'] = 0;
				StatShips['actual.dropped_capture_points'] = 0;
				StatShips['expected.wins'] = 0;
				StatShips['expected.damage_dealt'] = 0;
				StatShips['expected.frags'] = 0;
				StatShips['expected.planes_killed'] = 0;
				StatShips['expected.capture_points'] = 0;
				StatShips['expected.dropped_capture_points'] = 0;
				
				var StatShipsClass = [];
				for(var tS = 0; tS < typeShip.length; tS++){
					var typeS = typeShip[tS];
					StatShipsClass[typeS] = [];
					StatShipsClass[typeS]['damage_dealt'] = 0;
					StatShipsClass[typeS]['frags'] = 0;
					StatShipsClass[typeS]['planes_killed'] = 0;
					StatShipsClass[typeS]['expDamage'] = 0;
					StatShipsClass[typeS]['expFrags'] = 0;
					StatShipsClass[typeS]['expPlanesKilled'] = 0;
					
					StatShipsClass[typeS]['actual.wins'] = 0;
					StatShipsClass[typeS]['actual.damage_dealt'] = 0;
					StatShipsClass[typeS]['actual.frags'] = 0;
					StatShipsClass[typeS]['actual.planes_killed'] = 0;
					StatShipsClass[typeS]['actual.capture_points'] = 0;
					StatShipsClass[typeS]['actual.dropped_capture_points'] = 0;
					StatShipsClass[typeS]['expected.wins'] = 0;
					StatShipsClass[typeS]['expected.damage_dealt'] = 0;
					StatShipsClass[typeS]['expected.frags'] = 0;
					StatShipsClass[typeS]['expected.planes_killed'] = 0;
					StatShipsClass[typeS]['expected.capture_points'] = 0;
					StatShipsClass[typeS]['expected.dropped_capture_points'] = 0;
				}
				
				for(var shipI = 0; shipI < MembersArray[index]['ships'].length; shipI++){
					if(type == 'pvp_div' && MembersArray[index]['ships'][shipI]['pvp_div'] == undefined){
						MembersArray[index]['ships'][shipI]['pvp_div'] = [];
						
						if(MembersArray[index]['ships'][shipI]['pvp_div2']['max_xp'] > MembersArray[index]['ships'][shipI]['pvp_div3']['max_xp']){
							MembersArray[index]['ships'][shipI]['pvp_div']['max_xp'] = MembersArray[index]['ships'][shipI]['pvp_div2']['max_xp'];
						}else{
							MembersArray[index]['ships'][shipI]['pvp_div']['max_xp'] = MembersArray[index]['ships'][shipI]['pvp_div3']['max_xp'];
						}
						
						if(MembersArray[index]['ships'][shipI]['pvp_div2']['max_frags_battle'] > MembersArray[index]['ships'][shipI]['pvp_div3']['max_frags_battle']){
							MembersArray[index]['ships'][shipI]['pvp_div']['max_frags_battle'] = MembersArray[index]['ships'][shipI]['pvp_div2']['max_frags_battle'];
						}else{
							MembersArray[index]['ships'][shipI]['pvp_div']['max_frags_battle'] = MembersArray[index]['ships'][shipI]['pvp_div3']['max_frags_battle'];
						}
						
						if(MembersArray[index]['ships'][shipI]['pvp_div2']['max_planes_killed'] > MembersArray[index]['ships'][shipI]['pvp_div3']['max_planes_killed']){
							MembersArray[index]['ships'][shipI]['pvp_div']['max_planes_killed'] = MembersArray[index]['ships'][shipI]['pvp_div2']['max_planes_killed'];
						}else{
							MembersArray[index]['ships'][shipI]['pvp_div']['max_planes_killed'] = MembersArray[index]['ships'][shipI]['pvp_div3']['max_planes_killed'];
						}
						
						if(MembersArray[index]['ships'][shipI]['pvp_div2']['max_damage_dealt'] > MembersArray[index]['ships'][shipI]['pvp_div3']['max_damage_dealt']){
							MembersArray[index]['ships'][shipI]['pvp_div']['max_damage_dealt'] = MembersArray[index]['ships'][shipI]['pvp_div2']['max_damage_dealt'];
						}else{
							MembersArray[index]['ships'][shipI]['pvp_div']['max_damage_dealt'] = MembersArray[index]['ships'][shipI]['pvp_div3']['max_damage_dealt'];
						}
						
						MembersArray[index]['ships'][shipI]['pvp_div']['xp'] = MembersArray[index]['ships'][shipI]['pvp_div2']['xp'] + MembersArray[index]['ships'][shipI]['pvp_div3']['xp'];
						MembersArray[index]['ships'][shipI]['pvp_div']['survived_battles'] = MembersArray[index]['ships'][shipI]['pvp_div2']['survived_battles'] + MembersArray[index]['ships'][shipI]['pvp_div3']['survived_battles'];
						MembersArray[index]['ships'][shipI]['pvp_div']['dropped_capture_points'] = MembersArray[index]['ships'][shipI]['pvp_div2']['dropped_capture_points'] + MembersArray[index]['ships'][shipI]['pvp_div3']['dropped_capture_points'];
						MembersArray[index]['ships'][shipI]['pvp_div']['draws'] = MembersArray[index]['ships'][shipI]['pvp_div2']['draws'] + MembersArray[index]['ships'][shipI]['pvp_div3']['draws'];
						MembersArray[index]['ships'][shipI]['pvp_div']['wins'] = MembersArray[index]['ships'][shipI]['pvp_div2']['wins'] + MembersArray[index]['ships'][shipI]['pvp_div3']['wins'];
						MembersArray[index]['ships'][shipI]['pvp_div']['damage_dealt'] = MembersArray[index]['ships'][shipI]['pvp_div2']['damage_dealt'] + MembersArray[index]['ships'][shipI]['pvp_div3']['damage_dealt'];
						MembersArray[index]['ships'][shipI]['pvp_div']['losses'] = MembersArray[index]['ships'][shipI]['pvp_div2']['losses'] + MembersArray[index]['ships'][shipI]['pvp_div3']['losses'];
						MembersArray[index]['ships'][shipI]['pvp_div']['frags'] = MembersArray[index]['ships'][shipI]['pvp_div2']['frags'] + MembersArray[index]['ships'][shipI]['pvp_div3']['frags'];
						MembersArray[index]['ships'][shipI]['pvp_div']['capture_points'] = MembersArray[index]['ships'][shipI]['pvp_div2']['capture_points'] + MembersArray[index]['ships'][shipI]['pvp_div3']['capture_points'];
						MembersArray[index]['ships'][shipI]['pvp_div']['survived_wins'] = MembersArray[index]['ships'][shipI]['pvp_div2']['survived_wins'] + MembersArray[index]['ships'][shipI]['pvp_div3']['survived_wins'];
						MembersArray[index]['ships'][shipI]['pvp_div']['battles'] = MembersArray[index]['ships'][shipI]['pvp_div2']['battles'] + MembersArray[index]['ships'][shipI]['pvp_div3']['battles'];
						MembersArray[index]['ships'][shipI]['pvp_div']['planes_killed'] = MembersArray[index]['ships'][shipI]['pvp_div2']['planes_killed'] + MembersArray[index]['ships'][shipI]['pvp_div3']['planes_killed'];
					}
				
					var Ship = MembersArray[index]['ships'][shipI];
					var ship_id = Ship['ship_id'];
					var Statistics = Ship[type];
					
					MembersArray[index]['ships'][shipI][type]['avg_xp'] = Statistics['xp'] / Statistics['battles'];
					if(isNaN(MembersArray[index]['ships'][shipI][type]['avg_xp'])){MembersArray[index]['ships'][shipI][type]['avg_xp'] = 0;}
					
					MembersArray[index]['ships'][shipI][type]['avg_damage_dealt'] = Statistics['damage_dealt'] / Statistics['battles'];
					if(isNaN(MembersArray[index]['ships'][shipI][type]['avg_damage_dealt'])){MembersArray[index]['ships'][shipI][type]['avg_damage_dealt'] = 0;}
					
					MembersArray[index]['ships'][shipI][type]['avg_frags'] = Statistics['frags'] / Statistics['battles'];
					if(isNaN(MembersArray[index]['ships'][shipI][type]['avg_frags'])){MembersArray[index]['ships'][shipI][type]['avg_frags'] = 0;}
					
					MembersArray[index]['ships'][shipI][type]['avg_planes_killed'] = Statistics['planes_killed'] / Statistics['battles'];
					if(isNaN(MembersArray[index]['ships'][shipI][type]['avg_planes_killed'])){MembersArray[index]['ships'][shipI][type]['avg_planes_killed'] = 0;}
					
					MembersArray[index]['ships'][shipI][type]['avg_capture_points'] = Statistics['capture_points'] / Statistics['battles'];
					if(isNaN(MembersArray[index]['ships'][shipI][type]['avg_capture_points'])){MembersArray[index]['ships'][shipI][type]['avg_capture_points'] = 0;}
					
					MembersArray[index]['ships'][shipI][type]['avg_dropped_capture_points'] = Statistics['dropped_capture_points'] / Statistics['battles'];
					if(isNaN(MembersArray[index]['ships'][shipI][type]['avg_dropped_capture_points'])){MembersArray[index]['ships'][shipI][type]['avg_dropped_capture_points'] = 0;}
					
					MembersArray[index]['ships'][shipI][type]['wins_percents'] = (Statistics['wins']/Statistics['battles'])*100;
					if(isNaN(MembersArray[index]['ships'][shipI][type]['wins_percents'])){MembersArray[index]['ships'][shipI][type]['wins_percents'] = 0;}
					
					MembersArray[index]['ships'][shipI][type]['survived_battles_percents'] = (Statistics['survived_battles']/Statistics['battles'])*100;
					if(isNaN(MembersArray[index]['ships'][shipI][type]['survived_battles_percents'])){MembersArray[index]['ships'][shipI][type]['survived_battles_percents'] = 0;}
					
					if(Statistics['battles'] == Statistics['survived_battles']){
						MembersArray[index]['ships'][shipI][type]['kill_dead'] = Statistics['frags']/Statistics['battles'];
					}else{
						MembersArray[index]['ships'][shipI][type]['kill_dead'] = Statistics['frags']/(Statistics['battles']-Statistics['survived_battles']);
					}
					if(isNaN(MembersArray[index]['ships'][shipI][type]['kill_dead'])){MembersArray[index]['ships'][shipI][type]['kill_dead'] = 0;}
					
					if(Encyclopedia != null){
						var ship_type = Encyclopedia[ship_id]['type'];
						var ship_tier = Encyclopedia[ship_id]['tier'];
						
						if(ExpShips[ship_id] !== undefined){
							var battles = Statistics['battles'];
							var damage_dealt = Statistics['damage_dealt'];
							var frags = Statistics['frags'];
							var planes_killed = Statistics['planes_killed'];
							
							var StatShip = [];
							StatShip['damage_dealt'] = damage_dealt;
							StatShip['frags'] = frags;
							StatShip['planes_killed'] = planes_killed;
							StatShip['expDamage'] = battles * ExpShips[ship_id]['expDamage'];
							StatShip['expFrags'] = battles * ExpShips[ship_id]['expFrags'];
							StatShip['expPlanesKilled'] = battles * ExpShips[ship_id]['expPlanesKilled'];
							
							MembersArray[index]['ships'][shipI][type]['wr'] = calcWR(StatShip);
							
							StatShipsClass[ship_type]['damage_dealt'] += damage_dealt;
							StatShipsClass[ship_type]['frags'] += frags;
							StatShipsClass[ship_type]['planes_killed'] += planes_killed;
							StatShipsClass[ship_type]['expDamage'] += StatShip['expDamage'];
							StatShipsClass[ship_type]['expFrags'] += StatShip['expFrags'];
							StatShipsClass[ship_type]['expPlanesKilled'] += StatShip['expPlanesKilled'];
							
							StatShips['damage_dealt'] += damage_dealt;
							StatShips['frags'] += frags;
							StatShips['planes_killed'] += planes_killed;
							StatShips['expDamage'] += StatShip['expDamage'];
							StatShips['expFrags'] += StatShip['expFrags'];
							StatShips['expPlanesKilled'] += StatShip['expPlanesKilled'];
						}else{
							MembersArray[index]['ships'][shipI][type]['wr'] = 0;
						}
						
						if(ExpWTR['expected'][ship_id] !== undefined){
							var battles = Statistics['battles'];
							var wins = Statistics['wins'];
							var damage_dealt = Statistics['damage_dealt'];
							var frags = Statistics['frags'];
							var planes_killed = Statistics['planes_killed'];
							var capture_points = Statistics['capture_points'];
							var dropped_capture_points = Statistics['dropped_capture_points'];
							
							var StatShip = [];
							StatShip['actual.wins'] = wins;
							StatShip['actual.damage_dealt'] = damage_dealt;
							StatShip['actual.frags'] = frags;
							StatShip['actual.planes_killed'] = planes_killed;
							StatShip['actual.capture_points'] = capture_points;
							StatShip['actual.dropped_capture_points'] = dropped_capture_points;
							StatShip['expected.wins'] = battles * ExpWTR['expected'][ship_id]['wins'];
							StatShip['expected.damage_dealt'] = battles * ExpWTR['expected'][ship_id]['damage_dealt'];
							StatShip['expected.frags'] = battles * ExpWTR['expected'][ship_id]['frags'];
							StatShip['expected.planes_killed'] = battles * ExpWTR['expected'][ship_id]['planes_killed'];
							StatShip['expected.capture_points'] = battles * ExpWTR['expected'][ship_id]['capture_points'];
							StatShip['expected.dropped_capture_points'] = battles * ExpWTR['expected'][ship_id]['dropped_capture_points'];
							
							MembersArray[index]['ships'][shipI][type]['wtr'] = calcWTR(StatShip);
							
							StatShipsClass[ship_type]['actual.wins'] += wins;
							StatShipsClass[ship_type]['actual.damage_dealt'] += damage_dealt;
							StatShipsClass[ship_type]['actual.frags'] += frags;
							StatShipsClass[ship_type]['actual.planes_killed'] += planes_killed;
							StatShipsClass[ship_type]['actual.capture_points'] += capture_points;
							StatShipsClass[ship_type]['actual.dropped_capture_points'] += dropped_capture_points;
							StatShipsClass[ship_type]['expected.wins'] += StatShip['expected.wins'];
							StatShipsClass[ship_type]['expected.damage_dealt'] += StatShip['expected.damage_dealt'];
							StatShipsClass[ship_type]['expected.frags'] += StatShip['expected.frags'];
							StatShipsClass[ship_type]['expected.planes_killed'] += StatShip['expected.planes_killed'];
							StatShipsClass[ship_type]['expected.capture_points'] += StatShip['expected.capture_points'];
							StatShipsClass[ship_type]['expected.dropped_capture_points'] += StatShip['expected.dropped_capture_points'];
							
							StatShips['actual.wins'] += wins;
							StatShips['actual.damage_dealt'] += damage_dealt;
							StatShips['actual.frags'] += frags;
							StatShips['actual.planes_killed'] += planes_killed;
							StatShips['actual.capture_points'] += capture_points;
							StatShips['actual.dropped_capture_points'] += dropped_capture_points;
							StatShips['expected.wins'] += StatShip['expected.wins'];
							StatShips['expected.damage_dealt'] += StatShip['expected.damage_dealt'];
							StatShips['expected.frags'] += StatShip['expected.frags'];
							StatShips['expected.planes_killed'] += StatShip['expected.planes_killed'];
							StatShips['expected.capture_points'] += StatShip['expected.capture_points'];
							StatShips['expected.dropped_capture_points'] += StatShip['expected.dropped_capture_points'];
						}else{
							MembersArray[index]['ships'][shipI][type]['wtr'] = 0;
						}
						
						if(Statistics['battles'] > 0){
							MembersArray[index]['info']['statistics'][type]['avg_battles_level'] += ship_tier * Statistics['battles'] / MembersArray[index]['info']['statistics'][type]['battles'];
						}
						
						if(Encyclopedia[ship_id]['tier'] == 10 && type == 'pvp'){
							MembersArray[index]['info']['ships_x_level']++;
						}
						
						if(MembersArray[index]['info']['statistics'][type]['max_ship_level'] < Encyclopedia[ship_id]['tier']){
							MembersArray[index]['info']['statistics'][type]['max_ship_level'] = Encyclopedia[ship_id]['tier'];
						}
					}
				}
				
				for(var tS = 0; tS < typeShip.length; tS++){
					var typeS = typeShip[tS];
					
					MembersArray[index]['info']['statistics'][type]['wr_'+typeS] = calcWR(StatShipsClass[typeS]);
					MembersArray[index]['info']['statistics'][type]['wtr_'+typeS] = calcWTR(StatShipsClass[typeS]);
				}
				
				MembersArray[index]['info']['statistics'][type]['wr'] = calcWR(StatShips);
				MembersArray[index]['info']['statistics'][type]['wtr'] = calcWTR(StatShips);
			}
			
			if(MembersArray[index]['info']['statistics']['pvp_div2']['avg_battles_level'] > MembersArray[index]['info']['statistics']['pvp_div3']['avg_battles_level']){
				MembersArray[index]['info']['statistics']['pvp_div']['avg_battles_level'] = MembersArray[index]['info']['statistics']['pvp_div2']['avg_battles_level'];
			}else{
				MembersArray[index]['info']['statistics']['pvp_div']['avg_battles_level'] = MembersArray[index]['info']['statistics']['pvp_div3']['avg_battles_level'];
			}
			
			if(MembersArray[index]['info']['statistics']['pvp_div2']['max_ship_level'] > MembersArray[index]['info']['statistics']['pvp_div3']['max_ship_level']){
				MembersArray[index]['info']['statistics']['pvp_div']['max_ship_level'] = MembersArray[index]['info']['statistics']['pvp_div2']['max_ship_level'];
			}else{
				MembersArray[index]['info']['statistics']['pvp_div']['max_ship_level'] = MembersArray[index]['info']['statistics']['pvp_div3']['max_ship_level'];
			}
			
			for(var key in MembersArray[index]['achievements']['battle']){
				var battle = MembersArray[index]['info']['statistics']['pvp']['battles'];
				var achievements = MembersArray[index]['achievements']['battle'];
				MembersArray[index]['achievements']['battle'][key+'_battle'] = (battle / achievements[key]).toFixed(0);
			}
			
			return true;
		}
		function calcWR(Stat){
			var rDamage = Stat['damage_dealt'] / Stat['expDamage']; if(isNaN(rDamage)){rDamage = 0;}
			var rFrags = Stat['frags'] / Stat['expFrags']; if(isNaN(rFrags)){rFrags = 0;}
			var rPlanesKilled = Stat['planes_killed'] / Stat['expPlanesKilled']; if(isNaN(rPlanesKilled)){rPlanesKilled = 0;}
			
			var rDamagec = Math.max(0, (rDamage - 0.25) / (1 - 0.25));
			var rFragsc = Math.max(0, Math.min(rDamagec + 0.2, (rFrags - 0.12) / (1 - 0.12)));
			var rPlanesKilledc = Math.max(0, Math.min(rDamagec + 0.1, (rPlanesKilled - 0.15) / (1 - 0.15)));
			
			var wr = 650 * rDamagec + 150 * rFragsc * rDamagec + 80 * rPlanesKilledc;
			if(isNaN(wr)){wr = 0;}
			
			return wr;
		}
		function calcWTR(Stat){
			var wins = Stat['actual.wins'] / Stat['expected.wins'];
			var damage_dealt = Stat['actual.damage_dealt'] / Stat['expected.damage_dealt'];
			var ship_frags = Stat['actual.frags'] / Stat['expected.frags'];
			var capture_points = Stat['actual.capture_points'] / Stat['expected.capture_points'];
			var dropped_capture_points = Stat['actual.dropped_capture_points'] / Stat['expected.dropped_capture_points'];
			var planes_killed = Stat['actual.planes_killed'] / Stat['expected.planes_killed'];
			var ship_frags_importance_weight = ExpWTR['coefficients']['ship_frags_importance_weight'];
			var frags = 1.0; // fallback to avoid division by zero
			if (Stat['expected.planes_killed'] + Stat['expected.frags'] > 0) { // this should be happening virtually always
				var aircraft_frags_coef = Stat['expected.planes_killed'] / (Stat['expected.planes_killed'] + ship_frags_importance_weight * Stat['expected.frags']);
				var ship_frags_coef = 1 - aircraft_frags_coef;
				if (aircraft_frags_coef == 1) {
					frags = planes_killed;
				} else if (ship_frags_coef == 1) {
					frags = ship_frags;
				} else {
					frags = ship_frags * ship_frags_coef + planes_killed * aircraft_frags_coef;
				}
			}
			var wins_weight = ExpWTR['coefficients']['wins_weight'];
			var damage_weight = ExpWTR['coefficients']['damage_weight'];
			var frags_weight = ExpWTR['coefficients']['frags_weight'];
			var capture_weight = ExpWTR['coefficients']['capture_weight'];
			var dropped_capture_weight = ExpWTR['coefficients']['dropped_capture_weight'];
			var wtr =
				wins                    * wins_weight             +
				damage_dealt            * damage_weight           +
				frags                   * frags_weight            +
				capture_points          * capture_weight          +
				dropped_capture_points  * dropped_capture_weight;
			var nominal_rating = ExpWTR['coefficients']['nominal_rating'];
			wtr = wtr * nominal_rating;
			if(isNaN(wtr)){wtr = 0;}
			
			return wtr;
		}
		function doneEncyclopedia(url, response){
			if(response.status && response.status == "error"){
				errorEncyclopedia();
				return;
			}
			
			Encyclopedia = response['data'];
			
			Encyclopedia['null'] = [];
			Encyclopedia['null']['name'] = '';
		}
		function errorEncyclopedia(url){
			Encyclopedia = null;
			
			console.log('Get Encyclopedia Error');
		}
		function findColorASC(value, stat_type, type){
			if(colorStat[stat_type] === undefined){
				return '';
			}
			
			if(type == 'main'){
				if(isNaN(value) || parseFloat(value) <= parseFloat(colorStat[stat_type][0])){
					return color['very_bad'];
				}else if(parseFloat(value) <= parseFloat(colorStat[stat_type][1])){
					return color['bad'];
				}else if(parseFloat(value) <= parseFloat(colorStat[stat_type][2])){
					return color['normal'];
				}else if(parseFloat(value) <= parseFloat(colorStat[stat_type][3])){
					return color['good'];
				}else if(parseFloat(value) <= parseFloat(colorStat[stat_type][4])){
					return color['very_good'];
				}else if(parseFloat(value) <= parseFloat(colorStat[stat_type][5])){
					return color['unique'];
				}else{
					return color['very_bad'];
				}
			}else{
				return '';
			}
		}
		function findColorDESC(value, stat_type, type){
			if(type == 'main'){
				var return_color = color['very_bad'];
				
				if(isNaN(value)){
					return_color = color['very_bad'];
				}
				if(parseFloat(value) <= parseFloat(colorStat[stat_type][0])){
					return_color = color['very_bad'];
				}
				if(parseFloat(value) <= parseFloat(colorStat[stat_type][1])){
					return_color = color['bad'];
				}
				if(parseFloat(value) <= parseFloat(colorStat[stat_type][2])){
					return_color = color['normal'];
				}
				if(parseFloat(value) <= parseFloat(colorStat[stat_type][3])){
					return_color = color['good'];
				}
				if(parseFloat(value) <= parseFloat(colorStat[stat_type][4])){
					return_color = color['very_good'];
				}
				if(parseFloat(value) <= parseFloat(colorStat[stat_type][5])){
					return_color = color['unique'];
				}
				
				return return_color;
			}else{
				return '';
			}
		}
		function ASC(a){
			return function (b, c){
				if(a == 'account_name'){
					return b[a].toLowerCase() < c[a].toLowerCase() ? -1 : b[a].toLowerCase() > c[a].toLowerCase() ? 1 : 0;
				}else if(a == 'role_i18n'){
					a = 'role_sort_num';
					return parseFloat(b[a]) < parseFloat(c[a]) ? -1 : parseFloat(b[a]) > parseFloat(c[a]) ? 1 : 0;
				}else{
					var attr = a.split('.');
					for(var i = 0; i < attr.length; i++){
						b = b[attr[i]];
						c = c[attr[i]];
					}				
					return parseFloat(b) < parseFloat(c) ? -1 : parseFloat(b) > parseFloat(c) ? 1 : 0;
				}
			}
		};
		function DESC(a){
			return function (b, c){
				if(a == 'account_name'){
					return b[a].toLowerCase() > c[a].toLowerCase() ? -1 : b[a].toLowerCase() < c[a].toLowerCase() ? 1 : 0;
				}else if(a == 'role_i18n'){
					a = 'role_sort_num';
					return parseFloat(b[a]) > parseFloat(c[a]) ? -1 : parseFloat(b[a]) < parseFloat(c[a]) ? 1 : 0;
				}else{
					var attr = a.split('.');
					for(var i = 0; i < attr.length; i++){
						b = b[attr[i]];
						c = c[attr[i]];
					}					
					return parseFloat(b) > parseFloat(c) ? -1 : parseFloat(b) < parseFloat(c) ? 1 : 0;
				}
			}
		};
		function valueFormat(value){
			if(isInt(value)){value = value.toString();}
			var newValue = '';
			var valueSplit = value.split('.');
			
			var numChar = 0;
			for(var i = valueSplit[0].length; i > 0; i--){
				if(numChar < 3){
					newValue = valueSplit[0].substr(i - 1, 1)+''+newValue;
					numChar++;
				}else{
					newValue = valueSplit[0].substr(i - 1, 1)+''+localizationText['num-separator']+''+newValue;
					numChar = 1;
				}
			}
			
			if(valueSplit.length > 1){newValue += localizationText['num-fractional']+''+valueSplit[1];}
			
			return newValue;
		}
		function isInt(value){
			return !isNaN(value) && (function(x){return (x | 0) === x;})(parseFloat(value));
		}
		function htmlParseMemberStatistic(element){
			var value = element.textContent.trim().replace(new RegExp(' ', 'g'), '');
			value = value.replace(/[^0-9,.()% ]/g, "");
			
			value = value.replace('%', '');
			
			value = value.split(localizationText['num-separator']).join('');
			value = value.replace(localizationText['num-fractional'], '.');
			
			if(value.indexOf('(') > -1 && value.indexOf(')') > -1){
				value = (value.split('('))[0];
			}
			
			return value;
		}		
		function getJson(url, onDone, onError){
			if(MaxProcess > Process){
				Process++;
				jQ.getJSON(url).done(function(result){
					Process--;
					onDone(url, result);
				}).fail(function(jqxhr, textStatus, error){
					Process--;
					onError(url);
				});
			}else{
				setTimeout(function(){getJson(url, onDone, onError);}, 1000);
			}
		}
		function getApplicationId(realm){
			var applicationId = [];
			
			applicationId['ru'] = '7149a13b5f5fb7109c5b2400d31b7d42';
			applicationId['na'] = '7225f37baa8385ced2469cbaca8aeaba';
			applicationId['asia'] = '4ce71a5573f43c721806b4b4dadc822d';
			applicationId['eu'] = '953df86f6bca01a7af80c3bdedd9c1d9';
			
			return applicationId[realm].split("").reverse().join("");
		}
		function getUrlVars(url){
			var vars = {};
			var parts = url.replace(/[?&]+([^=&]+)=([^&]*)/gi, function(m, key, value){
				vars[key] = value;
			});
			return vars;
		}
		function TimeToDate(time){
			var date = new Date(time * 1000);
			var seconds = date.getSeconds();
			var minutes = date.getMinutes();
			var hour = date.getHours();
			var day = date.getDate();
			var month = (date.getMonth() + 1);
			var year = date.getFullYear();
			
			if(seconds < 10){
				seconds = '0'+seconds;
			}
			if(minutes < 10){
				minutes = '0'+minutes;
			}
			if(hour < 10){
				hour = '0'+hour;
			}
			if(day < 10){
				day = '0'+day;
			}
			if(month < 10){
				month = '0'+month;
			}
			
			return 	day+'.'+month+'.'+year+' '+hour+':'+minutes;
		}
		function dateDiffInDays(a, b){
			var date1 = new Date(a);
			var date2 = new Date(b);
			if(b == null){
				date2 = new Date();
			}
			var timeDiff = date2.getTime() - date1.getTime(); 
			var diffDays = timeDiff / (1000 * 3600 * 24); 
			if(diffDays < 0){diffDays = 0;}
			return (diffDays).toFixed(0);
		}		
		function getUserScriptDeveloperBlock(){
			var html = '' +
				'' +
					'
' +
						'UserScript WoWsStatInfo ' + VersionWoWsStatInfo +
						'
'+
					' ' +
				'
' +
					'
' +
						localizationText['userscript-developer'] +
						' Vov_chiK  ' +
						localizationText['userscript-alliance'] +
						' Walkure ,' +
						' '+localizationText['userscript-support'] +
						' vzhabin.ru ' +
						'' +
							WoWsStatInfoLinkName +
						' ' +
						''+localizationText['userscript-developer-support']+' Web-Money WMR  R295712009837 Web-Money WMZ  Z226959724402 Yandex Money  41001290117791 RBK Money  RU353257918  ' +
				'
 tag.
				//
				// @constructor
				// @param {string} cls List of classes to apply to the canvas.
				// @param {element} container Element onto which to append the canvas.
				//
				// Requiring a container is a little iffy, but unfortunately canvas
				// operations don't work unless the canvas is attached to the DOM.
				function Canvas(cls, container) {
					var element = container.children("." + cls)[0];
					if (element == null) {
						element = document.createElement("canvas");
						element.className = cls;
						$(element).css({ direction: "ltr", position: "absolute", left: 0, top: 0 })
							.appendTo(container);
						// If HTML5 Canvas isn't available, fall back to [Ex|Flash]canvas
						if (!element.getContext) {
							if (window.G_vmlCanvasManager) {
								element = window.G_vmlCanvasManager.initElement(element);
							} else {
								throw new Error("Canvas is not available. If you're using IE with a fall-back such as Excanvas, then there's either a mistake in your conditional include, or the page has no DOCTYPE and is rendering in Quirks Mode.");
							}
						}
					}
					this.element = element;
					var context = this.context = element.getContext("2d");
					// Determine the screen's ratio of physical to device-independent
					// pixels.  This is the ratio between the canvas width that the browser
					// advertises and the number of pixels actually present in that space.
					// The iPhone 4, for example, has a device-independent width of 320px,
					// but its screen is actually 640px wide.  It therefore has a pixel
					// ratio of 2, while most normal devices have a ratio of 1.
					var devicePixelRatio = window.devicePixelRatio || 1,
						backingStoreRatio =
							context.webkitBackingStorePixelRatio ||
							context.mozBackingStorePixelRatio ||
							context.msBackingStorePixelRatio ||
							context.oBackingStorePixelRatio ||
							context.backingStorePixelRatio || 1;
					this.pixelRatio = devicePixelRatio / backingStoreRatio;
					// Size the canvas to match the internal dimensions of its container
					this.resize(container.width(), container.height());
					// Collection of HTML div layers for text overlaid onto the canvas
					this.textContainer = null;
					this.text = {};
					// Cache of text fragments and metrics, so we can avoid expensively
					// re-calculating them when the plot is re-rendered in a loop.
					this._textCache = {};
				}
				// Resizes the canvas to the given dimensions.
				//
				// @param {number} width New width of the canvas, in pixels.
				// @param {number} width New height of the canvas, in pixels.
				Canvas.prototype.resize = function(width, height) {
					if (width <= 0 || height <= 0) {
						throw new Error("Invalid dimensions for plot, width = " + width + ", height = " + height);
					}
					var element = this.element,
						context = this.context,
						pixelRatio = this.pixelRatio;
					// Resize the canvas, increasing its density based on the display's
					// pixel ratio; basically giving it more pixels without increasing the
					// size of its element, to take advantage of the fact that retina
					// displays have that many more pixels in the same advertised space.
					// Resizing should reset the state (excanvas seems to be buggy though)
					if (this.width != width) {
						element.width = width * pixelRatio;
						element.style.width = width + "px";
						this.width = width;
					}
					if (this.height != height) {
						element.height = height * pixelRatio;
						element.style.height = height + "px";
						this.height = height;
					}
					// Save the context, so we can reset in case we get replotted.  The
					// restore ensure that we're really back at the initial state, and
					// should be safe even if we haven't saved the initial state yet.
					context.restore();
					context.save();
					// Scale the coordinate space to match the display density; so even though we
					// may have twice as many pixels, we still want lines and other drawing to
					// appear at the same size; the extra pixels will just make them crisper.
					context.scale(pixelRatio, pixelRatio);
				};
				// Clears the entire canvas area, not including any overlaid HTML text
				Canvas.prototype.clear = function() {
					this.context.clearRect(0, 0, this.width, this.height);
				};
				// Finishes rendering the canvas, including managing the text overlay.
				Canvas.prototype.render = function() {
					var cache = this._textCache;
					// For each text layer, add elements marked as active that haven't
					// already been rendered, and remove those that are no longer active.
					for (var layerKey in cache) {
						if (hasOwnProperty.call(cache, layerKey)) {
							var layer = this.getTextLayer(layerKey),
								layerCache = cache[layerKey];
							layer.hide();
							for (var styleKey in layerCache) {
								if (hasOwnProperty.call(layerCache, styleKey)) {
									var styleCache = layerCache[styleKey];
									for (var key in styleCache) {
										if (hasOwnProperty.call(styleCache, key)) {
											var positions = styleCache[key].positions;
											for (var i = 0, position; position = positions[i]; i++) {
												if (position.active) {
													if (!position.rendered) {
														layer.append(position.element);
														position.rendered = true;
													}
												} else {
													positions.splice(i--, 1);
													if (position.rendered) {
														position.element.detach();
													}
												}
											}
											if (positions.length == 0) {
												delete styleCache[key];
											}
										}
									}
								}
							}
							layer.show();
						}
					}
				};
				// Creates (if necessary) and returns the text overlay container.
				//
				// @param {string} classes String of space-separated CSS classes used to
				//     uniquely identify the text layer.
				// @return {object} The jQuery-wrapped text-layer div.
				Canvas.prototype.getTextLayer = function(classes) {
					var layer = this.text[classes];
					// Create the text layer if it doesn't exist
					if (layer == null) {
						// Create the text layer container, if it doesn't exist
						if (this.textContainer == null) {
							this.textContainer = $("
")
								.css({
									position: "absolute",
									top: 0,
									left: 0,
									bottom: 0,
									right: 0,
									'font-size': "smaller",
									color: "#545454"
								})
								.insertAfter(this.element);
						}
						layer = this.text[classes] = $("
")
							.addClass(classes)
							.css({
								position: "absolute",
								top: 0,
								left: 0,
								bottom: 0,
								right: 0
							})
							.appendTo(this.textContainer);
					}
					return layer;
				};
				// Creates (if necessary) and returns a text info object.
				//
				// The object looks like this:
				//
				// {
				//     width: Width of the text's wrapper div.
				//     height: Height of the text's wrapper div.
				//     element: The jQuery-wrapped HTML div containing the text.
				//     positions: Array of positions at which this text is drawn.
				// }
				//
				// The positions array contains objects that look like this:
				//
				// {
				//     active: Flag indicating whether the text should be visible.
				//     rendered: Flag indicating whether the text is currently visible.
				//     element: The jQuery-wrapped HTML div containing the text.
				//     x: X coordinate at which to draw the text.
				//     y: Y coordinate at which to draw the text.
				// }
				//
				// Each position after the first receives a clone of the original element.
				//
				// The idea is that that the width, height, and general 'identity' of the
				// text is constant no matter where it is placed; the placements are a
				// secondary property.
				//
				// Canvas maintains a cache of recently-used text info objects; getTextInfo
				// either returns the cached element or creates a new entry.
				//
				// @param {string} layer A string of space-separated CSS classes uniquely
				//     identifying the layer containing this text.
				// @param {string} text Text string to retrieve info for.
				// @param {(string|object)=} font Either a string of space-separated CSS
				//     classes or a font-spec object, defining the text's font and style.
				// @param {number=} angle Angle at which to rotate the text, in degrees.
				//     Angle is currently unused, it will be implemented in the future.
				// @param {number=} width Maximum width of the text before it wraps.
				// @return {object} a text info object.
				Canvas.prototype.getTextInfo = function(layer, text, font, angle, width) {
					var textStyle, layerCache, styleCache, info;
					// Cast the value to a string, in case we were given a number or such
					text = "" + text;
					// If the font is a font-spec object, generate a CSS font definition
					if (typeof font === "object") {
						textStyle = font.style + " " + font.variant + " " + font.weight + " " + font.size + "px/" + font.lineHeight + "px " + font.family;
					} else {
						textStyle = font;
					}
					// Retrieve (or create) the cache for the text's layer and styles
					layerCache = this._textCache[layer];
					if (layerCache == null) {
						layerCache = this._textCache[layer] = {};
					}
					styleCache = layerCache[textStyle];
					if (styleCache == null) {
						styleCache = layerCache[textStyle] = {};
					}
					info = styleCache[text];
					// If we can't find a matching element in our cache, create a new one
					if (info == null) {
						var element = $("
").html(text)
							.css({
								position: "absolute",
								'max-width': width,
								top: -9999
							})
							.appendTo(this.getTextLayer(layer));
						if (typeof font === "object") {
							element.css({
								font: textStyle,
								color: font.color
							});
						} else if (typeof font === "string") {
							element.addClass(font);
						}
						info = styleCache[text] = {
							width: element.outerWidth(true),
							height: element.outerHeight(true),
							element: element,
							positions: []
						};
						element.detach();
					}
					return info;
				};
				// Adds a text string to the canvas text overlay.
				//
				// The text isn't drawn immediately; it is marked as rendering, which will
				// result in its addition to the canvas on the next render pass.
				//
				// @param {string} layer A string of space-separated CSS classes uniquely
				//     identifying the layer containing this text.
				// @param {number} x X coordinate at which to draw the text.
				// @param {number} y Y coordinate at which to draw the text.
				// @param {string} text Text string to draw.
				// @param {(string|object)=} font Either a string of space-separated CSS
				//     classes or a font-spec object, defining the text's font and style.
				// @param {number=} angle Angle at which to rotate the text, in degrees.
				//     Angle is currently unused, it will be implemented in the future.
				// @param {number=} width Maximum width of the text before it wraps.
				// @param {string=} halign Horizontal alignment of the text; either "left",
				//     "center" or "right".
				// @param {string=} valign Vertical alignment of the text; either "top",
				//     "middle" or "bottom".
				Canvas.prototype.addText = function(layer, x, y, text, font, angle, width, halign, valign) {
					var info = this.getTextInfo(layer, text, font, angle, width),
						positions = info.positions;
					// Tweak the div's position to match the text's alignment
					if (halign == "center") {
						x -= info.width / 2;
					} else if (halign == "right") {
						x -= info.width;
					}
					if (valign == "middle") {
						y -= info.height / 2;
					} else if (valign == "bottom") {
						y -= info.height;
					}
					// Determine whether this text already exists at this position.
					// If so, mark it for inclusion in the next render pass.
					for (var i = 0, position; position = positions[i]; i++) {
						if (position.x == x && position.y == y) {
							position.active = true;
							return;
						}
					}
					// If the text doesn't exist at this position, create a new entry
					// For the very first position we'll re-use the original element,
					// while for subsequent ones we'll clone it.
					position = {
						active: true,
						rendered: false,
						element: positions.length ? info.element.clone() : info.element,
						x: x,
						y: y
					};
					positions.push(position);
					// Move the element to its final position within the container
					position.element.css({
						top: Math.round(y),
						left: Math.round(x),
						'text-align': halign	// In case the text wraps
					});
				};
				// Removes one or more text strings from the canvas text overlay.
				//
				// If no parameters are given, all text within the layer is removed.
				//
				// Note that the text is not immediately removed; it is simply marked as
				// inactive, which will result in its removal on the next render pass.
				// This avoids the performance penalty for 'clear and redraw' behavior,
				// where we potentially get rid of all text on a layer, but will likely
				// add back most or all of it later, as when redrawing axes, for example.
				//
				// @param {string} layer A string of space-separated CSS classes uniquely
				//     identifying the layer containing this text.
				// @param {number=} x X coordinate of the text.
				// @param {number=} y Y coordinate of the text.
				// @param {string=} text Text string to remove.
				// @param {(string|object)=} font Either a string of space-separated CSS
				//     classes or a font-spec object, defining the text's font and style.
				// @param {number=} angle Angle at which the text is rotated, in degrees.
				//     Angle is currently unused, it will be implemented in the future.
				Canvas.prototype.removeText = function(layer, x, y, text, font, angle) {
					if (text == null) {
						var layerCache = this._textCache[layer];
						if (layerCache != null) {
							for (var styleKey in layerCache) {
								if (hasOwnProperty.call(layerCache, styleKey)) {
									var styleCache = layerCache[styleKey];
									for (var key in styleCache) {
										if (hasOwnProperty.call(styleCache, key)) {
											var positions = styleCache[key].positions;
											for (var i = 0, position; position = positions[i]; i++) {
												position.active = false;
											}
										}
									}
								}
							}
						}
					} else {
						var positions = this.getTextInfo(layer, text, font, angle).positions;
						for (var i = 0, position; position = positions[i]; i++) {
							if (position.x == x && position.y == y) {
								position.active = false;
							}
						}
					}
				};
				///////////////////////////////////////////////////////////////////////////
				// The top-level container for the entire plot.
				function Plot(placeholder, data_, options_, plugins) {
					// data is on the form:
					//   [ series1, series2 ... ]
					// where series is either just the data as [ [x1, y1], [x2, y2], ... ]
					// or { data: [ [x1, y1], [x2, y2], ... ], label: "some label", ... }
					var series = [],
						options = {
							// the color theme used for graphs
							colors: ["#edc240", "#afd8f8", "#cb4b4b", "#4da74d", "#9440ed"],
							legend: {
								show: true,
								noColumns: 1, // number of colums in legend table
								labelFormatter: null, // fn: string -> string
								labelBoxBorderColor: "#ccc", // border color for the little label boxes
								container: null, // container (as jQuery object) to put legend in, null means default on top of graph
								position: "ne", // position of default legend container within plot
								margin: 5, // distance from grid edge to default legend container within plot
								backgroundColor: null, // null means auto-detect
								backgroundOpacity: 0.85, // set to 0 to avoid background
								sorted: null    // default to no legend sorting
							},
							xaxis: {
								show: null, // null = auto-detect, true = always, false = never
								position: "bottom", // or "top"
								mode: null, // null or "time"
								font: null, // null (derived from CSS in placeholder) or object like { size: 11, lineHeight: 13, style: "italic", weight: "bold", family: "sans-serif", variant: "small-caps" }
								color: null, // base color, labels, ticks
								tickColor: null, // possibly different color of ticks, e.g. "rgba(0,0,0,0.15)"
								transform: null, // null or f: number -> number to transform axis
								inverseTransform: null, // if transform is set, this should be the inverse function
								min: null, // min. value to show, null means set automatically
								max: null, // max. value to show, null means set automatically
								autoscaleMargin: null, // margin in % to add if auto-setting min/max
								ticks: null, // either [1, 3] or [[1, "a"], 3] or (fn: axis info -> ticks) or app. number of ticks for auto-ticks
								tickFormatter: null, // fn: number -> string
								labelWidth: null, // size of tick labels in pixels
								labelHeight: null,
								reserveSpace: null, // whether to reserve space even if axis isn't shown
								tickLength: null, // size in pixels of ticks, or "full" for whole line
								alignTicksWithAxis: null, // axis number or null for no sync
								tickDecimals: null, // no. of decimals, null means auto
								tickSize: null, // number or [number, "unit"]
								minTickSize: null // number or [number, "unit"]
							},
							yaxis: {
								autoscaleMargin: 0.02,
								position: "left" // or "right"
							},
							xaxes: [],
							yaxes: [],
							series: {
								points: {
									show: false,
									radius: 3,
									lineWidth: 2, // in pixels
									fill: true,
									fillColor: "#ffffff",
									symbol: "circle" // or callback
								},
								lines: {
									// we don't put in show: false so we can see
									// whether lines were actively disabled
									lineWidth: 2, // in pixels
									fill: false,
									fillColor: null,
									steps: false
									// Omit 'zero', so we can later default its value to
									// match that of the 'fill' option.
								},
								bars: {
									show: false,
									lineWidth: 2, // in pixels
									barWidth: 1, // in units of the x axis
									fill: true,
									fillColor: null,
									align: "left", // "left", "right", or "center"
									horizontal: false,
									zero: true
								},
								shadowSize: 3,
								highlightColor: null
							},
							grid: {
								show: true,
								aboveData: false,
								color: "#545454", // primary color used for outline and labels
								backgroundColor: null, // null for transparent, else color
								borderColor: null, // set if different from the grid color
								tickColor: null, // color for the ticks, e.g. "rgba(0,0,0,0.15)"
								margin: 0, // distance from the canvas edge to the grid
								labelMargin: 5, // in pixels
								axisMargin: 8, // in pixels
								borderWidth: 2, // in pixels
								minBorderMargin: null, // in pixels, null means taken from points radius
								markings: null, // array of ranges or fn: axes -> array of ranges
								markingsColor: "#f4f4f4",
								markingsLineWidth: 2,
								// interactive stuff
								clickable: false,
								hoverable: false,
								autoHighlight: true, // highlight in case mouse is near
								mouseActiveRadius: 10 // how far the mouse can be away to activate an item
							},
							interaction: {
								redrawOverlayInterval: 1000/60 // time between updates, -1 means in same flow
							},
							hooks: {}
						},
					surface = null,     // the canvas for the plot itself
					overlay = null,     // canvas for interactive stuff on top of plot
					eventHolder = null, // jQuery object that events should be bound to
					ctx = null, octx = null,
					xaxes = [], yaxes = [],
					plotOffset = { left: 0, right: 0, top: 0, bottom: 0},
					plotWidth = 0, plotHeight = 0,
					hooks = {
						processOptions: [],
						processRawData: [],
						processDatapoints: [],
						processOffset: [],
						drawBackground: [],
						drawSeries: [],
						draw: [],
						bindEvents: [],
						drawOverlay: [],
						shutdown: []
					},
					plot = this;
					// public functions
					plot.setData = setData;
					plot.setupGrid = setupGrid;
					plot.draw = draw;
					plot.getPlaceholder = function() { return placeholder; };
					plot.getCanvas = function() { return surface.element; };
					plot.getPlotOffset = function() { return plotOffset; };
					plot.width = function () { return plotWidth; };
					plot.height = function () { return plotHeight; };
					plot.offset = function () {
						var o = eventHolder.offset();
						o.left += plotOffset.left;
						o.top += plotOffset.top;
						return o;
					};
					plot.getData = function () { return series; };
					plot.getAxes = function () {
						var res = {}, i;
						$.each(xaxes.concat(yaxes), function (_, axis) {
							if (axis)
								res[axis.direction + (axis.n != 1 ? axis.n : "") + "axis"] = axis;
						});
						return res;
					};
					plot.getXAxes = function () { return xaxes; };
					plot.getYAxes = function () { return yaxes; };
					plot.c2p = canvasToAxisCoords;
					plot.p2c = axisToCanvasCoords;
					plot.getOptions = function () { return options; };
					plot.highlight = highlight;
					plot.unhighlight = unhighlight;
					plot.triggerRedrawOverlay = triggerRedrawOverlay;
					plot.pointOffset = function(point) {
						return {
							left: parseInt(xaxes[axisNumber(point, "x") - 1].p2c(+point.x) + plotOffset.left, 10),
							top: parseInt(yaxes[axisNumber(point, "y") - 1].p2c(+point.y) + plotOffset.top, 10)
						};
					};
					plot.shutdown = shutdown;
					plot.destroy = function () {
						shutdown();
						placeholder.removeData("plot").empty();
						series = [];
						options = null;
						surface = null;
						overlay = null;
						eventHolder = null;
						ctx = null;
						octx = null;
						xaxes = [];
						yaxes = [];
						hooks = null;
						highlights = [];
						plot = null;
					};
					plot.resize = function () {
						var width = placeholder.width(),
							height = placeholder.height();
						surface.resize(width, height);
						overlay.resize(width, height);
					};
					// public attributes
					plot.hooks = hooks;
					// initialize
					initPlugins(plot);
					parseOptions(options_);
					setupCanvases();
					setData(data_);
					setupGrid();
					draw();
					bindEvents();
					function executeHooks(hook, args) {
						args = [plot].concat(args);
						for (var i = 0; i < hook.length; ++i)
							hook[i].apply(this, args);
					}
					function initPlugins() {
						// References to key classes, allowing plugins to modify them
						var classes = {
							Canvas: Canvas
						};
						for (var i = 0; i < plugins.length; ++i) {
							var p = plugins[i];
							p.init(plot, classes);
							if (p.options)
								$.extend(true, options, p.options);
						}
					}
					function parseOptions(opts) {
						$.extend(true, options, opts);
						// $.extend merges arrays, rather than replacing them.  When less
						// colors are provided than the size of the default palette, we
						// end up with those colors plus the remaining defaults, which is
						// not expected behavior; avoid it by replacing them here.
						if (opts && opts.colors) {
							options.colors = opts.colors;
						}
						if (options.xaxis.color == null)
							options.xaxis.color = $.color.parse(options.grid.color).scale('a', 0.22).toString();
						if (options.yaxis.color == null)
							options.yaxis.color = $.color.parse(options.grid.color).scale('a', 0.22).toString();
						if (options.xaxis.tickColor == null) // grid.tickColor for back-compatibility
							options.xaxis.tickColor = options.grid.tickColor || options.xaxis.color;
						if (options.yaxis.tickColor == null) // grid.tickColor for back-compatibility
							options.yaxis.tickColor = options.grid.tickColor || options.yaxis.color;
						if (options.grid.borderColor == null)
							options.grid.borderColor = options.grid.color;
						if (options.grid.tickColor == null)
							options.grid.tickColor = $.color.parse(options.grid.color).scale('a', 0.22).toString();
						// Fill in defaults for axis options, including any unspecified
						// font-spec fields, if a font-spec was provided.
						// If no x/y axis options were provided, create one of each anyway,
						// since the rest of the code assumes that they exist.
						var i, axisOptions, axisCount,
							fontSize = placeholder.css("font-size"),
							fontSizeDefault = fontSize ? +fontSize.replace("px", "") : 13,
							fontDefaults = {
								style: placeholder.css("font-style"),
								size: Math.round(0.8 * fontSizeDefault),
								variant: placeholder.css("font-variant"),
								weight: placeholder.css("font-weight"),
								family: placeholder.css("font-family")
							};
						axisCount = options.xaxes.length || 1;
						for (i = 0; i < axisCount; ++i) {
							axisOptions = options.xaxes[i];
							if (axisOptions && !axisOptions.tickColor) {
								axisOptions.tickColor = axisOptions.color;
							}
							axisOptions = $.extend(true, {}, options.xaxis, axisOptions);
							options.xaxes[i] = axisOptions;
							if (axisOptions.font) {
								axisOptions.font = $.extend({}, fontDefaults, axisOptions.font);
								if (!axisOptions.font.color) {
									axisOptions.font.color = axisOptions.color;
								}
								if (!axisOptions.font.lineHeight) {
									axisOptions.font.lineHeight = Math.round(axisOptions.font.size * 1.15);
								}
							}
						}
						axisCount = options.yaxes.length || 1;
						for (i = 0; i < axisCount; ++i) {
							axisOptions = options.yaxes[i];
							if (axisOptions && !axisOptions.tickColor) {
								axisOptions.tickColor = axisOptions.color;
							}
							axisOptions = $.extend(true, {}, options.yaxis, axisOptions);
							options.yaxes[i] = axisOptions;
							if (axisOptions.font) {
								axisOptions.font = $.extend({}, fontDefaults, axisOptions.font);
								if (!axisOptions.font.color) {
									axisOptions.font.color = axisOptions.color;
								}
								if (!axisOptions.font.lineHeight) {
									axisOptions.font.lineHeight = Math.round(axisOptions.font.size * 1.15);
								}
							}
						}
						// backwards compatibility, to be removed in future
						if (options.xaxis.noTicks && options.xaxis.ticks == null)
							options.xaxis.ticks = options.xaxis.noTicks;
						if (options.yaxis.noTicks && options.yaxis.ticks == null)
							options.yaxis.ticks = options.yaxis.noTicks;
						if (options.x2axis) {
							options.xaxes[1] = $.extend(true, {}, options.xaxis, options.x2axis);
							options.xaxes[1].position = "top";
							// Override the inherit to allow the axis to auto-scale
							if (options.x2axis.min == null) {
								options.xaxes[1].min = null;
							}
							if (options.x2axis.max == null) {
								options.xaxes[1].max = null;
							}
						}
						if (options.y2axis) {
							options.yaxes[1] = $.extend(true, {}, options.yaxis, options.y2axis);
							options.yaxes[1].position = "right";
							// Override the inherit to allow the axis to auto-scale
							if (options.y2axis.min == null) {
								options.yaxes[1].min = null;
							}
							if (options.y2axis.max == null) {
								options.yaxes[1].max = null;
							}
						}
						if (options.grid.coloredAreas)
							options.grid.markings = options.grid.coloredAreas;
						if (options.grid.coloredAreasColor)
							options.grid.markingsColor = options.grid.coloredAreasColor;
						if (options.lines)
							$.extend(true, options.series.lines, options.lines);
						if (options.points)
							$.extend(true, options.series.points, options.points);
						if (options.bars)
							$.extend(true, options.series.bars, options.bars);
						if (options.shadowSize != null)
							options.series.shadowSize = options.shadowSize;
						if (options.highlightColor != null)
							options.series.highlightColor = options.highlightColor;
						// save options on axes for future reference
						for (i = 0; i < options.xaxes.length; ++i)
							getOrCreateAxis(xaxes, i + 1).options = options.xaxes[i];
						for (i = 0; i < options.yaxes.length; ++i)
							getOrCreateAxis(yaxes, i + 1).options = options.yaxes[i];
						// add hooks from options
						for (var n in hooks)
							if (options.hooks[n] && options.hooks[n].length)
								hooks[n] = hooks[n].concat(options.hooks[n]);
						executeHooks(hooks.processOptions, [options]);
					}
					function setData(d) {
						series = parseData(d);
						fillInSeriesOptions();
						processData();
					}
					function parseData(d) {
						var res = [];
						for (var i = 0; i < d.length; ++i) {
							var s = $.extend(true, {}, options.series);
							if (d[i].data != null) {
								s.data = d[i].data; // move the data instead of deep-copy
								delete d[i].data;
								$.extend(true, s, d[i]);
								d[i].data = s.data;
							}
							else
								s.data = d[i];
							res.push(s);
						}
						return res;
					}
					function axisNumber(obj, coord) {
						var a = obj[coord + "axis"];
						if (typeof a == "object") // if we got a real axis, extract number
							a = a.n;
						if (typeof a != "number")
							a = 1; // default to first axis
						return a;
					}
					function allAxes() {
						// return flat array without annoying null entries
						return $.grep(xaxes.concat(yaxes), function (a) { return a; });
					}
					function canvasToAxisCoords(pos) {
						// return an object with x/y corresponding to all used axes
						var res = {}, i, axis;
						for (i = 0; i < xaxes.length; ++i) {
							axis = xaxes[i];
							if (axis && axis.used)
								res["x" + axis.n] = axis.c2p(pos.left);
						}
						for (i = 0; i < yaxes.length; ++i) {
							axis = yaxes[i];
							if (axis && axis.used)
								res["y" + axis.n] = axis.c2p(pos.top);
						}
						if (res.x1 !== undefined)
							res.x = res.x1;
						if (res.y1 !== undefined)
							res.y = res.y1;
						return res;
					}
					function axisToCanvasCoords(pos) {
						// get canvas coords from the first pair of x/y found in pos
						var res = {}, i, axis, key;
						for (i = 0; i < xaxes.length; ++i) {
							axis = xaxes[i];
							if (axis && axis.used) {
								key = "x" + axis.n;
								if (pos[key] == null && axis.n == 1)
									key = "x";
								if (pos[key] != null) {
									res.left = axis.p2c(pos[key]);
									break;
								}
							}
						}
						for (i = 0; i < yaxes.length; ++i) {
							axis = yaxes[i];
							if (axis && axis.used) {
								key = "y" + axis.n;
								if (pos[key] == null && axis.n == 1)
									key = "y";
								if (pos[key] != null) {
									res.top = axis.p2c(pos[key]);
									break;
								}
							}
						}
						return res;
					}
					function getOrCreateAxis(axes, number) {
						if (!axes[number - 1])
							axes[number - 1] = {
								n: number, // save the number for future reference
								direction: axes == xaxes ? "x" : "y",
								options: $.extend(true, {}, axes == xaxes ? options.xaxis : options.yaxis)
							};
						return axes[number - 1];
					}
					function fillInSeriesOptions() {
						var neededColors = series.length, maxIndex = -1, i;
						// Subtract the number of series that already have fixed colors or
						// color indexes from the number that we still need to generate.
						for (i = 0; i < series.length; ++i) {
							var sc = series[i].color;
							if (sc != null) {
								neededColors--;
								if (typeof sc == "number" && sc > maxIndex) {
									maxIndex = sc;
								}
							}
						}
						// If any of the series have fixed color indexes, then we need to
						// generate at least as many colors as the highest index.
						if (neededColors <= maxIndex) {
							neededColors = maxIndex + 1;
						}
						// Generate all the colors, using first the option colors and then
						// variations on those colors once they're exhausted.
						var c, colors = [], colorPool = options.colors,
							colorPoolSize = colorPool.length, variation = 0;
						for (i = 0; i < neededColors; i++) {
							c = $.color.parse(colorPool[i % colorPoolSize] || "#666");
							// Each time we exhaust the colors in the pool we adjust
							// a scaling factor used to produce more variations on
							// those colors. The factor alternates negative/positive
							// to produce lighter/darker colors.
							// Reset the variation after every few cycles, or else
							// it will end up producing only white or black colors.
							if (i % colorPoolSize == 0 && i) {
								if (variation >= 0) {
									if (variation < 0.5) {
										variation = -variation - 0.2;
									} else variation = 0;
								} else variation = -variation;
							}
							colors[i] = c.scale('rgb', 1 + variation);
						}
						// Finalize the series options, filling in their colors
						var colori = 0, s;
						for (i = 0; i < series.length; ++i) {
							s = series[i];
							// assign colors
							if (s.color == null) {
								s.color = colors[colori].toString();
								++colori;
							}
							else if (typeof s.color == "number")
								s.color = colors[s.color].toString();
							// turn on lines automatically in case nothing is set
							if (s.lines.show == null) {
								var v, show = true;
								for (v in s)
									if (s[v] && s[v].show) {
										show = false;
										break;
									}
								if (show)
									s.lines.show = true;
							}
							// If nothing was provided for lines.zero, default it to match
							// lines.fill, since areas by default should extend to zero.
							if (s.lines.zero == null) {
								s.lines.zero = !!s.lines.fill;
							}
							// setup axes
							s.xaxis = getOrCreateAxis(xaxes, axisNumber(s, "x"));
							s.yaxis = getOrCreateAxis(yaxes, axisNumber(s, "y"));
						}
					}
					function processData() {
						var topSentry = Number.POSITIVE_INFINITY,
							bottomSentry = Number.NEGATIVE_INFINITY,
							fakeInfinity = Number.MAX_VALUE,
							i, j, k, m, length,
							s, points, ps, x, y, axis, val, f, p,
							data, format;
						function updateAxis(axis, min, max) {
							if (min < axis.datamin && min != -fakeInfinity)
								axis.datamin = min;
							if (max > axis.datamax && max != fakeInfinity)
								axis.datamax = max;
						}
						$.each(allAxes(), function (_, axis) {
							// init axis
							axis.datamin = topSentry;
							axis.datamax = bottomSentry;
							axis.used = false;
						});
						for (i = 0; i < series.length; ++i) {
							s = series[i];
							s.datapoints = { points: [] };
							executeHooks(hooks.processRawData, [ s, s.data, s.datapoints ]);
						}
						// first pass: clean and copy data
						for (i = 0; i < series.length; ++i) {
							s = series[i];
							data = s.data;
							format = s.datapoints.format;
							if (!format) {
								format = [];
								// find out how to copy
								format.push({ x: true, number: true, required: true });
								format.push({ y: true, number: true, required: true });
								if (s.bars.show || (s.lines.show && s.lines.fill)) {
									var autoscale = !!((s.bars.show && s.bars.zero) || (s.lines.show && s.lines.zero));
									format.push({ y: true, number: true, required: false, defaultValue: 0, autoscale: autoscale });
									if (s.bars.horizontal) {
										delete format[format.length - 1].y;
										format[format.length - 1].x = true;
									}
								}
								s.datapoints.format = format;
							}
							if (s.datapoints.pointsize != null)
								continue; // already filled in
							s.datapoints.pointsize = format.length;
							ps = s.datapoints.pointsize;
							points = s.datapoints.points;
							var insertSteps = s.lines.show && s.lines.steps;
							s.xaxis.used = s.yaxis.used = true;
							for (j = k = 0; j < data.length; ++j, k += ps) {
								p = data[j];
								var nullify = p == null;
								if (!nullify) {
									for (m = 0; m < ps; ++m) {
										val = p[m];
										f = format[m];
										if (f) {
											if (f.number && val != null) {
												val = +val; // convert to number
												if (isNaN(val))
													val = null;
												else if (val == Infinity)
													val = fakeInfinity;
												else if (val == -Infinity)
													val = -fakeInfinity;
											}
											if (val == null) {
												if (f.required)
													nullify = true;
												if (f.defaultValue != null)
													val = f.defaultValue;
											}
										}
										points[k + m] = val;
									}
								}
								if (nullify) {
									for (m = 0; m < ps; ++m) {
										val = points[k + m];
										if (val != null) {
											f = format[m];
											// extract min/max info
											if (f.autoscale !== false) {
												if (f.x) {
													updateAxis(s.xaxis, val, val);
												}
												if (f.y) {
													updateAxis(s.yaxis, val, val);
												}
											}
										}
										points[k + m] = null;
									}
								}
								else {
									// a little bit of line specific stuff that
									// perhaps shouldn't be here, but lacking
									// better means...
									if (insertSteps && k > 0
										&& points[k - ps] != null
										&& points[k - ps] != points[k]
										&& points[k - ps + 1] != points[k + 1]) {
										// copy the point to make room for a middle point
										for (m = 0; m < ps; ++m)
											points[k + ps + m] = points[k + m];
										// middle point has same y
										points[k + 1] = points[k - ps + 1];
										// we've added a point, better reflect that
										k += ps;
									}
								}
							}
						}
						// give the hooks a chance to run
						for (i = 0; i < series.length; ++i) {
							s = series[i];
							executeHooks(hooks.processDatapoints, [ s, s.datapoints]);
						}
						// second pass: find datamax/datamin for auto-scaling
						for (i = 0; i < series.length; ++i) {
							s = series[i];
							points = s.datapoints.points;
							ps = s.datapoints.pointsize;
							format = s.datapoints.format;
							var xmin = topSentry, ymin = topSentry,
								xmax = bottomSentry, ymax = bottomSentry;
							for (j = 0; j < points.length; j += ps) {
								if (points[j] == null)
									continue;
								for (m = 0; m < ps; ++m) {
									val = points[j + m];
									f = format[m];
									if (!f || f.autoscale === false || val == fakeInfinity || val == -fakeInfinity)
										continue;
									if (f.x) {
										if (val < xmin)
											xmin = val;
										if (val > xmax)
											xmax = val;
									}
									if (f.y) {
										if (val < ymin)
											ymin = val;
										if (val > ymax)
											ymax = val;
									}
								}
							}
							if (s.bars.show) {
								// make sure we got room for the bar on the dancing floor
								var delta;
								switch (s.bars.align) {
									case "left":
										delta = 0;
										break;
									case "right":
										delta = -s.bars.barWidth;
										break;
									default:
										delta = -s.bars.barWidth / 2;
								}
								if (s.bars.horizontal) {
									ymin += delta;
									ymax += delta + s.bars.barWidth;
								}
								else {
									xmin += delta;
									xmax += delta + s.bars.barWidth;
								}
							}
							updateAxis(s.xaxis, xmin, xmax);
							updateAxis(s.yaxis, ymin, ymax);
						}
						$.each(allAxes(), function (_, axis) {
							if (axis.datamin == topSentry)
								axis.datamin = null;
							if (axis.datamax == bottomSentry)
								axis.datamax = null;
						});
					}
					function setupCanvases() {
						// Make sure the placeholder is clear of everything except canvases
						// from a previous plot in this container that we'll try to re-use.
						placeholder.css("padding", 0) // padding messes up the positioning
							.children().filter(function(){
								return !$(this).hasClass("flot-overlay") && !$(this).hasClass('flot-base');
							}).remove();
						if (placeholder.css("position") == 'static')
							placeholder.css("position", "relative"); // for positioning labels and overlay
						surface = new Canvas("flot-base", placeholder);
						overlay = new Canvas("flot-overlay", placeholder); // overlay canvas for interactive features
						ctx = surface.context;
						octx = overlay.context;
						// define which element we're listening for events on
						eventHolder = $(overlay.element).unbind();
						// If we're re-using a plot object, shut down the old one
						var existing = placeholder.data("plot");
						if (existing) {
							existing.shutdown();
							overlay.clear();
						}
						// save in case we get replotted
						placeholder.data("plot", plot);
					}
					function bindEvents() {
						// bind events
						if (options.grid.hoverable) {
							eventHolder.mousemove(onMouseMove);
							// Use bind, rather than .mouseleave, because we officially
							// still support jQuery 1.2.6, which doesn't define a shortcut
							// for mouseenter or mouseleave.  This was a bug/oversight that
							// was fixed somewhere around 1.3.x.  We can return to using
							// .mouseleave when we drop support for 1.2.6.
							eventHolder.bind("mouseleave", onMouseLeave);
						}
						if (options.grid.clickable)
							eventHolder.click(onClick);
						executeHooks(hooks.bindEvents, [eventHolder]);
					}
					function shutdown() {
						if (redrawTimeout)
							clearTimeout(redrawTimeout);
						eventHolder.unbind("mousemove", onMouseMove);
						eventHolder.unbind("mouseleave", onMouseLeave);
						eventHolder.unbind("click", onClick);
						executeHooks(hooks.shutdown, [eventHolder]);
					}
					function setTransformationHelpers(axis) {
						// set helper functions on the axis, assumes plot area
						// has been computed already
						function identity(x) { return x; }
						var s, m, t = axis.options.transform || identity,
							it = axis.options.inverseTransform;
						// precompute how much the axis is scaling a point
						// in canvas space
						if (axis.direction == "x") {
							s = axis.scale = plotWidth / Math.abs(t(axis.max) - t(axis.min));
							m = Math.min(t(axis.max), t(axis.min));
						}
						else {
							s = axis.scale = plotHeight / Math.abs(t(axis.max) - t(axis.min));
							s = -s;
							m = Math.max(t(axis.max), t(axis.min));
						}
						// data point to canvas coordinate
						if (t == identity) // slight optimization
							axis.p2c = function (p) { return (p - m) * s; };
						else
							axis.p2c = function (p) { return (t(p) - m) * s; };
						// canvas coordinate to data point
						if (!it)
							axis.c2p = function (c) { return m + c / s; };
						else
							axis.c2p = function (c) { return it(m + c / s); };
					}
					function measureTickLabels(axis) {
						var opts = axis.options,
							ticks = axis.ticks || [],
							labelWidth = opts.labelWidth || 0,
							labelHeight = opts.labelHeight || 0,
							maxWidth = labelWidth || (axis.direction == "x" ? Math.floor(surface.width / (ticks.length || 1)) : null),
							legacyStyles = axis.direction + "Axis " + axis.direction + axis.n + "Axis",
							layer = "flot-" + axis.direction + "-axis flot-" + axis.direction + axis.n + "-axis " + legacyStyles,
							font = opts.font || "flot-tick-label tickLabel";
						for (var i = 0; i < ticks.length; ++i) {
							var t = ticks[i];
							if (!t.label)
								continue;
							var info = surface.getTextInfo(layer, t.label, font, null, maxWidth);
							labelWidth = Math.max(labelWidth, info.width);
							labelHeight = Math.max(labelHeight, info.height);
						}
						axis.labelWidth = opts.labelWidth || labelWidth;
						axis.labelHeight = opts.labelHeight || labelHeight;
					}
					function allocateAxisBoxFirstPhase(axis) {
						// find the bounding box of the axis by looking at label
						// widths/heights and ticks, make room by diminishing the
						// plotOffset; this first phase only looks at one
						// dimension per axis, the other dimension depends on the
						// other axes so will have to wait
						var lw = axis.labelWidth,
							lh = axis.labelHeight,
							pos = axis.options.position,
							isXAxis = axis.direction === "x",
							tickLength = axis.options.tickLength,
							axisMargin = options.grid.axisMargin,
							padding = options.grid.labelMargin,
							innermost = true,
							outermost = true,
							first = true,
							found = false;
						// Determine the axis's position in its direction and on its side
						$.each(isXAxis ? xaxes : yaxes, function(i, a) {
							if (a && (a.show || a.reserveSpace)) {
								if (a === axis) {
									found = true;
								} else if (a.options.position === pos) {
									if (found) {
										outermost = false;
									} else {
										innermost = false;
									}
								}
								if (!found) {
									first = false;
								}
							}
						});
						// The outermost axis on each side has no margin
						if (outermost) {
							axisMargin = 0;
						}
						// The ticks for the first axis in each direction stretch across
						if (tickLength == null) {
							tickLength = first ? "full" : 5;
						}
						if (!isNaN(+tickLength))
							padding += +tickLength;
						if (isXAxis) {
							lh += padding;
							if (pos == "bottom") {
								plotOffset.bottom += lh + axisMargin;
								axis.box = { top: surface.height - plotOffset.bottom, height: lh };
							}
							else {
								axis.box = { top: plotOffset.top + axisMargin, height: lh };
								plotOffset.top += lh + axisMargin;
							}
						}
						else {
							lw += padding;
							if (pos == "left") {
								axis.box = { left: plotOffset.left + axisMargin, width: lw };
								plotOffset.left += lw + axisMargin;
							}
							else {
								plotOffset.right += lw + axisMargin;
								axis.box = { left: surface.width - plotOffset.right, width: lw };
							}
						}
						 // save for future reference
						axis.position = pos;
						axis.tickLength = tickLength;
						axis.box.padding = padding;
						axis.innermost = innermost;
					}
					function allocateAxisBoxSecondPhase(axis) {
						// now that all axis boxes have been placed in one
						// dimension, we can set the remaining dimension coordinates
						if (axis.direction == "x") {
							axis.box.left = plotOffset.left - axis.labelWidth / 2;
							axis.box.width = surface.width - plotOffset.left - plotOffset.right + axis.labelWidth;
						}
						else {
							axis.box.top = plotOffset.top - axis.labelHeight / 2;
							axis.box.height = surface.height - plotOffset.bottom - plotOffset.top + axis.labelHeight;
						}
					}
					function adjustLayoutForThingsStickingOut() {
						// possibly adjust plot offset to ensure everything stays
						// inside the canvas and isn't clipped off
						var minMargin = options.grid.minBorderMargin,
							axis, i;
						// check stuff from the plot (FIXME: this should just read
						// a value from the series, otherwise it's impossible to
						// customize)
						if (minMargin == null) {
							minMargin = 0;
							for (i = 0; i < series.length; ++i)
								minMargin = Math.max(minMargin, 2 * (series[i].points.radius + series[i].points.lineWidth/2));
						}
						var margins = {
							left: minMargin,
							right: minMargin,
							top: minMargin,
							bottom: minMargin
						};
						// check axis labels, note we don't check the actual
						// labels but instead use the overall width/height to not
						// jump as much around with replots
						$.each(allAxes(), function (_, axis) {
							if (axis.reserveSpace && axis.ticks && axis.ticks.length) {
								if (axis.direction === "x") {
									margins.left = Math.max(margins.left, axis.labelWidth / 2);
									margins.right = Math.max(margins.right, axis.labelWidth / 2);
								} else {
									margins.bottom = Math.max(margins.bottom, axis.labelHeight / 2);
									margins.top = Math.max(margins.top, axis.labelHeight / 2);
								}
							}
						});
						plotOffset.left = Math.ceil(Math.max(margins.left, plotOffset.left));
						plotOffset.right = Math.ceil(Math.max(margins.right, plotOffset.right));
						plotOffset.top = Math.ceil(Math.max(margins.top, plotOffset.top));
						plotOffset.bottom = Math.ceil(Math.max(margins.bottom, plotOffset.bottom));
					}
					function setupGrid() {
						var i, axes = allAxes(), showGrid = options.grid.show;
						// Initialize the plot's offset from the edge of the canvas
						for (var a in plotOffset) {
							var margin = options.grid.margin || 0;
							plotOffset[a] = typeof margin == "number" ? margin : margin[a] || 0;
						}
						executeHooks(hooks.processOffset, [plotOffset]);
						// If the grid is visible, add its border width to the offset
						for (var a in plotOffset) {
							if(typeof(options.grid.borderWidth) == "object") {
								plotOffset[a] += showGrid ? options.grid.borderWidth[a] : 0;
							}
							else {
								plotOffset[a] += showGrid ? options.grid.borderWidth : 0;
							}
						}
						$.each(axes, function (_, axis) {
							var axisOpts = axis.options;
							axis.show = axisOpts.show == null ? axis.used : axisOpts.show;
							axis.reserveSpace = axisOpts.reserveSpace == null ? axis.show : axisOpts.reserveSpace;
							setRange(axis);
						});
						if (showGrid) {
							var allocatedAxes = $.grep(axes, function (axis) {
								return axis.show || axis.reserveSpace;
							});
							$.each(allocatedAxes, function (_, axis) {
								// make the ticks
								setupTickGeneration(axis);
								setTicks(axis);
								snapRangeToTicks(axis, axis.ticks);
								// find labelWidth/Height for axis
								measureTickLabels(axis);
							});
							// with all dimensions calculated, we can compute the
							// axis bounding boxes, start from the outside
							// (reverse order)
							for (i = allocatedAxes.length - 1; i >= 0; --i)
								allocateAxisBoxFirstPhase(allocatedAxes[i]);
							// make sure we've got enough space for things that
							// might stick out
							adjustLayoutForThingsStickingOut();
							$.each(allocatedAxes, function (_, axis) {
								allocateAxisBoxSecondPhase(axis);
							});
						}
						plotWidth = surface.width - plotOffset.left - plotOffset.right;
						plotHeight = surface.height - plotOffset.bottom - plotOffset.top;
						// now we got the proper plot dimensions, we can compute the scaling
						$.each(axes, function (_, axis) {
							setTransformationHelpers(axis);
						});
						if (showGrid) {
							drawAxisLabels();
						}
						insertLegend();
					}
					function setRange(axis) {
						var opts = axis.options,
							min = +(opts.min != null ? opts.min : axis.datamin),
							max = +(opts.max != null ? opts.max : axis.datamax),
							delta = max - min;
						if (delta == 0.0) {
							// degenerate case
							var widen = max == 0 ? 1 : 0.01;
							if (opts.min == null)
								min -= widen;
							// always widen max if we couldn't widen min to ensure we
							// don't fall into min == max which doesn't work
							if (opts.max == null || opts.min != null)
								max += widen;
						}
						else {
							// consider autoscaling
							var margin = opts.autoscaleMargin;
							if (margin != null) {
								if (opts.min == null) {
									min -= delta * margin;
									// make sure we don't go below zero if all values
									// are positive
									if (min < 0 && axis.datamin != null && axis.datamin >= 0)
										min = 0;
								}
								if (opts.max == null) {
									max += delta * margin;
									if (max > 0 && axis.datamax != null && axis.datamax <= 0)
										max = 0;
								}
							}
						}
						axis.min = min;
						axis.max = max;
					}
					function setupTickGeneration(axis) {
						var opts = axis.options;
						// estimate number of ticks
						var noTicks;
						if (typeof opts.ticks == "number" && opts.ticks > 0)
							noTicks = opts.ticks;
						else
							// heuristic based on the model a*sqrt(x) fitted to
							// some data points that seemed reasonable
							noTicks = 0.3 * Math.sqrt(axis.direction == "x" ? surface.width : surface.height);
						var delta = (axis.max - axis.min) / noTicks,
							dec = -Math.floor(Math.log(delta) / Math.LN10),
							maxDec = opts.tickDecimals;
						if (maxDec != null && dec > maxDec) {
							dec = maxDec;
						}
						var magn = Math.pow(10, -dec),
							norm = delta / magn, // norm is between 1.0 and 10.0
							size;
						if (norm < 1.5) {
							size = 1;
						} else if (norm < 3) {
							size = 2;
							// special case for 2.5, requires an extra decimal
							if (norm > 2.25 && (maxDec == null || dec + 1 <= maxDec)) {
								size = 2.5;
								++dec;
							}
						} else if (norm < 7.5) {
							size = 5;
						} else {
							size = 10;
						}
						size *= magn;
						if (opts.minTickSize != null && size < opts.minTickSize) {
							size = opts.minTickSize;
						}
						axis.delta = delta;
						axis.tickDecimals = Math.max(0, maxDec != null ? maxDec : dec);
						axis.tickSize = opts.tickSize || size;
						// Time mode was moved to a plug-in in 0.8, and since so many people use it
						// we'll add an especially friendly reminder to make sure they included it.
						if (opts.mode == "time" && !axis.tickGenerator) {
							throw new Error("Time mode requires the flot.time plugin.");
						}
						// Flot supports base-10 axes; any other mode else is handled by a plug-in,
						// like flot.time.js.
						if (!axis.tickGenerator) {
							axis.tickGenerator = function (axis) {
								var ticks = [],
									start = floorInBase(axis.min, axis.tickSize),
									i = 0,
									v = Number.NaN,
									prev;
								do {
									prev = v;
									v = start + i * axis.tickSize;
									ticks.push(v);
									++i;
								} while (v < axis.max && v != prev);
								return ticks;
							};
							axis.tickFormatter = function (value, axis) {
								var factor = axis.tickDecimals ? Math.pow(10, axis.tickDecimals) : 1;
								var formatted = "" + Math.round(value * factor) / factor;
								// If tickDecimals was specified, ensure that we have exactly that
								// much precision; otherwise default to the value's own precision.
								if (axis.tickDecimals != null) {
									var decimal = formatted.indexOf(".");
									var precision = decimal == -1 ? 0 : formatted.length - decimal - 1;
									if (precision < axis.tickDecimals) {
										return (precision ? formatted : formatted + ".") + ("" + factor).substr(1, axis.tickDecimals - precision);
									}
								}
								return formatted;
							};
						}
						if ($.isFunction(opts.tickFormatter))
							axis.tickFormatter = function (v, axis) { return "" + opts.tickFormatter(v, axis); };
						if (opts.alignTicksWithAxis != null) {
							var otherAxis = (axis.direction == "x" ? xaxes : yaxes)[opts.alignTicksWithAxis - 1];
							if (otherAxis && otherAxis.used && otherAxis != axis) {
								// consider snapping min/max to outermost nice ticks
								var niceTicks = axis.tickGenerator(axis);
								if (niceTicks.length > 0) {
									if (opts.min == null)
										axis.min = Math.min(axis.min, niceTicks[0]);
									if (opts.max == null && niceTicks.length > 1)
										axis.max = Math.max(axis.max, niceTicks[niceTicks.length - 1]);
								}
								axis.tickGenerator = function (axis) {
									// copy ticks, scaled to this axis
									var ticks = [], v, i;
									for (i = 0; i < otherAxis.ticks.length; ++i) {
										v = (otherAxis.ticks[i].v - otherAxis.min) / (otherAxis.max - otherAxis.min);
										v = axis.min + v * (axis.max - axis.min);
										ticks.push(v);
									}
									return ticks;
								};
								// we might need an extra decimal since forced
								// ticks don't necessarily fit naturally
								if (!axis.mode && opts.tickDecimals == null) {
									var extraDec = Math.max(0, -Math.floor(Math.log(axis.delta) / Math.LN10) + 1),
										ts = axis.tickGenerator(axis);
									// only proceed if the tick interval rounded
									// with an extra decimal doesn't give us a
									// zero at end
									if (!(ts.length > 1 && /\..*0$/.test((ts[1] - ts[0]).toFixed(extraDec))))
										axis.tickDecimals = extraDec;
								}
							}
						}
					}
					function setTicks(axis) {
						var oticks = axis.options.ticks, ticks = [];
						if (oticks == null || (typeof oticks == "number" && oticks > 0))
							ticks = axis.tickGenerator(axis);
						else if (oticks) {
							if ($.isFunction(oticks))
								// generate the ticks
								ticks = oticks(axis);
							else
								ticks = oticks;
						}
						// clean up/labelify the supplied ticks, copy them over
						var i, v;
						axis.ticks = [];
						for (i = 0; i < ticks.length; ++i) {
							var label = null;
							var t = ticks[i];
							if (typeof t == "object") {
								v = +t[0];
								if (t.length > 1)
									label = t[1];
							}
							else
								v = +t;
							if (label == null)
								label = axis.tickFormatter(v, axis);
							if (!isNaN(v))
								axis.ticks.push({ v: v, label: label });
						}
					}
					function snapRangeToTicks(axis, ticks) {
						if (axis.options.autoscaleMargin && ticks.length > 0) {
							// snap to ticks
							if (axis.options.min == null)
								axis.min = Math.min(axis.min, ticks[0].v);
							if (axis.options.max == null && ticks.length > 1)
								axis.max = Math.max(axis.max, ticks[ticks.length - 1].v);
						}
					}
					function draw() {
						surface.clear();
						executeHooks(hooks.drawBackground, [ctx]);
						var grid = options.grid;
						// draw background, if any
						if (grid.show && grid.backgroundColor)
							drawBackground();
						if (grid.show && !grid.aboveData) {
							drawGrid();
						}
						for (var i = 0; i < series.length; ++i) {
							executeHooks(hooks.drawSeries, [ctx, series[i]]);
							drawSeries(series[i]);
						}
						executeHooks(hooks.draw, [ctx]);
						if (grid.show && grid.aboveData) {
							drawGrid();
						}
						surface.render();
						// A draw implies that either the axes or data have changed, so we
						// should probably update the overlay highlights as well.
						triggerRedrawOverlay();
					}
					function extractRange(ranges, coord) {
						var axis, from, to, key, axes = allAxes();
						for (var i = 0; i < axes.length; ++i) {
							axis = axes[i];
							if (axis.direction == coord) {
								key = coord + axis.n + "axis";
								if (!ranges[key] && axis.n == 1)
									key = coord + "axis"; // support x1axis as xaxis
								if (ranges[key]) {
									from = ranges[key].from;
									to = ranges[key].to;
									break;
								}
							}
						}
						// backwards-compat stuff - to be removed in future
						if (!ranges[key]) {
							axis = coord == "x" ? xaxes[0] : yaxes[0];
							from = ranges[coord + "1"];
							to = ranges[coord + "2"];
						}
						// auto-reverse as an added bonus
						if (from != null && to != null && from > to) {
							var tmp = from;
							from = to;
							to = tmp;
						}
						return { from: from, to: to, axis: axis };
					}
					function drawBackground() {
						ctx.save();
						ctx.translate(plotOffset.left, plotOffset.top);
						ctx.fillStyle = getColorOrGradient(options.grid.backgroundColor, plotHeight, 0, "rgba(255, 255, 255, 0)");
						ctx.fillRect(0, 0, plotWidth, plotHeight);
						ctx.restore();
					}
					function drawGrid() {
						var i, axes, bw, bc;
						ctx.save();
						ctx.translate(plotOffset.left, plotOffset.top);
						// draw markings
						var markings = options.grid.markings;
						if (markings) {
							if ($.isFunction(markings)) {
								axes = plot.getAxes();
								// xmin etc. is backwards compatibility, to be
								// removed in the future
								axes.xmin = axes.xaxis.min;
								axes.xmax = axes.xaxis.max;
								axes.ymin = axes.yaxis.min;
								axes.ymax = axes.yaxis.max;
								markings = markings(axes);
							}
							for (i = 0; i < markings.length; ++i) {
								var m = markings[i],
									xrange = extractRange(m, "x"),
									yrange = extractRange(m, "y");
								// fill in missing
								if (xrange.from == null)
									xrange.from = xrange.axis.min;
								if (xrange.to == null)
									xrange.to = xrange.axis.max;
								if (yrange.from == null)
									yrange.from = yrange.axis.min;
								if (yrange.to == null)
									yrange.to = yrange.axis.max;
								// clip
								if (xrange.to < xrange.axis.min || xrange.from > xrange.axis.max ||
									yrange.to < yrange.axis.min || yrange.from > yrange.axis.max)
									continue;
								xrange.from = Math.max(xrange.from, xrange.axis.min);
								xrange.to = Math.min(xrange.to, xrange.axis.max);
								yrange.from = Math.max(yrange.from, yrange.axis.min);
								yrange.to = Math.min(yrange.to, yrange.axis.max);
								var xequal = xrange.from === xrange.to,
									yequal = yrange.from === yrange.to;
								if (xequal && yequal) {
									continue;
								}
								// then draw
								xrange.from = Math.floor(xrange.axis.p2c(xrange.from));
								xrange.to = Math.floor(xrange.axis.p2c(xrange.to));
								yrange.from = Math.floor(yrange.axis.p2c(yrange.from));
								yrange.to = Math.floor(yrange.axis.p2c(yrange.to));
								if (xequal || yequal) {
									var lineWidth = m.lineWidth || options.grid.markingsLineWidth,
										subPixel = lineWidth % 2 ? 0.5 : 0;
									ctx.beginPath();
									ctx.strokeStyle = m.color || options.grid.markingsColor;
									ctx.lineWidth = lineWidth;
									if (xequal) {
										ctx.moveTo(xrange.to + subPixel, yrange.from);
										ctx.lineTo(xrange.to + subPixel, yrange.to);
									} else {
										ctx.moveTo(xrange.from, yrange.to + subPixel);
										ctx.lineTo(xrange.to, yrange.to + subPixel);                            
									}
									ctx.stroke();
								} else {
									ctx.fillStyle = m.color || options.grid.markingsColor;
									ctx.fillRect(xrange.from, yrange.to,
												 xrange.to - xrange.from,
												 yrange.from - yrange.to);
								}
							}
						}
						// draw the ticks
						axes = allAxes();
						bw = options.grid.borderWidth;
						for (var j = 0; j < axes.length; ++j) {
							var axis = axes[j], box = axis.box,
								t = axis.tickLength, x, y, xoff, yoff;
							if (!axis.show || axis.ticks.length == 0)
								continue;
							ctx.lineWidth = 1;
							// find the edges
							if (axis.direction == "x") {
								x = 0;
								if (t == "full")
									y = (axis.position == "top" ? 0 : plotHeight);
								else
									y = box.top - plotOffset.top + (axis.position == "top" ? box.height : 0);
							}
							else {
								y = 0;
								if (t == "full")
									x = (axis.position == "left" ? 0 : plotWidth);
								else
									x = box.left - plotOffset.left + (axis.position == "left" ? box.width : 0);
							}
							// draw tick bar
							if (!axis.innermost) {
								ctx.strokeStyle = axis.options.color;
								ctx.beginPath();
								xoff = yoff = 0;
								if (axis.direction == "x")
									xoff = plotWidth + 1;
								else
									yoff = plotHeight + 1;
								if (ctx.lineWidth == 1) {
									if (axis.direction == "x") {
										y = Math.floor(y) + 0.5;
									} else {
										x = Math.floor(x) + 0.5;
									}
								}
								ctx.moveTo(x, y);
								ctx.lineTo(x + xoff, y + yoff);
								ctx.stroke();
							}
							// draw ticks
							ctx.strokeStyle = axis.options.tickColor;
							ctx.beginPath();
							for (i = 0; i < axis.ticks.length; ++i) {
								var v = axis.ticks[i].v;
								xoff = yoff = 0;
								if (isNaN(v) || v < axis.min || v > axis.max
									// skip those lying on the axes if we got a border
									|| (t == "full"
										&& ((typeof bw == "object" && bw[axis.position] > 0) || bw > 0)
										&& (v == axis.min || v == axis.max)))
									continue;
								if (axis.direction == "x") {
									x = axis.p2c(v);
									yoff = t == "full" ? -plotHeight : t;
									if (axis.position == "top")
										yoff = -yoff;
								}
								else {
									y = axis.p2c(v);
									xoff = t == "full" ? -plotWidth : t;
									if (axis.position == "left")
										xoff = -xoff;
								}
								if (ctx.lineWidth == 1) {
									if (axis.direction == "x")
										x = Math.floor(x) + 0.5;
									else
										y = Math.floor(y) + 0.5;
								}
								ctx.moveTo(x, y);
								ctx.lineTo(x + xoff, y + yoff);
							}
							ctx.stroke();
						}
						// draw border
						if (bw) {
							// If either borderWidth or borderColor is an object, then draw the border
							// line by line instead of as one rectangle
							bc = options.grid.borderColor;
							if(typeof bw == "object" || typeof bc == "object") {
								if (typeof bw !== "object") {
									bw = {top: bw, right: bw, bottom: bw, left: bw};
								}
								if (typeof bc !== "object") {
									bc = {top: bc, right: bc, bottom: bc, left: bc};
								}
								if (bw.top > 0) {
									ctx.strokeStyle = bc.top;
									ctx.lineWidth = bw.top;
									ctx.beginPath();
									ctx.moveTo(0 - bw.left, 0 - bw.top/2);
									ctx.lineTo(plotWidth, 0 - bw.top/2);
									ctx.stroke();
								}
								if (bw.right > 0) {
									ctx.strokeStyle = bc.right;
									ctx.lineWidth = bw.right;
									ctx.beginPath();
									ctx.moveTo(plotWidth + bw.right / 2, 0 - bw.top);
									ctx.lineTo(plotWidth + bw.right / 2, plotHeight);
									ctx.stroke();
								}
								if (bw.bottom > 0) {
									ctx.strokeStyle = bc.bottom;
									ctx.lineWidth = bw.bottom;
									ctx.beginPath();
									ctx.moveTo(plotWidth + bw.right, plotHeight + bw.bottom / 2);
									ctx.lineTo(0, plotHeight + bw.bottom / 2);
									ctx.stroke();
								}
								if (bw.left > 0) {
									ctx.strokeStyle = bc.left;
									ctx.lineWidth = bw.left;
									ctx.beginPath();
									ctx.moveTo(0 - bw.left/2, plotHeight + bw.bottom);
									ctx.lineTo(0- bw.left/2, 0);
									ctx.stroke();
								}
							}
							else {
								ctx.lineWidth = bw;
								ctx.strokeStyle = options.grid.borderColor;
								ctx.strokeRect(-bw/2, -bw/2, plotWidth + bw, plotHeight + bw);
							}
						}
						ctx.restore();
					}
					function drawAxisLabels() {
						$.each(allAxes(), function (_, axis) {
							var box = axis.box,
								legacyStyles = axis.direction + "Axis " + axis.direction + axis.n + "Axis",
								layer = "flot-" + axis.direction + "-axis flot-" + axis.direction + axis.n + "-axis " + legacyStyles,
								font = axis.options.font || "flot-tick-label tickLabel",
								tick, x, y, halign, valign;
							// Remove text before checking for axis.show and ticks.length;
							// otherwise plugins, like flot-tickrotor, that draw their own
							// tick labels will end up with both theirs and the defaults.
							surface.removeText(layer);
							if (!axis.show || axis.ticks.length == 0)
								return;
							for (var i = 0; i < axis.ticks.length; ++i) {
								tick = axis.ticks[i];
								if (!tick.label || tick.v < axis.min || tick.v > axis.max)
									continue;
								if (axis.direction == "x") {
									halign = "center";
									x = plotOffset.left + axis.p2c(tick.v);
									if (axis.position == "bottom") {
										y = box.top + box.padding;
									} else {
										y = box.top + box.height - box.padding;
										valign = "bottom";
									}
								} else {
									valign = "middle";
									y = plotOffset.top + axis.p2c(tick.v);
									if (axis.position == "left") {
										x = box.left + box.width - box.padding;
										halign = "right";
									} else {
										x = box.left + box.padding;
									}
								}
								surface.addText(layer, x, y, tick.label, font, null, null, halign, valign);
							}
						});
					}
					function drawSeries(series) {
						if (series.lines.show)
							drawSeriesLines(series);
						if (series.bars.show)
							drawSeriesBars(series);
						if (series.points.show)
							drawSeriesPoints(series);
					}
					function drawSeriesLines(series) {
						function plotLine(datapoints, xoffset, yoffset, axisx, axisy) {
							var points = datapoints.points,
								ps = datapoints.pointsize,
								prevx = null, prevy = null;
							ctx.beginPath();
							for (var i = ps; i < points.length; i += ps) {
								var x1 = points[i - ps], y1 = points[i - ps + 1],
									x2 = points[i], y2 = points[i + 1];
								if (x1 == null || x2 == null)
									continue;
								// clip with ymin
								if (y1 <= y2 && y1 < axisy.min) {
									if (y2 < axisy.min)
										continue;   // line segment is outside
									// compute new intersection point
									x1 = (axisy.min - y1) / (y2 - y1) * (x2 - x1) + x1;
									y1 = axisy.min;
								}
								else if (y2 <= y1 && y2 < axisy.min) {
									if (y1 < axisy.min)
										continue;
									x2 = (axisy.min - y1) / (y2 - y1) * (x2 - x1) + x1;
									y2 = axisy.min;
								}
								// clip with ymax
								if (y1 >= y2 && y1 > axisy.max) {
									if (y2 > axisy.max)
										continue;
									x1 = (axisy.max - y1) / (y2 - y1) * (x2 - x1) + x1;
									y1 = axisy.max;
								}
								else if (y2 >= y1 && y2 > axisy.max) {
									if (y1 > axisy.max)
										continue;
									x2 = (axisy.max - y1) / (y2 - y1) * (x2 - x1) + x1;
									y2 = axisy.max;
								}
								// clip with xmin
								if (x1 <= x2 && x1 < axisx.min) {
									if (x2 < axisx.min)
										continue;
									y1 = (axisx.min - x1) / (x2 - x1) * (y2 - y1) + y1;
									x1 = axisx.min;
								}
								else if (x2 <= x1 && x2 < axisx.min) {
									if (x1 < axisx.min)
										continue;
									y2 = (axisx.min - x1) / (x2 - x1) * (y2 - y1) + y1;
									x2 = axisx.min;
								}
								// clip with xmax
								if (x1 >= x2 && x1 > axisx.max) {
									if (x2 > axisx.max)
										continue;
									y1 = (axisx.max - x1) / (x2 - x1) * (y2 - y1) + y1;
									x1 = axisx.max;
								}
								else if (x2 >= x1 && x2 > axisx.max) {
									if (x1 > axisx.max)
										continue;
									y2 = (axisx.max - x1) / (x2 - x1) * (y2 - y1) + y1;
									x2 = axisx.max;
								}
								if (x1 != prevx || y1 != prevy)
									ctx.moveTo(axisx.p2c(x1) + xoffset, axisy.p2c(y1) + yoffset);
								prevx = x2;
								prevy = y2;
								ctx.lineTo(axisx.p2c(x2) + xoffset, axisy.p2c(y2) + yoffset);
							}
							ctx.stroke();
						}
						function plotLineArea(datapoints, axisx, axisy) {
							var points = datapoints.points,
								ps = datapoints.pointsize,
								bottom = Math.min(Math.max(0, axisy.min), axisy.max),
								i = 0, top, areaOpen = false,
								ypos = 1, segmentStart = 0, segmentEnd = 0;
							// we process each segment in two turns, first forward
							// direction to sketch out top, then once we hit the
							// end we go backwards to sketch the bottom
							while (true) {
								if (ps > 0 && i > points.length + ps)
									break;
								i += ps; // ps is negative if going backwards
								var x1 = points[i - ps],
									y1 = points[i - ps + ypos],
									x2 = points[i], y2 = points[i + ypos];
								if (areaOpen) {
									if (ps > 0 && x1 != null && x2 == null) {
										// at turning point
										segmentEnd = i;
										ps = -ps;
										ypos = 2;
										continue;
									}
									if (ps < 0 && i == segmentStart + ps) {
										// done with the reverse sweep
										ctx.fill();
										areaOpen = false;
										ps = -ps;
										ypos = 1;
										i = segmentStart = segmentEnd + ps;
										continue;
									}
								}
								if (x1 == null || x2 == null)
									continue;
								// clip x values
								// clip with xmin
								if (x1 <= x2 && x1 < axisx.min) {
									if (x2 < axisx.min)
										continue;
									y1 = (axisx.min - x1) / (x2 - x1) * (y2 - y1) + y1;
									x1 = axisx.min;
								}
								else if (x2 <= x1 && x2 < axisx.min) {
									if (x1 < axisx.min)
										continue;
									y2 = (axisx.min - x1) / (x2 - x1) * (y2 - y1) + y1;
									x2 = axisx.min;
								}
								// clip with xmax
								if (x1 >= x2 && x1 > axisx.max) {
									if (x2 > axisx.max)
										continue;
									y1 = (axisx.max - x1) / (x2 - x1) * (y2 - y1) + y1;
									x1 = axisx.max;
								}
								else if (x2 >= x1 && x2 > axisx.max) {
									if (x1 > axisx.max)
										continue;
									y2 = (axisx.max - x1) / (x2 - x1) * (y2 - y1) + y1;
									x2 = axisx.max;
								}
								if (!areaOpen) {
									// open area
									ctx.beginPath();
									ctx.moveTo(axisx.p2c(x1), axisy.p2c(bottom));
									areaOpen = true;
								}
								// now first check the case where both is outside
								if (y1 >= axisy.max && y2 >= axisy.max) {
									ctx.lineTo(axisx.p2c(x1), axisy.p2c(axisy.max));
									ctx.lineTo(axisx.p2c(x2), axisy.p2c(axisy.max));
									continue;
								}
								else if (y1 <= axisy.min && y2 <= axisy.min) {
									ctx.lineTo(axisx.p2c(x1), axisy.p2c(axisy.min));
									ctx.lineTo(axisx.p2c(x2), axisy.p2c(axisy.min));
									continue;
								}
								// else it's a bit more complicated, there might
								// be a flat maxed out rectangle first, then a
								// triangular cutout or reverse; to find these
								// keep track of the current x values
								var x1old = x1, x2old = x2;
								// clip the y values, without shortcutting, we
								// go through all cases in turn
								// clip with ymin
								if (y1 <= y2 && y1 < axisy.min && y2 >= axisy.min) {
									x1 = (axisy.min - y1) / (y2 - y1) * (x2 - x1) + x1;
									y1 = axisy.min;
								}
								else if (y2 <= y1 && y2 < axisy.min && y1 >= axisy.min) {
									x2 = (axisy.min - y1) / (y2 - y1) * (x2 - x1) + x1;
									y2 = axisy.min;
								}
								// clip with ymax
								if (y1 >= y2 && y1 > axisy.max && y2 <= axisy.max) {
									x1 = (axisy.max - y1) / (y2 - y1) * (x2 - x1) + x1;
									y1 = axisy.max;
								}
								else if (y2 >= y1 && y2 > axisy.max && y1 <= axisy.max) {
									x2 = (axisy.max - y1) / (y2 - y1) * (x2 - x1) + x1;
									y2 = axisy.max;
								}
								// if the x value was changed we got a rectangle
								// to fill
								if (x1 != x1old) {
									ctx.lineTo(axisx.p2c(x1old), axisy.p2c(y1));
									// it goes to (x1, y1), but we fill that below
								}
								// fill triangular section, this sometimes result
								// in redundant points if (x1, y1) hasn't changed
								// from previous line to, but we just ignore that
								ctx.lineTo(axisx.p2c(x1), axisy.p2c(y1));
								ctx.lineTo(axisx.p2c(x2), axisy.p2c(y2));
								// fill the other rectangle if it's there
								if (x2 != x2old) {
									ctx.lineTo(axisx.p2c(x2), axisy.p2c(y2));
									ctx.lineTo(axisx.p2c(x2old), axisy.p2c(y2));
								}
							}
						}
						ctx.save();
						ctx.translate(plotOffset.left, plotOffset.top);
						ctx.lineJoin = "round";
						var lw = series.lines.lineWidth,
							sw = series.shadowSize;
						// FIXME: consider another form of shadow when filling is turned on
						if (lw > 0 && sw > 0) {
							// draw shadow as a thick and thin line with transparency
							ctx.lineWidth = sw;
							ctx.strokeStyle = "rgba(0,0,0,0.1)";
							// position shadow at angle from the mid of line
							var angle = Math.PI/18;
							plotLine(series.datapoints, Math.sin(angle) * (lw/2 + sw/2), Math.cos(angle) * (lw/2 + sw/2), series.xaxis, series.yaxis);
							ctx.lineWidth = sw/2;
							plotLine(series.datapoints, Math.sin(angle) * (lw/2 + sw/4), Math.cos(angle) * (lw/2 + sw/4), series.xaxis, series.yaxis);
						}
						ctx.lineWidth = lw;
						ctx.strokeStyle = series.color;
						var fillStyle = getFillStyle(series.lines, series.color, 0, plotHeight);
						if (fillStyle) {
							ctx.fillStyle = fillStyle;
							plotLineArea(series.datapoints, series.xaxis, series.yaxis);
						}
						if (lw > 0)
							plotLine(series.datapoints, 0, 0, series.xaxis, series.yaxis);
						ctx.restore();
					}
					function drawSeriesPoints(series) {
						function plotPoints(datapoints, radius, fillStyle, offset, shadow, axisx, axisy, symbol) {
							var points = datapoints.points, ps = datapoints.pointsize;
							for (var i = 0; i < points.length; i += ps) {
								var x = points[i], y = points[i + 1];
								if (x == null || x < axisx.min || x > axisx.max || y < axisy.min || y > axisy.max)
									continue;
								ctx.beginPath();
								x = axisx.p2c(x);
								y = axisy.p2c(y) + offset;
								if (symbol == "circle")
									ctx.arc(x, y, radius, 0, shadow ? Math.PI : Math.PI * 2, false);
								else
									symbol(ctx, x, y, radius, shadow);
								ctx.closePath();
								if (fillStyle) {
									ctx.fillStyle = fillStyle;
									ctx.fill();
								}
								ctx.stroke();
							}
						}
						ctx.save();
						ctx.translate(plotOffset.left, plotOffset.top);
						var lw = series.points.lineWidth,
							sw = series.shadowSize,
							radius = series.points.radius,
							symbol = series.points.symbol;
						// If the user sets the line width to 0, we change it to a very 
						// small value. A line width of 0 seems to force the default of 1.
						// Doing the conditional here allows the shadow setting to still be 
						// optional even with a lineWidth of 0.
						if( lw == 0 )
							lw = 0.0001;
						if (lw > 0 && sw > 0) {
							// draw shadow in two steps
							var w = sw / 2;
							ctx.lineWidth = w;
							ctx.strokeStyle = "rgba(0,0,0,0.1)";
							plotPoints(series.datapoints, radius, null, w + w/2, true,
									   series.xaxis, series.yaxis, symbol);
							ctx.strokeStyle = "rgba(0,0,0,0.2)";
							plotPoints(series.datapoints, radius, null, w/2, true,
									   series.xaxis, series.yaxis, symbol);
						}
						ctx.lineWidth = lw;
						ctx.strokeStyle = series.color;
						plotPoints(series.datapoints, radius,
								   getFillStyle(series.points, series.color), 0, false,
								   series.xaxis, series.yaxis, symbol);
						ctx.restore();
					}
					function drawBar(x, y, b, barLeft, barRight, fillStyleCallback, axisx, axisy, c, horizontal, lineWidth) {
						var left, right, bottom, top,
							drawLeft, drawRight, drawTop, drawBottom,
							tmp;
						// in horizontal mode, we start the bar from the left
						// instead of from the bottom so it appears to be
						// horizontal rather than vertical
						if (horizontal) {
							drawBottom = drawRight = drawTop = true;
							drawLeft = false;
							left = b;
							right = x;
							top = y + barLeft;
							bottom = y + barRight;
							// account for negative bars
							if (right < left) {
								tmp = right;
								right = left;
								left = tmp;
								drawLeft = true;
								drawRight = false;
							}
						}
						else {
							drawLeft = drawRight = drawTop = true;
							drawBottom = false;
							left = x + barLeft;
							right = x + barRight;
							bottom = b;
							top = y;
							// account for negative bars
							if (top < bottom) {
								tmp = top;
								top = bottom;
								bottom = tmp;
								drawBottom = true;
								drawTop = false;
							}
						}
						// clip
						if (right < axisx.min || left > axisx.max ||
							top < axisy.min || bottom > axisy.max)
							return;
						if (left < axisx.min) {
							left = axisx.min;
							drawLeft = false;
						}
						if (right > axisx.max) {
							right = axisx.max;
							drawRight = false;
						}
						if (bottom < axisy.min) {
							bottom = axisy.min;
							drawBottom = false;
						}
						if (top > axisy.max) {
							top = axisy.max;
							drawTop = false;
						}
						left = axisx.p2c(left);
						bottom = axisy.p2c(bottom);
						right = axisx.p2c(right);
						top = axisy.p2c(top);
						// fill the bar
						if (fillStyleCallback) {
							c.fillStyle = fillStyleCallback(bottom, top);
							c.fillRect(left, top, right - left, bottom - top)
						}
						// draw outline
						if (lineWidth > 0 && (drawLeft || drawRight || drawTop || drawBottom)) {
							c.beginPath();
							// FIXME: inline moveTo is buggy with excanvas
							c.moveTo(left, bottom);
							if (drawLeft)
								c.lineTo(left, top);
							else
								c.moveTo(left, top);
							if (drawTop)
								c.lineTo(right, top);
							else
								c.moveTo(right, top);
							if (drawRight)
								c.lineTo(right, bottom);
							else
								c.moveTo(right, bottom);
							if (drawBottom)
								c.lineTo(left, bottom);
							else
								c.moveTo(left, bottom);
							c.stroke();
						}
					}
					function drawSeriesBars(series) {
						function plotBars(datapoints, barLeft, barRight, fillStyleCallback, axisx, axisy) {
							var points = datapoints.points, ps = datapoints.pointsize;
							for (var i = 0; i < points.length; i += ps) {
								if (points[i] == null)
									continue;
								drawBar(points[i], points[i + 1], points[i + 2], barLeft, barRight, fillStyleCallback, axisx, axisy, ctx, series.bars.horizontal, series.bars.lineWidth);
							}
						}
						ctx.save();
						ctx.translate(plotOffset.left, plotOffset.top);
						// FIXME: figure out a way to add shadows (for instance along the right edge)
						ctx.lineWidth = series.bars.lineWidth;
						ctx.strokeStyle = series.color;
						var barLeft;
						switch (series.bars.align) {
							case "left":
								barLeft = 0;
								break;
							case "right":
								barLeft = -series.bars.barWidth;
								break;
							default:
								barLeft = -series.bars.barWidth / 2;
						}
						var fillStyleCallback = series.bars.fill ? function (bottom, top) { return getFillStyle(series.bars, series.color, bottom, top); } : null;
						plotBars(series.datapoints, barLeft, barLeft + series.bars.barWidth, fillStyleCallback, series.xaxis, series.yaxis);
						ctx.restore();
					}
					function getFillStyle(filloptions, seriesColor, bottom, top) {
						var fill = filloptions.fill;
						if (!fill)
							return null;
						if (filloptions.fillColor)
							return getColorOrGradient(filloptions.fillColor, bottom, top, seriesColor);
						var c = $.color.parse(seriesColor);
						c.a = typeof fill == "number" ? fill : 0.4;
						c.normalize();
						return c.toString();
					}
					function insertLegend() {
						if (options.legend.container != null) {
							$(options.legend.container).html("");
						} else {
							placeholder.find(".legend").remove();
						}
						if (!options.legend.show) {
							return;
						}
						var fragments = [], entries = [], rowStarted = false,
							lf = options.legend.labelFormatter, s, label;
						// Build a list of legend entries, with each having a label and a color
						for (var i = 0; i < series.length; ++i) {
							s = series[i];
							if (s.label) {
								label = lf ? lf(s.label, s) : s.label;
								if (label) {
									entries.push({
										label: label,
										color: s.color
									});
								}
							}
						}
						// Sort the legend using either the default or a custom comparator
						if (options.legend.sorted) {
							if ($.isFunction(options.legend.sorted)) {
								entries.sort(options.legend.sorted);
							} else if (options.legend.sorted == "reverse") {
								entries.reverse();
							} else {
								var ascending = options.legend.sorted != "descending";
								entries.sort(function(a, b) {
									return a.label == b.label ? 0 : (
										(a.label < b.label) != ascending ? 1 : -1   // Logical XOR
									);
								});
							}
						}
						// Generate markup for the list of entries, in their final order
						for (var i = 0; i < entries.length; ++i) {
							var entry = entries[i];
							if (i % options.legend.noColumns == 0) {
								if (rowStarted)
									fragments.push('');
								fragments.push('');
								rowStarted = true;
							}
							fragments.push(
								'' + entry.label + ' '
							);
						}
						if (rowStarted)
							fragments.push(' ');
						if (fragments.length == 0)
							return;
						var table = '' + fragments.join("") + '
';
						if (options.legend.container != null)
							$(options.legend.container).html(table);
						else {
							var pos = "",
								p = options.legend.position,
								m = options.legend.margin;
							if (m[0] == null)
								m = [m, m];
							if (p.charAt(0) == "n")
								pos += 'top:' + (m[1] + plotOffset.top) + 'px;';
							else if (p.charAt(0) == "s")
								pos += 'bottom:' + (m[1] + plotOffset.bottom) + 'px;';
							if (p.charAt(1) == "e")
								pos += 'right:' + (m[0] + plotOffset.right) + 'px;';
							else if (p.charAt(1) == "w")
								pos += 'left:' + (m[0] + plotOffset.left) + 'px;';
							var legend = $('' + table.replace('style="', 'style="position:absolute;' + pos +';') + '
').appendTo(placeholder);
							if (options.legend.backgroundOpacity != 0.0) {
								// put in the transparent background
								// separately to avoid blended labels and
								// label boxes
								var c = options.legend.backgroundColor;
								if (c == null) {
									c = options.grid.backgroundColor;
									if (c && typeof c == "string")
										c = $.color.parse(c);
									else
										c = $.color.extract(legend, 'background-color');
									c.a = 1;
									c = c.toString();
								}
								var div = legend.children();
								$(' 
').prependTo(legend).css('opacity', options.legend.backgroundOpacity);
							}
						}
					}
					// interactive features
					var highlights = [],
						redrawTimeout = null;
					// returns the data item the mouse is over, or null if none is found
					function findNearbyItem(mouseX, mouseY, seriesFilter) {
						var maxDistance = options.grid.mouseActiveRadius,
							smallestDistance = maxDistance * maxDistance + 1,
							item = null, foundPoint = false, i, j, ps;
						for (i = series.length - 1; i >= 0; --i) {
							if (!seriesFilter(series[i]))
								continue;
							var s = series[i],
								axisx = s.xaxis,
								axisy = s.yaxis,
								points = s.datapoints.points,
								mx = axisx.c2p(mouseX), // precompute some stuff to make the loop faster
								my = axisy.c2p(mouseY),
								maxx = maxDistance / axisx.scale,
								maxy = maxDistance / axisy.scale;
							ps = s.datapoints.pointsize;
							// with inverse transforms, we can't use the maxx/maxy
							// optimization, sadly
							if (axisx.options.inverseTransform)
								maxx = Number.MAX_VALUE;
							if (axisy.options.inverseTransform)
								maxy = Number.MAX_VALUE;
							if (s.lines.show || s.points.show) {
								for (j = 0; j < points.length; j += ps) {
									var x = points[j], y = points[j + 1];
									if (x == null)
										continue;
									// For points and lines, the cursor must be within a
									// certain distance to the data point
									if (x - mx > maxx || x - mx < -maxx ||
										y - my > maxy || y - my < -maxy)
										continue;
									// We have to calculate distances in pixels, not in
									// data units, because the scales of the axes may be different
									var dx = Math.abs(axisx.p2c(x) - mouseX),
										dy = Math.abs(axisy.p2c(y) - mouseY),
										dist = dx * dx + dy * dy; // we save the sqrt
									// use <= to ensure last point takes precedence
									// (last generally means on top of)
									if (dist < smallestDistance) {
										smallestDistance = dist;
										item = [i, j / ps];
									}
								}
							}
							if (s.bars.show && !item) { // no other point can be nearby
								var barLeft, barRight;
								switch (s.bars.align) {
									case "left":
										barLeft = 0;
										break;
									case "right":
										barLeft = -s.bars.barWidth;
										break;
									default:
										barLeft = -s.bars.barWidth / 2;
								}
								barRight = barLeft + s.bars.barWidth;
								for (j = 0; j < points.length; j += ps) {
									var x = points[j], y = points[j + 1], b = points[j + 2];
									if (x == null)
										continue;
									// for a bar graph, the cursor must be inside the bar
									if (series[i].bars.horizontal ?
										(mx <= Math.max(b, x) && mx >= Math.min(b, x) &&
										 my >= y + barLeft && my <= y + barRight) :
										(mx >= x + barLeft && mx <= x + barRight &&
										 my >= Math.min(b, y) && my <= Math.max(b, y)))
											item = [i, j / ps];
								}
							}
						}
						if (item) {
							i = item[0];
							j = item[1];
							ps = series[i].datapoints.pointsize;
							return { datapoint: series[i].datapoints.points.slice(j * ps, (j + 1) * ps),
									 dataIndex: j,
									 series: series[i],
									 seriesIndex: i };
						}
						return null;
					}
					function onMouseMove(e) {
						if (options.grid.hoverable)
							triggerClickHoverEvent("plothover", e,
												   function (s) { return s["hoverable"] != false; });
					}
					function onMouseLeave(e) {
						if (options.grid.hoverable)
							triggerClickHoverEvent("plothover", e,
												   function (s) { return false; });
					}
					function onClick(e) {
						triggerClickHoverEvent("plotclick", e,
											   function (s) { return s["clickable"] != false; });
					}
					// trigger click or hover event (they send the same parameters
					// so we share their code)
					function triggerClickHoverEvent(eventname, event, seriesFilter) {
						var offset = eventHolder.offset(),
							canvasX = event.pageX - offset.left - plotOffset.left,
							canvasY = event.pageY - offset.top - plotOffset.top,
						pos = canvasToAxisCoords({ left: canvasX, top: canvasY });
						pos.pageX = event.pageX;
						pos.pageY = event.pageY;
						var item = findNearbyItem(canvasX, canvasY, seriesFilter);
						if (item) {
							// fill in mouse pos for any listeners out there
							item.pageX = parseInt(item.series.xaxis.p2c(item.datapoint[0]) + offset.left + plotOffset.left, 10);
							item.pageY = parseInt(item.series.yaxis.p2c(item.datapoint[1]) + offset.top + plotOffset.top, 10);
						}
						if (options.grid.autoHighlight) {
							// clear auto-highlights
							for (var i = 0; i < highlights.length; ++i) {
								var h = highlights[i];
								if (h.auto == eventname &&
									!(item && h.series == item.series &&
									  h.point[0] == item.datapoint[0] &&
									  h.point[1] == item.datapoint[1]))
									unhighlight(h.series, h.point);
							}
							if (item)
								highlight(item.series, item.datapoint, eventname);
						}
						placeholder.trigger(eventname, [ pos, item ]);
					}
					function triggerRedrawOverlay() {
						var t = options.interaction.redrawOverlayInterval;
						if (t == -1) {      // skip event queue
							drawOverlay();
							return;
						}
						if (!redrawTimeout)
							redrawTimeout = setTimeout(drawOverlay, t);
					}
					function drawOverlay() {
						redrawTimeout = null;
						// draw highlights
						octx.save();
						overlay.clear();
						octx.translate(plotOffset.left, plotOffset.top);
						var i, hi;
						for (i = 0; i < highlights.length; ++i) {
							hi = highlights[i];
							if (hi.series.bars.show)
								drawBarHighlight(hi.series, hi.point);
							else
								drawPointHighlight(hi.series, hi.point);
						}
						octx.restore();
						executeHooks(hooks.drawOverlay, [octx]);
					}
					function highlight(s, point, auto) {
						if (typeof s == "number")
							s = series[s];
						if (typeof point == "number") {
							var ps = s.datapoints.pointsize;
							point = s.datapoints.points.slice(ps * point, ps * (point + 1));
						}
						var i = indexOfHighlight(s, point);
						if (i == -1) {
							highlights.push({ series: s, point: point, auto: auto });
							triggerRedrawOverlay();
						}
						else if (!auto)
							highlights[i].auto = false;
					}
					function unhighlight(s, point) {
						if (s == null && point == null) {
							highlights = [];
							triggerRedrawOverlay();
							return;
						}
						if (typeof s == "number")
							s = series[s];
						if (typeof point == "number") {
							var ps = s.datapoints.pointsize;
							point = s.datapoints.points.slice(ps * point, ps * (point + 1));
						}
						var i = indexOfHighlight(s, point);
						if (i != -1) {
							highlights.splice(i, 1);
							triggerRedrawOverlay();
						}
					}
					function indexOfHighlight(s, p) {
						for (var i = 0; i < highlights.length; ++i) {
							var h = highlights[i];
							if (h.series == s && h.point[0] == p[0]
								&& h.point[1] == p[1])
								return i;
						}
						return -1;
					}
					function drawPointHighlight(series, point) {
						var x = point[0], y = point[1],
							axisx = series.xaxis, axisy = series.yaxis,
							highlightColor = (typeof series.highlightColor === "string") ? series.highlightColor : $.color.parse(series.color).scale('a', 0.5).toString();
						if (x < axisx.min || x > axisx.max || y < axisy.min || y > axisy.max)
							return;
						var pointRadius = series.points.radius + series.points.lineWidth / 2;
						octx.lineWidth = pointRadius;
						octx.strokeStyle = highlightColor;
						var radius = 1.5 * pointRadius;
						x = axisx.p2c(x);
						y = axisy.p2c(y);
						octx.beginPath();
						if (series.points.symbol == "circle")
							octx.arc(x, y, radius, 0, 2 * Math.PI, false);
						else
							series.points.symbol(octx, x, y, radius, false);
						octx.closePath();
						octx.stroke();
					}
					function drawBarHighlight(series, point) {
						var highlightColor = (typeof series.highlightColor === "string") ? series.highlightColor : $.color.parse(series.color).scale('a', 0.5).toString(),
							fillStyle = highlightColor,
							barLeft;
						switch (series.bars.align) {
							case "left":
								barLeft = 0;
								break;
							case "right":
								barLeft = -series.bars.barWidth;
								break;
							default:
								barLeft = -series.bars.barWidth / 2;
						}
						octx.lineWidth = series.bars.lineWidth;
						octx.strokeStyle = highlightColor;
						drawBar(point[0], point[1], point[2] || 0, barLeft, barLeft + series.bars.barWidth,
								function () { return fillStyle; }, series.xaxis, series.yaxis, octx, series.bars.horizontal, series.bars.lineWidth);
					}
					function getColorOrGradient(spec, bottom, top, defaultColor) {
						if (typeof spec == "string")
							return spec;
						else {
							// assume this is a gradient spec; IE currently only
							// supports a simple vertical gradient properly, so that's
							// what we support too
							var gradient = ctx.createLinearGradient(0, top, 0, bottom);
							for (var i = 0, l = spec.colors.length; i < l; ++i) {
								var c = spec.colors[i];
								if (typeof c != "string") {
									var co = $.color.parse(defaultColor);
									if (c.brightness != null)
										co = co.scale('rgb', c.brightness);
									if (c.opacity != null)
										co.a *= c.opacity;
									c = co.toString();
								}
								gradient.addColorStop(i / (l - 1), c);
							}
							return gradient;
						}
					}
				}
				// Add the plot function to the top level of the jQuery object
				$.plot = function(placeholder, data, options) {
					//var t0 = new Date();
					var plot = new Plot($(placeholder), data, options, $.plot.plugins);
					//(window.console ? console.log : alert)("time used (msecs): " + ((new Date()).getTime() - t0.getTime()));
					return plot;
				};
				$.plot.version = "0.8.3";
				$.plot.plugins = [];
				// Also add the plot function as a chainable property
				$.fn.plot = function(data, options) {
					return this.each(function() {
						$.plot(this, data, options);
					});
				};
				// round to nearby lower multiple of base
				function floorInBase(n, base) {
					return base * Math.floor(n / base);
				}
			})(jQuery);
		}
	}
	
	/* ===== Load UserScript ===== */
	function addJQuery(callback){
		var script = document.createElement("script");
		script.async = true;
		script.setAttribute("src", "http://ajax.googleapis.com/ajax/libs/jquery/2.0.1/jquery.min.js");
		script.addEventListener(
			'load', 
			function(){
				var script = document.createElement("script");
				script.async = true;
				script.textContent = "window.jQ = jQuery.noConflict(true);"+
				"("+callback.toString()+")();";
				document.head.appendChild(script);
			}, 
			false
		);
		document.head.appendChild(script);
	}
	
	if(window.location.host.indexOf("worldofwarships") > -1 || window.location.host.indexOf(".wargaming.net") > -1){
		addJQuery(WoWsStatInfo);
	}
})(window);