Multiple Windows Live IDs

Easy login with multiple Microsoft accounts.

目前為 2016-02-20 提交的版本,檢視 最新版本

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