IDEAL Scripts

All IDEAL Scripts for idealclan.eu merged into one

2019-02-03 يوللانغان نەشرى. ئەڭ يېڭى نەشرىنى كۆرۈش.

// ==UserScript==
// @name        IDEAL Scripts
// @version     1.04
// @description All IDEAL Scripts for idealclan.eu merged into one
// @author      Orrie
// @namespace   http://idealclan.eu/viewtopic.php?f=7&t=1791#p57645
// @icon        http://i.imgur.com/9GeRjIp.png
// @include     http://*idealclan.eu/*
// @grant       GM_xmlhttpRequest
// @grant       GM.xmlHttpRequest
// @connect     eu.wargaming.net
// @connect     api.worldoftanks.eu
// @connect     api.twitch.tv
// @require     https://greasyfork.org/scripts/18946-tablesort/code/Tablesort.js?version=120660
// @require     https://greasemonkey.github.io/gm4-polyfill/gm4-polyfill.js
// @license     Mr Sexii License
// ==/UserScript==
(function() {
	// global variables
	const date = new Date();

	let streamTable, battleTable, updateInterval, style, sortTable = false, time, styleDynBattles, styleDynFinales, inputSpan, timeJump = false, timeInterval, styleDynTimezone;

	// script variables
	const bs = {
		href: document.location.href,
		vers: ((GM.info) ? GM.info.script.version : ""),
		tw: {
			api: "https://api.twitch.tv/kraken/streams/?channel=",
			key: "fbs3puqef627klk0wf9jrgjach2h3y9"
		},
		cw: {
			status: JSON.parse(localStorage.getItem("battles_force")) || false,
			event: false,
			gold: true,
			fame: 400,
			tier: "Ɵ",
			bats: "Ɵ",
			elo: "Ɵ",
			current: 0,
			globalmap: false
		},
		dyn: {
			conc: {},
			prov: [],
			plan: 0,
			check: 0,
			gold: 0
		},
		clan: {
			ideal: 500010805,
			id: JSON.parse(localStorage.getItem("battles_clanid")) || 500010805,
			tag: "Ɵ",
			emblem: "Ɵ",
			color: "Ɵ"
		},
		table: {
			static: 10,
			eu: [17, 18, 19, 20, 21, 22, 23, 24],
			c: "",
			s: []
		},
		front_loc: {
			"arms_race_eu_league1": " (BF)",
			"arms_race_eu_league2": " (AF)",
			"arms_race_eu_league3": " (EF)",
			"soldiers_of_fortune_eu_league1": " (BF)",
			"soldiers_of_fortune_eu_league2": " (AF)",
			"soldiers_of_fortune_eu_league3": " (EF)"
		},
		app_id: "a7595640a90bf2d19065f3f2683b171c",
		web: {
			// browser detection
			gecko: typeof InstallTrigger !== 'undefined',
			chrome: !!window.chrome && !!window.chrome.webstore
		},
		time: {
			h: date.getHours(),
			m: date.getMinutes(),
			o: (date.getTimezoneOffset() > 0 ? -Math.abs(date.getTimezoneOffset()) : Math.abs(date.getTimezoneOffset())) / 60
		},
		debug: document.getElementsByClassName("headerbnl")[0].innerHTML.match(/Orrie/) ? true : false
	};
	bs.api = {
		event: `https://api.worldoftanks.eu/wot/globalmap/events/?application_id=${bs.app_id}&limit=1`,
		clan: `https://eu.wargaming.net/globalmap/game_api/clan/${bs.clan.id}/battles`,
		bats: "https://eu.wargaming.net/globalmap/game_api/map_fill_info?aliases=",
		tourney: "https://eu.wargaming.net/globalmap/game_api/tournament_info?alias=",
		prov: "https://eu.wargaming.net/globalmap/game_api/province_info?alias=",
		provs: `https://api.worldoftanks.eu/wot/globalmap/clanprovinces/?application_id=${bs.app_id}&clan_id=${bs.clan.id}`,
		divs: "https://eu.wargaming.net/globalmap/game_api/wot/clan_tactical_data", // need authentication from global map
		raids: "https://eu.wargaming.net/globalmap/game_api/wot/raids" // need authentication from global map
	};
	bs.time.r = ((bs.time.m >= 15 && bs.time.m <= 45) ? [bs.time.h, "30"] : ((bs.time.m <= 15) ? [bs.time.h, "00"] : [(bs.time.h + 1), "00"]));
	bs.time.t = `${bs.time.r[0]}_${bs.time.r[1]}`;

	// image codes and stream list
	const img = {
		main: "",
		flags: "",
		clans: ""
	},
	streams = [
		// IDEAD
		["Hamu", "drhamu", "ideal", ["en", "no"], "0"],
		["HowTheStoryEnds", "howthestoryends", "ideal", "en", "0"],
		["Kriegerseele", "kriegerseele_", "ideal", ["en", "de"], "0"],
		["Lilsaah", "lilsaah", "ideal", ["en", "no"], "0"],
		["Malgorn", "malgorn", "ideal", "en", "0"],
		["McMokka", "mcmokka", "ideal", ["en", "de"], "0"],
		["MrOmen", "mromenx", "ideal", ["en", "pl"], "0"],
		["Orrie", "orrie_", "ideal", ["en", "no"], "0"],
		["Otto_von_HEATmarck", "otto_von_heatmarck", "ideal", ["en", "de"], "0"],
		["R0XJi", "roxjiwot", "ideal", ["en", "fi"], "0"],
		["Remmke", "remmke", "ideal", ["en", "de"], "0"],
		["Sekundenkleber", "sekundenkleber", "ideal", ["en", "de"], "0"],
		["Schnueres", "schnueres", "ideal", ["en", "de"], "0"],
		["Stratikat", "stratikat", "ideal", "en", "0"],
		["Taylor_Swift", "gashting", "ideal", "en", "0"],
		["TheGamingLion", "thegaminglion7", "ideal", "en", "0"],
		["veitileiN", "veitilein", "ideal", ["en", "de"], "0"],
		// IDEAD friends
		["Evidence", "evidence__", ["fnd", "vole"], ["en", "cz"], "0"],
		["Fjellreven", "fjellreven_tv", "fnd", ["en", "no"], "0"],
		["Keezoo", "keezoo__", ["fnd", "fame"], "pl", "0"],
		["Rudko", "rudko44", ["fnd", "s3al"], "en", "0"],
		["Weenis", "weenis__", ["fnd", "fame"], "en", "0"],
		// S3AL
		["Andgus", "andgus_swe", "s3al", ["en", "se"], "1"],
		["Atrevir", "atrevir", "s3al", "en", "1"],
		["DMminion", "dmminion", "s3al", "en", "1"],
		["Dukkerz", "dukkerz", "s3al", "en", "1"],
		["Elecktr", "elecktr_pl", "s3al", "en", "1"],
		["Faptastic_Panda", "faptastic_panda", "s3al", "en", "1"],
		["Fosco", "fosc0", "s3al", "en", "1"],
		["Hallucinogen", "hailucinogen", "s3al", "en", "1"],
		["Hassh", "hashonline", "s3al", "en", "1"],
		["Jona", "jonasb7", "s3al", "en", "1"],
		["Kapz93", "kapz93", "s3al", "en", "1"],
		["OffCerium", "offcerium", "s3al", "en", "1"],
		["POtkanzoR", "potkanzor", "s3al", "en", "1"],
		["Richblaster", "richblaster", "ptsd", "ptsd", "1"],
		["RoyM18", "roym18", "s3al", "en", "1"],
		["Sukkamurmeli", "sukkamurmeli", "s3al", "en", "1"],
		["Tarantula", "kridtv", "s3al", ["en", "se"], "1"],
		["Trechy", "trechy", "s3al", ["en", "se"], "1"],
		// FAME
		["1stWarlord", "1stwarlord", "fame", "en", "1"],
		["CarryBarry", "swift_m0ti0n", "fame", "en", "1"],
		["Dakillzor", "dakillzor", "fame", "en", "1"],
		["Decha", "dechapl", "fame", "en", "1"],
		["Deekay", "dkns_eu", "fame", "en", "1"],
		["Dyaebl", "dyaebl", "fame", ["en", "pl"], "1"],
		["Failware", "failwarelp", "fame", "en", "1"],
		["FieserFettsakk", "fieserfettsakk_", "fame", ["en", "de"], "1"],
		["Floris", "florisgg", "fame", "en", "1"],
		["Kajzoo", "kajzoo", "fame", "en", "1"],
		["Mailand", "mailand121", "fame", "en", "1"],
		["Nicktast1c", "nicktast1c", "fame", "en", "1"],
		["RavenHR", "ravenhr", "fame", "en", "1"],
		["Schockisch", "schockisch", "fame", "en", "1"],
		["Shishx", "shishx", "fame", "en", "1"],
		["Skarium", "skarium1", "fame", "en", "1"],
		["TCAmir", "tcamir", "fame", "en", "1"],
		["Tyne", "k1ll_5t3al_tyne", "fame", ["en", "fi"], "1"],
		["X3N4", "x3n4", "fame", "en", "1"],
		["Xaneleon", "xaneleon", "fame", "en", "1"],
		["Yung", "yung_xd", "fame", ["en", "se"], "1"],
		// RSOP
		["CharCharro", "charcharro", "rsop", ["en", "fi"], "1"],
		["Choppy", "choppyyxd", "rsop", ["en", "fi"], "1"],
		["Count_TR", "count_tr", "rsop", ["en", "fi"], "1"],
		["Muppetti", "muppetti", "rsop", ["en", "fi"], "1"],
		// OMNI
		["Genghiswolves", "genghiswolves", "omni", "en", "1"],
		["Tomosa", "tomosasamosa", "omni", "en", "1"],
		["TotallyMeYou", "totallymeyou", "omni", "en", "1"],
		// Other Clans (Only 1 streamer)
		["Marty_vole", "marty_vole", "vole", ["en", "cz"], "1"],
		["Morgotz", "morgotz", "e50m", "en", "1"],
		["No6ody", "no6ody", "s4", ["en", "de"], "1"],
		["skill4ltu", "skill4ltu", "santi", "en", "1"],
		// WGL
		["Knäckebröd WGL", "knackebrod_wgl", "kb", "en", "1"],
		// Amerifats/AZNs/Austrians
		["Barks_Internally", "barks_internally", "bulba", "au", "2"],
		["Overlord_Prime", "overlord_prime", "bulba", "us", "2"],
		["Eural", "x3eural", "ptsd", "ca", "2"],
		["Sela", "selawot", "sela", "us", "2"],
		["Vetro", "vetro_", "us", "us", "2"],
		["Zeven", "zeven_na", "us", "us", "2"],
		// Eastern Overlords
		["LuciqueII", "pukanuragan", "mamb", "ru", "3"],
		["Near_You", "near_you", "mamb", "ru", "3"],
		["Skylex", "skylex_pro", "mamb", "ru", "3"],
		["LeBwa", "lebwa_wot", "navi", "ru", "3"],
		["Straik", "straikoid", "navi", "ru", "3"],
		["StoleYourBike", "viilageidiot", "syb", "ru", "3"],
		// Honourable Mentions & Wargaming Affiliates, Brought to you by PENTA™
		["WGL English", "wgl_en", "wg", "en", "4"],
		["WGL Europe", "wgleu", "wg", "en", "4"],
		["Ectar", "murphy1up", "wg", "en", "4"],
		["Circon", "circon", "circ", "en", "4"],
		["SirFoch", "sirfoch", "foch", "en", "4"],
		["Quickybaby", "quickybaby", "qb", "en", "4"]
	],
	clans = [
		"<option value='500010805'>IDEAL</option>",
		"<option value='500003585'>RSOP</option>",
		"<option value='500012667'>-PJ-</option>",
		"<option value='500025989'>FAME</option>",
		"<option value='500000013'>CSA</option>"
	];

	// script functions
	const sf = {
		elem(tag, attributes, children) {
			// element creation
			const element = document.createElement(tag);
			if (attributes) {
				for (let _e_k = Object.keys(attributes), _e = _e_k.length; _e > 0; _e--) {
					element[_e_k[_e - 1]] = attributes[_e_k[_e - 1]];
				}
			}
			if (children) {
				if (children.nodeType) {
					element.appendChild(children);
				}
				else {
					for (let _c = 0, _c_len = children.length; _c < _c_len; _c++) {
						element.appendChild(children[_c]);
					}
				}
			}
			return element;
		},
		tw: {
			// twich stream status
			handlerMain(data) {
				// data handler
				const onlineStreams = [];
				// empty online tables
				for (let _t = 0, _t_len = (streamTable.length - 2); _t < _t_len; _t++) {
					if (streamTable[_t].childElementCount > 0) {
						streamTable[_t].innerHTML = "";
					}
				}
				// roll through streams and append rows to specified table
				if (data.streams.length) {
					for (let _l = 0, _l_len = data.streams.length; _l < _l_len; _l++) {
						const stream = data.streams[_l],
						streamName = stream.channel.name,
						offlineRow = document.getElementById(streamName),
						onlineRow = offlineRow.cloneNode(true);
						onlineRow.classList.remove('streamHide');
						onlineRow.cells[0].innerHTML = `<div class='b-info'><span class='s-status'>${stream.channel.status}</span><span class='s-game'>Playing <a target='_blank' href='https://www.twitch.tv/directory/game/${stream.game}'>${stream.game}</a></span><span class='s-img'><img src='${stream.preview.medium}'></span></div>`;
						if (stream.game == "World of Tanks") {
							onlineRow.cells[1].appendChild(sf.elem("span", {
								className: "icon icon_main icon_wot"
							}));
						}
						onlineRow.cells[3].className = "td_status s-online";
						onlineRow.cells[3].innerHTML = stream.viewers;
						offlineRow.classList.add('streamHide');
						streamTable[onlineRow.className].appendChild(onlineRow);
						onlineStreams.push(streamName);
					}
					// check if any streams went offline
					const hiddenStreams = document.getElementsByClassName("streamHide");
					for (let _h = 0, _h_len = hiddenStreams.length; _h < _h_len; _h++) {
						const streamID = hiddenStreams[_h].id;
						if (!onlineStreams.includes(streamID)) {
							hiddenStreams[_h].classList.remove('streamHide');
						}
					}
				}
				else {
					streamTable[1].appendChild(sf.elem("tr", {
						innerHTML: "<td colspan='5'>No Streams Online</td>"
					}));
				}
				// update date and stream amount
				document.getElementById('js-streamsOnline').innerHTML = onlineStreams.length;
				document.getElementById('js-streamUpdate').innerHTML = new Date().toLocaleTimeString("en-GB");
			},
			handlerError() {
				for (let _t = 0, _t_len = (streamTable.length - 2); _t < _t_len; _t++) {
					if (streamTable[_t].childElementCount > 0) {
						streamTable[_t].innerHTML = "";
					}
				}
				streamTable[1].appendChild(sf.elem("tr", {
					innerHTML: "<td colspan='5'>Error Fetching Streams</td>"
				}));
				document.getElementById('js-streamUpdate').innerHTML = new Date().toLocaleTimeString("en-GB");
			},
			updater() {
				// updater handler
				sf.request("twitchDataInterval", bs.tw.api, sf.tw.handlerMain, "tw");
			}
		},
		cw: {
			// battle scheduler
			handlerEvent(data) {
				// event checker
				// check if active event exists
				const event = (data.data) ? data.data[0] : "error";
				if (event.status == "ACTIVE" || !bs.cw.status) {
					if (event.status == "ACTIVE") {
						bs.cw.event = true;
						bs.cw.tier = 10;
						document.getElementById('js-eventName').innerHTML = `${event.event_name.replace("_"," ")} &#8226; `;
					}
					sf.request("mainData", bs.api.clan, sf.cw.handlerMain, "cw");
					document.getElementById('js-tableStats').classList.remove("b-display-none");
				}
				else {
					// empty table
					battleTable.children[1].innerHTML = "";
					battleTable.children[1].appendChild(sf.elem("tr", {
						className: "t-cwText",
						innerHTML: "<td colspan='23'>See you next time.</td>"
					}));
					clearInterval(updateInterval);
				}
				// insert update timestamp
				document.getElementById('js-batttleUpdate').textContent = new Date().toLocaleTimeString("en-GB");
			},
			handlerMain({clan, battles, planned_battles}) {
				// main handler
				const battleProvinces = [],
				battleFragment = document.createDocumentFragment();
				let battle, battleTime;
				// store data
				bs.clan.tag = clan.tag;
				bs.clan.emblem = clan.emblem_url;
				bs.clan.color = clan.color;
				bs.cw.bats = clan.appointed_battles_count;
				bs.cw.current = battles.length;
				bs.cw.elo = {
					6: clan.elo_rating_6,
					8: clan.elo_rating_8,
					10: clan.elo_rating_10
				};
				style.textContent += `.t-clantag {color: ${bs.clan.color};}`;
				// go through battles and planned battles
				for (let _b = 0, _b_len = battles.length; _b < _b_len; _b++) {
					battle = battles[_b];
					if (!battleProvinces.includes(battle.province_id)) {
						battleProvinces.push(battle.province_id);
						battleTime = [sf.cw.time(parseFloat(battle.battle_time.match(/\d+/g)[3])), battle.battle_time.match(/\d+/g)[4], battle.battle_time.match(/\d+/g)[5]];
						battleFragment.appendChild(sf.elem("tr", {
							className: `battle ${battle.province_id} attack`,
							innerHTML: `<td><a target='_blank' href='https://eu.wargaming.net/globalmap/#province/${battle.province_id}'>${battle.province_name}</a> [<a class='t-tournament' target='_blank' href='https://eu.wargaming.net/globalmap/#tournament/${battle.province_id}'>&#9876;</a>]</td><td>${sf.cw.mapFix(battle.arena_name)}</td><td></td><td class='t-gold'><span></span><span class='icon icon_main icon_gold'></span></td><td class='t-fame'>Ɵ</td><td></td><td><span></span><span></span></td><td id='${battle.enemy.id}'><a target='_blank' href='https://eu.wargaming.net/clans/${battle.enemy.id}/'>[${battle.enemy.tag}] <img src='${battle.enemy.emblem_url}'></a><span class='t-elo'>(${battle.enemy[`elo_rating_${(bs.cw.event) ? bs.cw.tier : "10"}`]})</span></td><td class='t-nextBattle' data-sort='${battleTime[0] === 0 ? battleTime[0]+25 : battleTime[0]}${battleTime[1]}.${battleTime[2]}'>${battleTime[0]}:${battleTime[1]}:${battleTime[2]}</td><td class='t-battle'>Ɵ</td><td class='t-battle t-border'>Ɵ</td>${bs.table.c}`
						}));
						if (battle.round_number) {
							sf.request("freepassData", `${bs.api.tourney+battle.province_id}&round=${battle.round_number}`, sf.cw.handlerFreepass, "cw");
						}
					}
				}
				for (let _bp = 0, _bp_len = planned_battles.length; _bp < _bp_len; _bp++) {
					battle = planned_battles[_bp];
					if (!battleProvinces.includes(battle.province_id)) {
						battleProvinces.push(battle.province_id);
						battleTime = [sf.cw.time(parseFloat(battle.battle_time.match(/\d+/g)[3])), battle.battle_time.match(/\d+/g)[4], battle.battle_time.match(/\d+/g)[5]];
						battleFragment.appendChild(sf.elem("tr", {
							className: `battle ${battle.province_id} attack`,
							innerHTML: `<td><a target='_blank' href='https://eu.wargaming.net/globalmap/#province/${battle.province_id}'>${battle.province_name}</a> [<a class='t-tournament' target='_blank' href='https://eu.wargaming.net/globalmap/#tournament/${battle.province_id}'>&#9876;</a>]</td><td>${sf.cw.mapFix(battle.arena_name)}</td><td></td><td class='t-gold'><span>${battle.province_revenue}</span><span class='icon icon_main icon_gold'></span></td><td class='t-fame'>Ɵ</td><td></td><td><span></span><span></span></td><td>Not Started</td><td class='t-nextBattle' data-sort='${battleTime[0] === 0 ? battleTime[0]+25 : battleTime[0]}${battleTime[1]}.${battleTime[2]}'>${battleTime[0] === 0 ? "00" : battleTime[0]}:${battleTime[1]}:${battleTime[2]}</td><td class='t-battle'>Ɵ</td><td class='t-battle t-border'>Ɵ</td>${bs.table.c}`
						}));
						if (battle.is_attacker) {
							sf.request("freepassData", `${bs.api.tourney+battle.province_id}&round=1`, sf.cw.handlerFreepass, "cw");
						}
					}
				}
				// show foes and battle count if clan has any battles and remove loading indicator
				if (bs.cw.bats > 0) {
					// style.textContent += ".t-battle {display: table-cell !important;}";
					battleTable.children[1].innerHTML = ""; // empty table
				}
				battleTable.children[1].appendChild(battleFragment);
				// insert clan name
				document.getElementById('js-clan').innerHTML = `<span class='t-clantag'>[${bs.clan.tag}]</span> ${clan.name} <img src='${bs.clan.emblem}'> &#8226; `;
				// insert battle count
				document.getElementById('js-battles').textContent = bs.cw.current;
				// insert battle count
				document.getElementById('js-battlesPlan').textContent = bs.cw.bats;
				// send request for detailed battle information
				if (bs.cw.bats > 0) {
					sf.request("batsData", bs.api.bats + battleProvinces.join("&aliases="), sf.cw.handlerBats, "cw");
				}
				// send request for clan provinces
				sf.request("provsData", bs.api.provs, sf.cw.handlerProvs, "cw");
			},
			handlerBats(data, isPlanned) {
				// battles handler
				for (let _bd = 0, _bd_len = data.data.length; _bd < _bd_len; _bd++) {
					const battle = data.data[_bd],
					battleRow = document.getElementsByClassName(battle.alias)[0],
					enemyID = battleRow.children[7].id,
					battleType = ((battle.owner_clan_id == bs.clan.id) ? "Defense" : ((battle.owner_clan_id == enemyID) ? "Owner" : "Attack")),
					primeTime = [sf.cw.time(parseFloat(battle.primetime.match(/\d+/g)[0])), battle.primetime.match(/\d+/g)[1], parseFloat(battle.primetime.match(/\d+/g)[0])];
					battleRow.dataset.primeTime = JSON.stringify(primeTime);
					if (battleType == "Defense") {
						battleRow.classList.remove("attack");
						battleRow.classList.add("defense");
					}
					// modify cells
					battleRow.children[2].textContent = `${primeTime[0]}:${primeTime[1]}`;
					battleRow.children[2].dataset.sort = `${primeTime[0]}${primeTime[1]}`;
					battleRow.children[3].firstElementChild.textContent = battle.revenue;
					battleRow.children[6].firstElementChild.textContent = battleType;
					if (battleRow.children[8].textContent == "") {
						battleRow.children[8].textContent = `${primeTime[0]}:${primeTime[1]}:00`;
						battleRow.children[8].dataset.sort = `${primeTime[0]}${primeTime[1]}`;
					}
					// get correct battle count and schedule
					sf.request("tourneyData", `${bs.api.tourney+battle.alias}&round=1`, sf.cw.handlerTourney, "cw");
					// get province famepoints if event
					if (bs.cw.event) {
						sf.request("provData", bs.api.prov + battle.alias, sf.cw.handlerProv, "cw", (battle.is_wagon_train) ? "×20" : (battle.is_enclave_ring) ? "×5" : (battle.is_enclave_center) ? "×3" : "");
					}
				}
				// refresh table
				sortTable.refresh();
			},
			handlerTourney(data, isPlanned) {
				// tournament handler
				const battleRow = document.getElementsByClassName(data.province_id)[0],
				ownerClan = (data.owner) ? (data.owner.id == bs.clan.id) || false : false,
				isAuction = battleRow.classList.contains("auction");
				let primeTime,
					cellOwnerTime = false,
					attackers = [data.pretenders, 0];
				bs.dyn.check++;
				primeTime = [sf.cw.time(parseFloat(data.start_time.match(/\d+/g)[0])), data.start_time.match(/\d+/g)[1], parseFloat(data.start_time.match(/\d+/g)[0])];
				battleRow.dataset.primeTime = JSON.stringify(primeTime);
				// check attackers
				if (attackers[0]) {
					const _pre_len = attackers[0].length,
					attacker_ids = [];
					for (let _pre = 0; _pre < _pre_len; _pre++) {
						attacker_ids.push(attackers[0][_pre].id);
					}
					if (!ownerClan && !attacker_ids.includes(bs.clan.id)) {
						//battleRow.remove();
					}
				}
				else if (data.battles) {
					const _bat_len = data.battles.length,
					attacker_ids = [];
					for (let _bat = 0; _bat < _bat_len; _bat++) {
						const battle = data.battles[_bat];
						attacker_ids.push(battle.first_competitor.id);
						if (battle.second_competitor) {
							attacker_ids.push(data.battles[_bat].second_competitor.id);
						}
					}
					if (!ownerClan && !attacker_ids.includes(bs.clan.id)) {
						//battleRow.remove();
					}
				}
				if (data.is_superfinal) {
					attackers = 1;
				}
				else if (attackers[0]) {
					attackers = attackers[0].length;
				}
				else {
					attackers = attackers[1];
					for (let _bc = 0, _bc_len = data.battles.length; _bc < _bc_len; _bc++) {
						attackers += ((data.battles[_bc].is_fake) ? 1 : 2);
					}
				}
				if (isAuction && battleRow.children[7].innerHTML == "Not Started") {
					attackers = data.size;
				}
				if (isPlanned) {
					attackers++;
				}
				// find how many battles
				const battles = (attackers !== 0) ? Math.ceil(Math.log2(attackers)) + 1 : 0;
				// modify cells
				battleRow.children[1].textContent = sf.cw.mapFix(data.arena_name);
				battleRow.children[2].textContent = `${primeTime[0]}:${primeTime[1]}`;
				battleRow.children[2].dataset.sort = `${primeTime[0]}${primeTime[1]}`;
				battleRow.children[3].firstElementChild.textContent = data.province_revenue;
				battleRow.children[5].innerHTML = (data.owner) ? `<a target='_blank' href='https://eu.wargaming.net/clans/${data.owner.id}/'><span class='t-clantag' style='color: ${data.owner.color};'>[${data.owner.tag}]</span> <img src='${data.owner.emblem_url}'></a>` : "No Owner";
				if (data.owner && bs.cw.tier !== "Ɵ") {
					battleRow.children[5].appendChild(sf.elem("span", {
						className: "t-elo",
						innerHTML: `(${data.owner[`elo_rating_${(bs.cw.event) ? bs.cw.tier : "10"}`]})`
					}));
				}
				if (isAuction) {
					battleRow.children[8].textContent = `${primeTime[0]}:${primeTime[1]}:00`;
					battleRow.children[8].dataset.sort = `${primeTime[0]}${primeTime[1]}.${battles}`;
				}
				// only continue if there are any attackers
				if (attackers) {
					const emptyCells = ((primeTime[2] - bs.table.eu[0]) * 2) + bs.table.static,
					lastBattle = battles + emptyCells;
					battleRow.children[9].textContent = attackers;
					battleRow.children[10].textContent = battles;
					for (let _conc = 0, _cell = bs.table.static + 1; _cell < battleRow.childElementCount; _cell++) {
						const cell = battleRow.children[_cell];
						if (_cell > emptyCells && _cell <= lastBattle) {
							const timeClass = `.${cell.classList.item(1)}`,
							timePrevClass = `.${cell.previousElementSibling.classList.item(1)}`,
							primeTimeConc = `${Number(primeTime[0]+primeTime[1])+([0,30,100,130,200,230,300,330][_conc])}`;
							if (!bs.table.s.includes(timeClass) || !bs.table.s.includes(timePrevClass)) {
								bs.table.s.push(timePrevClass, timeClass);
								if (_cell == lastBattle) {
									bs.table.s.push(`${timePrevClass} + th`, `${timePrevClass} + td`, `${timeClass} + th`, `${timeClass} + td`);
								}
							}
							if (ownerClan && _cell !== lastBattle) {
								cell.classList.add("t-noFight");
							}
							else {
								cell.classList.add("t-fight");
								if (isAuction) {
									cell.classList.add("t-auction");
								}
								if (!bs.dyn.prov.includes(data.province_id)) {
									if (bs.dyn.conc[primeTimeConc]) {
										bs.dyn.conc[primeTimeConc]++;
									}
									else {
										bs.dyn.conc[primeTimeConc] = 1;
									}
								}
							}
							if (_cell == lastBattle) {
								if (ownerClan) {
									cell.classList.add("js-last");
									let nextBattle = cell.dataset.time;
									if (primeTime[1] == 15) {
										time = nextBattle.match(/(\d+)/g);
										nextBattle = `${time[0]}:${parseFloat(time[1])+15}`;
									}
									//battleRow.children[8].textContent = `${nextBattle}:00`;
									//battleRow.children[8].dataset.sort = `${nextBattle.replace(":","")}`;
								}
								if (bs.cw.globalmap && (!data.owner || !data.owner.division_id) && (battleRow.children[7].innerHTML !== "Not Started" || battleRow.children[6].firstElementChild.innerHTML == "No Division")) {
									cell.classList.add("t-noOwner");
									bs.dyn.conc[primeTimeConc]--;
								}
								cell.innerHTML = "♖";
								cellOwnerTime = [parseFloat(cell.classList.item(1).match(/\d+/g)[0]), parseFloat(cell.classList.item(1).match(/\d+/g)[1])];
								if (cellOwnerTime[0] < 5) {
									cellOwnerTime[0] += 24;
								}
							}
							else {
								cell.innerHTML = "&#9876;";
							}
							if (primeTime[1] == "15") {
								battleRow.classList.add("timeShift");
								cell.innerHTML += "<span class='t-timeShift'>+</span>";
							}
							_conc++;
						}
					}
					if (!bs.dyn.prov.includes(data.province_id)) {
						bs.dyn.prov.push(data.province_id);
					}
					if (bs.dyn.check == bs.cw.bats || bs.dyn.plan > 0) {
						const conc = Object.entries(bs.dyn.conc).reduce(function(max, arr) {
							return max[1] >= arr[1] ? max : arr;
						}),
						conc_table = {};
						document.getElementById('js-battlesConc').textContent = `${conc[1]} [${conc[0].replace(/^(\d{2})/,"$1:")}]`;
						styleDynBattles.textContent = `${bs.table.s.join(", ")} {display: table-cell !important;}`;
						for (let _c_k = Object.keys(bs.dyn.conc), _c = _c_k.length; _c > 0; _c--) {
							const key = _c_k[_c - 1],
							hour = key.slice(0, 2),
							min = key.slice(2),
							newKey = `t-${hour >= 24 ? hour-24 : hour}_${min == 15 ? "00" : (min == 45 ? "30" : min)}`,
							time = bs.dyn.conc[key],
							timeShift = /15|45/.test(min) ? true : false;
							if (conc_table[newKey]) {
								conc_table[newKey][timeShift ? 1 : 0] += time;
							}
							else {
								conc_table[newKey] = timeShift ? [0, time] : [time, 0];
							}
						}
						const footer = document.getElementById('js-footer');
						for (let _ct_k = Object.keys(conc_table), _ct = _ct_k.length; _ct > 0; _ct--) {
							const key = _ct_k[_ct - 1],
							times = conc_table[key],
							foot = footer.getElementsByClassName(key)[0];
							if (foot) {
								foot.textContent = `${times[0] ? times[0] : ""}${times[1] ? ` +${times[1]}` : ""}`;
							}
							else {
								console.error("Foot doesn't exist", key, times);
							}
						}
					}
					// check if battle is planned or not started and change state to ongoing
					if (!/\[|Free Round/i.test(battleRow.children[7].textContent) && new Date().getHours() >= primeTime[0] - 1 && new Date().getHours() < cellOwnerTime[0]) {
						switch (battleRow.children[6].firstElementChild.innerHTML) {
							case ("Attack"):
								battleRow.children[7].textContent = "Ongoing";
								break;
							case ("Planned"):
								battleRow.children[6].firstElementChild.textContent = "Defense";
								battleRow.children[7].textContent = "Ongoing";
								break;
							case ("Defense"):
								battleRow.children[7].textContent = "Ongoing";
								sf.request("provData", bs.api.prov + data.province_id, sf.cw.handlerProv, "cw");
								break;
							default:
								break;
						}
					}
					else if (battleRow.children[7].innerHTML == "Planned" && bs.time.r[0] > cellOwnerTime[0]) {
						battleRow.children[7].textContent = "Completed";
					}
				}
				else {
					const lastBattle = battleRow.getElementsByClassName(`t-${primeTime[0]}_00`)[0];
					if (primeTime[1] == "15") {
						battleRow.classList.add("timeShift");
						lastBattle.innerHTML = "♖<span class='t-timeShift'>+</span>";
					}
					else {
						lastBattle.innerHTML = "♖";
					}
					lastBattle.classList.add("t-noFight");
				}
				if (ownerClan) {
					bs.dyn.gold += data.province_revenue;
					document.getElementById('js-gold').textContent = bs.dyn.gold; // insert gold count
				}
				if (bs.cw.gold && data.province_revenue === 0) {
					document.getElementById('js-goldInfo').textContent = "(No Gold Revenue in Event!)";
					bs.cw.gold = false;
					style.textContent += "th.t-gold, td.t-gold {display: none;}";
				}
				// refresh table
				sortTable.refresh();
			},
			handlerProvs(data) {
				// clan provinces handler
				const provs = data.data[bs.clan.id],
				ownedProvinces = [],
				provTimes = [],
				provFragment = document.createDocumentFragment();
				if (battleTable.rows[1] && battleTable.rows[1].classList.contains("t-cwText")) {
					battleTable.children[1].innerHTML = ""; // empty table
				}
				if (provs) {
					for (let _p = 0, _p_len = provs.length; _p < _p_len; _p++) {
						const prov = provs[_p],
						battleRow = document.getElementsByClassName(prov.province_id)[0];
						if (!battleRow) {
							const primeTime = [sf.cw.time(parseFloat(prov.prime_time.match(/\d+/g)[0])), prov.prime_time.match(/\d+/g)[1], parseFloat(prov.prime_time.match(/\d+/g)[0]) + bs.table.eu.length],
							provRow = sf.elem("tr", {
								className: `province ${prov.province_id} defense`,
								innerHTML: `<td><a target='_blank' href='https://eu.wargaming.net/globalmap/#province/${prov.province_id}'>${prov.province_name}</a> [<a class='t-tournament' target='_blank' href='https://eu.wargaming.net/globalmap/#tournament/${prov.province_id}'>&#9876;</a>]</td><td>${sf.cw.mapFix(prov.arena_name)}</td><td data-sort='${primeTime[2]}${primeTime[1]}'>${primeTime[0]}:${primeTime[1]}</td><td class='t-gold'><span>${prov.daily_revenue}</span><span class='icon icon_main icon_gold'></span></td><td class='t-fame'>Ɵ</td><td><a target='_blank' href='https://eu.wargaming.net/clans/${bs.clan.id}/'><span class='t-clantag'>[${bs.clan.tag}]</span> <img src='${bs.clan.emblem}'></a><span class='t-elo'>(${bs.cw.elo[prov.max_vehicle_level]})</span></td><td><span>Defense</span><span></span></td><td>No Attacks</td><td class='t-nextBattle' data-sort='9999'></td><td class='t-battle' data-sort='99'>Ɵ</td><td class='t-battle t-border' data-sort='99'>Ɵ</td>${bs.table.c}`
							}),
							provTime = `t-${primeTime[0]}_00`,
							provTimeClass = `.${provTime}, .${provTime} + td, .${provTime} + th`,
								lastBattle = provRow.getElementsByClassName(provTime)[0];
							provRow.dataset.primeTime = JSON.stringify(primeTime);
							ownedProvinces.push(prov.province_id);
							bs.dyn.gold += prov.daily_revenue;
							bs.cw.tier = prov.max_vehicle_level;
							if (primeTime[1] == "15") {
								provRow.classList.add("timeShift");
								lastBattle.innerHTML = "♖<span class='t-timeShift'>+</span>";
							}
							else {
								lastBattle.innerHTML = "♖";
							}
							lastBattle.classList.add("t-noFight");
							if (!provTimes.includes(provTimeClass)) {
								provTimes.push(provTimeClass);
							}
							if (bs.cw.gold && prov.daily_revenue === 0) {
								document.getElementById('js-goldInfo').textContent = "(No Gold Revenue in Event!)";
								bs.cw.gold = false;
								style.textContent += "th.t-gold, td.t-gold {display: none;}";
							}
							// get province famepoints if event
							if (bs.cw.event) {
								sf.request("provData", bs.api.prov + prov.province_id, sf.cw.handlerProv, "cw");
							}
							provFragment.appendChild(provRow);
						}
					}
					// display finals column
					styleDynFinales.textContent += `${provTimes.join(", ")} {display: table-cell !important;}`;
					// insert province count
					document.getElementById('js-provs').textContent = provs.length;
					// insert gold count
					document.getElementById('js-gold').textContent = bs.dyn.gold;
					// send request for divisions
					sf.request("divsData", bs.api.divs, sf.cw.handlerDivs, "cw");
					// send request for raids if event
					//if (bs.cw.event) {
					//  sf.request("raidsData", bs.api.raids, sf.cw.handlerRaids, "cw");
					//}
				}
				else if (bs.cw.bats === 0) {
					battleTable.children[1].innerHTML = ""; // empty table
					provFragment.appendChild(sf.elem("tr", {
						className: "t-cwText",
						innerHTML: "<td colspan='23'>No Battles</td>"
					}));
				}
				battleTable.children[1].appendChild(provFragment);
				// add expand button
				const tableExpand = document.getElementById('js-expand');
				tableExpand.textContent = "❐";
				battleTable.parentNode.classList.add("b-fullHeight");
				tableExpand.addEventListener('click', function() {
					battleTable.parentNode.classList.toggle("b-fullHeight");
				}, false);
				// refresh table
				sortTable.refresh();
			},
			handlerProv({province, owner}, coef) {
				// province info handler
				const battleRow = document.getElementsByClassName(province.alias)[0];
				let fame_style = "t-green",
				fame = "Ɵ",
				sort = 0;
				if (bs.cw.event) {
					if (province.fame_points) {
						fame = province.fame_points;
						sort = fame;
					}
					else if (province.money_box) {
						if (owner.id == bs.clan.id) {
							fame_style = "t-red";
							fame = -Math.abs(province.money_box.risky_fame_points);
						}
						else {
							fame = `+${province.money_box.capture_fame_points}`;
						}
						sort = fame;
					}
					else if (province.is_enclave) {
						fame = province.single_province_fp;
						coef = province.fame_points_coefficient;
						sort = fame;
					}
					else if (province.enclave_neighbours_number) {
						fame = `${province.enclave_neighbours_number}/${province.number_of_enclave_provinces}`;
						sort = province.enclave_neighbours_number / province.number_of_enclave_provinces;
						// change fame column title
						document.getElementById('js-fame').textContent = "Fame & Enclaves";
					}
					else if (province.raids) {
						if (province.raids.secondary_mission_reward) {
							fame = province.raids.secondary_mission_reward;
						}
						else {
							fame_style = "";
						}
					}
					if (province.type !== "landing" || owner.id == bs.clan.id) {
						coef = "x5";
					}
				}
				if (battleRow) {
					const fameHTML = `<span class='${fame_style}'>${fame}</span>${(coef) ? ` ${coef}` : ""}`;
					battleRow.children[4].innerHTML = fameHTML;
					battleRow.children[4].dataset.sort = sort;
					if (bs.front_loc[province.front_id]) {
						battleRow.children[6].lastElementChild.innerHTML = bs.front_loc[province.front_id];
					}
					if (province.type == "auction" && battleRow.children[7].innerHTML == "Not Started") {
						battleTable.children[1].appendChild(sf.elem("tr", {
							className: `battle ${province.alias} attack auction`,
							innerHTML: `<td><a target='_blank' href='https://eu.wargaming.net/globalmap/#province/${province.alias}'>${province.name}</a> [<a class='t-tournament' target='_blank' href='https://eu.wargaming.net/globalmap/#tournament/${province.alias}'>&#9876;</a>]</td><td></td><td></td><td class='t-gold'><span></span><span class='icon icon_main icon_gold'></span></td><td class='t-fame'>${fameHTML}</td><td></td><td class='t-auction'><span>Auction</span><span>${bs.front_loc[province.front_id] ? bs.front_loc[province.front_id] : ""}</span></td><td>Not Started</td><td class='t-nextBattle' data-sort='9999'></td><td class='t-battle'>Ɵ</td><td class='t-battle t-border'>Ɵ</td>${bs.table.c}`
						}));
						battleRow.remove();
						// request tourneyData to be sure of proper visuals
						sf.request("tourneyData -- auction", `${bs.api.tourney+province.alias}&round=1`, sf.cw.handlerTourney, "cw");
					}
				}
			},
			handlerDivs(data) {
				// divisions handler
				const divsId = JSON.stringify(data).match(/\d{9}/g);
				if (divsId.includes(bs.clan.id.toString())) {
					bs.cw.globalmap = true;
					for (let _p = 0, _p_len = data.data.length; _p < _p_len; _p++) {
						const div = data.data[_p],
						battleRow = document.getElementsByClassName(div.alias)[0];
						if (!div.division) {
							if (battleRow && battleRow.classList.contains('defense')) {
								const defBattle = battleRow.getElementsByClassName("js-last")[0],
								primeTime = JSON.parse(battleRow.dataset.primeTime),
								primeTimeConc = `${Number(primeTime[0]+primeTime[1])+([0,30,100,130,200,230,300,330][Number(battleRow.children[10].innerHTML) || 0])}`;
								battleRow.children[2].dataset.sort = 5000;
								battleRow.children[8].dataset.sort = 5000;
								battleRow.children[6].firstElementChild.textContent = "No Division";
								battleRow.children[6].classList.add("t-bold");
								if (defBattle) {
									defBattle.classList.remove("t-fight");
									defBattle.classList.add("t-noFight");
									if (bs.dyn.conc[primeTimeConc]) {
										bs.dyn.conc[primeTimeConc]--;
									}
								}
							}
							else if (!battleRow) {
								bs.dyn.plan++;
								battleTable.children[1].appendChild(sf.elem("tr", {
									className: `planned ${div.alias}`,
									innerHTML: `<td><a target='_blank' href='https://eu.wargaming.net/globalmap/#province/${div.alias}'>${div.name}</a> [<a class='t-tournament' target='_blank' href='https://eu.wargaming.net/globalmap/#tournament/${div.alias}'>&#9876;</a>]</td><td></td><td></td><td class='t-gold'><span></span><span class='icon icon_main icon_gold'></span></td><td class='t-fame'>Ɵ</td><td></td><td><span>Planned</span><span></span></td><td>Planned</td><td class='t-nextBattle' data-sort='9999'></td><td class='t-battle'>Ɵ</td><td class='t-battle t-border'>Ɵ</td>${bs.table.c}`
								}));
								sf.request("batsData", bs.api.bats + div.alias, sf.cw.handlerBats, "cw", true);
							}
						}
						else {
							// sometimes future defenses wont show up in planned battles
							if (div.attackers.length > 0 && battleRow && battleRow.classList.contains('province')) {
								bs.dyn.plan++;
								battleTable.children[1].appendChild(sf.elem("tr", {
									className: `planned ${div.alias}`,
									innerHTML: `<td><a target='_blank' href='https://eu.wargaming.net/globalmap/#province/${div.alias}'>${div.name}</a> [<a class='t-tournament' target='_blank' href='https://eu.wargaming.net/globalmap/#tournament/${div.alias}'>&#9876;</a>]</td><td></td><td></td><td class='t-gold'><span></span><span class='icon icon_main icon_gold'></span></td><td class='t-fame'>Ɵ</td><td></td><td><span>Defense</span><span></span></td><td>Planned</td><td class='t-nextBattle' data-sort='9999'></td><td class='t-battle'>Ɵ</td><td class='t-battle t-border'>Ɵ</td>${bs.table.c}`
								}));
								battleRow.remove();
								sf.request("tourneyData", `${bs.api.tourney+div.alias}&round=1`, sf.cw.handlerTourney, "cw", true);
							}
						}
					}
					if (bs.dyn.plan > 0) {
						// style.textContent += ".t-battle {display: table-cell !important;}";
					}
					// refresh table
					sortTable.refresh();
				}
				else {
					document.getElementById('js-error').textContent = "Division Data not Available!";
				}
			},
			handlerFreepass(data) {
				const battleRow = document.getElementsByClassName(data.province_id)[0],
				lastGroup = data.battles[data.battles.length - 1],
				freeRound = battleRow.children[7].innerHTML == "Not Started" && lastGroup && lastGroup.is_fake && lastGroup.first_competitor.id == bs.clan.id;

				// check if no opponent - free round
				if (freeRound) {
					battleRow.children[7].textContent = "Free Round";
					battleRow.children[7].classList.add("t-bold");
					battleRow.classList.add("freePass");
				}
			},
			handlerRaids(data) {
				// raids handler for campaign events
				for (let _r = 0, _r_len = data.length; _r < _r_len; _r++) {
					const raid = data[_r],
					battleRow = document.getElementsByClassName(raid.province.id)[0],
					fame = raid.fame_points,
					bonus = raid.bonus_fame_points,
					sort = fame + bonus,
					coef = raid.battle_coef;
					let denyFame = 0;
					if (battleRow) {
						denyFame = parseFloat(battleRow.children[4].innerHTML.match(/\d+/));
						battleRow.children[4].innerHTML = `<span>${fame} + ${bonus}${(denyFame) ? ` + ${denyFame}` : ""}</span> ${coef}`;
						battleRow.children[4].dataset.sort = sort + denyFame;
					}
				}
			},
			handlerError(name, data) {
				// error handler
				switch (name) {
					case ("mainData"):
						battleTable.children[1].appendChild(sf.elem("tr", {
							className: "t-cwText",
							innerHTML: "<td colspan='23'>Clan ID Error</td>"
						}));
						inputSpan.lastElementChild.textContent = "Clan ID invalid!";
						localStorage.removeItem("battles_clanid");
						break;
					case ("divsData"):
						document.getElementById('js-error').textContent = "Division Data not Available!";
						break;
					default:
						break;
				}
			},
			time(hour, min, type) {
				// time converter
				let time = hour + bs.time.o;
				if (time >= 24) {
					time -= 24;
				}
				else if (time <= 0) {
					time += 24;
				}
				if (type == "s") {
					time = `t-${time}_${min}${(time === 0 && min == "00") ? " t-24_00" : ""}`;
				}
				return time;
			},
			timer() {
				// timestamp handler
				const dateNow = new Date(),
					time = {
						h: sf.cw.time(16) - dateNow.getHours(),
						m: 60 - dateNow.getMinutes() - 1,
						s: 60 - dateNow.getSeconds() - 1
					},
					timeSpan = document.getElementById('js-timePrime');
				if (timeJump && !bs.cw.event && bs.cw.status) {
					timeSpan.textContent = "No Event Running";
					timeSpan.classList.add("t-bold");
					clearInterval(timeInterval);
				}
				else if (time.h >= 0 && (time.s > 0 || time.m < 15)) {
					timeSpan.textContent = `${((time.h > 0) ? `${time.h} Hours, ` : "")+((time.m > 0) ? `${time.m} Mins, ` : "")+time.s} Secs`;
				}
				else if (time.h < 0 && bs.cw.bats !== "Ɵ") {
					if (bs.cw.bats === 0) {
						timeSpan.textContent = "No Planned Battles";
						timeSpan.classList.add("t-bold");
						clearInterval(timeInterval);
					}
					else {
						timeSpan.classList.add("h-shadow");
						timeSpan.innerHTML = "<span style='color:#ff0000;'>X</span><span style='color:#ff2a00;'></span><span style='color:#ff5500;'>e</span><span style='color:#ff7f00;'></span><span style='color:#ffaa00;'>n</span> <span style='color:#ffff00;'>i</span><span style='color:#aaff00;'></span><span style='color:#55ff00;'>s</span> <span style='color:#00ff80;'>a</span> <span style='color:#00aaff;'>f</span><span style='color:#0055ff;'></span><span style='color:#0000ff;'>g</span><span style='color:#2e00ff;'></span><span style='color:#5d00ff;'>t</span><span style='color:#8b00ff;'>.</span>";
						if (bs.cw.current > 0) {
							document.getElementById('js-provStatus').textContent = "Next Opponent (ELO)";
						}
						clearInterval(timeInterval);
					}
				}
				else {
					timeSpan.textContent = "Hold on a sec...";
				}
				timeJump = true;
			},
			mapFix(name) {
				// map name fixer
				const fixedNames = {
					"114_czech/name": "Pilsen"
				};
				return (fixedNames[name]) ? fixedNames[name] : name;
			},
			updater(force) {
				// updater handler
				const dateNow = new Date(),
					newDate = {
						h: dateNow.getHours(),
						m: dateNow.getMinutes(),
						o: (dateNow.getTimezoneOffset() > 0 ? -Math.abs(dateNow.getTimezoneOffset()) : Math.abs(dateNow.getTimezoneOffset())) / 60
					},
					newTime = ((bs.time.m >= 15 && bs.time.m <= 45) ? [bs.time.h, "30"] : ((bs.time.m <= 15) ? [bs.time.h, "00"] : [(bs.time.h + 1), "00"]));
				if (force || (bs.time.r[0] !== newTime[0] || bs.time.r[1] !== newTime[1])) {
					bs.time = newDate;
					bs.time.r = newTime;
					bs.time.t = `${bs.time.r[0]}_${bs.time.r[1]}`;
					battleTable.lastElementChild.firstElementChild.innerHTML = `<td></td><td></td><td></td><td></td><td></td><td></td><td></td><td></td><td></td><td></td>${bs.table.c}`;
					styleDynTimezone.textContent = `
					.b-battles .b-battles-holder .t-battles tr .t-${bs.time.t} {background-color: rgba(254,252,223, 0.5); border-left: 1px solid #808080; border-right: 1px solid #808080;}
					.b-battles .b-battles-holder .t-battles tr .t-${bs.time.t} + th, .b-battles .b-battles-holder .t-battles tr .t-${bs.time.t} + td {background-color: rgba(224,223,218, 0.5); border-right: 1px solid #808080;}
				`;
					bs.dyn = {
						conc: {},
						prov: [],
						plan: 0,
						check: 0,
						gold: 0
					};
					sf.request("mainData", bs.api.clan, sf.cw.handlerMain, "cw");
					// insert update timestamp
					document.getElementById('js-batttleUpdate').textContent = new Date().toLocaleTimeString("en-GB");
				}
			}
		},
		buttonHide(name, item, img, {classList}) {
			const buttonDiv = sf.elem("div", {
				className: "b-idealscript b-buttonHide",
				innerHTML: `<a target='_blank' href='#' onClick='return false;'><span class='icon icon_main icon_${img}'></span><span>Hide</span> ${name} Panel</a><span> &#8226; </span>`
			});
			let buttonState = JSON.parse(localStorage.getItem(item)) || false;
			buttonDiv.firstElementChild.addEventListener('click', function() {
				this.children[1].textContent = (buttonState) ? "Hide" : "Show";
				classList.toggle("b-display-none");
				buttonState = (buttonState) ? false : true;
				localStorage.setItem(item, buttonState);
			}, false);
			if (buttonState) {
				buttonDiv.firstElementChild.children[1].textContent = "Show";
				classList.toggle("b-display-none");
			}
			return buttonDiv;
		},
		request(name, api, handler, mode, extra) {
			// request handler
			GM.xmlHttpRequest({
				method: "GET",
				url: api,
				headers: {
					"Accept": "application/json",
					"Client-ID": mode == "tw" ? bs.tw.key : undefined
				},
				onload(resp) {
					const data = JSON.parse(resp.responseText);
					if (resp.status == 200) {
						if (bs.debug) {
							console.info(name, data, api, new Date().toLocaleTimeString("en-GB"));
						}
						handler(data, extra);
					}
					else {
						console.error("Error with API", name, api, resp);
						sf[mode].handlerError(name);
					}
				},
				onerror(resp) {
					console.error("Error accessing API", name, api, resp);
				}
			});
		}
	};

// inserting style into head
style = sf.elem("style", {
	className: "idealScript",
	type: "text/css",
	textContent: `
.b-panel {display: table; margin: 15px auto 15px;}
.p-helper {display: table-cell; width: 100%;}
.b-buttonHide {display: inline;}
/* icon rules */
.b-idealscript .icon {display: inline-block; height: 16px; width: 16px; vertical-align: text-top;}
.b-idealscript.b-buttonHide .icon {display: inline-block; height: 16px; width: 16px; margin-right: 3px; vertical-align: sub;}
.icon_main {background: url(${img.main}) no-repeat 0 0;}
.b-idealscript .icon_flag {background: url(${img.flags}) no-repeat 0 0; height: 11px; margin: 0 1px; vertical-align: baseline;}
.b-idealscript .icon_clan {background: url(${img.clans}) no-repeat 0 0; margin: 0 1px; vertical-align: text-bottom;}
/* icons */
/* main */
.b-idealscript .icon_chat {}
.b-idealscript .icon_twitch {background-position: -16px 0;}
.b-idealscript .icon_fork {background-position: -32px 0;}
.b-idealscript .icon_wg {background: url(${img.main}) no-repeat -48px 0;}
.b-idealscript .icon_cw {background-position: -64px 0;}
.b-idealscript .icon_gold {background-position: -82px 0; margin-left: 2px;}
.b-idealscript .icon_wot {background-position: -98px 0;}
/* eu */
.b-idealscript .icon_ideal {}
.b-idealscript .icon_s3al {background-position: -16px 0;}
.b-idealscript .icon_fame {background-position: -32px 0;}
.b-idealscript .icon_rsop {background-position: -48px 0;}
.b-idealscript .icon_ptsd {background-position: -64px 0;}
.b-idealscript .icon_fnd {background-position: 0 -16px;}
.b-idealscript .icon_s4 {background-position: -16px -16px;}
.b-idealscript .icon_omni {background-position: -32px -16px;}
.b-idealscript .icon_santi {background-position: -48px -16px;}
.b-idealscript .icon_vter {background-position: -64px -16px;}
.b-idealscript .icon_vole {background-position: 0 -32px;}
.b-idealscript .icon_e50m {background-position: -16px -32px;}
/* ru */
.b-idealscript .icon_mamb {background-position: -32px -32px;}
.b-idealscript .icon_syb {background-position: -48px -32px;}
/* na */
.b-idealscript .icon_bulba {background-position: -64px -32px;}
.b-idealscript .icon_sela {background-position: 0 -48px;}
/* esl teams */
.b-idealscript .icon_kb {background-position: -16px -48px;}
.b-idealscript .icon_penta {background-position: -32px -48px;}
.b-idealscript .icon_navi {background-position: -48px -48px;}
/* other */
.b-idealscript .icon_circ {background-position: 0 -64px;}
.b-idealscript .icon_foch {background-position: -16px -64px;}
.b-idealscript .icon_qb {background-position: -32px -64px;}
/* flags */
.b-idealscript .icon_en {}
.b-idealscript .icon_de {background-position: -16px 0;}
.b-idealscript .icon_no {background-position: -32px 0;}
.b-idealscript .icon_se {background-position: -48px 0;}
.b-idealscript .icon_fi {background-position: -64px 0;}
.b-idealscript .icon_cz {background-position: 0 -11px;}
.b-idealscript .icon_pl {background-position: -16px -11px;}
.b-idealscript .icon_ru {background-position: -32px -11px;}
.b-idealscript .icon_us {background: url(${img.flags}) no-repeat 0 -22px; height: 11px; margin: 0 1px; vertical-align: baseline;}
.b-idealscript .icon_au {background-position: -16px -22px;}
.b-idealscript .icon_ca {background-position: -32px -22px;}
/* twitch stream status */
.b-streams {display: table-cell; font-size: 11px; max-width: 400px; min-width: 340px; padding-left: 10px;}
.b-streams .s-helper {display: inline-block;}
.b-streams .h-streams {background-color: #D1E8FE; border: 1px solid;border-radius: 10px 10px 0 0; margin-top: -2px; text-align: center;}
.b-streams .h-streams .h-streams-info {border-bottom: 1px solid #F5F5F5; margin: 0 auto; padding: 4px 0px; width: 325px;}
.b-streams .h-streams .h-streams-info > * {display: inline-block;}
.b-streams .h-streams .h-streams-info .h-left {width: 10%;}
.b-streams .h-streams .h-streams-info .h-mid {width: 80%;}
.b-streams .h-streams .h-streams-info .h-right {width: 10%;}
.b-streams .h-streams .h-streams-info img {vertical-align: bottom;}
.b-streams .h-streams .h-streams-head {border-spacing: 0; text-align: center; width: 323px;}
.b-streams .h-streams .h-streams-head tr th {line-height: 20px;}
.b-streams .h-streams .h-streams-head tr th:nth-of-type(1) {width: 0px;}
.b-streams .h-streams .h-streams-head tr th:nth-of-type(2) {width: 145px;}
.b-streams .h-streams .h-streams-head tr th:nth-of-type(3) {width: 75px;}
.b-streams .h-streams .h-streams-head tr th:nth-of-type(4) {width: 60px;}
.b-streams .h-streams .h-streams-head tr th:nth-of-type(5) {width: 40px;}
.b-streams .b-streams-holder {border-left: 1px solid; border-right: 1px solid; max-height: ${(bs.web.chrome) ? "641" : "650"}px; overflow-y: auto;}
.b-streams .b-streams-holder .t-streams {border-spacing: 0; text-align: center; width: 100%;}
.b-streams .b-streams-holder .t-streams tr:nth-child(even) {background-color: #FBFBFB;}
.b-streams .b-streams-holder .t-streams tr:nth-child(odd) {background-color: #EFEFEF;}
.b-streams .b-streams-holder .t-streams tr:hover {background-color: #FEFAC0;}
.b-streams .b-streams-holder .t-streams tr td {line-height: 19px; border-bottom: 1px dotted #B1B1B1;}
.b-streams .b-streams-holder .t-streams tr td.td_info {width: 0px;}
.b-streams .b-streams-holder .t-streams tr td.td_name {width: 145px; position: relative;}
.b-streams .b-streams-holder .t-streams tr td.td_clan {width: 75px;}
.b-streams .b-streams-holder .t-streams tr td.td_status {width: 60px;}
.b-streams .b-streams-holder .t-streams tr td.td_flag {width: 40px;}
.b-streams .b-streams-holder .t-streams .streamIdeal tr:last-of-type td {border-bottom: 1px solid #CC5BFF;}
.b-streams .b-streams-holder .t-streams .streamSeperator td {border-top: 1px solid #CD2911;}
.b-streams .b-streams-holder .t-streams td.s-online {color: #4D7326;}
.b-streams .b-streams-holder .t-streams td.s-offline {color: #930D0D;}
.b-streams .b-streams-holder .t-streams .icon_wot {margin-top: 1px; opacity: 0.25; position: absolute; right: 0;}
.b-streams .b-streams-holder .t-streams .b-info {background-color: #FEFAC0; border: 1px dotted #B1B1B1; display: none; margin: -11px 0px 0px -282px; padding-bottom: 8px; position: absolute; width: 280px;}
.b-streams .b-streams-holder .t-streams .b-info > span {display: block;}
.b-streams .b-streams-holder .t-streams .b-info .s-status {font-weight: bold;}
.b-streams .b-streams-holder .t-streams .b-info .s-game {}
.b-streams .b-streams-holder .t-streams .b-info .s-img img {width: 200px;}
.b-streams .b-streams-holder .t-streams tr:hover .b-info {display: block;}
.b-streams .b-streams-holder .streamHide {display: none;}
.b-streams .b-streams-holder img {display: block; margin: 0 auto;}
.b-streams .f-streams {background-color: #D1E8FE; border: 1px solid; border-top: 0; border-radius: 0 0 10px 10px; padding: 3px 0px; text-align: center;}
.b-streams .f-streams .f-battles-info > * {display: inline-block;}
/* battle scheduler */
.b-battles {border: 1px solid; border-radius: 10px; display: table; font-size: 11px; width: 90vw; min-width: 1280px; margin: 15px auto 15px;}
.b-battles .h-battles {background-color: #D1E8FE; border-radius: 10px 10px 0 0;}
.b-battles .h-battles .h-battles-info {border-bottom: 1px solid #F5F5F5; margin: 0 auto; padding: 4px 0px; width: 98%;}
.b-battles .h-battles .h-battles-info > * {display: inline-block;}
.b-battles .h-battles .h-battles-info .h-left {width: 17%;}
.b-battles .h-battles .h-battles-info .h-left a {margin-right: 5px;}
.b-battles .h-battles .h-battles-info .h-mid {text-align: center; width: 66%;}
.b-battles .h-battles .h-battles-info .h-mid #js-eventName {text-transform: capitalize;}
.b-battles .h-battles .h-battles-info .h-right {text-align: right; width: 17%;}
.b-battles .h-battles .h-battles-info .h-right .h-expand {cursor: pointer; font-size: 2em; line-height: 10px; vertical-align: sub;}
.b-battles .h-battles .h-battles-info .h-right .h-force {cursor: pointer; margin-right: 15px;}
.b-battles .h-battles .h-battles-info .h-right .h-force input {height: 15px; vertical-align: bottom;}
.b-battles .h-battles .h-battles-info .h-right #js-update {cursor: pointer; margin-right: 17px;}
.b-battles .h-battles .h-battles-info img {max-height: 16px; vertical-align: bottom;}
.b-battles .h-battles .h-battles-info .h-shadow {font-weight: bold; text-shadow: 0px 0px 1px rgba(27,27,28, 1), 0px 0px 2px rgba(27,27,28, 1);}
.b-battles .b-battles-holder {background-color: #FBFBFB; max-height: 221px; overflow-y: auto;}
.b-battles .b-battles-holder.b-fullHeight {max-height: unset;}
.b-battles .b-battles-holder .t-battles {border-spacing: 0; text-align: center; width: 100%;}
.b-battles .b-battles-holder .t-battles .t-time {display: none; position: relative;}
.b-battles .b-battles-holder .t-battles thead tr, .b-battles .b-battles-holder .t-battles tfoot tr {background-color: #D1E8FE;}
.b-battles .b-battles-holder .t-battles .sort-up, .b-battles .b-battles-holder .t-battles .sort-down {color: #C600C6;}
.b-battles .b-battles-holder .t-battles tbody tr:nth-child(even) {background-color: #FBFBFB;}
.b-battles .b-battles-holder .t-battles tbody tr:nth-child(odd) {background-color: #EFEFEF;}
.b-battles .b-battles-holder .t-battles tbody tr:hover {background-color: #FEFAC0;}
.b-battles .b-battles-holder .t-battles tr .t-border {border-right: 2px solid rgba(194, 173, 173, 0.5);}
.b-battles .b-battles-holder .t-battles tr .t-tournament {font-size: 15px; font-weight: bold;}
.b-battles .b-battles-holder .t-battles tr th {border-bottom: 1px solid #808080; line-height: 20px;}
.b-battles .b-battles-holder .t-battles tr td {line-height: 19px; border-bottom: 1px dotted #B1B1B1;}
.b-battles .b-battles-holder .t-battles tr td:first-of-type {max-width: 100px; overflow: hidden; padding: 0 5px; text-overflow: ellipsis; white-space: nowrap;}
.b-battles .b-battles-holder .t-battles tr td.t-title {font-weight: bold;}
.b-battles .b-battles-holder .t-battles tr td.t-good {color: #4D7326;}
.b-battles .b-battles-holder .t-battles tr td.t-bad {color: #930D0D;}
.b-battles .b-battles-holder .t-battles tr td.t-plan {color: #FFE400;}
.b-battles .b-battles-holder .t-battles tr td.t-fight {color: #4D7326; font-size: 15px; font-weight: bold;}
.b-battles .b-battles-holder .t-battles tr td.t-auction {color: #2F396A; font-weight: bold;}
.b-battles .b-battles-holder .t-battles tr td.t-noFight {color: #808080; font-size: 14px;}
.b-battles .b-battles-holder .t-battles tr td.t-fight.t-noOwner {color: #808080;}
.b-battles .b-battles-holder .t-battles tr td.t-error {color: #CD2911;}
.b-battles .b-battles-holder .t-battles tr.t-cwText td {font-size: 26px; line-height: 54px;}
.b-battles .b-battles-holder .t-battles tr.timeShift td:nth-child(3), .b-battles .b-battles-holder .t-battles tr.timeShift td:nth-child(9) {color: #25931A;}
.b-battles .b-battles-holder .t-battles tr.timeShift td .t-timeShift {font-size: 8px; position: absolute;}
.b-battles .b-battles-holder .t-battles img {height: 16px; margin-bottom: 2px; vertical-align: bottom;}
.b-battles .b-battles-holder .t-battles tfoot tr td {border-bottom: 1px solid #F5F5F5; border-top: 1px solid #808080; font-weight: bold;}
.b-battles .f-battles {background-color: #D1E8FE; border-radius: 0 0 10px 10px; padding: 3px 0px;}
.b-battles .f-battles .f-battles-info {margin: 0 auto; width: 98%;}
.b-battles .f-battles .f-battles-info > * {display: inline-block; vertical-align: middle;}
.b-battles .f-battles .f-battles-info .f-left {width: 20%;}
.b-battles .f-battles .f-battles-info .f-left > span {margin-left: 5px;}
.b-battles .f-battles .f-battles-info .f-left .f-inputText {color: #CD2911;}
.b-battles .f-battles .f-battles-info .f-left .f-inputText.t-green {color: #4D7326;}
.b-battles .f-battles .f-battles-info .f-mid {text-align: center; width: 60%;}
.b-battles .f-battles .f-battles-info .f-mid table {margin: 0 auto;}
.b-battles .f-battles .f-battles-info .f-mid table td {padding: 0 10px;}
.b-battles .f-battles .f-battles-info .f-mid #js-goldInfo {margin-left: 3px;}
.b-battles .f-battles .f-battles-info .f-mid #js-error {display: block;}
.b-battles .f-battles .f-battles-info .f-right {text-align: right; width: 20%;}
.b-battles .f-battles .f-battles-info img {max-height: 16px; vertical-align: bottom;}
/* toggle overrides */
.t-elo {margin-left: 3px;}
.t-gold {color: #FFC364;}
.b-display-none {display: none;}
.b-display-block {display: block;}
/* .t-battle {display: none;} */
.t-bold {font-weight: bold;}
.t-green {color: #4D7326;}
.t-red {color: #930D0D;}
`
});
document.head.appendChild(style);
	
	if (/games.php/.test(window.location.href)) {
		styleDynTimezone = sf.elem("style", {
			className: "styleDynTimezone",
			type: "text/css",
			textContent: `
				.b-battles .b-battles-holder .t-battles tr .t-${bs.time.t} {background-color: rgba(254,252,223, 0.5); border-left: 1px solid #808080; border-right: 1px solid #808080;}
				.b-battles .b-battles-holder .t-battles tr .t-${bs.time.t} + th, .b-battles .b-battles-holder .t-battles tr .t-${bs.time.t} + td {background-color: rgba(224,223,218, 0.5); border-right: 1px solid #808080;}
			`
		});
		styleDynBattles = sf.elem("style", {
			className: "styleDynBattles",
			type: "text/css"
		});
		styleDynFinales = sf.elem("style", {
			className: "styleDynFinales",
			type: "text/css"
		});
		document.head.appendChild(styleDynTimezone);
		document.head.appendChild(styleDynBattles);
		document.head.appendChild(styleDynFinales);

		// prepare static html
		const page_body = document.getElementById("page-body"),
		battlePanel = sf.elem("div", {
			className: "b-idealscript b-battles",
			innerHTML: `<div class='h-battles'><div class='h-battles-info'><div class='h-left'><a target='_blank' href='https://greasyfork.org/en/scripts/20094-ideal-scripts'><span class='icon icon_main icon_fork'></span></a><a target='_blank' href='https://eu.wargaming.net/clans/wot/${bs.clan.id}/globalmap'><span class='icon icon_main icon_wg'></span></a><a target='_blank' href='https://eu.wargaming.net/globalmap/'><span class='icon icon_main icon_cw'></span></a></div><div class='h-mid'><span id='js-eventName'></span><span id='js-clan'></span><span id='js-timePrime'>Ɵ</span></div><div class='h-right'><span id='js-update'>Update</span><span class='h-force'><label id='js-force'>Campaign Only: <input type='checkbox' name='force'></label></span><span class='h-expand' id='js-expand'></div></div></div><div class='b-battles-holder'><table class='t-battles sortable'><thead><tr><th>Province</th><th>Map</th><th data-sort-method='number' data-sort-order='desc'>Timezone</th><th class='t-gold'>Gold</th><th class='t-fame' id='js-fame'>Fame</th><th>Owner (ELO)</th><th>Type</th><th id='js-provStatus'>Status</th><th class='t-battle sort-default'  data-sort-method='number' data-sort-order='desc'>Next Battle</th><th class='t-battle'>Attackers</th><th class='t-battle t-border'>Turns</th></tr></thead><tbody></tbody><tfoot id ='js-footer'><tr><td></td><td></td><td></td><td></td><td></td><td></td><td></td><td></td><td></td><td></td></tr></tfoot></table></div><div class='f-battles'><div class='f-battles-info'><div class='f-left' id='js-input'><input type='text' name='id' placeholder='Clan ID' value='${bs.clan.id}' list='clanid' maxlength='9' size='9'><datalist id='clanid'>${clans.join("")}</datalist><span class='f-inputImage'><img src='https://wot-life.com/img/clanicon/${bs.clan.id}.png'></span><span class='f-inputText'>${(bs.clan.id !== bs.clan.ideal) ? "</span><span class='f-inputBack'>Back to <a target='_blank' href='#' onclick='localStorage.removeItem(\"battles_clanid\"); location.reload();'>[IDEAL]</a></span>" : ""}</div><div class='f-mid'><table id='js-tableStats' class='b-display-none'><tr class='t-bold'><td>Current Battles: <span id='js-battles'>0</span></td><td>Simultaneously: <span id='js-battlesConc'>0</span></td><td>Planned: <span id='js-battlesPlan'>0</span></td></tr><td>Owned Provinces: <span id='js-provs'>0</span></td><td colspan='2'>Gold Income: <span class='t-gold' id='js-gold'>0</span><span class='icon icon_main icon_gold'></span><span id='js-goldInfo'></span></td></tr></tr><td><span class='t-bold t-green'>&#9876;</span> Tournament Battle</td><td><span class='t-bold t-green'>♖</span> Battle with Owner</td><td><span class='t-bold t-green'>&#9876;+</span> Timeshift</td></tr></tr><tr><td colspan='3'>Last Updated: <span id='js-batttleUpdate'>Ɵ</span> [UTC${(bs.time.o >= 0) ? "+" : ""}${bs.time.o}]</td></tr><td colspan='3'><span id='js-error'></span></td></tr></table></div><div class='f-right'>v${bs.vers}</span></div></div></div>`
		});
		page_body.innerHTML = "";
		page_body.appendChild(battlePanel);

		// toggle button to hide/show new panels and chat
		// const rightSide = document.getElementsByClassName("rightside")[3];
		// rightSide.prepend(sf.buttonHide("Chat", "chat_panelState", "chat", parentPanel.firstElementChild));
		// rightSide.prepend(sf.buttonHide("Stream", "stream_panelState", "twitch", parentPanel.lastElementChild));
		// rightSide.prepend(sf.buttonHide("Battles", "battles_panelState", "cw", battlePanel));

		// table references
		battleTable = battlePanel.children[1].firstElementChild;


		// time cells for header and body rows
		const timeFragment = document.createDocumentFragment();
		for (let _tc = 0, _tc_len = bs.table.eu.length; _tc < _tc_len; _tc++) {
			const t = bs.table.eu[_tc],
			times = [sf.cw.time(t, "00", "s"), `${sf.cw.time(t)}:00`, sf.cw.time(t, "30", "s"), `${sf.cw.time(t)}:30`];
			timeFragment.appendChild(sf.elem("th", {
				className: `t-time ${times[0]}`,
				innerHTML: times[1]
			}));
			bs.table.c += `<td class='t-time ${times[0]}' data-time='${times[1]}'></td>`;
			if (_tc !== _tc_len - 1) {
				timeFragment.appendChild(sf.elem("th", {
					className: `t-time ${times[2]}`,
					innerHTML: times[3]
				}));
				bs.table.c += `<td class='t-time ${times[2]}' data-time='${times[3]}'></td>`;
			}
		}
		battleTable.firstElementChild.firstElementChild.appendChild(timeFragment);
		battleTable.lastElementChild.firstElementChild.innerHTML += bs.table.c;

		// button to update
		const updateSpan = document.getElementById('js-update');
		updateSpan.addEventListener('click', function() {
			sf.cw.updater(true);
		}, false);

		// button to force load cw battles
		const forceSpan = document.getElementById('js-force');
		forceSpan.lastElementChild.checked = (bs.cw.status) ? true : false;
		forceSpan.addEventListener('click', function() {
			localStorage.setItem("battles_force", this.lastElementChild.checked);
			location.reload();
		}, false);

		// add custom input for checking other clans
		inputSpan = document.getElementById('js-input');
		inputSpan.firstElementChild.addEventListener('input', function() {
			const value = this.value;
			if (value.length == 9 && value[0] == "5") {
				inputSpan.children[2].firstElementChild.src = `https://wot-life.com/img/clanicon/${value}.png`;
				inputSpan.children[3].innerHTML = "Stored! <a target='_blank' href='#' onclick='location.reload();'>Refresh?</a>";
				inputSpan.children[3].classList.add("t-green");
				if (inputSpan.children.length == 5) {
					inputSpan.children[4].textContent = "";
				}
				localStorage.setItem("battles_clanid", value);
			}
			else if (value.length > 0 && value[0] !== "5") {
				inputSpan.children[3].textContent = "Not Valid!";
			}
			else {
				inputSpan.children[3].textContent = "Missing";
			}
		}, false);

		// add intervals for time and round updater
		timeInterval = setInterval(sf.cw.timer, 1000); // 1 second
		updateInterval = setInterval(sf.cw.updater, 120000); // 2 minutes

		// activate tablesort function for battle scheduler
		if (Tablesort) {
			// Numeric sort
			Tablesort.extend('number', function(item) {
				return item.match(/^-?(\d)*-?([,\.]){0,1}-?(\d)+([E,e][\-+][\d]+)?%?$/); // Number
			}, function(a, b) {
				a = parseFloat(a);
				b = parseFloat(b);

				a = isNaN(a) ? 0 : a;
				b = isNaN(b) ? 0 : b;
				return a - b;
			});
			sortTable = new Tablesort(battleTable);
		}
		else {
			window.alert("Error activating tablesort, please refresh - if this shit continues, poke Orrie");
		}

		// insert update status - will crash the tablesorter if inserted earlier
		battleTable.children[1].appendChild(sf.elem("tr", {
			className: "t-cwText",
			innerHTML: "<td colspan='23'>Updating...</td>"
		}));

		// activate handler functions through xmlhttp request
		// send request to wargaming api to see if an event is running
		sf.request("eventData", bs.api.event, sf.cw.handlerEvent, "cw");
	}
	else {
		// prepare static html
		const page_body = document.getElementById("page-body"),
		streamers = streams.length,
		parentPanel = sf.elem("div", {
			className: "b-panel"
		}, [
			document.getElementsByClassName("tablebg")[0],
			sf.elem("div", {
				className: "b-idealscript b-streams",
				innerHTML: `<span class='s-helper'></span><div class='h-streams'><div class='h-streams-info'><div class='h-left'><a target='_blank' href='https://www.twitch.tv/directory/game/World%20of%20Tanks'><span class='icon icon_main icon_twitch'></span></a></div><div class='h-mid'>Total Streams: ${streamers} &#8226; Streams Online: <span id='js-streamsOnline'>0</span></div><div class='h-right'><a target='_blank' href='https://greasyfork.org/en/scripts/20094-ideal-scripts'><span class='icon icon_main icon_fork'></span></a></div></div><table class='h-streams-head'><thead><tr><th></th><th>Name</th><th>Clan/Team</th><th>Viewers</th><th>Flag</th></thead></table></div><div class='b-streams-holder'><table class='t-streams'><tbody class='streamIdeal'></tbody><tbody class='streamOnline'></tbody><tbody class='streamAmerica'><tbody class='streamRussia'><tbody class='streamCommunity'></tbody><tbody class='streamSeperator'><tr><td colspan='5'></td></tr></tbody><tbody class='streamOffline'></tbody></table></div><div class='f-streams'><div class='f-mid'>Updates Every 2 Mins &#8226; Last Updated: <span id='js-streamUpdate'>Updating...</span></div></div>`
			})
		]);
		page_body.insertBefore(parentPanel, page_body.children[3]);
		// table references
		streamTable = parentPanel.lastElementChild.children[2].firstElementChild.childNodes;

		// fill stream table
		const streamString = [],
		streamFragment = document.createDocumentFragment();
		for (let _s = 0; _s < streamers; _s++) {
			const stream = streams[_s],
				streamRow = sf.elem("tr", {
					className: stream[4],
					id: stream[1],
					innerHTML: `<td class='td_info'></td><td class='td_name'><a target='_blank' href='https://www.twitch.tv/${stream[1]}'>${stream[0]}</a></td><td class='td_clan'>${(stream[2].constructor === Array) ? `<span class='icon icon_clan icon_${stream[2][0]}' alt='clan'></span><span class='icon icon_clan icon_${stream[2][1]}' alt='team'></span>` : `<span class='icon icon_clan icon_${stream[2]}' alt='clan'></span>`}</td><td class='td_status s-offline'>Offline</td><td class='td_flag'>${(stream[3].constructor === Array) ? `<span class='icon icon_flag icon_${stream[3][0]}' alt='flag'></span><span class='icon icon_flag icon_${stream[3][1]}' alt='flag'></span>` : `<span class='icon icon_flag icon_${stream[3]}' alt='flag'></span>`}</td>`
				});
			streamString.push(stream[1]);
			streamFragment.appendChild(streamRow);
		}
		streamTable[6].appendChild(streamFragment);
		// create twitch api url
		bs.tw.api += streamString.join(",");

		// toggle button to hide/show new panels and chat
		const rightSide = document.getElementsByClassName("rightside")[3];
		rightSide.prepend(sf.elem("div", {
			className: "b-idealscript b-buttonHide",
			innerHTML: `<a target='_blank' href='http://idealclan.eu/games.php'><span class="icon icon_main icon_cw"></span>Battle Schedule</a><span> &#8226; </span>`
		}));

		// activate handler functions through xmlhttp request
		// send request to twitch api with 2mins frequency
		sf.request("twitchData", bs.tw.api, sf.tw.handlerMain, "tw");
		window.setInterval(sf.tw.updater, 120000);
	}
}(window));