Multiple Windows Live IDs

Easy login with multiple Microsoft accounts.

  1. // ==UserScript==
  2. // @name Multiple Windows Live IDs
  3. // @id Multiple_Windows_Live_IDs@https://github.com/jerone/UserScripts
  4. // @namespace https://github.com/jerone/UserScripts
  5. // @description Easy login with multiple Microsoft accounts.
  6. // @author jerone
  7. // @copyright 2014+, jerone (https://github.com/jerone)
  8. // @license CC-BY-NC-SA-4.0; https://creativecommons.org/licenses/by-nc-sa/4.0/legalcode
  9. // @license GPL-3.0-or-later; http://www.gnu.org/licenses/gpl-3.0.txt
  10. // @homepage https://github.com/jerone/UserScripts/tree/master/Multiple_Windows_Live_IDs
  11. // @homepageURL https://github.com/jerone/UserScripts/tree/master/Multiple_Windows_Live_IDs
  12. // @supportURL https://github.com/jerone/UserScripts/issues
  13. // @contributionURL https://www.paypal.com/cgi-bin/webscr?cmd=_s-xclick&hosted_button_id=VCYMHWQ7ZMBKW
  14. // @version 0.2.0
  15. // @grant GM_getValue
  16. // @grant GM_setValue
  17. // @run-at document-end
  18. // @include http*://login.live.com*
  19. // ==/UserScript==
  20.  
  21. // cSpell:ignore MWLID, maincontent, phholder, transform
  22. /* eslint security/detect-object-injection: "off" */
  23.  
  24. (function () {
  25. var autoLogin = true;
  26. var addPassMask = true;
  27.  
  28. window.setTimeout(function () {
  29. var profileString = GM_getValue("MWLID.profiles"),
  30. profiles = [
  31. { name: "Account 1", mail: "test1@live.com", pass: "P@ssw0rd" },
  32. {
  33. name: "Account 2",
  34. mail: "test2@live.com",
  35. pass: "P@ssw0rd",
  36. photo: "",
  37. },
  38. {
  39. name: "Account 3",
  40. mail: "test3@live.com",
  41. pass: "P@ssw0rd",
  42. photo: "http://my.pictu.re/img.png",
  43. },
  44. {
  45. name: "Account 4",
  46. mail: "test4@live.com",
  47. pass: "P@ssw0rd",
  48. color: "#EB008B",
  49. },
  50. ];
  51. if (profileString == null) {
  52. GM_setValue("MWLID.profiles", JSON.stringify(profiles));
  53. } else {
  54. profiles = JSON.parse(profileString);
  55. }
  56.  
  57. var image = {
  58. photoLight:
  59. "",
  60. photoDark:
  61. "",
  62.  
  63. leftLight:
  64. "",
  65. leftDark:
  66. "",
  67.  
  68. rightLight:
  69. "",
  70. rightDark:
  71. "",
  72.  
  73. editLight:
  74. "",
  75. editDark:
  76. "",
  77.  
  78. deleteLight:
  79. "",
  80. deleteDark:
  81. "",
  82.  
  83. addLight:
  84. "",
  85.  
  86. header: "",
  87.  
  88. passMask:
  89. "",
  90. };
  91.  
  92. function proxy(fn) {
  93. return function () {
  94. var that = this;
  95. return function (e) {
  96. var args = that.slice(0); // clone;
  97. args.unshift(e); // prepend event;
  98. fn.apply(this, args);
  99. };
  100. }.call([].slice.call(arguments, 1));
  101. }
  102.  
  103. function fireEvent(elm, eventName) {
  104. var event = document.createEvent("HTMLEvents");
  105. event.initEvent(eventName, true, true);
  106. elm.dispatchEvent(event);
  107. }
  108.  
  109. function addEventListeners(elm, eventNames, fn) {
  110. Array.forEach(eventNames, function (event) {
  111. elm.addEventListener(event, fn);
  112. });
  113. }
  114.  
  115. function getContrastYIQ(hexcolor) {
  116. hexcolor = hexcolor.replace("#", "");
  117. var r = parseInt(hexcolor.substr(0, 2), 16),
  118. g = parseInt(hexcolor.substr(2, 2), 16),
  119. b = parseInt(hexcolor.substr(4, 2), 16),
  120. yiq = (r * 299 + g * 587 + b * 114) / 1000;
  121. return yiq >= 200;
  122. }
  123.  
  124. var metroColors = [
  125. "#00AEDB",
  126. "#00B159",
  127. "#F37735",
  128. "#7C4199",
  129. "#FFC425",
  130. "#EC098C",
  131. "#D11141",
  132. "#000000",
  133. ],
  134. metroColorsIndex = -1;
  135.  
  136. var css =
  137. // layout;
  138. "#maincontent, #accountTD { display: inline-block; }" +
  139. // accounts;
  140. "#accountTD { font-size: 12px; width: 500px; min-height: 400px; margin: 5px; }" +
  141. "#accountTD .profile { text-transform: uppercase; color: #FFFFFF; cursor: pointer; float: left; height: 150px; position: relative; margin: 5px; padding: 5px; text-align: center; width: 150px; }" +
  142. "#accountTD .profile:hover{ opacity: 0.85; }" +
  143. "#accountTD .profile.dark { color: #000000; }" +
  144. "#accountTD .profile > img { max-height: 100px; max-width: 100px; vertical-align: middle; }" +
  145. "#accountTD .profile > span { bottom: 0; left: 0; margin: 5px; overflow: hidden; position: absolute; text-overflow: ellipsis; white-space: nowrap; width: 140px; }" +
  146. "#accountTD .profile > div { display: none; position: absolute; right: 0; top: 0; }" +
  147. "#accountTD .profile > div img { opacity: 0.3; margin: 4px 4px 0 0; }" +
  148. "#accountTD .profile:hover > div { display: block; }" +
  149. "#accountTD .profile:hover > div img:hover { opacity: 1; }" +
  150. // add account button;
  151. "#accountTD .addAccountBtn { opacity: 0.6; width: 100px; height: 100px; }" +
  152. "#accountTD .addAccountBtn:hover { opacity: 1; }" +
  153. "#accountTD .addAccountBtn > img { max-height: 40px; max-width: 40px; }" +
  154. "#accountTD .addAccountBtn > span { width: 90px; }" +
  155. "#accountTD .addAccountBtn > div { float: right; }" +
  156. // edit account;
  157. "#editAccountTD { display: none; position: relative; }" +
  158. "#editAccountTD .signInHeader img { position: relative; left: -34px; }" +
  159. "#editAccountTD .phholder { left: 0px; top: 0px; width: 100%; position: absolute; z-index: 5; cursor: text; }" +
  160. "#editAccountTD .alert-error { display: none; }" +
  161. "#editAccountCancel { background-color: #D11141; margin-left: 8px; }" +
  162. // password mask;
  163. ".passMask { position: absolute; right: 8px; top: 8px; width: 16px; height: 16px; cursor: pointer; }";
  164. var stylesheet = document.createElement("style");
  165. stylesheet.type = "text/css";
  166. if (stylesheet.styleSheet) {
  167. stylesheet.styleSheet.cssText = css;
  168. } else {
  169. stylesheet.appendChild(document.createTextNode(css));
  170. }
  171. (document.head || document.getElementsByTagName("head")[0]).appendChild(
  172. stylesheet,
  173. );
  174.  
  175. var accountTD = document.createElement("div");
  176. accountTD.id = "accountTD";
  177.  
  178. var mainTD = document.getElementById("maincontent");
  179. mainTD.parentNode.insertBefore(accountTD, mainTD);
  180.  
  181. function paint() {
  182. profiles.forEach(function (profile, i) {
  183. if (!profile.color) {
  184. profile.color =
  185. metroColors[
  186. (metroColorsIndex =
  187. ++metroColorsIndex >= metroColors.length
  188. ? 0
  189. : metroColorsIndex)
  190. ];
  191. GM_setValue("MWLID.profiles", JSON.stringify(profiles));
  192. }
  193. var contrastDark = getContrastYIQ(profile.color);
  194.  
  195. var profileDiv = document.createElement("div");
  196. profileDiv.classList.add(
  197. "profile",
  198. contrastDark ? "dark" : "light",
  199. );
  200. profileDiv.setAttribute("title", profile.mail);
  201. profileDiv.style.backgroundColor = profile.color;
  202. profileDiv.addEventListener(
  203. "click",
  204. proxy(function (_event, _profile) {
  205. document.getElementById("i0116").value = _profile.mail;
  206. fireEvent(document.getElementById("i0116"), "change");
  207.  
  208. document.getElementById("i0118").value = _profile.pass;
  209. fireEvent(document.getElementById("i0118"), "change");
  210.  
  211. if (autoLogin) {
  212. document.getElementById("idSIButton9").click();
  213. }
  214. }, profile),
  215. );
  216.  
  217. var profileImg = document.createElement("img");
  218. profileImg.classList.add("profileImg");
  219. profileImg.setAttribute(
  220. "src",
  221. profile.photo ||
  222. profile.img ||
  223. (contrastDark ? image.photoDark : image.photoLight),
  224. );
  225.  
  226. var profileName = document.createElement("span");
  227. profileName.classList.add("profileName");
  228. profileName.appendChild(document.createTextNode(profile.name));
  229.  
  230. var profileManage = document.createElement("div");
  231. profileManage.classList.add("profileManage");
  232.  
  233. if (i !== 0) {
  234. var profileManageLeft = document.createElement("img");
  235. profileManageLeft.setAttribute("title", "Move to the left");
  236. profileManageLeft.setAttribute(
  237. "src",
  238. contrastDark ? image.leftDark : image.leftLight,
  239. );
  240. profileManage.appendChild(profileManageLeft);
  241. profileManageLeft.addEventListener(
  242. "click",
  243. proxy(
  244. function (_event, _profile, _i) {
  245. _event.stopPropagation();
  246.  
  247. var index = parseInt(_i, 10);
  248.  
  249. if (
  250. parseInt(editAccountId.value, 10) === index
  251. ) {
  252. editAccountId.value = index - 1;
  253. }
  254.  
  255. var tmp = profiles[index];
  256. profiles[index] = profiles[index - 1];
  257. profiles[index - 1] = tmp;
  258.  
  259. GM_setValue(
  260. "MWLID.profiles",
  261. JSON.stringify(profiles),
  262. );
  263.  
  264. repaint();
  265. },
  266. profile,
  267. i,
  268. ),
  269. );
  270. }
  271.  
  272. if (i !== profiles.length - 1) {
  273. var profileManageRight = document.createElement("img");
  274. profileManageRight.setAttribute(
  275. "title",
  276. "Move to the right",
  277. );
  278. profileManageRight.setAttribute(
  279. "src",
  280. contrastDark ? image.rightDark : image.rightLight,
  281. );
  282. profileManage.appendChild(profileManageRight);
  283. profileManageRight.addEventListener(
  284. "click",
  285. proxy(
  286. function (_event, _profile, _i) {
  287. _event.stopPropagation();
  288.  
  289. var index = parseInt(_i, 10);
  290.  
  291. if (
  292. parseInt(editAccountId.value, 10) === index
  293. ) {
  294. editAccountId.value = index + 1;
  295. }
  296.  
  297. var tmp = profiles[index];
  298. profiles[index] = profiles[index + 1];
  299. profiles[index + 1] = tmp;
  300.  
  301. GM_setValue(
  302. "MWLID.profiles",
  303. JSON.stringify(profiles),
  304. );
  305.  
  306. repaint();
  307. },
  308. profile,
  309. i,
  310. ),
  311. );
  312. }
  313.  
  314. var profileManageEdit = document.createElement("img");
  315. profileManageEdit.setAttribute(
  316. "title",
  317. "Click to edit this account...",
  318. );
  319. profileManageEdit.setAttribute(
  320. "src",
  321. contrastDark ? image.editDark : image.editLight,
  322. );
  323. profileManage.appendChild(profileManageEdit);
  324. profileManageEdit.addEventListener(
  325. "click",
  326. proxy(
  327. function (_event, _profile, _i) {
  328. _event.stopPropagation();
  329.  
  330. document.querySelector(
  331. "#maincontent > section",
  332. ).style.display = "none";
  333.  
  334. document.getElementById(
  335. "editAccountTD",
  336. ).style.display = "block";
  337.  
  338. setAccount(_i, _profile);
  339. },
  340. profile,
  341. i,
  342. ),
  343. );
  344.  
  345. var profileManageDelete = document.createElement("img");
  346. profileManageDelete.setAttribute(
  347. "title",
  348. "Delete this account!",
  349. );
  350. profileManageDelete.setAttribute(
  351. "src",
  352. contrastDark ? image.deleteDark : image.deleteLight,
  353. );
  354. profileManage.appendChild(profileManageDelete);
  355. profileManageDelete.addEventListener(
  356. "click",
  357. proxy(
  358. function (_event, _profile, _i) {
  359. _event.stopPropagation();
  360.  
  361. if (
  362. window.confirm(
  363. "Are you sure you want to delete this account?",
  364. )
  365. ) {
  366. profiles.splice(_i, 1);
  367.  
  368. GM_setValue(
  369. "MWLID.profiles",
  370. JSON.stringify(profiles),
  371. );
  372.  
  373. repaint();
  374.  
  375. setAccount();
  376. }
  377. },
  378. profile,
  379. i,
  380. ),
  381. );
  382.  
  383. accountTD.appendChild(profileDiv);
  384. profileDiv.appendChild(profileImg);
  385. profileDiv.appendChild(profileName);
  386. profileDiv.appendChild(profileManage);
  387. });
  388.  
  389. var addAccountBtnDiv = document.createElement("div");
  390. addAccountBtnDiv.classList.add("profile", "addAccountBtn");
  391. addAccountBtnDiv.setAttribute("title", "Add account");
  392. addAccountBtnDiv.style.backgroundColor = "#0072C6";
  393. addAccountBtnDiv.addEventListener("click", function () {
  394. document.querySelector("#maincontent > section").style.display =
  395. "none";
  396.  
  397. document.getElementById("editAccountTD").style.display =
  398. "block";
  399.  
  400. setAccount();
  401. });
  402.  
  403. var addAccountBtnImg = document.createElement("img");
  404. addAccountBtnImg.classList.add("profileImg");
  405. addAccountBtnImg.setAttribute("src", image.addLight);
  406.  
  407. var addAccountBtnName = document.createElement("span");
  408. addAccountBtnName.classList.add("profileName");
  409. addAccountBtnName.appendChild(
  410. document.createTextNode("Add account"),
  411. );
  412.  
  413. accountTD.appendChild(addAccountBtnDiv);
  414. addAccountBtnDiv.appendChild(addAccountBtnImg);
  415. addAccountBtnDiv.appendChild(addAccountBtnName);
  416. }
  417. function repaint() {
  418. while (accountTD.hasChildNodes()) {
  419. accountTD.removeChild(accountTD.lastChild);
  420. }
  421. metroColorsIndex = -1;
  422. paint();
  423. }
  424. paint();
  425.  
  426. var editAccountDiv = document.createElement("div");
  427. editAccountDiv.id = "editAccountTD";
  428. editAccountDiv.classList.add("floatLeft");
  429. editAccountDiv.innerHTML =
  430. '<div style="height: 40px;"></div>' +
  431. '<div id="i0272" class="signInHeader" style="height: 80px;">' +
  432. '<h1><img src="' +
  433. image.header +
  434. '" alt="Multiple Windows Live IDs" /></h1>' +
  435. "</div>" +
  436. '<input id="editAccountId" type="hidden" />' +
  437. "<div>" +
  438. "<div>" +
  439. '<div id="editAccountHeader1" class="row text-subheader">Add account</div>' +
  440. '<div id="editAccountHeader2" class="row text-subheader">Edit account</div>' +
  441. '<div class="form-group">' +
  442. ' <div class="placeholderContainer"><input id="editAccountName" class="form-control" type="text" /><div class="phholder"><div class="placeholder">Name</div></div></div></div>' +
  443. '<div class="form-group">' +
  444. ' <div class="alert alert-error" id="editAccountMailError">Please enter your email address in the format someone@example.com.</div>' +
  445. ' <div class="placeholderContainer"><input id="editAccountMail" class="form-control" type="email" /><div class="phholder"><div class="placeholder">someone@example.com</div></div></div></div>' +
  446. '<div class="form-group">' +
  447. ' <div class="alert alert-error" id="editAccountPassError">Please enter the password for your Microsoft account.</div>' +
  448. ' <div class="placeholderContainer"><input id="editAccountPass" class="form-control" type="password"/><div class="phholder"><div class="placeholder">Password</div></div></div></div>' +
  449. '<div class="form-group">' +
  450. ' <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>' +
  451. '<div class="form-group">' +
  452. ' <div class="placeholderContainer"><input id="editAccountColor" class="form-control" type="text" /><div class="phholder"><div class="placeholder">#AB12CD</div></div></div></div>' +
  453. "</div>" +
  454. '<div class="section"><input id="editAccountSubmit" value="Submit" class="default" type="submit"/><input id="editAccountCancel" value="Cancel" class="default" type="submit"/></div>' +
  455. '<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>' +
  456. "</div>";
  457. mainTD.appendChild(editAccountDiv);
  458.  
  459. var editAccountHeader1 = document.getElementById("editAccountHeader1"),
  460. editAccountHeader2 = document.getElementById("editAccountHeader2"),
  461. editAccountId = document.getElementById("editAccountId"),
  462. editAccountName = document.getElementById("editAccountName"),
  463. editAccountMail = document.getElementById("editAccountMail"),
  464. editAccountPass = document.getElementById("editAccountPass"),
  465. editAccountPhoto = document.getElementById("editAccountPhoto"),
  466. editAccountColor = document.getElementById("editAccountColor"),
  467. editAccountMailError = document.getElementById(
  468. "editAccountMailError",
  469. ),
  470. editAccountPassError = document.getElementById(
  471. "editAccountPassError",
  472. );
  473.  
  474. addPlaceHolders(editAccountName);
  475. addPlaceHolders(editAccountMail);
  476. addPlaceHolders(editAccountPass);
  477. addPlaceHolders(editAccountPhoto);
  478. addPlaceHolders(editAccountColor);
  479.  
  480. if (addPassMask) {
  481. addPassMaskFn(editAccountPass);
  482. }
  483.  
  484. document
  485. .getElementById("editAccountSubmit")
  486. .addEventListener("click", function (e) {
  487. e.preventDefault();
  488.  
  489. editAccountMailError.style.display = !editAccountMail.value
  490. ? "block"
  491. : "none";
  492. editAccountPassError.style.display = !editAccountPass.value
  493. ? "block"
  494. : "none";
  495.  
  496. if (!editAccountPass.value || !editAccountMail.value) {
  497. return;
  498. }
  499.  
  500. var index = parseInt(editAccountId.value, 10);
  501. profiles[index === -1 ? profiles.length : index] = {
  502. name: editAccountName.value,
  503. mail: editAccountMail.value,
  504. pass: editAccountPass.value,
  505. photo: editAccountPhoto.value,
  506. color: editAccountColor.value,
  507. };
  508.  
  509. GM_setValue("MWLID.profiles", JSON.stringify(profiles));
  510.  
  511. repaint();
  512.  
  513. document.querySelector("#maincontent > section").style.display =
  514. "block";
  515.  
  516. document.getElementById("editAccountTD").style.display = "none";
  517.  
  518. setAccount();
  519. });
  520.  
  521. document
  522. .getElementById("editAccountCancel")
  523. .addEventListener("click", function (e) {
  524. e.preventDefault();
  525.  
  526. document.querySelector("#maincontent > section").style.display =
  527. "block";
  528.  
  529. document.getElementById("editAccountTD").style.display = "none";
  530.  
  531. setAccount();
  532. });
  533.  
  534. function setAccount(id, profile) {
  535. profile = profile || {};
  536.  
  537. editAccountHeader1.style.display = !id ? "block" : "none";
  538. editAccountHeader2.style.display = id ? "block" : "none";
  539.  
  540. editAccountId.value = id != null ? id : -1;
  541. editAccountName.value = profile.name || "";
  542. editAccountMail.value = profile.mail || "";
  543. editAccountPass.value = profile.pass || "";
  544. editAccountPhoto.value = profile.photo || profile.img || "";
  545. editAccountColor.value =
  546. profile.color ||
  547. metroColors[profiles.length % metroColors.length];
  548.  
  549. fireEvent(editAccountName, "change");
  550. fireEvent(editAccountMail, "change");
  551. fireEvent(editAccountPass, "change");
  552. fireEvent(editAccountPhoto, "change");
  553. fireEvent(editAccountColor, "change");
  554.  
  555. editAccountMailError.style.display = "";
  556. editAccountPassError.style.display = "";
  557.  
  558. editAccountName.focus();
  559. }
  560.  
  561. function addPlaceHolders(elm) {
  562. elm.parentNode
  563. .getElementsByClassName("phholder")[0]
  564. .addEventListener("mouseup", function () {
  565. elm.focus();
  566. });
  567. addEventListeners(
  568. elm,
  569. ["change", "keyup", "keydown", "keypress"],
  570. function () {
  571. elm.parentNode.getElementsByClassName(
  572. "phholder",
  573. )[0].style.display = !elm.value ? "block" : "none";
  574. },
  575. );
  576. }
  577.  
  578. function addPassMaskFn(elm) {
  579. var img = document.createElement("img");
  580. img.classList.add("passMask");
  581. img.setAttribute("src", image.passMask);
  582. img.setAttribute("title", "Click to hide/show the password");
  583. img.style.display = elm.value ? "block" : "none";
  584. img.addEventListener("click", function () {
  585. elm.setAttribute(
  586. "type",
  587. elm.getAttribute("type") === "password"
  588. ? "text"
  589. : "password",
  590. );
  591. });
  592. addEventListeners(
  593. elm,
  594. ["change", "keyup", "keydown", "keypress"],
  595. function () {
  596. img.style.display = elm.value ? "block" : "none";
  597. },
  598. );
  599. elm.parentNode.insertBefore(img, elm.nextSibling);
  600. }
  601.  
  602. if (addPassMask) {
  603. addPassMaskFn(document.getElementById("i0118")); // Microsoft password;
  604. }
  605. }, 500);
  606. })();