BvS Infinite Quickteams

Allows you to have more Quickteams

// ==UserScript==
// @name           BvS Infinite Quickteams
// @namespace      rvQuick
// @description    Allows you to have more Quickteams
// @include        http*://*animecubed.com/billy/bvs/team.html
// @include        http*://*animecubedgaming.com/billy/bvs/team.html
// @version        1.4.3
// @history        1.4.3 New domain - animecubedgaming.com - Channel28
// @history        1.4.2 Now https compatible (Updated by Channel28)
// @history        1.4.1 (9/9/2014) Added grant permissions.
// @history        1.4  Added underline coloring for allies that couldn't be found, making a team unusable.
// @history        1.3g Fixed a rare bug.
// @history        1.3f Fixed a bug that I created from 1.3e.
// @history        1.3e Fixed some of turn the page jutsu.
// @history        1.3d Fixed bug where you couldn't drag n' drop newly created teams until you refreshed the page.
// @history        1.3c Bug Fix
// @history        1.3b Added Drag n' Drop support to the table
// @history        1.3a Fixed up Export display.
// @history        1.3  Added Export/Import functionality
// @history        1.2e Fixed "None" quickteam to bypass confirm screen
// @history        1.2d Temporary fix for FF4 bug.
// @history        1.2c Improved cross browser support.
// @history        1.2b Bug fixes. Hide Team MisAlignment dialog on team rename.
// @history        1.2a Confirmed support for Chrome, but Escape doesn't work.
// @history        1.2  Fixed more scrolling issues and added script support for other browsers besides FF.
// @history        1.1c Fixed scrolling issues when renaming a QT.
// @history        1.1b Fixed move QT down exception when there was only 1 QT.
// @history        1.1a Fixed up final bug.
// @history        1.1  Fixed a bug where you couldn't work with a QT that you didn't have one of the allies for.
// @history        1.0  Start
// @grant          GM_getValue
// @grant          GM_setValue
// @grant          GM_addStyle
// ==/UserScript==

const VERSION = "1.4.1";
const GMSTORAGE_PATH = "BvS_Quickteams_";

var playerData;
var playerName;

var allies;
var startPoint;
var currentTeam;
var quickteam;
var testDiv;
var turnThePage;

function load(ev) {
	try {
		var temp = document.getElementByName("player");
		if ((temp == null) || (temp.localName.toLowerCase() == "text") || (temp.value.length == 0))
			return;
		playerName = temp.value;

		getGM();
		parsePage();
		applyGM();
	} catch(e) {
		alert("Exception!\n\nError name: " + e.name + "\nError message: " + e.message);
	}
}

function getGM() {
	var value = GM_getValue(playerName, "{}");

	if ((value == undefined) || (value == null) || (value == ""))
		value = "{}";
	
	playerData = JSON.parse(value);
}

function saveGM() {
	GM_setValue(playerName, JSON.stringify(playerData));
}

