Multiple Windows Live IDs

Easy login with multiple Microsoft accounts.

当前为 2024-02-03 提交的版本,查看 最新版本

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