当前位置:   article > 正文

js密码生成器小工具_js.tjorjs.cn

js.tjorjs.cn

此小工具是我突发奇想想的一个密码加密算法, 通过对一个小数的幂指数的不断迭代运算, 可以取得完全打乱字符排列的效果, 用户只要略微更改这密码初值,密码模板,计算指数中的任何一个, 所运算的结果都千差万别,毫无规律。

该小工具完全采用原生js编写,方便用户更改代码, 没有应用任何外部资源, 完全可以断网使用。

  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <meta charset="UTF-8">
  5. <meta name="viewport" content="width=device-width, initial-scale=1.0,maximum-scale=1.0, user-scalable=no" />
  6. <title>简单密码生成器</title>
  7. <style type="text/css">
  8. html,
  9. body {
  10. padding: 0;
  11. margin: 0;
  12. }
  13. .password-create {
  14. margin: 5px auto;
  15. border: 1px solid gray;
  16. box-shadow: rgba(0, 0, 0, .5) 2px 3px 10px;
  17. border-radius: 0;
  18. width: calc(100% - 10px);
  19. box-sizing: border-box;
  20. overflow-x: hidden;
  21. padding: 10px;
  22. min-width: 340px;
  23. max-width: 500px;
  24. }
  25. .password-create-title {
  26. color: black;
  27. font-size: 18px;
  28. font-weight: bold;
  29. padding: 5px;
  30. margin-bottom: 5px;
  31. }
  32. .row {
  33. width: 100%;
  34. height: 55px;
  35. margin-bottom: 0;
  36. display: flex;
  37. justify-content: flex-start;
  38. align-items: center;
  39. }
  40. .input_label {
  41. height: 32px;
  42. display: flex;
  43. justify-content: flex-end;
  44. align-items: center;
  45. text-align: right;
  46. width: 80px;
  47. font-size: 16px;
  48. }
  49. .input-text {
  50. height: 32px;
  51. display: flex;
  52. justify-content: flex-start;
  53. align-items: center;
  54. padding-left: 5px;
  55. margin-left: 5px;
  56. width: calc(100% - 160px);
  57. font-size: 16px;
  58. }
  59. .btn {
  60. width: 120px;
  61. height: 40px;
  62. display: inline-flex;
  63. justify-content: center;
  64. align-items: center;
  65. color: white;
  66. background: rgb(54, 211, 153);
  67. font-size: 16px;
  68. cursor: pointer;
  69. border: 0px solid rgb(54, 211, 153);
  70. outline: none;
  71. }
  72. .btn:active {
  73. opacity: 0.8;
  74. }
  75. .input-checkbox {
  76. width: 20px;
  77. height: 20px;
  78. }
  79. a {
  80. text-decoration: none;
  81. }
  82. .hint-span {
  83. color: red;
  84. }
  85. .msg {
  86. color: red;
  87. font-size: 20px;
  88. font-weight: bold;
  89. padding-left: 20px;
  90. box-sizing: border-box;
  91. }
  92. .btn-row {
  93. justify-content: flex-end;
  94. text-align: right;
  95. padding-right: 2%;
  96. box-sizing: border-box;
  97. margin-top: 10px;
  98. }
  99. .readme p {
  100. margin: 5px 0;
  101. padding: 0;
  102. font-size: 18px;
  103. font-weight: bold;
  104. color: #333;
  105. }
  106. .readme ul {
  107. list-style-type: none;
  108. padding: 0;
  109. margin: 0;
  110. }
  111. .readme li {
  112. padding: 5px;
  113. margin: 0;
  114. font-size: 16px;
  115. color: #666;
  116. line-height: 26px;
  117. }
  118. .readme .demo-img {
  119. width: 100%;
  120. }
  121. .input-select {
  122. position: relative;
  123. margin-left: 5px;
  124. width: calc(100% - 160px);
  125. font-size: 16px;
  126. box-sizing: border-box;
  127. }
  128. .input-select input {
  129. height: 36px;
  130. display: flex;
  131. justify-content: flex-start;
  132. align-items: center;
  133. width: 100%;
  134. font-size: 16px;
  135. padding-left: 10px;
  136. box-sizing: border-box;
  137. }
  138. .input-select input:focus+.select-wrap {
  139. display: block;
  140. }
  141. .input-select:focus .select-wrap {
  142. display: block;
  143. }
  144. .input-select .select-wrap {
  145. display: none;
  146. box-sizing: border-box;
  147. position: absolute;
  148. width: 100%;
  149. z-index: 1;
  150. top: 100%;
  151. left: 0;
  152. padding: 5px;
  153. background-color: white;
  154. box-shadow: rgba(0, 0, 0, .5) 2px 3px 10px;
  155. }
  156. .input-select .select-wrap .select-option {
  157. height: 32px;
  158. display: flex;
  159. justify-content: flex-start;
  160. align-items: center;
  161. font-size: 16px;
  162. color: #333;
  163. padding: 5px 10px;
  164. cursor: pointer;
  165. user-select: none;
  166. border-bottom: 1px dashed #eee;
  167. }
  168. .input-select .select-wrap .select-option:active {
  169. opacity: 0.8;
  170. background-color: #f2f2f2;
  171. }
  172. </style>
  173. </head>
  174. <body>
  175. <div class="password-create">
  176. <div class="password-create-title">
  177. 密码生成器
  178. </div>
  179. <div id="msg" class="row msg"></div>
  180. <div class="row">
  181. <label for="seedValue" class="input_label">
  182. <span class="hint-span">*</span>
  183. 密码初值:
  184. </label>
  185. <div class="input-select" tabindex="0">
  186. <input type="text" id="seedValueInput" value="" placeholder="请输入密码初始值" />
  187. </div>
  188. <a href="javascript:;" id="clearSeedValueBtn">&nbsp;&nbsp;清空</a>
  189. </div>
  190. <div class="row">
  191. <label for="pwdTpl" class="input_label">
  192. <span class="hint-span">*</span>
  193. 密码模板:
  194. </label>
  195. <div class="input-select" tabindex="0">
  196. <input type="text" id="pwdTplInput" value="" list="pwdTplDataList" placeholder="请输入密码生成模板" />
  197. <div class="select-wrap" id="pwdTplDataList"></div>
  198. </div>
  199. <a href="javascript:;" id="clearPwdTplBtn">&nbsp;&nbsp;清空</a>
  200. </div>
  201. <div class="row">
  202. <label for="createPowInput" class="input_label">
  203. <span class="hint-span">*</span>
  204. 计算指数:
  205. </label>
  206. <div class="input-select" tabindex="0">
  207. <input type="text" id="createPowInput" value="" placeholder="计算指数为小于10的8-15位小数" />
  208. <div class="select-wrap" id="createPowList"></div>
  209. </div>
  210. <a href="javascript:;" id="clearCreatePowBtn">&nbsp;&nbsp;清空</a>
  211. </div>
  212. <div class="row btn-row">
  213. <input type="button" id="createBtn" class="btn" value="生成密码" />
  214. </div>
  215. <div class="readme">
  216. <p>下载地址: &nbsp;&nbsp;<a style="font-size: 16px;" href="./pwd.html"
  217. download='pwd.html'>点此下载该工具</a></p>
  218. <p>使用说明: </p>
  219. <ul>
  220. <li>
  221. 1. 密码初值: 用于生成密码的种子, 可以看做为明密, 可以为任何字符, 可以包含汉字, 长度限制30个字符
  222. </li>
  223. <li>
  224. 2. 密码模板: 用于生成密码的模板, 最终生成的密码格式, 包含数字, 大写字母, 小写字母和特殊字符, 长度限制30个字符
  225. </li>
  226. <li>
  227. 3. 计算指数: 用于密码计算的指数, 该数字为一个小于108-15位小数, 最好为一个无理数的计算结果取前15
  228. </li>
  229. <li>
  230. 4. 把这个文件保存自己的手机或电脑上, 首次使用时一定要自己修改这个文件的模板数据, 不要用这个默认的示例
  231. </li>
  232. <li>
  233. <img class="demo-img"
  234. src="" />
  235. </li>
  236. </ul>
  237. </div>
  238. </div>
  239. <script type="text/javascript">
  240. "use strict";
  241. let PasswordUtil = function () {
  242. // 此处的配置一定要自行修改, 不要用这个默认的配置
  243. let pwdConfig = {
  244. numberMap: "1234567890",// 所有的数字
  245. uppercaseMap: "abcdefghijklmnopqrstuvwxyz",// 所有的小写字母
  246. lowercaseMap: "ABCDEFGHIJKLMNOPQRSTUVWXYZ",// 所有的大写字母
  247. specialCharacterMap: "~!@#$%^&*()_+-=|/\<>,.;:[]{}",// 所有的特殊字符
  248. randomPow: Math.PI + 1, // 打乱排序的计算指数, 一个杂乱无章的小数 如: 3.141592653589793, 小数位数最多为15
  249. createPow: Math.E + 2, // 生成密码计算指数, 一个杂乱无章的小数 如: 2.718281828459045, 小数位数最多为15
  250. }
  251. try {
  252. charMapRandom()
  253. } catch (e) {
  254. console.error(e)
  255. }
  256. function getRandomChar(s, randomIndex) {
  257. let ch = "";
  258. if (pwdConfig.numberMap.indexOf(s) > -1) {
  259. ch = "" + pwdConfig.numberMap[randomIndex % pwdConfig.numberMap.length];
  260. } else if (pwdConfig.uppercaseMap.indexOf(s) > -1) {
  261. ch = pwdConfig.uppercaseMap[randomIndex % pwdConfig.uppercaseMap.length];
  262. } else if (pwdConfig.lowercaseMap.indexOf(s) > -1) {
  263. ch = pwdConfig.lowercaseMap[randomIndex % pwdConfig.lowercaseMap.length];
  264. } else if (pwdConfig.specialCharacterMap.indexOf(s) > -1) {
  265. ch = pwdConfig.specialCharacterMap[randomIndex % pwdConfig.specialCharacterMap.length];
  266. }
  267. return ch;
  268. }
  269. function getCharWeight(s) {
  270. let weight = 1;
  271. if (pwdConfig.numberMap.indexOf(s) > -1) {
  272. weight = 2;
  273. } else if (pwdConfig.uppercaseMap.indexOf(s) > -1) {
  274. weight = 3;
  275. } else if (pwdConfig.lowercaseMap.indexOf(s) > -1) {
  276. weight = 4;
  277. } else if (pwdConfig.specialCharacterMap.indexOf(s) > -1) {
  278. weight = 5;
  279. }
  280. return weight;
  281. }
  282. // 打乱字符顺序
  283. function charMapRandom() {
  284. randowMap("numberMap")
  285. randowMap("uppercaseMap")
  286. randowMap("lowercaseMap")
  287. randowMap("specialCharacterMap")
  288. function randowMap(filedName) {
  289. let charArr = pwdConfig[filedName].split("")
  290. let len = charArr.length
  291. let precision = 12
  292. for (let i = 0; i < len; i++) {
  293. let randomNum = Math.pow(i + 1, pwdConfig.randomPow).toFixed(precision)
  294. let randomNumArr = randomNum.split(".")[1].split("").reverse()
  295. let randomIndex = parseInt(randomNumArr[0] + randomNumArr[precision / 2] + randomNumArr[precision - 1])
  296. let toIndex = randomIndex % len
  297. let temp = charArr[toIndex]
  298. charArr[toIndex] = charArr[i]
  299. charArr[i] = temp
  300. }
  301. pwdConfig[filedName] = charArr.join("")
  302. }
  303. }
  304. // 生成密码
  305. function createPwd(seedStr, pwdTpl, createPow) {
  306. let seedArr = seedStr.split("")
  307. let pwdTplArr = pwdTpl.split("")
  308. createPow = createPow || pwdConfig.createPow
  309. let seedCodeArr = [];
  310. seedArr.forEach(function (ch, index) {
  311. seedCodeArr.push(ch.charCodeAt());
  312. });
  313. let seedSum = 0;
  314. seedCodeArr.forEach(function (item, index) {
  315. seedSum += item * 1 + index;
  316. });
  317. let pwd = ""
  318. let pwdTplLen = pwdTplArr.length
  319. let seedLen = seedArr.length
  320. let precision = 12
  321. let charWeightSum = pwdTplLen
  322. for (let i = 0; i < pwdTplLen; i++) {
  323. let type = pwdTplArr[i];
  324. charWeightSum += getCharWeight(type);
  325. let randomNum = Math.pow((seedSum + charWeightSum) / (3 * i + 1), pwdConfig.createPow).toFixed(precision);
  326. let randomNumArr = randomNum.split(".")[1].split("").reverse();
  327. let randomIndex = parseInt(randomNumArr[0] + randomNumArr[precision / 2] + randomNumArr[precision - 1]);
  328. pwd += getRandomChar(type, randomIndex);
  329. }
  330. return pwd;
  331. }
  332. return {
  333. pwdConfig,
  334. createPwd,
  335. };
  336. }();
  337. setTimeout(() => {
  338. appInit()
  339. }, 0)
  340. function appInit() {
  341. // 密码模板选项数据
  342. let allPwdTplDataOptions = [
  343. "123456",
  344. "12345678",
  345. "aaa123456",
  346. "aa#123456",
  347. "Aaa123456",
  348. "Aaa#123456",
  349. "Aaa#12345678"
  350. ]
  351. // 计算指数选项数据
  352. let allCreatePowOptions = [
  353. 3.141592653589793,
  354. 2.718281828459045,
  355. 1.414213562373095,
  356. 1.732050807568877,
  357. 2.645751311064591,
  358. ]
  359. let seedValueInputEl = document.getElementById("seedValueInput")
  360. let clearSeedValueBtnEl = document.getElementById("clearSeedValueBtn")
  361. let pwdTplInputEl = document.getElementById("pwdTplInput")
  362. let clearPwdTplBtnEl = document.getElementById("clearPwdTplBtn")
  363. let createPowInputEl = document.getElementById("createPowInput")
  364. let clearCreatePowBtnEl = document.getElementById("clearCreatePowBtn")
  365. let createBtn = document.getElementById("createBtn")
  366. let pwdTplInputSelect = createSelect("pwdTplInput", "pwdTplDataList")
  367. let createPowSelect = createSelect("createPowInput", "createPowList")
  368. pwdTplInputSelect.setData(allPwdTplDataOptions, false)
  369. createPowSelect.setData(allCreatePowOptions, true)
  370. // 清空输入框
  371. clearSeedValueBtnEl.addEventListener("click", function (e) {
  372. seedValueInputEl.value = ""
  373. }, false)
  374. clearPwdTplBtnEl.addEventListener("click", function (e) {
  375. pwdTplInputEl.value = ""
  376. }, false)
  377. clearCreatePowBtnEl.addEventListener("click", function (e) {
  378. createPowInputEl.value = ""
  379. }, false)
  380. // 点击生成密码
  381. createBtn.addEventListener("click", function (e) {
  382. let seedValue = seedValueInputEl.value.trim()
  383. let pwdTpl = pwdTplInputEl.value.trim()
  384. let createPow = createPowInputEl.value
  385. if (!seedValue) {
  386. printMsg("请输入密码初始值")
  387. return false;
  388. }
  389. if (!pwdTpl) {
  390. printMsg("请输入密码生成模板")
  391. return false;
  392. }
  393. if (!checkInputValue(pwdTpl)) {
  394. printMsg("密码生成模板不能包含非法字符")
  395. return false;
  396. }
  397. if (seedValue.length > 30) {
  398. printMsg("密码初始值不能超过30个字符!")
  399. return false;
  400. }
  401. if (pwdTpl.length > 30) {
  402. printMsg("密码模板不能超过30个字符!")
  403. return false;
  404. }
  405. if (!myUtil.isNumeric(createPow)) {
  406. printMsg("请输入计算指数!")
  407. return false;
  408. }
  409. let createPowDotIndex = createPow.indexOf(".")
  410. if (createPowDotIndex == -1 || createPow * 1 > 10) {
  411. printMsg("计算指数必须为小于10的小数!")
  412. return false;
  413. }
  414. let createPowDecimal = createPow.substring(createPowDotIndex + 1)
  415. if (createPowDecimal.length > 15 || createPowDecimal.length < 8) {
  416. printMsg("计算指数小数位数必须为8-15位!")
  417. return false;
  418. }
  419. PasswordUtil.pwdConfig.createPow = createPow * 1
  420. let pwdStr = PasswordUtil.createPwd(seedValue, pwdTpl)
  421. printMsg(`<span style="color:#36d49a">${pwdStr}</span>`)
  422. }, false)
  423. function printMsg(htmlMsg) {
  424. let msgEl = document.getElementById("msg")
  425. if (msgEl) {
  426. msgEl.innerHTML = htmlMsg
  427. }
  428. }
  429. function checkInputValue(value) {
  430. let valueArr = value.split("")
  431. let allStr = PasswordUtil.pwdConfig.numberMap + PasswordUtil.pwdConfig.uppercaseMap +
  432. PasswordUtil.pwdConfig.lowercaseMap + PasswordUtil.pwdConfig.specialCharacterMap;
  433. for (let i = 0; i < valueArr.length; i++) {
  434. if (allStr.indexOf(valueArr[i]) == -1) {
  435. return false
  436. }
  437. }
  438. return true
  439. }
  440. function refreshDataList(id, dataList) {
  441. let html = "";
  442. dataList.forEach(function (val) {
  443. html += '<option value="' + val + '" />';
  444. });
  445. let dataListEl = document.getElementById(id);
  446. if (dataListEl) {
  447. dataListEl.innerHTML = html;
  448. }
  449. }
  450. function createSelect(inputId, inputListId) {
  451. let inputEl = document.getElementById(inputId)
  452. let inputListEl = document.getElementById(inputListId)
  453. let allDataList = []
  454. let showDataList = []
  455. inputEl.addEventListener("input", function (e) {
  456. refreshData()
  457. })
  458. inputEl.addEventListener("focus", function (e) {
  459. refreshData()
  460. })
  461. inputListEl.addEventListener("click", function (e) {
  462. if (e.target.classList.contains("select-option")) {
  463. let item = e.target.dataset.item
  464. inputEl.value = item
  465. inputListEl.parentNode.blur()
  466. }
  467. })
  468. function refreshData() {
  469. let inputValue = inputEl.value.trim()
  470. let showDataList = allDataList
  471. // 过滤选项
  472. // if (inputValue) {
  473. // showDataList = allDataList.filter((item) => {
  474. // return (item + "").indexOf(inputValue) > -1
  475. // })
  476. // }
  477. let html = "";
  478. showDataList.forEach(function (item, index) {
  479. html += `<div data-index="${index}" data-item="${item}" class="select-option">${item}</div>`;
  480. });
  481. if (showDataList.length == 0) {
  482. html += `<div data-index="-1" data-item="" class="select-option">暂无符合条件的数据</div>`;
  483. }
  484. inputListEl.innerHTML = html;
  485. }
  486. function setData(dataList, isInit) {
  487. allDataList = dataList
  488. if (isInit && !inputEl.value && dataList.length > 0) {
  489. inputEl.value = allDataList[0]
  490. }
  491. refreshData()
  492. }
  493. return {
  494. setData
  495. }
  496. }
  497. };
  498. </script>
  499. <script type="text/javascript">
  500. "use strict";
  501. window.myUtil = (function () {
  502. function isType(val) {
  503. return Object.prototype.toString.call(val).slice(8, -1);
  504. }
  505. function isObject(val) {
  506. return isType(val) === 'Object';
  507. }
  508. function isArray(val) {
  509. return isType(val) === 'Array';
  510. }
  511. function isFunction(val) {
  512. return isType(val) === 'Function';
  513. }
  514. function isNumeric(val) {
  515. return (!Number.isNaN(parseFloat(val))) && Number.isFinite(Number(val));
  516. }
  517. let sessionStorage = window.sessionStorage;
  518. let localStorage = window.localStorage;
  519. function saveDataToLocal(saveName, obj, isSessionStorage = true, expireTime) {
  520. if (obj === null) {
  521. if (isSessionStorage) {
  522. sessionStorage.removeItem(saveName)
  523. } else {
  524. localStorage.removeItem(saveName)
  525. }
  526. return;
  527. }
  528. if (obj && (isObject(obj) || isArray(obj))) {
  529. if (isSessionStorage) {
  530. sessionStorage.setItem(saveName, JSON.stringify(obj))
  531. } else {
  532. localStorage.setItem(saveName, JSON.stringify(obj))
  533. }
  534. } else {
  535. if (isSessionStorage) {
  536. sessionStorage.setItem(saveName, obj)
  537. } else {
  538. localStorage.setItem(saveName, obj)
  539. }
  540. }
  541. //存过期时间
  542. if (expireTime) {
  543. let expireTimeObjStr = JSON.stringify({
  544. time: new Date().getTime(),
  545. expireTime: expireTime
  546. });
  547. if (isSessionStorage) {
  548. sessionStorage.setItem(saveName + "ExpireTime", expireTimeObjStr)
  549. } else {
  550. localStorage.setItem(saveName + "ExpireTime", expireTimeObjStr)
  551. }
  552. }
  553. }
  554. function readDataFromLocal(readName, isSessionStorage = true) {
  555. let dataStr = null;
  556. let data = null;
  557. if (isSessionStorage) {
  558. dataStr = sessionStorage.getItem(readName);
  559. } else {
  560. dataStr = localStorage.getItem(readName);
  561. }
  562. if (dataStr && typeof dataStr == 'string') {
  563. //如果过期直接返回null
  564. let expireTimeObj = readDataFromLocal(readName + "ExpireTime", isSessionStorage);
  565. if (expireTimeObj) {
  566. if ((new Date().getTime() - expireTimeObj.time) >= expireTimeObj.expireTime) {
  567. saveDataToLocal(readName, null, isSessionStorage);
  568. return null;
  569. }
  570. }
  571. let first = dataStr.charAt(0);
  572. //如果是数组或对象
  573. if (first === '{' || first === '[') {
  574. data = JSON.parse(dataStr);
  575. } else {
  576. //尝试保持原始数据类型
  577. try {
  578. data = JSON.parse(dataStr);;
  579. } catch (e) {
  580. data = dataStr;
  581. }
  582. }
  583. } else {
  584. data = dataStr;
  585. }
  586. return data;
  587. }
  588. return {
  589. isType,
  590. isObject,
  591. isNumeric,
  592. saveDataToLocal,
  593. readDataFromLocal
  594. }
  595. })()
  596. </script>
  597. </body>
  598. </html>

demo地址: 简单密码生成器

github源码:  GitHub - chengxg/simple-create-password: 简单的密码生成器

  • 1. 密码初值: 用于生成密码的种子, 可以看做为明密, 可以为任何字符, 可以包含汉字, 长度限制30个字符
  • 2. 密码模板: 用于生成密码的模板, 最终生成的密码格式, 包含数字, 大写字母, 小写字母和特殊字符, 长度限制30个字符
  • 3. 计算指数: 用于密码计算的指数, 该数字为一个小于10的8-15位小数, 最好为一个无理数的计算结果取前15位
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Monodyee/article/detail/378368
推荐阅读
相关标签
  

闽ICP备14008679号