Multiple Windows Live IDs

Easy login with multiple Microsoft accounts.

当前为 2020-07-18 提交的版本,查看 最新版本

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