Greasy Fork is available in English.

IDEAL Scripts

All IDEAL Scripts for idealclan.eu merged into one

// ==UserScript==
// @name        IDEAL Scripts
// @version     1.29
// @description All IDEAL Scripts for idealclan.eu merged into one
// @author      Orrie
// @namespace   http://idealclan.eu/viewtopic.php?f=7&t=1791#p57645
// @icon        https://i.imgur.com/9GeRjIp.png
// @include     http*://*idealclan.eu/*
// @grant       GM_xmlhttpRequest
// @grant       GM.xmlHttpRequest
// @connect     eu.wargaming.net
// @connect     api.worldoftanks.eu
// @connect     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
	let battleTable, updateInterval, style, sortTable = false, time, styleDynBattles, styleDynFinales, inputSpan, timeJump = false, timeInterval, styleDynTimezone;
	// script variables
	const script = {
		vers: GM.info ? GM.info.script.version : "",
		cw: {
			status: JSON.parse(localStorage.getItem("battles_force")) || false,
			event: false,
			gold: true,
			tier: "Ɵ",
			battles: "Ɵ",
			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],
			html: "",
			classes: []
		},
		front_loc: {
			"league1": " (BF)",
			"league2": " (AF)",
			"league3": " (EF)"
		},
		app_id: "a7595640a90bf2d19065f3f2683b171c",
		time: (function () {
			const date = new Date(),
			hour = date.getHours(),
			min = date.getMinutes(),
			offset = (date.getTimezoneOffset() > 0 ? -Math.abs(date.getTimezoneOffset()) : Math.abs(date.getTimezoneOffset())) / 60,
			roundTime = min >= 15 && min <= 45 ? [hour, "30"] : min <= 15 ? [hour, "00"] : [hour + 1, "00"],
			classes = `${roundTime[0]}_${roundTime[1]}`;
			return { hour, min, offset, roundTime, classes };
		})(),
		api: {
			event: "https://api.worldoftanks.eu/wot/globalmap/events/?application_id=",
			clan: "https://eu.wargaming.net/globalmap/game_api/clan/",
			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=",
			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
			stats: "https://worldoftanks.eu/en/clanwars/rating/alley/users/stats/myclan/" // need authentication from worldoftanks.eu
		},
		img: {
			// image data uri
			main: "",
			bonds: ""
		},
		symbols: {
			owner:"🏰", // ♖
			attack: "⚔️", // ⚔ - &#9876;
			defence: "🛡️",
			tournament: "📋",
            time: "⌛"
		},
		clans: [
			"<option value='500010805'>IDEAL</option>",
			"<option value='500152606'>EDITS</option>",
			"<option value='500025989'>FAME</option>",
			"<option value='500138915'>W31RD</option>",
			"<option value='500073739'>GX</option>"
		],
		debug: document.getElementsByClassName("headerbnl")[0].innerHTML.match(/Orrie/) ? true : false,
		debugColors: {
			"handlerEvent":   "#BD00FF",
			"handlerMain":    "#BD00FF",
			"handlerClan":    "#BD00FF",
			"handlerBats":    "#BD00FF",
			"handlerTourney": "#00FF9F",
			"handlerProvs":   "#00B8FF",
			"handlerDivs":    "#00B8FF",
			"handlerFame":    "#6678FF",
			"handlerRaids":   "#6678FF"
		}
	};
	if (script.debug) {
		console.info("script", script, new Date().toLocaleTimeString("en-GB"));
	}
	// script functions
	const handlerEvent = function({data}) {
		// event checker
		// check if active event exists
		const event = data ? data[0] : "error";
		if (event.status == "ACTIVE" || !script.cw.status) {
			if (event.status == "ACTIVE") {
				script.cw.event = true;
				script.cw.tier = 10;
				document.getElementById('js-eventName').innerHTML = `${event.event_name.replace(/_/g," ")} • `;
				if (script.clan.id == script.clan.ideal) {
					request({name: "handlerClan", api: `${script.api.stats}?event_id=${event.event_id}&front_id=${event.event_id}_bg`, handler: handlerClan});
				}
			}
			request({name: "handlerMain", api: `${script.api.clan}${script.clan.id}/battles`, handler: handlerMain});
			document.getElementById('js-tableStats').classList.remove("b-display-none");
		}
		else {
			// empty table
			battleTable.children[1].innerHTML = "";
			battleTable.children[1].appendChild(_createElement("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");
	},
	handlerClan = function(resp) {
		if (resp.clan_stats) {
			document.getElementById('js-tableEvent').classList.remove("b-display-none");
			document.getElementById('js-famepoints').textContent = resp.clan_stats.fame_points.toLocaleString("no-NO");
			document.getElementById('js-eventPosition').textContent = resp.clan_stats.rank;
			if (resp.clan_stats.rewards) {
				const [bond, gold] = resp.clan_stats.rewards;
				document.getElementById('js-rewards').innerHTML = `Rewards: <span class='t-reward'><span class='icon icon_bonds'></span> x <span class='t-bonds t-bold'>${bond.value}</span></span><span class='t-reward'><span class='t-gold t-bold'>${gold.value}</span><span class='icon icon_main icon_gold'></span></span>`;
			}
			else {
				document.getElementById('js-rewards').innerHTML = "No reward data available"
			}
		}
		else {
			console.error("Error with API", "handlerClan", "User not logged in");
		}
	},
	handlerMain = function({clan, battles, planned_battles}) {
		// main handler
		const battleProvinces = [],
		battleFragment = document.createDocumentFragment();
		let battle, battleTime;
		// store data
		script.clan.tag = clan.tag;
		script.clan.emblem = clan.emblem_url;
		script.clan.color = clan.color;
		script.cw.battles = clan.appointed_battles_count;
		script.cw.current = battles.length;
		script.cw.elo = {
			6: clan.elo_rating_6,
			8: clan.elo_rating_8,
			10: clan.elo_rating_10
		};
		style.textContent += `.t-clantag {color: ${script.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 = [timeParser(parseFloat(battle.battle_time.match(/\d+/g)[3])), battle.battle_time.match(/\d+/g)[4], battle.battle_time.match(/\d+/g)[5]];
				battleFragment.appendChild(_createElement("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 target='_blank' href='https://eu.wargaming.net/globalmap/#tournament/${battle.province_id}'>${script.symbols.tournament}</a>]</td>
					<td>${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_${script.cw.event ? script.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>
					${script.table.html}
					`
				}));
				if (battle.round_number) {
					request({name: "freepassData", api:`${script.api.tourney+battle.province_id}&round=${battle.round_number}`, handler: handlerFreepass, province: battle.province_id});
				}
			}
		}
		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 = [timeParser(parseFloat(battle.battle_time.match(/\d+/g)[3])), battle.battle_time.match(/\d+/g)[4], battle.battle_time.match(/\d+/g)[5]];
				battleFragment.appendChild(_createElement("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 target='_blank' href='https://eu.wargaming.net/globalmap/#tournament/${battle.province_id}'>${script.symbols.tournament}</a>]</td>
					<td>${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>
					${script.table.html}
					`
				}));
				if (battle.is_attacker) {
					request({name: "freepassData", api: `${script.api.tourney+battle.province_id}&round=1`, handler: handlerFreepass, province: battle.province_id});
				}
			}
		}
		// show foes and battle count if clan has any battles and remove loading indicator
		if (script.cw.battles > 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'>[${script.clan.tag}]</span> ${clan.name} <img src='${script.clan.emblem}'> • `;
		// insert battle count
		document.getElementById('js-battles').textContent = script.cw.current;
		// insert battle count
		document.getElementById('js-battlesPlan').textContent = script.cw.battles;
		// send request for detailed battle information
		if (script.cw.battles > 0) {
			request({name: "handlerBats", api: script.api.bats + battleProvinces.join("&aliases="), handler: handlerBats});
		}
		// send request for clan provinces
		request({name: "handlerProvs", api: `${script.api.provs}${script.app_id}&clan_id=${script.clan.id}`, handler: handlerProvs});
	},
	handlerBats = function({data}, isPlanned) {
		// battles handler
		for (let _bd = 0, _bd_len = data.length; _bd < _bd_len; _bd++) {
			const battle = data[_bd],
			battleRow = document.getElementsByClassName(battle.alias)[0],
			enemyID = battleRow.children[7].id,
			battleType = battle.owner_clan_id == script.clan.id ? "Defence" : battle.owner_clan_id == enemyID ? "Owner" : "Attack",
			primeTime = [timeParser(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 == "Defence") {
				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
			request({name: "handlerTourney", api: `${script.api.tourney+battle.alias}&round=1`, handler: handlerTourney, extra: isPlanned, province: battle.alias});
		}
		// refresh table
		sortTable.refresh();
	},
	handlerTourney = function(data, isPlanned) {
		// tournament handler
		const battleRow = document.getElementsByClassName(data.province_id)[0],
		battleFront = data.front_id.match(/league\d/)[0],
		ownerClan = data.owner ? (data.owner.id == script.clan.id) || false : false,
		isAuction = data.auction_type == "auction" || battleRow.classList.contains("auction");
		let primeTime,
		cellOwnerTime = false,
		attackers = data.is_superfinal ? 1 : data.pretenders.length;
		script.dyn.check++;
		primeTime = [timeParser(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) {
			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 && !ownerClan) {
			attackers++;
		}
		// find how many battles
		const battles = attackers !== 0 ? Math.ceil(Math.log2(attackers)) + 1 : 0;
		// modify cells
		battleRow.children[1].textContent = 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 && script.cw.tier !== "Ɵ") {
			battleRow.children[5].appendChild(_createElement("span", {
				className: "t-elo",
				innerHTML: `(${data.owner.elo_rating ? data.owner.elo_rating : data.owner[`elo_rating_${script.cw.event ? script.cw.tier : "10"}`]})`
			}));
		}
		if (script.front_loc[battleFront]) {
			battleRow.children[6].lastElementChild.innerHTML = script.front_loc[battleFront];
		}
		if (isAuction && !ownerClan) {
			battleRow.classList.add("auction");
			battleRow.children[6].classList.add("t-auction");
			battleRow.children[6].firstElementChild.textContent = "Auction";
		}
		if (isPlanned) {
			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] - script.table.eu[0]) * 2 + script.table.static,
			lastBattle = battles + emptyCells;
			battleRow.children[9].textContent = attackers;
			battleRow.children[10].textContent = battles;
			for (let _conc = 0, _cell = script.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 (!script.table.classes.includes(timeClass) || !script.table.classes.includes(timePrevClass)) {
						script.table.classes.push(timePrevClass, timeClass);
						if (_cell == lastBattle) {
							script.table.classes.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 (!script.dyn.prov.includes(data.province_id)) {
							if (script.dyn.conc[primeTimeConc]) {
								script.dyn.conc[primeTimeConc]++;
							}
							else {
								script.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 (script.cw.globalmap && battleRow.children[6].firstElementChild.innerHTML == "No Division" || (!data.owner || data.owner && data.owner.division_id == false)) {
							cell.classList.add("t-noOwner");
							if (script.dyn.conc[primeTimeConc]) {
								script.dyn.conc[primeTimeConc]--;
							}
						}
						cell.innerHTML = ownerClan ? script.symbols.defence : script.symbols.owner;
						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 = script.symbols.attack;
					}
					if (primeTime[1] == "15") {
						battleRow.classList.add("timeShift");
						cell.innerHTML += `<span class='t-timeShift'>${script.symbols.time}</span>`;
					}
					_conc++;
				}
			}
			if (!script.dyn.prov.includes(data.province_id)) {
				script.dyn.prov.push(data.province_id);
			}
			if (script.dyn.check >= script.cw.battles + script.dyn.plan) {
				// send request for divisions
				request({name: "handlerDivs", api: script.api.divs, handler: handlerDivs});
				handlerFooter("tourney", data.province_id);
			}
			// check if battle is planned or not started and change state to ongoing
			if (battleRow.children[7].textContent == "Free Round" && 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 = "Defence";
						battleRow.children[7].textContent = "Ongoing";
						break;
					case "Defence":
						battleRow.children[7].textContent = "Ongoing";
						request({name: "handlerFame", comment: "ongoing", api: script.api.prov + data.province_id, handler: handlerFame, province: data.province_id});
						break;
					default:
						break;
				}
			}
			else if (battleRow.children[7].innerHTML == "Planned" && script.time.roundTime[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 = `${script.symbols.owner}<span class='t-timeShift'>${script.symbols.time}</span>`;
			}
			else {
				lastBattle.innerHTML = script.symbols.owner;
			}
			lastBattle.classList.add("t-noFight");
		}
		if (ownerClan) {
			script.dyn.gold += data.province_revenue;
			document.getElementById('js-gold').textContent = script.dyn.gold; // insert gold count
		}
		if (script.cw.gold && data.province_revenue === 0) {
			document.getElementById('js-goldInfo').textContent = "(No Gold Revenue in Event!)";
			script.cw.gold = false;
			style.textContent += "th.t-gold, td.t-gold {display: none;}";
		}
		// get province famepoints if event
		if (script.cw.event) {
			request({name: "handlerFame", api: script.api.prov + data.province_id, handler: handlerFame, province: data.province_id});
		}
		// refresh table
		sortTable.refresh();
	},
	handlerProvs = function({data}) {
		// clan provinces handler
		const provs = data[script.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 = [timeParser(parseFloat(prov.prime_time.match(/\d+/g)[0])), prov.prime_time.match(/\d+/g)[1], parseFloat(prov.prime_time.match(/\d+/g)[0]) + script.table.eu.length],
					battleFront = prov.front_id.match(/league\d/)[0],
					provRow = _createElement("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 target='_blank' href='https://eu.wargaming.net/globalmap/#tournament/${prov.province_id}'>${script.symbols.tournament}</a>]</td>
							<td>${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/${script.clan.id}/'><span class='t-clantag'>[${script.clan.tag}]</span> <img src='${script.clan.emblem}'></a><span class='t-elo'>(${script.cw.elo[prov.max_vehicle_level]})</span></td>
							<td><span>Defence</span><span>${script.front_loc[battleFront] ? script.front_loc[battleFront] : "(??)"}</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>
							${script.table.html}
						`
					}),
					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);
					script.dyn.gold += prov.daily_revenue;
					script.cw.tier = prov.max_vehicle_level;
					if (primeTime[1] == "15") {
						provRow.classList.add("timeShift");
						lastBattle.innerHTML = `${script.symbols.defence}<span class='t-timeShift'>${script.symbols.time}</span>`;
					}
					else {
						lastBattle.innerHTML = script.symbols.defence;
					}
					lastBattle.classList.add("t-noFight");
					if (!provTimes.includes(provTimeClass)) {
						provTimes.push(provTimeClass);
					}
					if (script.cw.gold && prov.daily_revenue === 0) {
						document.getElementById('js-goldInfo').textContent = "(No Gold Revenue in Event!)";
						script.cw.gold = false;
						style.textContent += "th.t-gold, td.t-gold {display: none;}";
					}
					// get province famepoints if event
					if (script.cw.event) {
						request({name: "handlerFame", api: script.api.prov + prov.province_id, handler: handlerFame, province: prov.province_id});
					}
					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 = script.dyn.gold;
			// send request for raids if event
			//if (script.cw.event) {
			//  request({name: "handlerRaids", api: script.api.raids, handler: handlerRaids});
			//}
		}
		else if (script.cw.battles === 0) {
			battleTable.children[1].innerHTML = ""; // empty table
			provFragment.appendChild(_createElement("tr", {
				className: "t-cwText",
				innerHTML: "<td colspan='23'>No Battles</td>"
			}));
		}
		battleTable.children[1].appendChild(provFragment);
		// refresh table
		sortTable.refresh();
	},
	handlerFame = function({province, owner}, coef) {
		// fame handler for global map events
		const battleRow = document.getElementsByClassName(province.alias)[0];
		let fame_style = "t-green",
		fame = "Ɵ",
		sort = 0;
		if (script.cw.event) {
			if (province.fame_points) {
				fame = province.fame_points;
				sort = fame;
			}
			else if (province.money_box) {
				if (owner.id == script.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 && owner.id == script.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 (province.type == "auction") {
			//  if (battleRow.children[7].innerHTML == "Not Started") {
			//      battleTable.children[1].appendChild(_createElement("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 target='_blank' href='https://eu.wargaming.net/globalmap/#tournament/${province.alias}'>${script.symbols.tournament}</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>${script.front_loc[battleFront] ? script.front_loc[battleFront] : ""}</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>
			//              ${script.table.html}
			//          `
			//      }));
			//      battleRow.remove();
			//      // request tourneyData to be sure of proper visuals
			//      request({name: "handlerTourney (auction)", api: `${script.api.tourney+province.alias}&round=1`, handler: handlerTourney, province: province.alias});
			//  }
			//  else {
			//      battleRow.classList.add("auction");
			//      battleRow.children[6].classList.add("t-auction");
			//      battleRow.children[6].firstElementChild.textContent = "Auction";
			//  }
			//}
		}
	},
	handlerDivs = function({data}) {
		// divisions handler
		if (data && data.length) {
			const divsId = JSON.stringify(data).match(/\d{9}/g);
			if (divsId.includes(script.clan.id.toString())) {
				script.cw.globalmap = true;
				for (let _p = 0, _p_len = data.length; _p < _p_len; _p++) {
					const div = 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 (script.dyn.conc[primeTimeConc]) {
									script.dyn.conc[primeTimeConc]--;
								}
							}
						}
						else if (!battleRow) {
							script.dyn.plan++;
							battleTable.children[1].appendChild(_createElement("tr", {
								className: `planned ${div.alias}`,
								innerHTML: `
									<td><a target='_blank' href='https://eu.wargaming.net/globalmap/#province/${div.alias}'>${div.name}</a> [<a target='_blank' href='https://eu.wargaming.net/globalmap/#tournament/${div.alias}'>${script.symbols.tournament}</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>
									${script.table.html}
								`
							}));
							request({name: "handlerBats", comment: "planned", api: script.api.bats + div.alias, handler: handlerBats, extra: true, province: div.alias});
						}
					}
					else {
						// sometimes future defenses wont show up in planned battles
						if (div.attackers.length > 0 && battleRow && battleRow.classList.contains('province')) {
							script.dyn.plan++;
							battleTable.children[1].appendChild(_createElement("tr", {
								className: `planned ${div.alias}`,
								innerHTML: `
									<td><a target='_blank' href='https://eu.wargaming.net/globalmap/#province/${div.alias}'>${div.name}</a> [<a target='_blank' href='https://eu.wargaming.net/globalmap/#tournament/${div.alias}'>${script.symbols.tournament}</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>Defence</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>
									${script.table.html}
								`
							}));
							battleRow.remove();
							request({name: "handlerTourney", comment: "planned def", api: `${script.api.tourney+div.alias}&round=1`, handler: handlerTourney, extra: true, province: div.alias});
						}
					}
					// update footer
					if (script.dyn.check >= script.cw.battles + script.dyn.plan) {
						handlerFooter("divisions", div.alias);
					}
				}
				if (script.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, as you're not a member of this clan.";
			}
		}
	},
	handlerFreepass = function(data) {
		// freepass handler
		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 == script.clan.id;
		//console.log(data.province_id, lastGroup);
		// 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");
		}
	},
	handlerFooter = function(mode, province) {
		const footer = document.getElementById('js-footer'),
		conc = Object.entries(script.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 = `${script.table.classes.join(", ")} {display: table-cell !important;}`;
		for (let _c_k = Object.keys(script.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 = script.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];
			}
		}
		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] ? `${script.symbols.attack}${times[0]}` : ""}${times[1] ? ` ${script.symbols.time}${times[1]}` : ""}`;
			}
			else {
				console.error("Foot doesn't exist", key, times);
			}
		}
	},
	handlerRaids = function(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 = function(name, data) {
		// error handler
		switch (name) {
			case "handlerMain":
				battleTable.children[1].appendChild(_createElement("tr", {
					className: "t-cwText",
					innerHTML: "<td colspan='23'>Clan ID Error</td>"
				}));
				inputSpan.lastElementChild.textContent = "Clan ID invalid!";
				localStorage.removeItem("battles_clanid");
				break;
			case "handlerDivs":
				document.getElementById('js-error').innerHTML = `Division data not available, because you need to be logged in on the <a target='_blank' href='https://eu.wargaming.net/globalmap/'>Global Map</a>.`;
				break;
			default:
				break;
		}
	},
	timeParser = function(hour, min, type) {
		// time converter
		let time = hour + script.time.offset;
		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 = function() {
		// timestamp handler
		const dateNow = new Date(),
		time = {
			hour: timeParser(16) - dateNow.getHours(),
			min: 60 - dateNow.getMinutes() - 1,
			sec: 60 - dateNow.getSeconds() - 1
		},
		timeSpan = document.getElementById('js-timePrime');
		if (timeJump && !script.cw.event && script.cw.status) {
			timeSpan.textContent = "No Event Running";
			timeSpan.classList.add("t-bold");
			clearInterval(timeInterval);
		}
		else if (time.hour >= 0 && (time.sec > 0 || time.min < 15)) {
			timeSpan.textContent = `${(time.hour > 0 ? `${time.hour} Hours, ` : "") + (time.min > 0 ? `${time.min} Mins, ` : "") + time.sec} Secs`;
		}
		else if (time.hour < 0 && script.cw.battles !== "Ɵ") {
			if (script.cw.battles === 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 (script.cw.current > 0) {
					document.getElementById('js-provStatus').textContent = "Next Opponent (ELO)";
				}
				clearInterval(timeInterval);
			}
		}
		else {
			timeSpan.textContent = "Hold on a sec...";
		}
		timeJump = true;
	},
	mapFix = function(name) {
		// map name fixer
		const fixedNames = {
			"112_eiffel_tower_ctf/name": "Paris",
			"114_czech/name": "Pilsen",
			"99_poland/name": "Studzianki"
		};
		return fixedNames[name] ? fixedNames[name] : name;
	},
	updater = function(force) {
		// updater handler
		const dateNow = new Date(),
		newDate = {
			hour: dateNow.getHours(),
			min: dateNow.getMinutes(),
			offset: (dateNow.getTimezoneOffset() > 0 ? -Math.abs(dateNow.getTimezoneOffset()) : Math.abs(dateNow.getTimezoneOffset())) / 60
		},
		newTime = script.time.min >= 15 && script.time.min <= 45 ? [script.time.hour, "30"] : script.time.min <= 15 ? [script.time.hour, "00"] : [script.time.hour + 1, "00"];
		if (force || script.time.roundTime[0] !== newTime[0] || script.time.roundTime[1] !== newTime[1]) {
			script.time = newDate;
			script.time.roundTime = newTime;
			script.time.classes = `${script.time.roundTime[0]}_${script.time.roundTime[1]}`;
			battleTable.lastElementChild.firstElementChild.innerHTML = `<td></td><td></td><td></td><td class='t-gold'></td><td></td><td></td><td></td><td></td><td></td><td></td><td class='t-border'></td>${script.table.html}`;
			styleDynTimezone.textContent = `
				.b-battles .b-battles-holder .t-battles tr .t-${script.time.classes} {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-${script.time.classes} + th, .b-battles .b-battles-holder .t-battles tr .t-${script.time.classes} + td {background-color: rgba(224,223,218, 0.5); border-right: 1px solid #808080;}
			`;
			script.dyn = {
				conc: {},
				prov: [],
				plan: 0,
				check: 0,
				gold: 0
			};
			request({name: "handlerMain", comment: "update", api: `${script.api.clan}${script.clan.id}/battles`, handler: handlerMain});
			// insert update timestamp
			document.getElementById('js-batttleUpdate').textContent = new Date().toLocaleTimeString("en-GB");
		}
	},
	request = function({name, api, handler, extra, province, comment}) {
		// request handler
		GM.xmlHttpRequest({
			method: "GET",
			url: api,
			headers: {
				"Accept": "application/json",
				"X-Requested-With": "XMLHttpRequest"
			},
			onload(resp) {
				const data = JSON.parse(resp.responseText);
				if (resp.status == 200) {
					if (script.debug) {
						console.info(`%c${name}${comment ? `: (${comment})` : ""}${province ? `: ${province}` : ""}`, `color: ${script.debugColors[name]}`, {data, api, "time": new Date().toLocaleTimeString("en-GB")});
					}
					handler(data, extra);
				}
				else {
					console.error("Error with API", name, api, resp);
					handlerError(name);
				}
			},
			onerror(resp) {
				console.error("Error accessing API", name, api, resp);
			}
		});
	},
	_createElement = function(tag, attributes, children) {
		// element creation
		const element = Object.assign(document.createElement(tag), attributes);
		if (children) {
			if (children.nodeType) {
				element.appendChild(children);
			}
			else {
				for (let _c=0, _c_len=children.length; _c<_c_len; _c++) {
					const child = children[_c];
					if (child && child.nodeType) {
						element.appendChild(child);
					}
				}
			}
		}
		return element;
	};
	// inserting style into head
	style = _createElement("style", {
		className: "battleScheduler",
		type: "text/css",
		textContent: `
/* scipt buttons */
.b-scriptButton {height: 25.2px}
.b-scriptButton a {vertical-align: middle;}
.t-fakeButton {cursor: pointer; border-radius: 5px; padding: 2px 5px; transition: 0.3s;}
.t-fakeButton:hover {background-color: rgba(57, 51, 49, 0.15);}
/* icon rules */
.b-battleScheduler .icon {display: inline-block; height: 16px; width: 16px; vertical-align: text-top;}
.b-battleScheduler.b-scriptButton .icon {display: inline-block; height: 16px; width: 16px; margin-right: 3px; vertical-align: sub;}
.icon_main {background: url(${script.img.main}) no-repeat 0 0;}
.b-battleScheduler .icon_chat {}
.b-battleScheduler .icon_twitch {background-position: -16px 0;}
.b-battleScheduler .icon_fork {background-position: -32px 0;}
.b-battleScheduler .icon_wg {background-position: -48px 0;}
.b-battleScheduler .icon_cw {background-position: -64px 0;}
.b-battleScheduler .icon_gold {background-position: -82px 0; margin-left: 2px;}
.b-battleScheduler .icon_wot {background-position: -98px 0;}
.b-battleScheduler .icon_bonds {background: url(${script.img.bonds}) no-repeat 0 0; background-size: contain; margin-left: 2px;}
/* battle scheduler */
.b-battles {background-color: #D1E8FE; border: 1px solid; border-radius: 10px; display: table; font-size: 11px; width: 90vw; min-width: 1280px; margin: 25px auto 50px;}
.b-battles .h-battles {}
.b-battles .h-battles .h-battles-info {border-bottom: 1px solid #F5F5F5; margin: 0 auto; padding: 4px 0px; width: 98%; display: flex; align-items: center; justify-content: space-between;}
.b-battles .h-battles .h-battles-info .h-left {width: 380px;}
.b-battles .h-battles .h-battles-info .h-left a {font-size: 13px;}
.b-battles .h-battles .h-battles-info .h-mid {text-align: center;}
.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: 380px;}
.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 label {cursor: pointer;}
.b-battles .h-battles .h-battles-info .h-right .h-force input {height: 13px; vertical-align: bottom;}
.b-battles .h-battles .h-battles-info .h-right #js-update {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 {max-height: 90vh; overflow-y: auto;}
.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 .sort-up, .b-battles .b-battles-holder .t-battles .sort-down {color: #C600C6;}
.b-battles .b-battles-holder .t-battles thead th {border-bottom: 1px solid #808080; line-height: 20px;}
.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 tbody td {border-bottom: 1px dotted #B1B1B1; line-height: 21px;}
.b-battles .b-battles-holder .t-battles tbody 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 tbody .t-good {color: #4D7326;}
.b-battles .b-battles-holder .t-battles tbody .t-bad {color: #930D0D;}
.b-battles .b-battles-holder .t-battles tbody .t-time {font-weight: bold;}
.b-battles .b-battles-holder .t-battles tbody .t-plan {color: #FFE400;}
.b-battles .b-battles-holder .t-battles tbody .t-fight {color: #4D7326;}
.b-battles .b-battles-holder .t-battles tbody .t-auction {color: #2F396A;}
.b-battles .b-battles-holder .t-battles tbody .t-noFight {color: transparent; text-shadow: rgba(64, 58, 58, 0.35) 0 0 0;}
.b-battles .b-battles-holder .t-battles tbody .t-fight.t-noOwner {color: transparent; text-shadow: rgba(64, 58, 58, 0.35) 0 0 0;}
.b-battles .b-battles-holder .t-battles tbody .t-error {color: #C42811;}
.b-battles .b-battles-holder .t-battles tbody .t-cwText td {font-size: 26px; line-height: 54px;}
.b-battles .b-battles-holder .t-battles tbody .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 tbody .timeShift td:nth-child(3)::after, .b-battles .b-battles-holder .t-battles tr.timeShift td:nth-child(9)::after {content: " ⌛";}
.b-battles .b-battles-holder .t-battles tbody .timeShift td .t-timeShift {font-size: 8px; position: absolute;}
.b-battles .b-battles-holder .t-battles tbody img {height: 16px; margin-bottom: 2px; vertical-align: bottom;}
.b-battles .b-battles-holder .t-battles tfoot td {border-top: 1px solid #808080; font-weight: bold; line-height: 20px; height: 20px;}
.b-battles .b-battles-holder .t-battles tr .t-border {border-right: 2px solid rgba(194, 173, 173, 0.5);}
.b-battles .f-battles {}
.b-battles .f-battles .f-battles-info {border-top: 1px solid #F5F5F5; margin: 0 auto; padding: 3px 0px; width: 98%; display: flex; align-items: center; justify-content: space-between;}
.b-battles .f-battles .f-battles-info .f-left {width: 380px;}
.b-battles .f-battles .f-battles-info .f-left .f-input > span {margin-left: 5px;}
.b-battles .f-battles .f-battles-info .f-left .f-inputText {color: #C42811;}
.b-battles .f-battles .f-battles-info .f-left .f-inputText.t-green {color: #4D7326;}
.b-battles .f-battles .f-battles-info .f-left #js-goldInfo {margin-left: 3px;}
.b-battles .f-battles .f-battles-info .f-mid {text-align: center; flex-grow: 1;}
.b-battles .f-battles .f-battles-info .f-mid-tables {display: flex; justify-content: center;}
.b-battles .f-battles .f-battles-info .f-mid table {margin: 0px 20px;}
.b-battles .f-battles .f-battles-info .f-mid table td {padding: 0 10px;}
.b-battles .f-battles .f-battles-info .f-mid table td {padding: 0 10px;}
.b-battles .f-battles .f-battles-info .f-mid .t-reward {margin: 0 5px;}
.b-battles .f-battles .f-battles-info .f-mid #js-error {display: block; color: #C42811;}
.b-battles .f-battles .f-battles-info .f-right {text-align: right; width: 380px;}
.b-battles .f-battles .f-battles-info img {max-height: 16px; vertical-align: bottom;}
/* toggle overrides */
.t-elo {margin-left: 3px;}
.t-gold {color: #AC813C;}
.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);
	// link to battle scheduler
	const rightSide = document.getElementsByClassName("linklist rightside")[0];
	rightSide.prepend(_createElement("li", {
		className: "b-battleScheduler b-scriptButton",
		innerHTML: `<a target='_blank' href='./games.php'><span class='icon icon_main icon_cw'></span>Battle Schedule</a>`
	}));
	rightSide.prepend(_createElement("li", {
		className: "b-battleScheduler b-scriptButton",
		innerHTML: `<a target='_blank' href='https://eu.wargaming.net/clans/wot/500010805/'><span class='icon icon_main icon_wg'></span>Clan Page</a>`
	}));
	if (/games.php/.test(window.location.href)) {
		styleDynTimezone = _createElement("style", {
			className: "styleDynTimezone",
			type: "text/css",
			textContent: `
				.b-battles .b-battles-holder .t-battles tr .t-${script.time.classes} {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-${script.time.classes} + th, .b-battles .b-battles-holder .t-battles tr .t-${script.time.classes} + td {background-color: rgba(224,223,218, 0.5); border-right: 1px solid #808080;}
			`
		});
		styleDynBattles = _createElement("style", {
			className: "styleDynBattles",
			type: "text/css"
		});
		styleDynFinales = _createElement("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"),
		chat = page_body.getElementsByClassName("tablebg")[0],
		battlePanel = _createElement("div", {
			className: "b-battleScheduler b-battles",
			innerHTML: `
<div class='h-battles'>
	<div class='h-battles-info'>
		<div class='h-left'>
			<a class='t-fakeButton' target='_blank' href='https://greasyfork.org/en/scripts/20094-ideal-scripts' title='Script Page'><span class='icon icon_main icon_fork'></span></a>
			<a class='t-fakeButton' target='_blank' href='https://eu.wargaming.net/clans/wot/${script.clan.id}/globalmap' title='Clan Page'><span class='icon icon_main icon_wg'></span></a>
			<a class='t-fakeButton' target='_blank' href='https://eu.wargaming.net/globalmap/' title='Global Map'><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 class='t-fakeButton' id='js-update'>Update</span>
			<span class='h-force t-fakeButton' id='js-force'><label>Campaign Only: <input type='checkbox' name='force'></label></span>
		</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 class='t-gold'></td><td></td><td></td><td></td><td></td><td></td><td></td><td class='t-border'></td></tr>
		</tfoot>
	</table>
</div>
<div class='f-battles'>
	<div class='f-battles-info'>
		<div class='f-left'>
			<div class='f-input' id='js-input'>
				<input type='text' name='id' placeholder='Clan ID' value='${script.clan.id}' list='clanid' maxlength='9' size='9'>
				<datalist id='clanid'>${script.clans.join("")}</datalist>
				<span class='f-inputImage'><img src='https://wot-life.com/img/clanicon/${script.clan.id}.png'></span>
				<span class='f-inputText'>${script.clan.id !== script.clan.ideal ? "<span class='f-inputBack'>Back to <a href='' onclick='localStorage.removeItem(\"battles_clanid\"); location.reload();'>[IDEAL]</a></span>" : ""}</span>
			</div>
			<table>
				<tr><td colspan='3'>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>
			</table>
		</div>
		<div class='f-mid'>
			<div id='js-tableStats' class='f-mid-tables b-display-none'>
				<table id='js-tableEvent' class='b-display-none'>
					<tr class='t-bold'><td>Famepoints: <span id='js-famepoints'>0</span></td><td>Position: <span id='js-eventPosition'>0</span></td></tr>
					<tr><td colspan='2' id='js-rewards'></td></tr>
				</table>
				<table>
					<tr class='t-bold'><td>Current Battles: <span id='js-battles'>0</span></td><td>Simultaneously: <span id='js-battlesConc'>0</span></td></tr>
					<tr><td>Planned: <span id='js-battlesPlan'>0</span></td><td>Owned Provinces: <span id='js-provs'>0</span></td></tr>
				</table>
			</div>
			<span id='js-error'></span>
		</div>
		<div class='f-right'>
			<div>${script.symbols.attack} Battle • ${script.symbols.defence} Defence • ${script.symbols.owner} Battle with Owner • ${script.symbols.attack}<span class='t-bold t-green'>${script.symbols.time}</span> Timeshift</div>
			<div>Last Updated: <span id='js-batttleUpdate'>Ɵ</span> [UTC${script.time.offset >= 0 ? "+" : ""}${script.time.offset}] • v${script.vers}</span></div>
		</div>
	</div>
</div>
`
		});
		page_body.parentNode.appendChild(chat);
		page_body.innerHTML = "";
		page_body.appendChild(battlePanel);
		page_body.appendChild(chat);
		// table references
		battleTable = battlePanel.children[1].firstElementChild;
		// time cells for header and body rows
		const timeFragment = document.createDocumentFragment();
		for (let _tc = 0, _tc_len = script.table.eu.length; _tc < _tc_len; _tc++) {
			const t = script.table.eu[_tc],
			times = [timeParser(t, "00", "s"), `${timeParser(t)}:00`, timeParser(t, "30", "s"), `${timeParser(t)}:30`];
			timeFragment.appendChild(_createElement("th", {
				className: `t-time ${times[0]}`,
				innerHTML: times[1]
			}));
			script.table.html += `<td class='t-time ${times[0]}' data-time='${times[1]}'></td>`;
			if (_tc !== _tc_len - 1) {
				timeFragment.appendChild(_createElement("th", {
					className: `t-time ${times[2]}`,
					innerHTML: times[3]
				}));
				script.table.html += `<td class='t-time ${times[2]}' data-time='${times[3]}'></td>`;
			}
		}
		battleTable.firstElementChild.firstElementChild.appendChild(timeFragment);
		battleTable.lastElementChild.firstElementChild.innerHTML += script.table.html;
		// button to update
		const updateSpan = document.getElementById('js-update');
		updateSpan.addEventListener('click', function() {
			updater(true);
		}, false);
		// button to force load cw battles
		const forceSpan = document.getElementById('js-force');
		forceSpan.firstElementChild.lastElementChild.checked = script.cw.status ? true : false;
		forceSpan.addEventListener('click', function() {
			localStorage.setItem("battles_force", this.firstElementChild.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 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(timer, 1000); // 1 second
		updateInterval = setInterval(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(_createElement("tr", {
			className: "t-cwText",
			innerHTML: "<td colspan='23'>Updating...</td>"
		}));
		// send request to wargaming api to see if an event is running
		request({name: "handlerEvent", api: `${script.api.event}${script.app_id}&limit=1`, handler: handlerEvent});
	}
}());