Multiple Windows Live IDs

Easy login with multiple Microsoft accounts.

Pada tanggal 20 Februari 2016. Lihat %(latest_version_link).

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @id          Multiple_Windows_Live_IDs@https://github.com/jerone/UserScripts
// @name        Multiple Windows Live IDs
// @namespace   https://github.com/jerone/UserScripts
// @description Easy login with multiple Microsoft accounts.
// @author      jerone
// @copyright   2014+, jerone (http://jeroenvanwarmerdam.nl)
// @license     GNU GPLv3
// @homepage    https://github.com/jerone/UserScripts/tree/master/Multiple_Windows_Live_IDs
// @homepageURL https://github.com/jerone/UserScripts/tree/master/Multiple_Windows_Live_IDs
// @supportURL  https://github.com/jerone/UserScripts/issues
// @contributionURL https://www.paypal.com/cgi-bin/webscr?cmd=_s-xclick&hosted_button_id=VCYMHWQ7ZMBKW
// @version     0.1.2
// @grant       GM_getValue
// @grant       GM_setValue
// @run-at      document-end
// @include     http*://login.live.com*
// ==/UserScript==
/* global GM_getValue,GM_setValue */

(function() {

	var autoLogin = true;
	var addPassMask = true;

	window.setTimeout(function() {

		var profileString = GM_getValue("MWLID.profiles"),
			profiles = [
				{ name: "Account 1", mail: "[email protected]", pass: "P@ssw0rd" },
				{ name: "Account 2", mail: "[email protected]", pass: "P@ssw0rd", photo: "" },
				{ name: "Account 3", mail: "[email protected]", pass: "P@ssw0rd", photo: "http://my.pictu.re/img.png" },
				{ name: "Account 4", mail: "[email protected]", pass: "P@ssw0rd", color: "#EB008B" }
			];
		if (profileString == null) {
			GM_setValue("MWLID.profiles", JSON.stringify(profiles));
		} else {
			profiles = JSON.parse(profileString);
		}

		var image = {
			photoLight: "",
			photoDark: "",

			leftLight: "",
			leftDark: "",

			rightLight: "",
			rightDark: "",

			editLight: "",
			editDark: "",

			deleteLight: "",
			deleteDark: "",

			addLight: "",

			header: "",

			passMask: "",
		};

		function proxy(fn) {
			return function() {
				var that = this;
				return function(e) {
					var args = that.slice(0);  // clone;
					args.unshift(e);  // prepend event;
					fn.apply(this, args);
				};
			}.call([].slice.call(arguments, 1));
		}

		function fireEvent(elm, eventName) {
			var event = document.createEvent("HTMLEvents");
			event.initEvent(eventName, true, true);
			elm.dispatchEvent(event);
		}

		function addEventListeners(elm, eventNames, fn) {
			Array.forEach(eventNames, function(event) {
				elm.addEventListener(event, fn);
			});
		}

		function getContrastYIQ(hexcolor) {
			hexcolor = hexcolor.replace("#", "");
			var r = parseInt(hexcolor.substr(0, 2), 16),
				g = parseInt(hexcolor.substr(2, 2), 16),
				b = parseInt(hexcolor.substr(4, 2), 16),
				yiq = ((r * 299) + (g * 587) + (b * 114)) / 1000;
			return (yiq >= 200);
		}

		var metroColors = ["#00AEDB", "#00B159", "#F37735", "#7C4199", "#FFC425", "#EC098C", "#D11141", "#000000"], metroColorsIndex = -1;

		var css =
			// hide ad;
			"#brandModeTD { display:none; }" +

			// accounts;
			"#accountTD { font-size: 12px; width: 475px; }" +
			"#accountTD .profile { text-transform: uppercase; color: #FFFFFF; cursor: pointer; float: left; height: 138px; position: relative; margin: 5px; padding: 5px; line-height: 150px; text-align: center; width: 138px; }" +
			"#accountTD .profile:hover{ opacity: 0.85; }" +
			"#accountTD .profile.dark { color: #000000; }" +
			"#accountTD .profile > img { margin-bottom: 30px; max-height: 100px; max-width: 100px; vertical-align: middle; }" +
			"#accountTD .profile > span { bottom: 0; left: 0; margin: 5px; overflow: hidden; position: absolute; text-overflow: ellipsis; white-space: nowrap; width: 138px; }" +
			"#accountTD .profile > div { display: none; position: absolute; right: 0; top: 0; }" +
			"#accountTD .profile > div img { opacity: 0.3; margin: 4px 4px 0 0; }" +
			"#accountTD .profile:hover > div { display: block; }" +
			"#accountTD .profile:hover > div img:hover { opacity: 1; }" +

			// add account button;
			"#accountTD .addAccountBtn { opacity: 0.6; width: 88px; height: 88px; }" +
			"#accountTD .addAccountBtn:hover { opacity: 1; }" +
			"#accountTD .addAccountBtn > img { margin-bottom: 80px; max-height: 40px; max-width: 40px; }" +
			"#accountTD .addAccountBtn > span { width: 88px; }" +
			"#accountTD .addAccountBtn > div { float: right; }" +

			// edit account;
			"#editAccountTD { display: none; width: 420px; position: relative; }" +
			"#editAccountTD .signInHeader img { position: relative; left: -34px; }" +
			"#editAccountTD .phholder { position: absolute; top: 0px; left: 0px; z-index: 5; width: 100%; cursor: text; }" +
			"#editAccountTD .row.textbox { position: relative; width: 100%; }" +
			"#editAccountTD .errorDiv { display: none; }" +
			"#editAccountCancel { background-color: #D11141; margin-left: 8px; }" +

			// password mask;
			".passMask { position: absolute; right: 8px; top: 8px; width: 16px; height: 16px; cursor: pointer; }" +

			// fix for Microsoft login;
			".placeholder { margin: 6px 9px; }";
		var stylesheet = document.createElement("style");
		stylesheet.type = "text/css";
		if (stylesheet.styleSheet) {
			stylesheet.styleSheet.cssText = css;
		} else {
			stylesheet.appendChild(document.createTextNode(css));
		}
		(document.head || document.getElementsByTagName("head")[0]).appendChild(stylesheet);

		var accountTD = document.createElement("div");
		accountTD.id = "accountTD";
		accountTD.classList.add("floatLeft");

		var mainTD = document.getElementById("mainTD");
		mainTD.insertBefore(accountTD, mainTD.firstChild);

		function paint() {
			profiles.forEach(function(profile, i) {
				if (!profile.color) {
					profile.color = metroColors[(metroColorsIndex = ++metroColorsIndex >= metroColors.length ? 0 : metroColorsIndex)];
					GM_setValue("MWLID.profiles", JSON.stringify(profiles));
				}
				var contrastDark = getContrastYIQ(profile.color);

				var profileDiv = document.createElement("div");
				profileDiv.classList.add("profile", contrastDark ? "dark" : "light");
				profileDiv.setAttribute("title", profile.mail);
				profileDiv.style.backgroundColor = profile.color;
				profileDiv.addEventListener("click", proxy(function(_event, _profile) {
					document.getElementById("i0116").value = _profile.mail;

					var idDiv_PWD_UsernameExample = document.getElementById("idDiv_PWD_UsernameExample");
					if (idDiv_PWD_UsernameExample) { idDiv_PWD_UsernameExample.style.display = "none"; }

					document.getElementById("i0118").value = _profile.pass;

					var idDiv_PWD_PasswordExample = document.getElementById("idDiv_PWD_PasswordExample");
					if (idDiv_PWD_PasswordExample) { idDiv_PWD_PasswordExample.style.display = "none"; }

					if (autoLogin) {
						document.getElementById("idSIButton9").click();
					}
				}, profile));

				var profileImg = document.createElement("img");
				profileImg.classList.add("profileImg");
				profileImg.setAttribute("src", profile.photo || profile.img || (contrastDark ? image.photoDark : image.photoLight));

				var profileName = document.createElement("span");
				profileName.classList.add("profileName");
				profileName.appendChild(document.createTextNode(profile.name));

				var profileManage = document.createElement("div");
				profileManage.classList.add("profileManage");

				if (i !== 0) {
					var profileManageLeft = document.createElement("img");
					profileManageLeft.setAttribute("title", "Move to the left");
					profileManageLeft.setAttribute("src", contrastDark ? image.leftDark : image.leftLight);
					profileManage.appendChild(profileManageLeft);
					profileManageLeft.addEventListener("click", proxy(function(_event, _profile, _i) {
						_event.stopPropagation();

						var index = parseInt(_i, 10);

						if (parseInt(editAccountId.value, 10) === index) { editAccountId.value = index - 1; }

						var tmp = profiles[index];
						profiles[index] = profiles[index - 1];
						profiles[index - 1] = tmp;

						GM_setValue("MWLID.profiles", JSON.stringify(profiles));

						repaint();
					}, profile, i));
				}

				if (i !== (profiles.length - 1)) {
					var profileManageRight = document.createElement("img");
					profileManageRight.setAttribute("title", "Move to the right");
					profileManageRight.setAttribute("src", contrastDark ? image.rightDark : image.rightLight);
					profileManage.appendChild(profileManageRight);
					profileManageRight.addEventListener("click", proxy(function(_event, _profile, _i) {
						_event.stopPropagation();

						var index = parseInt(_i, 10);

						if (parseInt(editAccountId.value, 10) === index) { editAccountId.value = index + 1; }

						var tmp = profiles[index];
						profiles[index] = profiles[index + 1];
						profiles[index + 1] = tmp;

						GM_setValue("MWLID.profiles", JSON.stringify(profiles));

						repaint();
					}, profile, i));
				}

				var profileManageEdit = document.createElement("img");
				profileManageEdit.setAttribute("title", "Click to edit this account...");
				profileManageEdit.setAttribute("src", contrastDark ? image.editDark : image.editLight);
				profileManage.appendChild(profileManageEdit);
				profileManageEdit.addEventListener("click", proxy(function(_event, _profile, _i) {
					_event.stopPropagation();

					document.getElementById("signInTD").style.display = "none";

					document.getElementById("editAccountTD").style.display = "block";

					setAccount(_i, _profile);
				}, profile, i));

				var profileManageDelete = document.createElement("img");
				profileManageDelete.setAttribute("title", "Delete this account!");
				profileManageDelete.setAttribute("src", contrastDark ? image.deleteDark : image.deleteLight);
				profileManage.appendChild(profileManageDelete);
				profileManageDelete.addEventListener("click", proxy(function(_event, _profile, _i) {
					_event.stopPropagation();

					if (window.confirm("Are you sure you want to delete this account?")) {
						profiles.splice(_i, 1);

						GM_setValue("MWLID.profiles", JSON.stringify(profiles));

						repaint();

						setAccount();
					}
				}, profile, i));

				accountTD.appendChild(profileDiv);
				profileDiv.appendChild(profileImg);
				profileDiv.appendChild(profileName);
				profileDiv.appendChild(profileManage);
			});

			var addAccountBtnDiv = document.createElement("div");
			addAccountBtnDiv.classList.add("profile", "addAccountBtn");
			addAccountBtnDiv.setAttribute("title", "Add account");
			addAccountBtnDiv.style.backgroundColor = "#0072C6";
			addAccountBtnDiv.addEventListener("click", function() {
				document.getElementById("signInTD").style.display = "none";

				document.getElementById("editAccountTD").style.display = "block";

				setAccount();
			});

			var addAccountBtnImg = document.createElement("img");
			addAccountBtnImg.classList.add("profileImg");
			addAccountBtnImg.setAttribute("src", image.addLight);

			var addAccountBtnName = document.createElement("span");
			addAccountBtnName.classList.add("profileName");
			addAccountBtnName.appendChild(document.createTextNode("Add account"));

			accountTD.appendChild(addAccountBtnDiv);
			addAccountBtnDiv.appendChild(addAccountBtnImg);
			addAccountBtnDiv.appendChild(addAccountBtnName);
		}
		function repaint() {
			while (accountTD.hasChildNodes()) {
				accountTD.removeChild(accountTD.lastChild);
			}
			metroColorsIndex = -1;
			paint();
		}
		paint();

		var editAccountDiv = document.createElement("div");
		editAccountDiv.id = "editAccountTD";
		editAccountDiv.classList.add("floatLeft");
		editAccountDiv.innerHTML =
			'<div style="height: 40px;"></div>' +
			'<div id="i0272" class="signInHeader" style="height: 80px;">' +
				'<h1><img src="' + image.header + '" alt="Multiple Windows Live IDs" /></h1>' +
			'</div>' +
			'<div style="width: 100px; height: ' + ((document.getElementById("signInTD").offsetHeight || 500) - 120) + 'px;" class="floatLeft"><input id="editAccountId" type="hidden" /></div>' +
			'<div style="width: 320px;" class="floatLeft">' +
				'<div class="section">' +
					'<div id="editAccountHeader1" class="row label">Add account</div>' +
					'<div id="editAccountHeader2" class="row label">Edit account</div>' +
					'<div class="row textbox"><input id="editAccountName"  type="text"    ><div class="phholder"><div class="placeholder">Name</div></div></div>' +
					'<div class="errorDiv" id="editAccountMailError">Please enter your email address in the format [email protected].</div>' +
					'<div class="row textbox"><input id="editAccountMail"  type="email"   ><div class="phholder"><div class="placeholder">[email protected]</div></div></div>' +
					'<div class="errorDiv" id="editAccountPassError">Please enter the password for your Microsoft account.</div>' +
					'<div class="row textbox"><input id="editAccountPass"  type="password"><div class="phholder"><div class="placeholder">Password</div></div></div>' +
					'<div class="row textbox"><input id="editAccountPhoto" type="text"    ><div class="phholder"><div class="placeholder">http://my.pictu.re/img.png</div></div></div>' +
					'<div class="row textbox"><input id="editAccountColor" type="text"    ><div class="phholder"><div class="placeholder">#AB12CD</div></div></div>' +
				'</div>' +
				'<div class="section"><input id="editAccountSubmit" value="Submit" class="default" type="submit"><input id="editAccountCancel" value="Cancel" class="default" type="submit"></div>' +
				'<div class="signUpFloat">Multiple Windows Live IDs. <a class="TextSemiBold" href="https://github.com/jerone/UserScripts/tree/master/Multiple_Windows_Live_IDs" target="_blank">More info...</a></div>' +
			'</div>';
		mainTD.appendChild(editAccountDiv);

		var editAccountHeader1 = document.getElementById("editAccountHeader1"),
			editAccountHeader2 = document.getElementById("editAccountHeader2"),

			editAccountId = document.getElementById("editAccountId"),
			editAccountName = document.getElementById("editAccountName"),
			editAccountMail = document.getElementById("editAccountMail"),
			editAccountPass = document.getElementById("editAccountPass"),
			editAccountPhoto = document.getElementById("editAccountPhoto"),
			editAccountColor = document.getElementById("editAccountColor"),

			editAccountMailError = document.getElementById("editAccountMailError"),
			editAccountPassError = document.getElementById("editAccountPassError");

		addPlaceHolders(editAccountName);
		addPlaceHolders(editAccountMail);
		addPlaceHolders(editAccountPass);
		addPlaceHolders(editAccountPhoto);
		addPlaceHolders(editAccountColor);

		if (addPassMask) { addPassMaskFn(editAccountPass); }

		document.getElementById("editAccountSubmit").addEventListener("click", function() {
			editAccountMailError.style.display = !editAccountMail.value ? "block" : "none";
			editAccountPassError.style.display = !editAccountPass.value ? "block" : "none";

			if (!editAccountPass.value || !editAccountMail.value) { return; }

			var index = parseInt(editAccountId.value, 10);
			profiles[index === -1 ? profiles.length : index] = {
				name: editAccountName.value,
				mail: editAccountMail.value,
				pass: editAccountPass.value,
				photo: editAccountPhoto.value,
				color: editAccountColor.value
			};

			GM_setValue("MWLID.profiles", JSON.stringify(profiles));

			repaint();

			document.getElementById("signInTD").style.display = "block";

			document.getElementById("editAccountTD").style.display = "none";

			setAccount();
		});

		document.getElementById("editAccountCancel").addEventListener("click", function() {
			document.getElementById("signInTD").style.display = "block";

			document.getElementById("editAccountTD").style.display = "none";

			setAccount();
		});

		function setAccount(id, profile) {
			profile = profile || {};

			editAccountHeader1.style.display = !id ? "block" : "none";
			editAccountHeader2.style.display = id ? "block" : "none";

			editAccountId.value = id || -1;
			editAccountName.value = profile.name || "";
			editAccountMail.value = profile.mail || "";
			editAccountPass.value = profile.pass || "";
			editAccountPhoto.value = profile.photo || profile.img || "";
			editAccountColor.value = profile.color || metroColors[profiles.length % metroColors.length];

			fireEvent(editAccountName, "change");
			fireEvent(editAccountMail, "change");
			fireEvent(editAccountPass, "change");
			fireEvent(editAccountPhoto, "change");
			fireEvent(editAccountColor, "change");

			editAccountMailError.style.display = "";
			editAccountPassError.style.display = "";

			editAccountName.focus();
		}

		function addPlaceHolders(elm) {
			elm.parentNode.getElementsByClassName("phholder")[0].addEventListener("mouseup", function() {
				elm.focus();
			});
			addEventListeners(elm, ["change", "keyup", "keydown", "keypress"], function() {
				elm.parentNode.getElementsByClassName("phholder")[0].style.display = !elm.value ? "block" : "none";
			});
		}

		function addPassMaskFn(elm) {
			var img = document.createElement("img");
			img.classList.add("passMask");
			img.setAttribute("src", image.passMask);
			img.setAttribute("title", "Click to hide/show the password");
			img.style.display = elm.value ? "block" : "none";
			img.addEventListener("click", function() {
				elm.setAttribute("type", elm.getAttribute("type") === "password" ? "text" : "password");
			});
			addEventListeners(elm, ["change", "keyup", "keydown", "keypress"], function() {
				img.style.display = elm.value ? "block" : "none";
			});
			elm.parentNode.insertBefore(img, elm.nextSibling);
		}

		if (addPassMask) {
			addPassMaskFn(document.getElementById("i0118"));  // Microsoft password;
		}

	}, 500);
})();