// ==UserScript==
// @name Multiple Windows Live IDs
// @id Multiple_Windows_Live_IDs@https://github.com/jerone/UserScripts
// @namespace https://github.com/jerone/UserScripts
// @description Easy login with multiple Microsoft accounts.
// @author jerone
// @copyright 2014+, jerone (https://github.com/jerone)
// @license CC-BY-NC-SA-4.0; https://creativecommons.org/licenses/by-nc-sa/4.0/legalcode
// @license GPL-3.0-or-later; http://www.gnu.org/licenses/gpl-3.0.txt
// @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.2.0
// @grant GM_getValue
// @grant GM_setValue
// @run-at document-end
// @include http*://login.live.com*
// ==/UserScript==
// cSpell:ignore MWLID, maincontent, phholder, transform
/* eslint security/detect-object-injection: "off" */
(function () {
var autoLogin = true;
var addPassMask = true;
window.setTimeout(function () {
var profileString = GM_getValue("MWLID.profiles"),
profiles = [
{ name: "Account 1", mail: "test1@live.com", pass: "P@ssw0rd" },
{
name: "Account 2",
mail: "test2@live.com",
pass: "P@ssw0rd",
photo: "",
},
{
name: "Account 3",
mail: "test3@live.com",
pass: "P@ssw0rd",
photo: "http://my.pictu.re/img.png",
},
{
name: "Account 4",
mail: "test4@live.com",
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 =
// layout;
"#maincontent, #accountTD { display: inline-block; }" +
// accounts;
"#accountTD { font-size: 12px; width: 500px; min-height: 400px; margin: 5px; }" +
"#accountTD .profile { text-transform: uppercase; color: #FFFFFF; cursor: pointer; float: left; height: 150px; position: relative; margin: 5px; padding: 5px; text-align: center; width: 150px; }" +
"#accountTD .profile:hover{ opacity: 0.85; }" +
"#accountTD .profile.dark { color: #000000; }" +
"#accountTD .profile > img { 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: 140px; }" +
"#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: 100px; height: 100px; }" +
"#accountTD .addAccountBtn:hover { opacity: 1; }" +
"#accountTD .addAccountBtn > img { max-height: 40px; max-width: 40px; }" +
"#accountTD .addAccountBtn > span { width: 90px; }" +
"#accountTD .addAccountBtn > div { float: right; }" +
// edit account;
"#editAccountTD { display: none; position: relative; }" +
"#editAccountTD .signInHeader img { position: relative; left: -34px; }" +
"#editAccountTD .phholder { left: 0px; top: 0px; width: 100%; position: absolute; z-index: 5; cursor: text; }" +
"#editAccountTD .alert-error { display: none; }" +
"#editAccountCancel { background-color: #D11141; margin-left: 8px; }" +
// password mask;
".passMask { position: absolute; right: 8px; top: 8px; width: 16px; height: 16px; cursor: pointer; }";
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";
var mainTD = document.getElementById("maincontent");
mainTD.parentNode.insertBefore(accountTD, mainTD);
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;
fireEvent(document.getElementById("i0116"), "change");
document.getElementById("i0118").value = _profile.pass;
fireEvent(document.getElementById("i0118"), "change");
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.querySelector(
"#maincontent > section",
).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.querySelector("#maincontent > section").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>" +
'<input id="editAccountId" type="hidden" />' +
"<div>" +
"<div>" +
'<div id="editAccountHeader1" class="row text-subheader">Add account</div>' +
'<div id="editAccountHeader2" class="row text-subheader">Edit account</div>' +
'<div class="form-group">' +
' <div class="placeholderContainer"><input id="editAccountName" class="form-control" type="text" /><div class="phholder"><div class="placeholder">Name</div></div></div></div>' +
'<div class="form-group">' +
' <div class="alert alert-error" id="editAccountMailError">Please enter your email address in the format someone@example.com.</div>' +
' <div class="placeholderContainer"><input id="editAccountMail" class="form-control" type="email" /><div class="phholder"><div class="placeholder">someone@example.com</div></div></div></div>' +
'<div class="form-group">' +
' <div class="alert alert-error" id="editAccountPassError">Please enter the password for your Microsoft account.</div>' +
' <div class="placeholderContainer"><input id="editAccountPass" class="form-control" type="password"/><div class="phholder"><div class="placeholder">Password</div></div></div></div>' +
'<div class="form-group">' +
' <div class="placeholderContainer"><input id="editAccountPhoto" class="form-control" type="text" /><div class="phholder"><div class="placeholder">http://my.pictu.re/img.png</div></div></div></div>' +
'<div class="form-group">' +
' <div class="placeholderContainer"><input id="editAccountColor" class="form-control" type="text" /><div class="phholder"><div class="placeholder">#AB12CD</div></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="section">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 (e) {
e.preventDefault();
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.querySelector("#maincontent > section").style.display =
"block";
document.getElementById("editAccountTD").style.display = "none";
setAccount();
});
document
.getElementById("editAccountCancel")
.addEventListener("click", function (e) {
e.preventDefault();
document.querySelector("#maincontent > section").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 != null ? 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);
})();