function parsePage() {
	turnThePage = document.getElementsByName("turnthepage");
	if ((turnThePage) && (turnThePage.length == 0))
		turnThePage = null;

	if (document.getElementByName("conteam") == null) {
		allies = new Array();

		snap = document.evaluate("//div/table/tbody/tr/td/label/img", document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
		if ((snap != null) && (snap.snapshotLength > 0)) {
			var i;
			for (i = 0; temp = snap.snapshotItem(i); i++) {
				allies.push(new Array(document.getElementByName("teammate" + i).value, Allies.getAllyName(temp.src)));
			}
		}
		
		startPoint = document.getElementsByName("quickteam");
		if ((startPoint != null) && (startPoint.length == 0))
			startPoint = null;
		
		currentTeam = new Array();

		snap = document.evaluate("//center/center/table/tbody/tr/td/img", document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
		if ((snap != null) && (snap.snapshotLength > 0)) {
			var i;
			for (i = 0; temp = snap.snapshotItem(i); i++) {
				currentTeam.push(Allies.getAllyName(temp.src));
			}
		}
	}
}

function applyGM() {
	if ((allies != null) && (startPoint == null)) {
		startPoint = document.getElementByName("qteam");
		if (startPoint == null)
			return;
		
		startPoint = startPoint.children[0];

		var element = document.createElement('DIV');
		var text = "<table><tbody><tr><td style=\"font-family: arial; border: 1px dotted black; padding: 3px;\">";
		text += "<b>QuickTeams</b>: <font style=\"font-size: 12px;\">(Select team, and hit \"Use Quickteam\" - bypasses confirmation screen)</font><br>";
		text += "<input type=\"radio\" id=\"noneqt\" checked=\"\" value=\"\" name=\"quickteam\"> <label for=\"noneqt\">None</label><br>";
		text += "<a style=\"color: rgb(161, 0, 0);\" onfocus=\"this.blur();\" href=\"javascript:document.qteam.submit();\"><b>Use Quickteam &gt;</b></a>";
		text += "<noscript><input type=\"submit\" VALUE=\"Use Quickteam\"></noscript>";
		text += "<font style=\"font-size: 10px;\"><br>(Teams in <font color=\"#cc0000\">red</font> have allies that are different Levels than listed, so may have different effects)</font>";
		text += "</td></tr></tbody></table>";
		text += "<br>";

		element.innerHTML = text;
		startPoint.parentNode.insertBefore(element, startPoint.children[startPoint.children - 1]);

		startPoint = document.getElementsByName("quickteam");
		if ((startPoint != null) && (startPoint.length == 0))
			startPoint = null;
	}
	if (startPoint == null)
		return;
	if (turnThePage) {
		var temp = turnThePage[0];

		// remove <br>
		temp.parentNode.removeChild(temp.nextSibling.nextSibling.nextSibling);

		// move 'use quickteam over some'
		var extra = document.createElement('SPAN');
		extra.innerHTML = "&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;";
		temp.parentNode.insertBefore(extra, temp.nextSibling.nextSibling.nextSibling);
	}

	GM_addStyle("div#rvDialog {word-wrap: break-word; overflow: auto; background-image: url('http://www.animecubed.com/billy/layout/scrollbg.jpg'); min-width: 240px; min-height: 100px; z-index: 9002; padding: 0; border: 1px solid black;} #rvDialog h1 {color: white; font-size: 16px; font-weight: bold; padding: 4px; margin: 0; background-image: url('http://www.animecubed.com/billy/layout/scrolldark.jpg');} img#closebutton {float: right; cursor: pointer; margin: 6px;} #rvDialog div {font-size: 12px; font-family: arial; padding: 8px;} #rvDialog a {font-weight: bold; color: #a10000;} #rvDialog ul {margin: 0;} .high { font-weight: bold; color: #FFFF00; text-decoration: blink; } .mhigh { font-weight: bold; color: #00FF00; } .normal { font-weight: bold; } .mlow { font-weight: bold; color: #FF0000; } .low { font-weight: bold; color: #7F0000; text-decoration: line-through; } .special { font-weight: bold; border: 1px dotted white; background-color: #333; }");

	document.getElementById("noneqt").value = "^";
	
	/////////////////
	
	var length = startPoint.length;
	startPoint = startPoint[startPoint.length - 1];

	if (length == 1) {
		startPoint = startPoint.nextSibling.nextSibling.nextSibling.nextSibling;
	} else {
		if (startPoint.parentNode.localName == "div") {
			// BvS Loop Helper no personal teams, 1 GM team
			startPoint = startPoint.parentNode.nextSibling;
		} else {
			startPoint = startPoint.nextSibling.nextSibling;
		}
	}
	
	quickteam = document.createElement('DIV');
	var text = "<table id='rvQT' cellpadding='0' cellspacing='0' width='100%'>";
	text += "</table><br><b>New QuickTeam</b>:<br>";

	var i;
	for (i = 0; i < 3; i++) {
		text += "<select id='rvAlly" + i + "'>";
		if (i)
			text += "<option value='' />";

		var j;
		for (j = 0; j < allies.length; j++) {
			var level = Allies.getAllyLevel(allies[j][1]);
			
			if (level > 1) {
				var name = Allies.getAllyWithoutLevel(allies[j][1]);
				var sname = escape(name);
				
				text += "<option value='" + sname + "'>" + name + "</option>";

				var k;
				for (k = 2; k < level; k++) {
					text += "<option value='" + sname + " " + k + "'>" + name + " " + k + "</option>";
				}
			}
			
			text += "<option value='" + escape(allies[j][1]) + "'>" + allies[j][1] + "</option>";
		}
		
		text += "</select>";
	}
	
	text += "&nbsp;&nbsp;<input id='rvAdd' type='image' src='' alt='Add' width='31' height='15'>";
	text += "&nbsp;&nbsp;<input id='rvForm' type='image' src='%3D' alt='Form' width='40' height='15'>";

	quickteam.style.borderTop = quickteam.style.borderBottom = "1px dotted black";
	quickteam.innerHTML = text;
	
	startPoint.parentNode.insertBefore(quickteam, startPoint);

	tableDnD = new DnDTable();
	tableDnD.init(document.getElementById('rvQT'));

	tableDnD.onStartDragging = function() {
		this.startRowIndex = this.dragObject.rowIndex;
	}
	tableDnD.onDragging = function() {
		this.dragObject.style.background = "yellow";
	}
	tableDnD.onDrop = function() {
		this.dragObject.style.background = "";

		var endRowIndex = this.dragObject.rowIndex;
		var pos = this.startRowIndex;

		if (pos != endRowIndex) {
			if (endRowIndex < pos) {
				// move up
				var diff = pos - endRowIndex;

				for (; diff > 0; diff--) {
					playerData["Quickteams"].shiftUp(pos);
					playerData["QuickteamNames"].shiftUp(pos);
					pos--;
				}
			} else {
				// move down
				var diff = endRowIndex - pos;

				for (; diff > 0; diff--) {
					playerData["Quickteams"].shiftDown(pos);
					playerData["QuickteamNames"].shiftDown(pos);
					pos++;
				}
			}

			saveGM();

			updateNumbers();
		}
	}

	var extra = document.createElement('SPAN');

	text = "&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;";
	text += "<a id='rvExport' href='javascript:;' style='color:A10000'><b>Export &gt;</b></a>";
	text += "&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;";
	text += "&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;";
	text += "<a id='rvImport' href='javascript:;' style='color:A10000'><b>Import &gt;</b></a>";

	if (turnThePage) {
		text += "<br />";
	}

	extra.innerHTML = text;

	if (turnThePage) {
		startPoint.parentNode.insertBefore(extra, startPoint.nextSibling);
	} else {
		startPoint.parentNode.insertBefore(extra, startPoint.nextSibling.nextSibling);
	}
	
	redoQuickteamList();

	document.getElementById("rvAdd").addEventListener("click", addNewQuickteam, true);
	document.getElementById("rvForm").addEventListener("click", formTeam, true);
	document.getElementById("rvExport").addEventListener("click", exportQuickteams, true);
	document.getElementById("rvImport").addEventListener("click", importQuickteams, true);
	
	/////////////////

	testDiv = document.createElement('DIV');
	testDiv.style.opacity = 0;
	testDiv.style.fontFamily = "arial";
	testDiv.style.fontWeight = "bold";
	testDiv.style.display = "inline";

	document.body.insertBefore(testDiv, null);

	/////////////////
	
	startPoint = document.getElementByName("quickteamnumber");
	
	if (startPoint != null) {
		var table = startPoint.parentNode.parentNode.parentNode.parentNode.parentNode;
		table.cellSpacing = "0";
		table.cellPadding = "0";
		
		var empty = table.insertRow(table.rows.length).insertCell(0);
		empty.style.height = "10";
		empty.style.backgroundColor = "#ead8c3";
		
		var row = table.insertRow(table.rows.length);
		var element = row.insertCell(0);
		
		element.align = "center";
		element.style.backgroundColor = "#ead8c3";
		element.style.fontFamily = "arial";
		
		element.innerHTML = "<form style='margin: 0pt;'>Save current team as a GM Quickteam: <input id='rvAddCurrent' type='image' src='' alt='Add' width='31' height='15'></form>";
		
		document.getElementById("rvAddCurrent").addEventListener("click", addCurrentQuickteam, true);
	}
}

var tableDnD = null;

function redoQuickteamList() {
	try {
		var table = document.getElementById("rvQT");

		while (table.rows.length > 0) {
			table.deleteRow(0);
		}

		if (playerData["Quickteams"] != null) {
			var i;
			for (i = 0; i < playerData["Quickteams"].length; i++) {
				createQuickTeam(playerData["Quickteams"][i], playerData["QuickteamNames"][i], i+1);

				tableDnD.makeDraggable(table.rows[i]);
			}
		}
	} catch(e) {
		alert("Exception when Loading QuickTeams!\n\nError name: " + e.name + "\nError message: " + e.message);
	}
}

var dialog = null;
var dialogHeader = null;
var dialogText = null;

function exportQuickteams(ev) {
	try {
		if (dialog == null) {
			dialog = document.createElement("div");
			dialog.style.visibility = "hidden";
			dialog.id = "rvDialog";
			dialog.innerHTML = "<img id='closebutton' onclick=\"document.getElementById('rvDialog').style.visibility='hidden';\" title='Close' src='data:image/png,%89PNG%0D%0A%1A%0A%00%00%00%0DIHDR%00%00%00%10%00%00%00%10%08%03%00%00%00(-%0FS%00%00%00%01sRGB%00%AE%CE%1C%E9%00%00%00TPLTE%BA%0C%0C%BF%1D%1D%C1%26%26%C1((%C1))%C8%3D%3D%C8%3F%3F%CBHH%CDPP%D6nn%DC%83%83%DD%89%89%DE%89%89%E2%98%98%E4%A0%A0%EC%BA%BA%F1%CB%CB%F3%D7%D7%F4%D7%D7%F6%E1%E1%F7%E3%E3%FB%F0%F0%FB%F2%F2%FB%F3%F3%FC%F6%F6%FE%FB%FB%FE%FE%FE%FF%FF%FFD%88%E1%E1%00%00%00rIDAT%18%D3%5D%CFY%0E%80%20%10%03%D0%BA%E0%8E%8A%0A(%BD%FF%3D%8Da%08%EA%FC%F5%25M%3A%08%FC%5C%00%F19%3EP%A5T%09lK%CC%CB%26%80%F9%EC%80%EE%9CS%05%188%8E%1C%90%01%139%E1%05%E5N%EEe%86%C6q%5D%E9%9A%04%B5%E7%01%1C%F4%B5%80%E5%A5%00u%D1Fh%03%F5S%D7%0C%AD%80%89%C3%8C%00%FA%22B%D1%E7%1D%AF%E7%FE%EF%DF%8Eb%0A%CB%F1%17%10%11%00%00%00%00IEND%AEB%60%82' />";

			dialogHeader = document.createElement("h1");
			dialogText = document.createElement("div");
			dialog.appendChild(dialogHeader);
			dialog.appendChild(dialogText);

			document.body.appendChild(dialog);
		}

		centerdiv(dialog, "300", "200");

		dialogText.innerHTML = "<textarea rows='3' id='rvExportText' readOnly='true' style='background: #ccc; width: 100%; height: 150px;'>" + JSON.stringify(playerData).sanitizeHTML() + "</textarea>";
		dialogHeader.innerHTML = "Infinite QuickTeams Export";

		dialog.style.visibility = "visible";

		selectAll("rvExportText");
	} catch(e) {
		alert("Exception!\n\nError name: " + e.name + "\nError message: " + e.message);
	}

	if (ev && ev.preventDefault)
		ev.preventDefault();
	return false;
}

function importQuickteams(ev) {
	try {
		var input = prompt("Enter your saved 'Infinite QuickTeams' Export:", "");

		if ((input != null) && (input != "")) {
			var failed = false;
			var newData = null;

			try {
				newData = JSON.parse(input);
				failed = ((newData == undefined) || (newData == null));
			} catch (e) {
				failed = true;
			}

			if (failed) {
				alert("Failed to understand the Import text!");
			} else {
				playerData = newData;
				saveGM();

				redoQuickteamList();
			}
		}
	} catch(e) {
		alert("Exception!\n\nError name: " + e.name + "\nError message: " + e.message);
	}

	if (ev && ev.preventDefault)
		ev.preventDefault();
	return false;
}

function formTeam(ev) {
	try {
		var ally1 = unescape(document.getElementById("rvAlly0").value);
		var ally2 = unescape(document.getElementById("rvAlly1").value);
		var ally3 = unescape(document.getElementById("rvAlly2").value);

		if ((ally2 == "") && (ally3 != "")) {
			ally2 = ally3;
			ally3 = "";
		}

		var spot = document.getElementByName("quickteam");

		spot = spot.nextSibling.nextSibling.nextSibling.nextSibling;

		var element = document.createElement('DIV');
		var text = "<input type='radio' id='rvTempQT' value='";

		ally1 = Allies.get(Allies.getAllyWithoutLevel(ally1));

		text += ally1[0];
		if (ally2 != "") {
			ally2 = Allies.get(Allies.getAllyWithoutLevel(ally2));
			text += "^" + ally2[0];
		}
		if (ally3 != "") {
			ally3 = Allies.get(Allies.getAllyWithoutLevel(ally3));
			text += "^" + ally3[0];
		}

		text += "' name='quickteam'><b>Temporary Quickteam</b><br>";

		element.innerHTML = text;
		spot.parentNode.insertBefore(element, spot);

		document.getElementById("rvTempQT").checked = true;
	} catch(e) {
		alert("Exception!\n\nError name: " + e.name + "\nError message: " + e.message);
	}

	return true;
}

function addCurrentQuickteam(ev) {
	try {
		var newQT = currentTeam;
		
		if (newQT.length != 0) {
			while (newQT.length < 3)
				newQT.push("");
			
			genericAddNewQuickteam(newQT);

			redoQuickteamList();
		}
	} catch(e) {
		alert("Exception!\n\nError name: " + e.name + "\nError message: " + e.message);
	}
	
	if (ev && ev.preventDefault)
		ev.preventDefault();
	return false;
}

function addNewQuickteam(ev) {
	try {
		var ally1 = unescape(document.getElementById("rvAlly0").value);
		var ally2 = unescape(document.getElementById("rvAlly1").value);
		var ally3 = unescape(document.getElementById("rvAlly2").value);
		var cancel = false;
		
		if ((ally2 == "") && (ally3 != "")) {
			ally2 = ally3;
			ally3 = "";
		}

		if ((Allies.equals(ally1, ally2))
				|| (Allies.equals(ally1, ally3))
				|| ((Allies.equals(ally2, ally3)) && (ally2 != ""))) {
			alert("Invalid QuickTeam.");
			cancel = true;
		}
		
		if (!cancel) {
			var newQT = new Array(ally1, ally2, ally3);

			genericAddNewQuickteam(newQT);

			redoQuickteamList();
		}
	} catch(e) {
		alert("Exception!\n\nError name: " + e.name + "\nError message: " + e.message);
	}
	
	if (ev && ev.preventDefault)
		ev.preventDefault();
	return false;
}

function genericAddNewQuickteam(newQT) {
	if (playerData["Quickteams"] == null) {
		playerData["Quickteams"] = new Array();
		playerData["QuickteamNames"] = new Array();
	}

	var existing = playerData["Quickteams"].getPos(newQT);
	if (existing != -1) {
		alert("Quickteam is already in use in position #" + existing + ", named '"
				+ (playerData["QuickteamNames"][existing] == "" ? "GM Quickteam " + (existing+1) : playerData["QuickteamNames"][existing]) + "'.");
		return;
	}

	var newName = "";
	
	createQuickTeam(newQT, newName, playerData["Quickteams"].length+1);
	
	playerData["QuickteamNames"].push(newName);
	playerData["Quickteams"].push(newQT);
	
	saveGM();
}

function createQuickTeam(qt, name, number) {
	var table = document.getElementById("rvQT");
	
	var row = table.insertRow(table.rows.length);
	var element = row.insertCell(0);
	var valid = true;
	var svalid = new Array(true, true, true)
	var same = new Array();
	var id = "";
	var allyNames = "";
	var allyNamesNow = "";
	var realAllyNames = new Array();

	var i;
	for (i = 0; i < 3; i++) {
		if (qt[i] == "") {
			same.push(true);
			continue;
		}

		var ally = Allies.get(Allies.getAllyWithoutLevel(qt[i]));

		if (i > 0)
			allyNames += ", ";
		allyNames += qt[i];

		if (ally == null) {
			valid = false;
			svalid[i] = false;
		} else {
			if (i > 0) {
				id += "^";
				allyNamesNow += ", ";
			}

			id += ally[0];
			allyNamesNow += ally[1];

			realAllyNames.push(ally[1]);

			if (qt[i] == ally[1]) {
				same.push(true);
			} else {
				
				same.push(false);
			}
		}
	}

	var allyNamesColor = "";

	for (i = 0; i < 3; i++) {
		if (qt[i] == "")
			continue;

		if (i > 0) {
			allyNamesColor += ", ";
		}

		if (valid) {
			if (same[i])
				allyNamesColor += realAllyNames[i];
			else
				allyNamesColor += "<font color='#cc0000'><i>" + realAllyNames[i] + "</i></font>";
		} else {
			if (!svalid[i])
				allyNamesColor += "<span style='border-bottom: 1px solid red;'>";
			allyNamesColor += qt[i];
			if (!svalid[i])
				allyNamesColor += "</span>";
		}
	}
	
	var enames = escape(allyNames);
	var teamName = (name == "" ? "GM Quickteam" : name);
	var allsame = (same[0]) && (same[1]) && (same[2]);
	var coloring = ((allsame) || (!valid) ? "#000000" : "#cc0000");

	var text = "";

	text += "<input type='radio' id='" + id + "' value='" + id + "' name='quickteam'" + (valid ? "" : " disabled") + ">";
 	text += "<input type='text' id='nname" + enames + "' value='" + teamName + "' style='color: " + coloring + "; outline: none; font-weight: bold; font-family: arial; font-size: 100%; display: none; border: 0px; background: none;' onfocus='var evt = new Object(); evt.target = document; moveMouse(evt); return true;'>";

	if (valid) {
		text += "<label for='" + id + "'>";
		if (!allsame)
			text += "<span title='header=[Team Misalignment&nbsp;] body=[Original Team: " + allyNames.replaceAll("'", "&#39;") + "&nbsp;<br>Changed to:&nbsp;&nbsp;&nbsp;&nbsp;" + allyNamesNow.replaceAll("'", "&#39;") + "&nbsp;] offsetx=[0] offsety=[18] redswitch=[1] offsety=[-10]'>";
	}

	text += "<b style='color: " + coloring + ";'><div id='name" + enames + "' style='display: inline'>";
	text += teamName.sanitizeHTML();
	text += "</div>";
	text += " <div name='rvQTNumber' style='display: inline'>";
	if (name != "")
		text += "(";
	text += number;
	if (name != "")
		text += ")";
	text += "</div></b>";

	if ((!allsame) && (valid))
		text += "</span>";

	text += ": <font style='font-size: 12px;'>";	
	text += allyNamesColor;
	text += "<br></font>";
	
	if (valid)
		text += "</label>";
	
	element.innerHTML = text;
	
	element = row.insertCell(1);
	element.style.borderLeftStyle = "dotted";
	element.style.textAlign = "right";
	element.setAttribute("NoDrag", "true", 0);
	element.innerHTML = "<img id='down" + enames + "' alt='Move Down' src='%3D' width='12' height='14' style='cursor: pointer;'>"
		+ "&nbsp;&nbsp;&nbsp;<img id='up" + enames + "' alt='Move Up' src='%3D%3D' width='12' height='14' style='cursor: pointer;'>"
		+ "&nbsp;&nbsp;&nbsp;<img id='delete" + enames + "' alt='Remove' src='/billy/layout/xremove.gif' style='cursor: pointer;'>";

	document.getElementById("name" + enames).addEventListener("dblclick", renameQuickteam, true);
	document.getElementById("nname" + enames).addEventListener("blur", endQuickteamEdit, true);
	document.getElementById("nname" + enames).addEventListener("keydown", keydownQuickteamEdit, true);
	document.getElementById("nname" + enames).addEventListener("keypress", keypressQuickteamEdit, true);
	document.getElementById("nname" + enames).addEventListener("keyup", keyupQuickteamEdit, true);
	document.getElementById("nname" + enames).addEventListener("cut", cutQuickteamEdit, true);
	//document.getElementById("nname" + enames).addEventListener("paste", changeQuickteamEdit, true);

	document.getElementById("down" + enames).addEventListener("click", moveQuickteamDown, true);
	document.getElementById("up" + enames).addEventListener("click", moveQuickteamUp, true);
	document.getElementById("delete" + enames).addEventListener("click", removeQuickteam, true);
}

var currentlyEditing;

function renameQuickteam(ev) {
	try {
		if (currentlyEditing != null) {
			endQuickteamEdit();
		}

		currentlyEditing = this.id;
		var width = this.offsetWidth;
		var height = this.offsetHeight;
		this.style.display = "none";

		var edit = document.getElementById("n" + currentlyEditing);
		edit.value = this.textContent;
		edit.style.height = height + "px";
		edit.style.width = (width+3) + "px";
		edit.style.display = "";
		edit.focus();
	} catch(e) {
		alert("Exception!\n\nError name: " + e.name + "\nError message: " + e.message);
	}

	if (ev && ev.preventDefault)
		ev.preventDefault();
	return false;
}

function endQuickteamEdit(save) {
	try {
		if (currentlyEditing == null)
			return;

		var id = currentlyEditing;
		currentlyEditing = null;

		var edit = document.getElementById("n" + id);
		var text = document.getElementById(id);

		var newName = edit.value;
		edit.style.display = 'none';

		if (newName == "GM Quickteam")
			newName = "";

		if (save == false) {
			//edit.value = text.textContent;	// this isn't possible with onkeypress
		} else if (newName != text.textContent) {
			var qt = getQTFromId(text, 4);
			var existing = playerData["Quickteams"].getPos(qt);

			if (existing != -1) {
				playerData["QuickteamNames"][existing] = newName;

				saveGM();
			}

			edit.value = (newName == "" ? "GM Quickteam" : newName);
			text.innerHTML = edit.value.sanitizeHTML();

			var num = text.parentNode;
			num = num.childNodes[num.childNodes.length - 1];

			if (num.textContent.charAt(0) != '(') {
				if (newName != "")
					num.innerHTML = "(" + num.textContent + ")";
			} else if (newName == "") {
				num.innerHTML = num.textContent.replace("(","").replace(")","");
			}
		}

		text.style.display = 'inline';
		text.focus();
	} catch(e) {
		alert("Exception!\n\nError name: " + e.name + "\nError message: " + e.message);
	}
}

function keydownQuickteamEdit(ev) {
	return keypressQuickteamEdit(ev, true);
}

function keypressQuickteamEdit(ev, keydown) {
	var rtrn = true;
	keydown = (keydown == true ? true : false);
	var obj = ev.currentTarget;

	try {
		if (ev != null) {
			switch (ev.keyCode) {
			case 27: // escape
				if (ev && ev.preventDefault)
					ev.preventDefault();
				rtrn = false;

				endQuickteamEdit(false);
				break;
			case 13: // enter
				if (ev && ev.preventDefault)
					ev.preventDefault();

				endQuickteamEdit();
				break;
			case 46: // delete
			case 8:  // backspace
				var start = obj.selectionStart;
				var end = obj.selectionEnd;

				if (start == end) {
					if (ev.keyCode == 46)
						end++;
					else
						start--;
				}

				testDiv.innerHTML = (obj.value.substring(0, start) + obj.value.substring(end)).sanitizeHTML();
				obj.style.width = (testDiv.offsetWidth+3) + "px";
				break;
			default:
				if (keydown)
					break;

				var char = String.fromCharCode((ev.charCode ? ev.charCode : ev.which));

//document.getElementByName("rvQTNumber").innerHTML
// = ev.keyCode + "," + ev.charCode + "," + ev.which + "," + ev.shiftKey + "," + ev.ctrlKey + "," + String.fromCharCode(ev.keyCode) + "," + char;

				if ((!ev.ctrlKey) && (char.isPrintable())) {
					testDiv.innerHTML = (obj.value + char).sanitizeHTML();
					obj.style.width = (testDiv.offsetWidth+3) + "px";
				}
			}
		}
	} catch(e) {
		alert("Exception!\n\nError name: " + e.name + "\nError message: " + e.message);
	}

	return rtrn;
}

function keyupQuickteamEdit(ev) {
	try {
		testDiv.innerHTML = this.value.sanitizeHTML();
		this.style.width = (testDiv.offsetWidth+3) + "px";
	} catch(e) {
		alert("Exception!\n\nError name: " + e.name + "\nError message: " + e.message);
	}
}

function cutQuickteamEdit(ev) {
	try {
		var start = this.selectionStart;
		var end = this.selectionEnd;

		if (start != end) {
			testDiv.innerHTML = (this.value.substring(0, start) + this.value.substring(end)).sanitizeHTML();
			this.style.width = (testDiv.offsetWidth+3) + "px";
		}
	} catch(e) {
		alert("Exception!\n\nError name: " + e.name + "\nError message: " + e.message);
	}
}

function moveQuickteamDown(ev) {
	try {
		var qt = getQTFromId(this, 4);
		var existing = playerData["Quickteams"].getPos(qt);
		
		moveTableRowDown(this.parentNode.parentNode);

		playerData["Quickteams"].shiftDown(existing);
		playerData["QuickteamNames"].shiftDown(existing);

		saveGM();
		
		updateNumbers();
	} catch(e) {
		alert("Exception!\n\nError name: " + e.name + "\nError message: " + e.message);
	}
	
	if (ev && ev.preventDefault)
		ev.preventDefault();
	return false;
}

function moveQuickteamUp(ev) {
	try {
		var qt = getQTFromId(this, 2);
		var existing = playerData["Quickteams"].getPos(qt);
		
		moveTableRowUp(this.parentNode.parentNode);

		playerData["Quickteams"].shiftUp(existing);
		playerData["QuickteamNames"].shiftUp(existing);

		saveGM();
		
		updateNumbers();
	} catch(e) {
		alert("Exception!\n\nError name: " + e.name + "\nError message: " + e.message);
	}
	
	if (ev && ev.preventDefault)
		ev.preventDefault();
	return false;
}

function removeQuickteam(ev) {
	try {
		var qt = getQTFromId(this, 6);
		var existing = playerData["Quickteams"].getPos(qt);

		if (existing != -1) {
			this.parentNode.parentNode.parentNode.deleteRow(existing);

			playerData["Quickteams"].splice(existing, 1);
			playerData["QuickteamNames"].splice(existing, 1);

			saveGM();
			
			updateNumbers();
		}
	} catch(e) {
		alert("Exception!\n\nError name: " + e.name + "\nError message: " + e.message);
	}
	
	if (ev && ev.preventDefault)
		ev.preventDefault();
	return false;
}

function getQTFromId(elem, cut) {
	var qt = unescape(elem.id.substring(cut)).split(", ");
	
	while (qt.length < 3)
		qt.push("");
	
	return qt;
}

function updateNumbers() {
	var elems = document.getElementsByName("rvQTNumber");
	if (elems == null)
		return;

	var i;
	for (i = 0; i < elems.length; i++) {
		if (elems[i].textContent.charAt(0) == '(')
			elems[i].textContent = "(" + (i+1) + ")";
		else
			elems[i].textContent = (i+1);
	}
}

function moveTableRowUp(node) {
	var prevRow = node.previousSibling;
	var tableNode = node.parentNode;
	
	var remove = tableNode.removeChild(node);
	tableNode.insertBefore(remove, prevRow);
	
	return (prevRow == null);
}

function moveTableRowDown(node) {
	var nextRow = node.nextSibling;
	var tableNode = node.parentNode;
	var rtrn = nextRow == null;
	
	if (rtrn)
		nextRow = tableNode.firstChild;
	else
		nextRow = nextRow.nextSibling;

	if (nextRow == node)
		return true;
	
	var remove = tableNode.removeChild(node);
	tableNode.insertBefore(remove, nextRow);
	
	return rtrn;
}

var Allies = new Object();

Allies.getAllyName = function(src) {
	var pos = src.lastIndexOf("/")+1;
	var text = src.substring(pos).replaceAll("_", " ").replaceAll(" Lvl. ", " ").replaceAll("%27", "'").replace("ss\\.gif", "").replace(/\.gif/, "");
	return text;
};

Allies.getAllyWithoutLevel = function(name) {
	if (name.match(" \\d+$")) {
		var pos = name.lastIndexOf(" ");
		return name.substring(0, pos);
	}

	return name;
};

Allies.getPos = function(name) {
	if (allies != null) {
		var i;
		for (i = 0; i < allies.length; i++) {
			if (Allies.equals(name, allies[i][1]))
				return i;
		}
	}

	return -1;
};

Allies.get = function(name) {
	var pos = Allies.getPos(name);
	if (pos == -1)
		return null;

	return allies[pos];
};

Allies.getAllyLevel = function(name) {
	var match;

	if (match = name.match(" (\\d+)")) {
		return parseInt(match[1]);
	}

	return 1;
};

Allies.equals = function(input, arraytest) {
	if (arraytest == null)
		return (input == null);

	return (arraytest.match("^" + input + "( \\d+)?$") != null);
};

function selectAll(id) {
    document.getElementById(id).focus();
    document.getElementById(id).select();
}

function centerdiv(div, Xwidth, Yheight) {
	// First, determine how much the visitor has scrolled

	/*var scrolledX, scrolledY;

	if( self.pageYoffset ) {
		scrolledX = self.pageXoffset;
		scrolledY = self.pageYoffset;
	} else if( document.documentElement && document.documentElement.scrollTop ) {
		scrolledX = document.documentElement.scrollLeft;
		scrolledY = document.documentElement.scrollTop;
	} else if( document.body ) {
		scrolledX = document.body.scrollLeft;
		scrolledY = document.body.scrollTop;
	}*/

	// Next, determine the coordinates of the center of browser's window

	var centerX, centerY;
	if( self.innerHeight ) {
		centerX = self.innerWidth;
		centerY = self.innerHeight;
	} else if( document.documentElement && document.documentElement.clientHeight ) {
		centerX = document.documentElement.clientWidth;
		centerY = document.documentElement.clientHeight;
	} else if( document.body ) {
		centerX = document.body.clientWidth;
		centerY = document.body.clientHeight;
	}

	if (Yheight.substring(Yheight.length-1) == "%")
		Yheight = parseInt(Yheight) / 100 * centerY;
	if (Xwidth.substring(Xwidth.length-1) == "%")
		Xwidth = parseInt(Xwidth) / 100 * centerX;

	// Xwidth is the width of the div, Yheight is the height of the
	// div passed as arguments to the function:
	var leftoffset = /*scrolledX + */(centerX - Xwidth) / 2;
	var topoffset = /*scrolledY + */(centerY - Yheight) / 2;
	// The initial width and height of the div can be set in the
	// style sheet with display:none; divid is passed as an argument to // the function
	var r = div.style;
	r.position = 'fixed';
	r.top = topoffset + 'px';
	r.left = leftoffset + 'px';
	r.width = Xwidth + 'px';
	r.height = Yheight + 'px';
}

document.getElementByName = function(str) {
	var rtrn = document.getElementsByName(str);
	if ((rtrn == null) || (rtrn.length == 0)) return null;
	return rtrn[0];
};

String.prototype.replaceAll	= function(str, str2) { return (this.replace(new RegExp(str, "g"), str2)); };
String.prototype.isPrintable	= function() { return (this.match(/[\x00-\x1F\x80-\xFF]/) == null); };
String.prototype.sanitizeHTML	= function() { return (this.replaceAll("&", "&amp;").replaceAll(" ", "&nbsp;").replaceAll("<", "&lt;").replaceAll(">", "&gt;")); };

Array.prototype.swap = function (x,y) {
	var b = this[x];
	this[x] = this[y];
	this[y] = b;

	return this;
}

Array.prototype.shiftUp = function (pos) {
	if (pos == 0) {
		this.push(this.shift());
	} else {
		this.swap(pos, pos - 1);
	}
}

Array.prototype.shiftDown = function (pos) {
	if (pos == this.length - 1) {
		this.unshift(this.pop());
	} else {
		this.swap(pos, pos + 1);
	}
}


Array.prototype.getPos = function(obj) {
	var i = this.length;

	while (i--) {
		if (isArray(this[i])) {
			if ((isArray(obj)) && (this[i].length == obj.length)) {
				var j;
				for (j = 0; j < obj.length; j++) {
					if (this[i][j] != obj[j])
						break;
				}
				
				if (j == obj.length)
					return i;
			}
		}
		if (this[i] === obj) {
			return i;
		}
	}
	
	return -1;
};

function isArray(obj) {
	return obj.constructor == Array;
}

function DnDTable() {
	this.init = false;
	this.table = null;
	this.dragObject = null;
	this.dragPosition = null;
	this.oldY = 0;

	// attach to table
	this.init = function (table) {
		// can't run this script without the event listeners
		if (!window.addEventListener) {
			return;
		}

		this.table = table;
		this.init = true;
		var self = this;

		// make rows draggable

		var rows = table.rows;
		var i;
		for (i = 0; i < rows.length; i++) {
			this.makeDraggable(rows[i]);
		}

		window.addEventListener('mousemove',
		function (ev) {
			return self.onMouseMove(ev || window.event);
		}
		,false);

		window.addEventListener('mouseup',
		function (ev) {
			return self.onMouseUp(ev || window.event);
		}
		,false);
	}

	// make a element in the table draggable
	this.makeDraggable = function (item) {
		if ((!item) || (!this.init))
			return;

		var nodrag = item.getAttribute("NoDrag");
		if ((nodrag != null) && (nodrag != "undefined"))
			return;

		var self = this;

		var cells = item.cells;
		var i;
		for (i = 0; i < cells.length; i++) {
			var cell = cells[i];
			var nodrag = cell.getAttribute("NoDrag");

			if ((nodrag == null) || (nodrag == "undefined")) {
				cell.addEventListener('mousedown',
					function (ev) {
						// // allow normal processing on form controls
						// var targetName = this.getEventSource(ev).tagName.toUpperCase();
						// 
						// if ((targetName == 'INPUT') || (targetName == 'SELECT') || (targetName == 'TEXTAREA'))
						// 	return true;

						if (self.dragObject != null) {
							return self.onMouseUp(ev);
						} else {
							self.dragObject = item;
							self.mouseOffset = self.getMouseOffset(this, ev);

							if (self.onStartDragging) {
								self.onStartDragging();
							}
						}

						if (ev && ev.preventDefault)
							ev.preventDefault();

						return false;
					}
				,false);

				cell.style.cursor = "move";
			}
		}
	}

	this.getEventSource = function getEventSource(evt) {
		if (window.event) {
			return window.event.srcElement; // For IE
		} else {
			return evt.target; // For Firefox
		}
	}

	this.onMouseMove = function (ev) {
		if (this.dragObject) {
			var mousePos = this.mouseCoords(ev);
			var y = mousePos.y - this.mouseOffset.y;

			if (y != this.oldY) {
				var movingDown = y > this.oldY;
				this.oldY = y;

				// If we're over a row then move the dragged row to there so that the user sees the effect dynamically
				var currentRow = this.findDropTargetRow(y);

				if (this.onDragging)
					this.onDragging(currentRow);

				if ((currentRow) && (this.dragObject != currentRow)) {
					if (movingDown) {
						this.dragObject.parentNode.insertBefore(this.dragObject, currentRow.nextSibling);
					} else {
						this.dragObject.parentNode.insertBefore(this.dragObject, currentRow);
					}
				}
			}
		}

		return false;
	}

	this.onMouseUp = function (ev) {
		if (this.dragObject) {
			if (this.onDrop)
				this.onDrop();

			this.dragObject = null;
		}

		if (ev && ev.preventDefault)
			ev.preventDefault();

		return false;
	}

	// get the mouse offset from the element's position
	this.getMouseOffset = function (target, ev) {
		ev = ev || window.event;

		var docPos    = this.getPosition(target);
		var mousePos  = this.mouseCoords(ev);

		return {x:mousePos.x - docPos.x, y:mousePos.y - docPos.y};
	}

	// get the position of an element by going up the DOM tree and adding up all the offsets
	this.getPosition = function (e) {
		var left = 0;
		var top  = 0;

		// Safari fix
		if (e.offsetHeight == 0) {
			/**
				Safari 2 doesn't correctly grab the offsetTop of a table row
				this is detailed here:
				http://jacob.peargrove.com/blog/2006/technical/table-row-offsettop-bug-in-safari/
				the solution is likewise noted there, grab the offset of a table cell in the row - the firstChild.
				note that firefox will return a text node as a first child, so designing a more thorough
				solution may need to take that into account, for now this seems to work in firefox, safari, ie
			*/

			e = e.firstChild; // a table cell
		}

		while (e.offsetParent) {
			left += e.offsetLeft;
			top += e.offsetTop;
			e = e.offsetParent;
		}

		left += e.offsetLeft;
		top += e.offsetTop;

		return {x:left, y:top};
	}

	// get the mouse coordinates from the event
	this.mouseCoords = function (ev) {
		if ((ev.pageX) || (ev.pageY)) {
			return {x:ev.pageX, y:ev.pageY};
		}

		return {x:ev.clientX + document.body.scrollLeft - document.body.clientLeft, y:ev.clientY + document.body.scrollTop  - document.body.clientTop};
	}

	// 
	this.findDropTargetRow = function (y) {
		var rows = this.table.rows;

		var i;
		for (i = 0; i < rows.length; i++) {
			var row = rows[i];

			var nodrop = row.getAttribute("NoDrop");
			if ((nodrop == null) || (nodrop == "undefined")) {
				var rowY = this.getPosition(row).y;
				var rowHeight;

				if (row.offsetHeight == 0) {
					rowY = this.getPosition(row.firstChild).y;
					rowHeight = parseInt(row.firstChild.offsetHeight);
				} else {
					rowHeight = parseInt(row.offsetHeight);
				}

				rowHeight /= 2;


				// we need to offset the height a bit because we are inserting before
				if ((y > rowY - rowHeight) && (y < rowY + rowHeight)) {
					return row;
				}
			}
		}

		return null;
	}

	this.endDragging = function () {
		if (this.dragObject != null)
			self.onMouseUp(ev);
	}

	// this function is called when you start dragging a row, so redefine it in your code to do whatever you want
	// takes 0 parameters
	this.onStartDragging = null;

	// this function is called when a row is being dragged, so redefine it in your code to do whatever you want
	// takes 1 parameters: row hovering over in table
	this.onDragging = null;

	// this function is called when you drop a row, so redefine it in your code to do whatever you want
	// takes 0 parameters
	this.onDrop = null;
}

var gvar = new Object();

function GM_ApiBrowserCheck() {
	if (typeof(unsafeWindow) == 'undefined') { unsafeWindow=window; }
	if (typeof(GM_log) == 'undefined') { GM_log = function(msg) { try { unsafeWindow.console.log('GM_log: ' + msg); } catch(e) {} }; }
	GM_clog = function(msg) { if (arguments.callee.counter) { arguments.callee.counter++; } else { arguments.callee.counter=1; } GM_log('('+arguments.callee.counter+') '+msg); }
	GM_addGlobalStyle = function(css) {
		var sel = document.createElement('style');
		sel.setAttribute('type','text/css');
		sel.appendChild(document.createTextNode(css));
		var hel = document.documentElement.firstChild;
		while (hel && hel.nodeName != 'HEAD') { hel=hel.nextSibling; }
		if (hel && hel.nodeName == 'HEAD') { hel.appendChild(sel); } else { document.body.insertBefore(sel,document.body.firstChild); }
		return sel;
	}
	
	var needApiUpgrade=false;

	if(window.navigator.appName.match(/^opera/i) && typeof(window.opera) != 'undefined') {
		needApiUpgrade=true; gvar.isOpera=true; GM_log=window.opera.postError; GM_log('Opera detected...');
	}

	if(typeof(GM_setValue) != 'undefined') {
		try {
			var gsv=GM_setValue.toString();
			if (gsv.indexOf('staticArgs') > 0) { gvar.isGreaseMonkey = true; GM_log('GreaseMonkey Api detected...'); }
			else if (gsv.match(/not\s+supported/)) { needApiUpgrade = true; gvar.isBuggedChrome = true; GM_log('Bugged Chrome GM Api detected...'); }
		} catch(e) {
			gvar.isGreaseMonkey = (typeof(GM_setValue) == 'function');
			if (gvar.isGreaseMonkey)
				GM_log('GreaseMonkey Api is assumed because of exception...');
			else
				needApiUpgrade = true;
		}
	} else {
		needApiUpgrade=true; GM_log('No GM Api detected...');
	}

	if(needApiUpgrade) {
		GM_log('Try to recreate needed GM Api...');
 		var ws = null;
		try { ws=typeof(unsafeWindow.localStorage); unsafeWindow.localStorage.length; } catch(e) { ws=null; } // Catch Security error

		if (ws=='object') {
			GM_log('Using localStorage for GM Api.');
			GM_getValue=function(name,defValue) { var value=unsafeWindow.localStorage.getItem(GMSTORAGE_PATH+name); if(value==null) { return defValue; } else { switch(value.substr(0,2)) { case 'S]': return value.substr(2); case 'N]': return parseInt(value.substr(2)); case 'B]': return value.substr(2)=='true'; } } return value; }
			GM_setValue=function(name,value) { switch (typeof(value)) { case 'string': unsafeWindow.localStorage.setItem(GMSTORAGE_PATH+name,'S]'+value); break; case 'number': if(value.toString().indexOf('.')<0) { unsafeWindow.localStorage.setItem(GMSTORAGE_PATH+name,'N]'+value); } break; case 'boolean': unsafeWindow.localStorage.setItem(GMSTORAGE_PATH+name,'B]'+value); break; } }
			GM_deleteValue=function(name) { unsafeWindow.localStorage.removeItem(GMSTORAGE_PATH+name); }
		} else if(!gvar.isOpera || typeof(GM_setValue)=='undefined') {
			GM_log('Using temporarilyStorage for GM Api.'); gvar.temporarilyStorage=new Array();
			GM_getValue=function(name,defValue) { if(typeof(gvar.temporarilyStorage[GMSTORAGE_PATH+name])=='undefined') { return defValue; } else { return gvar.temporarilyStorage[GMSTORAGE_PATH+name]; } }
			GM_setValue=function(name,value) { switch (typeof(value)) { case "string": case "boolean": case "number": gvar.temporarilyStorage[GMSTORAGE_PATH+name]=value; } }
			GM_deleteValue=function(name) { delete gvar.temporarilyStorage[GMSTORAGE_PATH+name]; };
		}
		
		if(typeof(GM_addStyle)=='undefined') { GM_addStyle=function(css) { var heads = document.getElementsByTagName("head"); if (heads.length > 0) { var node = document.createElement("style"); node.type = "text/css"; node.appendChild(document.createTextNode(css)); heads[0].appendChild(node); } } }
		if(typeof(GM_openInTab)=='undefined') { GM_openInTab=function(url) { unsafeWindow.open(url,""); } }
		if(typeof(GM_registerMenuCommand)=='undefined') { GM_registerMenuCommand=function(name,cmd) { GM_log("Notice: GM_registerMenuCommand is not supported."); } } // Dummy
		if(!gvar.isOpera || typeof(GM_xmlhttpRequest)=='undefined') {
			GM_log('Using XMLHttpRequest for GM Api.');
			GM_xmlhttpRequest=function(obj) {
				var request=new XMLHttpRequest();
				request.onreadystatechange=function() { if(obj.onreadystatechange) { obj.onreadystatechange(request); }; if(request.readyState==4 && obj.onload) { obj.onload(request); } }
				request.onerror=function() { if(obj.onerror) { obj.onerror(request); } }
				try { request.open(obj.method,obj.url,true); } catch(e) { if(obj.onerror) { obj.onerror( {readyState:4,responseHeaders:'',responseText:'',responseXML:'',status:403,statusText:'Forbidden'} ); }; return; }
				if(obj.headers) { for(name in obj.headers) { request.setRequestHeader(name,obj.headers[name]); } }
				request.send(obj.data); return request;
			}
		}
	}
}

function waitForReady(callback) {
	var docState;
	
	try { docState = unsafeWindow.document.readyState; } catch(e) { docState = null; }
	if(docState) {
		if ((docState != 'complete') && (docState != 'interactive')) {
			window.setTimeout(waitForReady, 150, callback);
			return;
		}
	}
	
	callback();
}

GM_ApiBrowserCheck();
waitForReady(load);