NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==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 // @downloadURL https://github.com/jerone/UserScripts/raw/master/Multiple_Windows_Live_IDs/Multiple_Windows_Live_IDs.user.js // @updateURL https://github.com/jerone/UserScripts/raw/master/Multiple_Windows_Live_IDs/Multiple_Windows_Live_IDs.user.js // @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); })();