<html> <head> <style> :root { --accent-color: #f35f5f; --text-dark: #333333; --text-light: #808080; } body { accent-color: var(--accent-color); background: #f9fafa; border: 0px; font-family: Montserrat, sans-serif; margin: 0px; padding: 0px; text-align: center; } input[type="file"] { display: none; } .accent { color: var(--accent-color); text-decoration: underline; } .center-left { display: inline-block; text-align: left; } .clickable:hover { cursor: pointer; } .content-container { border-bottom-left-radius: 32px; border-bottom-right-radius: 32px; color: var(--text-light); font-size: 16px; opacity: 1; overflow-y: hidden; padding: 24px 15%; transition: opacity 0.25s; } .content-container-button { background: var(--accent-color); border-radius: 16px; color: #ffffff; display: inline-block; padding: 16px 64px; transition: background 0.25s; } .content-container-button:hover { cursor: pointer; } .content-container-button.disabled { background: var(--text-light) !important; } .content-container-button.disabled:hover { cursor: not-allowed; } .content-container-title { color: var(--text-dark); font-size: 24px; font-weight: bold; } .icon-right-arrow { border: solid #ffffff; border-width: 0 2px 2px 0; display: inline-block; margin-left: 4px; padding: 4px; position: relative; transform: rotate(315deg); } .main-container { background: #ffffff; border-radius: 32px; margin-bottom: 64px; margin-left: 50%; margin-top: 64px; padding: 24px; transform: translateX(-50%); width: 50%; } .main-container img { height: 24px; } .title-container { color: var(--text-dark); font-size: 48px; font-weight: bold; padding: 24px; } .two-column-left { display: inline-block; text-align: right; width: 50%; } .two-column-right { display: inline-block; text-align: left; width: 50%; } .spacer { padding-top: 8px; } .spacer-dash { border-top: 8px solid var(--text-dark); margin: auto; width: 32px; } .spinner { display: inline-block; position: relative; width: 32px; height: 32px; } .spinner div { box-sizing: border-box; display: block; position: absolute; width: 32px; height: 32px; border: 4px solid var(--accent-color); border-radius: 50%; animation: spinner 1.2s cubic-bezier(0.5, 0, 0.5, 1) infinite; border-color: var(--accent-color) transparent transparent transparent; } .spinner div:nth-child(1) { animation-delay: -0.45s; } .spinner div:nth-child(2) { animation-delay: -0.3s; } .spinner div:nth-child(3) { animation-delay: -0.15s; } @keyframes spinner { 0% { transform: rotate(0deg); } 100% { transform: rotate(360deg); } } </style> <title>Versium File Hasher</title> <link rel="stylesheet" href="wstyle.css" /> </head> <body> <div class="main-container"> <div class="title-container"> <img src="data:image/svg+xml;base64, <svg xmlns="http://www.w3.org/2000/svg" width="691.74" height="129.51" viewBox="0 0 691.74 129.51"><defs><style>.a{fill:#f35f5f;}.b{fill:#ff7d7d;}.c{fill:#ffb68e;}.d{fill:#333;}</style></defs><path class="a" d="M97,111.13a12.42,12.42,0,0,1-1.93-.1,21.69,21.69,0,0,1-30.34-4.56,20.31,20.31,0,0,1-1.32-2L26.27,40.19a16.82,16.82,0,0,1,0-17.27,16.63,16.63,0,0,1,13.32-8.5l.29-.58c.39-.67.87-1.35,1.35-2A30.83,30.83,0,0,1,51,2.55H41.23A29.06,29.06,0,0,0,16,17a28.66,28.66,0,0,0,0,29.15L53.2,110.45a33.52,33.52,0,0,0,58.1,0l1.93-3.47a33.58,33.58,0,0,1-13.12,4C98.93,111,98,111.13,97,111.13Z"/><path class="b" d="M113.91,86.71h-.39c-.77,0-1.54-.1-2.31-.1A28,28,0,0,1,109.47,89a5.51,5.51,0,0,1-.68.77c-.38.39-.87.77-1.25,1.16a11.64,11.64,0,0,1-1.26,1,11.81,11.81,0,0,1-2.53,1.35,13.66,13.66,0,0,1-1.35.58,10.82,10.82,0,0,1-3.18.68,18.33,18.33,0,0,1-2.22.1,16,16,0,0,1-6.37-1.26,16.26,16.26,0,0,1-7.92-7L54.24,37.2a4.1,4.1,0,0,1-.48-.87,13.91,13.91,0,0,1-1.06-2.8,13.23,13.23,0,0,1-.48-3.86A13.38,13.38,0,0,1,53,24.85a10,10,0,0,1,1.25-2.71,14.86,14.86,0,0,1,8.21-6.75,15.17,15.17,0,0,1,2.89-.68,3.51,3.51,0,0,1,1-.09c.38,0,.67-.1,1-.1h2.9A7.88,7.88,0,0,1,71,13a5.51,5.51,0,0,1,.38-.67l.1-.19.1-.1.09-.19A30.64,30.64,0,0,1,81,2.45H67.2a31,31,0,0,0-6.66.78c-.58.09-1.07.29-1.64.38a16.77,16.77,0,0,0-3,1.16c-.48.19-1,.48-1.44.68A30.14,30.14,0,0,0,51,7.67l-1.16.86a30.2,30.2,0,0,0-5,5.51l-1.15,1.73a25.08,25.08,0,0,0-2.71,6.66,22.75,22.75,0,0,0-.67,3.48,27.87,27.87,0,0,0,0,7c.09.58.19,1.15.3,1.73.18,1.16.57,2.32.87,3.48a32.28,32.28,0,0,0,1.35,3.28c.29.58.58,1.06.87,1.64L72.13,92.21A28.6,28.6,0,0,0,87.57,105l1.74.58a22.89,22.89,0,0,0,3.66.68c1.26.09,2.51.19,3.77.19.87,0,1.83-.1,2.7-.1a28.07,28.07,0,0,0,21.91-14.09l5-8.78A26.62,26.62,0,0,1,113.91,86.71Z"/><path class="b" d="M45,14.23c-.39.58-.78,1.16-1.15,1.74C44.26,15.41,44.64,14.83,45,14.23Z"/><path class="b" d="M52.43,31.41a13.87,13.87,0,0,1-.1-1.93A6.6,6.6,0,0,0,52.43,31.41Z"/><path class="b" d="M108.58,89.8c-.39.39-.87.77-1.25,1.16A7.65,7.65,0,0,0,108.58,89.8Z"/><path class="b" d="M53.78,36.14a13.91,13.91,0,0,1-1.06-2.8A23.66,23.66,0,0,0,53.78,36.14Z"/><path class="b" d="M52.81,33.44a16.05,16.05,0,0,1-.38-1.93A7.59,7.59,0,0,0,52.81,33.44Z"/><path class="b" d="M106.19,91.83a13.5,13.5,0,0,1-2.51,1.35A9.22,9.22,0,0,0,106.19,91.83Z"/><path class="b" d="M82.73,86.33,54.26,37.1,82.73,86.33a16.34,16.34,0,0,0,7.92,6.94A16.62,16.62,0,0,1,82.73,86.33Z"/><path class="b" d="M67.39,14.42a2.89,2.89,0,0,0-1,.1A2.89,2.89,0,0,1,67.39,14.42Z"/><path class="b" d="M62.47,15.19a16,16,0,0,1,2.89-.67A16,16,0,0,0,62.47,15.19Z"/><path class="b" d="M51.17,7.76a19.75,19.75,0,0,1,3.48-2.22C53.49,6.22,52.33,7,51.17,7.76Z"/><path class="b" d="M50.11,8.63a30.48,30.48,0,0,0-5,5.5h0A27,27,0,0,1,50.11,8.63Z"/><path class="c" d="M153.09,28.9a1.28,1.28,0,0,0-.11-.58,6.82,6.82,0,0,1-.1-1.35.35.35,0,0,0-.1-.29,26.36,26.36,0,0,0-3.37-9.94.11.11,0,0,0-.1-.1h0a24.36,24.36,0,0,0-3.19-4.44l-.19-.19c-.1-.19-.29-.29-.38-.48a25.41,25.41,0,0,0-7.44-5.79,30.81,30.81,0,0,0-10.29-3.19,36.3,36.3,0,0,0-4.54-.29H97a13.79,13.79,0,0,0-1.93.1,3.7,3.7,0,0,0-.48.09h-.2a25.1,25.1,0,0,0-19,11.78.3.3,0,0,0-.1.19c-.09.2-.29.48-.38.68-.29.58-.58,1.06-.87,1.54a21.92,21.92,0,0,0-1,2.22l-.29.87a11.44,11.44,0,0,0-.48,1.64,14,14,0,0,0-.38,1.93,25.66,25.66,0,0,0,.38,11.2,11.83,11.83,0,0,0,.48,1.64,29.51,29.51,0,0,0,2.13,4.73L91.5,69.63a24.79,24.79,0,0,0,17.08,12.16c.77.19,1.64.19,2.41.39a18,18,0,0,0,2.32.09h.39a24.47,24.47,0,0,0,18.72-8.88.09.09,0,0,1,.1-.09l.09-.1c.1-.1.1-.1.1-.19a22.48,22.48,0,0,0,2.41-3.48l13.42-23.35a31.46,31.46,0,0,0,4.24-13.9v-.2a6.93,6.93,0,0,0,.1-1.44v-.58A2.11,2.11,0,0,0,153.09,28.9ZM138.33,40.19,124.91,63.55a13.22,13.22,0,0,1-11.39,6.56,13,13,0,0,1-11.39-6.56L85.53,34.79a13.71,13.71,0,0,1,0-13.61,13.44,13.44,0,0,1,10.81-6.76c.29,0,.58-.09,1-.09h26.15a16.93,16.93,0,0,1,14.87,8.59c3.07,5.4,2.8,13.22,0,17.27Z"/><path class="d" d="M196.24,104.18,178.49,57.56a8.87,8.87,0,0,1,5.1-11.47,8.78,8.78,0,0,1,3.17-.59h.77A8.92,8.92,0,0,1,196,51.68l14.09,43.43,14-43.43a8.92,8.92,0,0,1,8.49-6.18h0a8.83,8.83,0,0,1,8.3,12.06l-17.66,46.72a14.47,14.47,0,0,1-13.51,9.36h0A14.78,14.78,0,0,1,196.24,104.18Z"/><path class="d" d="M303.66,85.36h-38C268,94,274.52,99.07,283.2,99.07a23.91,23.91,0,0,0,12.74-3.67,7.33,7.33,0,0,1,9.17,1.16h0A7.52,7.52,0,0,1,304,108a40.35,40.35,0,0,1-22.29,6.18c-21.81,0-35.61-13.9-35.61-34.55,0-21,14.28-34.94,35-34.94,19.79,0,30.88,10.81,33.1,28.66A10.6,10.6,0,0,1,305,85.27,10.2,10.2,0,0,1,303.66,85.36Zm-7.43-11.49c-.29-9-6.08-14.76-15.05-14.76-8.59,0-14.48,5.79-15.93,14.76Z"/><path class="d" d="M346.9,57.66a24.82,24.82,0,0,1,11.49-10.81c5.21-2.32,11,1.74,11,7.43v1a8.1,8.1,0,0,1-7.14,7.91c-9.65,1.45-15.44,8.21-15.44,17.57v23.06a9.72,9.72,0,0,1-9.69,9.75h-.06a9.72,9.72,0,0,1-9.75-9.69V55.15A9.72,9.72,0,0,1,337,45.4h.06a9.73,9.73,0,0,1,9.75,9.69v.06Z"/><path class="d" d="M426.33,58.63l-.19.29a7.44,7.44,0,0,1-9.46,3.47,38,38,0,0,0-14.57-3.57c-4.54,0-7.82,1.54-7.82,5,0,10.23,36,5,35.81,29.14,0,13.9-12.26,21.24-27.7,21.24a50,50,0,0,1-23.45-5.6,7.41,7.41,0,0,1-3.19-9.75h0a7.35,7.35,0,0,1,9.8-3.45l.14.07A39.13,39.13,0,0,0,403,99.93c4.92,0,8.59-1.64,8.59-5.3,0-10.91-35.33-5-35.33-29,0-14.09,12.07-21.23,26.83-21.23a52.13,52.13,0,0,1,19.6,4,7.46,7.46,0,0,1,3.86,9.81C426.47,58.35,426.4,58.49,426.33,58.63Z"/><ellipse class="b" cx="452.97" cy="26" rx="10.52" ry="10.91"/><path class="d" d="M443.22,103.79V55.15a9.73,9.73,0,0,1,9.69-9.75H453a9.72,9.72,0,0,1,9.74,9.7v48.69a9.71,9.71,0,0,1-9.69,9.75h0A9.79,9.79,0,0,1,443.22,103.79Z"/><path class="d" d="M536.46,113.54h0a9.65,9.65,0,0,1-9.65-9.65v-2.32c-4.74,8.59-12.55,12.74-22.78,12.74-15.06,0-24.52-10-24.52-26V55.15a9.65,9.65,0,0,1,9.65-9.65h.09a9.65,9.65,0,0,1,9.65,9.65V82.66c0,8.59,4.93,13.8,12.74,13.8C521,96.36,526.7,89,526.7,79.09v-24a9.65,9.65,0,0,1,9.65-9.65h0A9.65,9.65,0,0,1,546,55.05v48.74a9.55,9.55,0,0,1-9.34,9.75Z"/><path class="d" d="M679.68,70.69v33.1a9.66,9.66,0,0,1-9.65,9.66h0a9.66,9.66,0,0,1-9.65-9.66V76.38c0-8.78-5-13.89-13.13-13.89-9.36.09-15,7.43-15,17.37v24a9.65,9.65,0,0,1-9.65,9.65h0a9.65,9.65,0,0,1-9.65-9.65V76.48c0-8.78-5-13.9-13-13.9-9.46.1-15.16,7.43-15.16,17.38v24a9.65,9.65,0,0,1-9.65,9.65H575a9.66,9.66,0,0,1-9.66-9.65V55.34A9.72,9.72,0,0,1,575,45.6h.06a9.72,9.72,0,0,1,9.74,9.7v2.07c4.83-8.4,12.74-12.45,23.08-12.55,11.48,0,19.78,5.79,23.06,15.74C635.38,50,644,45,655.46,44.82,670.13,44.63,679.68,54.77,679.68,70.69Z"/></svg>" alt="Versium Logo" /> <div class="spacer"></div> File Hasher </div> <div class="spacer-dash"></div> <div class="content-container" id="content-container"> </div> </div> <input type="file" id="file-input"> <script src="hash.js"></script> <script src="simple-tagger.js"></script> <script src="logic.js"></script> <script> step0(true); </script> </body> <input type="file" id="file-input"> <script> const hashingFunctions = { md5: function(str, textEncoder = null) { const T = new Uint32Array([ 0xd76aa478, 0xe8c7b756, 0x242070db, 0xc1bdceee, 0xf57c0faf, 0x4787c62a, 0xa8304613, 0xfd469501, 0x698098d8, 0x8b44f7af, 0xffff5bb1, 0x895cd7be, 0x6b901122, 0xfd987193, 0xa679438e, 0x49b40821, 0xf61e2562, 0xc040b340, 0x265e5a51, 0xe9b6c7aa, 0xd62f105d, 0x02441453, 0xd8a1e681, 0xe7d3fbc8, 0x21e1cde6, 0xc33707d6, 0xf4d50d87, 0x455a14ed, 0xa9e3e905, 0xfcefa3f8, 0x676f02d9, 0x8d2a4c8a, 0xfffa3942, 0x8771f681, 0x6d9d6122, 0xfde5380c, 0xa4beea44, 0x4bdecfa9, 0xf6bb4b60, 0xbebfbc70, 0x289b7ec6, 0xeaa127fa, 0xd4ef3085, 0x04881d05, 0xd9d4d039, 0xe6db99e5, 0x1fa27cf8, 0xc4ac5665, 0xf4292244, 0x432aff97, 0xab9423a7, 0xfc93a039, 0x655b59c3, 0x8f0ccc92, 0xffeff47d, 0x85845dd1, 0x6fa87e4f, 0xfe2ce6e0, 0xa3014314, 0x4e0811a1, 0xf7537e82, 0xbd3af235, 0x2ad7d2bb, 0xeb86d391 ]); const shift = new Uint8Array([ 7, 12, 17, 22, 7, 12, 17, 22, 7, 12, 17, 22, 7, 12, 17, 22, 5, 9, 14, 20, 5, 9, 14, 20, 5, 9, 14, 20, 5, 9, 14, 20, 4, 11, 16, 23, 4, 11, 16, 23, 4, 11, 16, 23, 4, 11, 16, 23, 6, 10, 15, 21, 6, 10, 15, 21, 6, 10, 15, 21, 6, 10, 15, 21 ]); const blocks = new Uint32Array(Math.ceil(str.length / 64) * 16); let i; for (i = 0; i < str.length; i++) { blocks[i >> 2] |= str.charCodeAt(i) << ((i % 4) * 8); } blocks[i >> 2] |= 0x80 << ((i % 4) * 8); blocks[blocks.length - 2] = str.length * 8; let a = 0x67452301; let b = 0xefcdab89; let c = 0x98badcfe; let d = 0x10325476; for (i = 0; i < blocks.length; i += 16) { const aa = a; const bb = b; const cc = c; const dd = d; for (let j = 0; j < 64; j++) { const idx = j < 16 ? j : (j % 16); const f = j < 16 ? ((b & c) | (~b & d)) : j < 32 ? ((d & b) | (~d & c)) : j < 48 ? (b ^ c ^ d) : (c ^ (b | ~d)); const g = j < 16 ? j : j < 32 ? (5 * idx + 1) % 16 : j < 48 ? (3 * idx + 5) % 16 : (7 * idx) % 16; const tmp = d; d = c; c = b; b = b + rotateLeft((a + f + T[j] + blocks[i + g]), shift[j]); a = tmp; } a += aa; b += bb; c += cc; d += dd; } const result = new Uint32Array([a, b, c, d]); const hex = Array.from(new Uint8Array(result.buffer)) .map(b => b.toString(16).padStart(2, '0')) .join(''); return hex; function rotateLeft(n, s) { return (n << s) | (n >>> (32 - s)); } }, sha1: async function(str, textEncoder) { var hashBuffer = await window.crypto.subtle.digest("SHA-1", textEncoder.encode(str)); let hashArray = Array.from(new Uint8Array(hashBuffer)); return hashArray.map(b => ('00' + b.toString(16)).slice(-2)).join(''); }, sha256: async function(str, textEncoder) { var hashBuffer = await window.crypto.subtle.digest("SHA-256", textEncoder.encode(str)); let hashArray = Array.from(new Uint8Array(hashBuffer)); return hashArray.map(b => ('00' + b.toString(16)).slice(-2)).join(''); } }; </script> <script> function generateHashedSimpleTags(first, last, address, zip, tagFormat, tagLength, bits, encoding) { var numTags = 0; if (address && zip) { numTags = 1; if (last) { numTags = 2; if (first) numTags = 3; } } else return []; var addressTag = generateStAddrHelper(address, tagFormat, tagLength); zip = generateStZipHelper(zip); var toReturn = [ hashInt64(zip + ':::' + addressTag, bits, encoding) ]; if (numTags > 1) { last = generateStNameHelper(last); toReturn[1] = hashInt64(zip + ':' + last + '::' + addressTag, bits, encoding); if (numTags > 2) { first = generateStNameHelper(first); toReturn[2] = hashInt64(zip + ':' + last + ':' + first + ':' + addressTag, bits, encoding); } } return toReturn; function generateStAddrHelper(address, tagFormat, tagLength) { var addrChunkZeroWeights = [ 'POB', 'PMB', 'BOX', 'ST', 'STE', 'AVE', 'AV', 'DR', 'RD', 'APT', 'LN', 'N', 'E', 'S', 'W', 'NW', 'NE', 'SE', 'SW', 'BLVD', 'CIR', 'WA', 'WAY', 'CT', 'PKWY', 'RR', 'UNIT', 'TER' ]; var numDecode = { zero: '0', one: '1', two: '2', three: '3', four: '4', five: '5', six: '6', seven: '7', eight: '8', nine: '9', ten: '10', eleven: '11', twelve: '12', thirteen: '13', fourteen: '14', fifteen: '15', sixteen: '16', seventeen: '17', eighteen: '18', nineteen: '19', twenty: '20', thirty: '30', forty: '40', fifty: '50', sixty: '60', seventy: '70', eighty: '80', ninety: '90', first: '1st', second: '2nd', third: '3rd', fourth: '4th', fifth: '5th', sixth: '6th', seventh: '7th', eighth: '8th', nineth: '9th', tenth: '10th', eleventh: '11th', twelfth: '12th', thirteenth: '13th', fourteenth: '14th', fifteenth: '15th', sixteenth: '16th', seventeenth: '17th', eighteenth: '18th', nineteenth: '19th', twentieth: '20th', }; address = address.toUpperCase().replace(/[^0-9A-Za-z ]/g, '').trim(); var tokWeight = []; var toks = address.split(' '); var cc = toks.length; for (var i = 0; i < cc; i++) { toks[i] = toks[i].trim(); if (tagFormat == 1) { if (numDecode[toks[i].toLowerCase()]) toks[i] = numDecode[toks[i].toLowerCase()]; } tokWeight[i] = toks[i].length; if (/^\d$/.test(toks[i].charAt(0))) tokWeight[i] = ((tokWeight[i] + 3) * 3) / 2; if (addrChunkZeroWeights.includes(toks[i])) tokWeight[i] = 0; } for (var i = 0; i < cc; i++) { for (var j = i + 1; j < cc; j++) { if (tokWeight[j] > tokWeight[i]) { var k = tokWeight[i]; var p = toks[i]; tokWeight[i] = tokWeight[j]; toks[i] = toks[j]; tokWeight[j] = k; toks[j] = p; } } } return toks.join(' ').toUpperCase().replace(/[^0-9A-Z]/g, '').substring(0, tagLength); } function generateStNameHelper(s) { return s.toUpperCase().replace(/[^A-Z]/g, ''); } function generateStZipHelper(s) { s = s.replace(/[^0-9]/g, '').padStart(5, '0'); if (s.length > 5) s = s.substring(0, 5); return s; } function hashInt64(x, bits = 64, encoding = 0) { var md5 = hashingFunctions.md5(x.trim().toLowerCase()); var x1 = md5.substring(0, 8); var x2 = md5.substring(8, 16); var x3 = md5.substring(16, 24); var x4 = md5.substring(24, 32); var i1 = BigInt(parseInt(x1, 16)); var i2 = BigInt(parseInt(x2, 16)); var i3 = BigInt(parseInt(x3, 16)); var i4 = BigInt(parseInt(x4, 16)); var overflow = 2n ** 63n; i1 = leftShiftOverflow(i1, 48n, overflow); i2 = leftShiftOverflow(i2, 32n, overflow); i3 = leftShiftOverflow(i3, 16n, overflow); var i = i1 ^ i2 ^ i3 ^ i4; if (i < 0) i *= -1n; if (bits == 32) i = i % BigInt(parseInt('7FFFFFED', 16)); else if (bits == 16) i = i % BigInt(parseInt('FFF1', 16)); else if (bits == 8) i = i % BigInt(parseInt('FB', 16)); else i = i % BigInt(parseInt('7FFFFFFFFFFFE96B', 16)); if (encoding == 1) return i.toString(16); else if (encoding == 2) return btoa(i.toString()); else return i.toString(); } function leftShiftOverflow(n, shift, overflow) { n = n * (2n ** shift); var sign = n / overflow; if (sign % 2n) return (n % overflow) - overflow; else return n % overflow; } } </script> <script> var cryptoAvailable = false; if (window.crypto && window.crypto.subtle) cryptoAvailable = true; function emptyGlobalVariables() { inputInfo = { delimiter: '', first: '', last: '', address: '', zip: '' }; outputInfo = { capitalization: 1, trim: 1, hashSalt: '', tagFormat: 1, tagLength: 8, hashLength: 64, hashEncoding: 0 } header = {}; output = ''; errors = 0; fileInputId = 'file-input'; fileInput = document.getElementById(fileInputId); contentContainerId = 'content-container'; contentContainer = document.getElementById(contentContainerId); chooseOptions = 1; hashingAlgorithm = 'md5'; } async function step0(pageLoad = false) { if (!pageLoad) await clearContentContainer(); emptyGlobalVariables(); fileInput.value = null; var title = document.createElement('div'); title.className = 'content-container-title'; title.innerHTML = 'File Selection'; contentContainer.appendChild(title); contentContainer.appendChild(createSpacer()); var content = document.createElement('div'); var fileInputLabel = document.createElement('label'); fileInputLabel.className = 'accent clickable'; fileInputLabel.htmlFor = 'file-input'; fileInputLabel.innerHTML = 'Click here'; content.appendChild(fileInputLabel); content.innerHTML += ' to select a file to hash.<br>Note that you must select a file that has headers.'; contentContainer.appendChild(content); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); var nextButton = createNextButton(); nextButton.classList.add('disabled'); nextButton.onclick = function() { if (this.classList.contains('disabled')) return; step1(); }; contentContainer.appendChild(nextButton); fileInput.addEventListener ('change', () => { if (fileInput.value) { header = getFileHeader(); nextButton.classList.remove('disabled'); } else nextButton.classList.add('disabled'); }); } async function step1() { await clearContentContainer(); var hashingAlgorithmInputId = 'input-hashalgorithm'; var title = document.createElement('div'); title.className = 'content-container-title'; title.innerHTML = 'Hashing Algorithm'; contentContainer.appendChild(title); contentContainer.appendChild(createSpacer()); var content = document.createElement('div'); content.innerHTML = 'What hashing algorithm would you like to use?'; contentContainer.appendChild(content); contentContainer.appendChild(createSpacer()); var hashingAlgorithmButtonsDiv = document.createElement('div'); hashingAlgorithmButtonsDiv.className = 'center-left'; var radioButtonMd5 = document.createElement('input'); radioButtonMd5.id = hashingAlgorithmInputId + '-md5'; radioButtonMd5.name = hashingAlgorithmInputId; radioButtonMd5.type = 'radio'; radioButtonMd5.value = 'md5'; radioButtonMd5.checked = true; var radioButtonMd5Label = document.createElement('label'); radioButtonMd5Label.htmlFor = hashingAlgorithmInputId + '-md5'; radioButtonMd5Label.innerHTML = 'MD5'; hashingAlgorithmButtonsDiv.appendChild(radioButtonMd5); hashingAlgorithmButtonsDiv.appendChild(radioButtonMd5Label); hashingAlgorithmButtonsDiv.appendChild(document.createElement('br')); var radioButtonSha1 = document.createElement('input'); radioButtonSha1.id = hashingAlgorithmInputId + '-sha1'; radioButtonSha1.name = hashingAlgorithmInputId; radioButtonSha1.onclick = () => { alert('SHA-1 is a complex hashing algorithm. Large files may take several minutes to complete hashing.'); }; radioButtonSha1.type = 'radio'; radioButtonSha1.value = 'sha1'; radioButtonSha1.checked = false; radioButtonSha1.disabled = !cryptoAvailable; var radioButtonSha1Label = document.createElement('label'); radioButtonSha1Label.htmlFor = hashingAlgorithmInputId + '-sha1'; radioButtonSha1Label.innerHTML = 'SHA-1'; hashingAlgorithmButtonsDiv.appendChild(radioButtonSha1); hashingAlgorithmButtonsDiv.appendChild(radioButtonSha1Label); hashingAlgorithmButtonsDiv.appendChild(document.createElement('br')); var radioButtonSha256 = document.createElement('input'); radioButtonSha256.id = hashingAlgorithmInputId + '-sha256'; radioButtonSha256.name = hashingAlgorithmInputId; radioButtonSha256.onclick = () => { alert('SHA-256 is a complex hashing algorithm. Large files may take several minutes to complete hashing.'); }; radioButtonSha256.type = 'radio'; radioButtonSha256.value = 'sha256'; radioButtonSha256.checked = false; radioButtonSha256.disabled = !cryptoAvailable; var radioButtonSha256Label = document.createElement('label'); radioButtonSha256Label.htmlFor = hashingAlgorithmInputId + '-sha256'; radioButtonSha256Label.innerHTML = 'SHA-256'; hashingAlgorithmButtonsDiv.appendChild(radioButtonSha256); hashingAlgorithmButtonsDiv.appendChild(radioButtonSha256Label); contentContainer.appendChild(hashingAlgorithmButtonsDiv); contentContainer.appendChild(createSpacer()); if (!cryptoAvailable) { var noCryptoButton = document.createElement('div'); noCryptoButton.className = 'accent clickable'; noCryptoButton.innerHTML = 'Why can\'t I select SHA-1 or SHA-256?'; noCryptoButton.onclick = () => { alert('SHA-1 and SHA-256 hashing is only available on web browsers that support the web crypto library. Try a browser like Chrome, Firefox or Edge to use these hashing algorithms.'); }; contentContainer.appendChild(noCryptoButton); } contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); var nextButton = createNextButton(); nextButton.onclick = function() { hashingAlgorithm = getRadioButtonValue(hashingAlgorithmInputId); step2(); }; contentContainer.appendChild(nextButton); } async function step2() { await clearContentContainer(); var chooseOptionsInputId = 'input-chooseoptions'; var title = document.createElement('div'); title.className = 'content-container-title'; title.innerHTML = 'Hashing Customization'; contentContainer.appendChild(title); contentContainer.appendChild(createSpacer()); var content = document.createElement('div'); content.innerHTML = 'Would you like to customize how your file is hashed?'; contentContainer.appendChild(content); contentContainer.appendChild(createSpacer()); var chooseOptionsButtonsDiv = document.createElement('div'); chooseOptionsButtonsDiv.className = 'center-left'; var radioButtonDefaultOptions = document.createElement('input'); radioButtonDefaultOptions.id = chooseOptionsInputId + '-no'; radioButtonDefaultOptions.name = chooseOptionsInputId; radioButtonDefaultOptions.type = 'radio'; radioButtonDefaultOptions.value = '0'; radioButtonDefaultOptions.checked = true; var radioButtonDefaultOptionsLabel = document.createElement('label'); radioButtonDefaultOptionsLabel.htmlFor = chooseOptionsInputId + '-no'; radioButtonDefaultOptionsLabel.innerHTML = 'Use default hashing options'; chooseOptionsButtonsDiv.appendChild(radioButtonDefaultOptions); chooseOptionsButtonsDiv.appendChild(radioButtonDefaultOptionsLabel); chooseOptionsButtonsDiv.appendChild(document.createElement('br')); var radioButtonChooseOptions = document.createElement('input'); radioButtonChooseOptions.id = chooseOptionsInputId + '-yes'; radioButtonChooseOptions.name = chooseOptionsInputId; radioButtonChooseOptions.type = 'radio'; radioButtonChooseOptions.value = '1'; var radioButtonChooseOptionsLabel = document.createElement('label'); radioButtonChooseOptionsLabel.htmlFor = chooseOptionsInputId + '-yes'; radioButtonChooseOptionsLabel.innerHTML = 'Choose my hashing options'; chooseOptionsButtonsDiv.appendChild(radioButtonChooseOptions); chooseOptionsButtonsDiv.appendChild(radioButtonChooseOptionsLabel); contentContainer.appendChild(chooseOptionsButtonsDiv); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); var nextButton = createNextButton(); nextButton.onclick = function() { chooseOptions = parseInt(getRadioButtonValue(chooseOptionsInputId)); step3(); }; contentContainer.appendChild(nextButton); } async function step3() { var numCsvFields = 0; var numTsvFields = 0; var splitFileName = fileInput.files[0].name.split('.'); if (splitFileName[splitFileName.length - 1].toLowerCase() == 'csv') { numCsvFields = 1; numTsvFields = 0; } else if (splitFileName[splitFileName.length - 1].toLowerCase() == 'tsv') { numCsvFields = 0; numTsvFields = 1; } else { numCsvFields = parseLine(header, ',', false).length; numTsvFields = parseLine(header, '\t', false).length; } if (!chooseOptions && numCsvFields != numTsvFields && numCsvFields != 0) { if (numCsvFields > numTsvFields) inputInfo.delimiter = ','; else inputInfo.delimiter = '\t'; header = parseLine(header, inputInfo.delimiter, false); step4(); } else { await clearContentContainer(); var delimiterInputId = 'input-delimiter'; var title = document.createElement('div'); title.className = 'content-container-title'; title.innerHTML = 'File Type'; contentContainer.appendChild(title); contentContainer.appendChild(createSpacer()); var content = document.createElement('div'); content.innerHTML += 'What type of file is this?'; contentContainer.appendChild(content); contentContainer.appendChild(createSpacer()); var radioButtonsDiv = document.createElement('div'); radioButtonsDiv.className = 'center-left'; var radioButtonCSV = document.createElement('input'); radioButtonCSV.id = delimiterInputId + '-csv'; radioButtonCSV.name = delimiterInputId; radioButtonCSV.type = 'radio'; radioButtonCSV.value = ','; radioButtonCSV.checked = (numCsvFields > numTsvFields); var radioButtonCSVLabel = document.createElement('label'); radioButtonCSVLabel.htmlFor = delimiterInputId + '-csv'; radioButtonCSVLabel.innerHTML = 'CSV'; radioButtonsDiv.appendChild(radioButtonCSV); radioButtonsDiv.appendChild(radioButtonCSVLabel); radioButtonsDiv.appendChild(document.createElement('br')); var radioButtonTSV = document.createElement('input'); radioButtonTSV.id = delimiterInputId + '-tsv'; radioButtonTSV.name = delimiterInputId; radioButtonTSV.type = 'radio'; radioButtonTSV.value = '\t'; radioButtonTSV.checked = (numTsvFields > numCsvFields); var radioButtonTSVLabel = document.createElement('label'); radioButtonTSVLabel.htmlFor = delimiterInputId + '-tsv'; radioButtonTSVLabel.innerHTML = 'TSV'; radioButtonsDiv.appendChild(radioButtonTSV); radioButtonsDiv.appendChild(radioButtonTSVLabel); contentContainer.appendChild(radioButtonsDiv); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); var nextButton = createNextButton(); if (!((numCsvFields > numTsvFields) || (numTsvFields > numCsvFields))) nextButton.classList.add('disabled'); nextButton.onclick = function() { if (this.classList.contains('disabled')) return; inputInfo.delimiter = getRadioButtonValue(delimiterInputId); header = parseLine(header, inputInfo.delimiter, false); step4(); }; contentContainer.appendChild(nextButton); document.getElementsByName(delimiterInputId).forEach(e => { e.addEventListener('change', function () { nextButton.classList.remove('disabled'); }); }); } } async function step4() { var fieldGuesses = { first: { index: undefined, guesses: 0 }, last: { index: undefined, guesses: 0 }, address: { index: undefined, guesses: 0 }, zip: { index: undefined, guesses: 0 }, } for (var i = 0; i < header.length; i++) { var guesses = 0; var guess = ''; if (header[i].match(/.*first.*/i)) { guesses++; guess = 'first'; } if (header[i].match(/.*last.*/i)) { guesses++; guess = 'last'; } if (header[i].match(/^(postal|mailing|mail|street|corp|owner)?\s?addr(ess)?$/i)) { guesses++; guess = 'address'; } if (header[i].match(/.*(zip|postal\scode).*/i)) { guesses++; guess = 'zip'; } if (guesses == 1) { fieldGuesses[guess].index = i; fieldGuesses[guess].guesses++; } } Object.keys(fieldGuesses).forEach(k => { if (fieldGuesses[k].guesses != 1) fieldGuesses[k] = false; }); if (!chooseOptions) { Object.keys(fieldGuesses).forEach(k => { if (fieldGuesses[k]) inputInfo[k] = fieldGuesses[k].index; else inputInfo[k] = '-1'; }); step6(); } else { await clearContentContainer(); var relevantFields = { first: 'First name', last: 'Last name', address: 'Address', zip: 'Zip' }; var headerInputId = 'input-header'; var title = document.createElement('div'); title.className = 'content-container-title'; title.innerHTML = 'File Header'; contentContainer.appendChild(title); contentContainer.appendChild(createSpacer()); var content = document.createElement('div'); content.innerHTML += 'What columns in your file correspond to these field types? Identifying these allows us to add Versium SimpleTags to your output file.'; contentContainer.appendChild(content); contentContainer.appendChild(createSpacer()); var twoColumnContainer = document.createElement('div'); Object.keys(relevantFields).forEach(k => { var twoColumnLeft = document.createElement('div'); twoColumnLeft.className = 'two-column-left'; twoColumnLeft.innerHTML = relevantFields[k] + ': '; twoColumnContainer.appendChild(twoColumnLeft); var twoColumnRight = document.createElement('div'); twoColumnRight.className = 'two-column-right'; var twoColumnRightInput = document.createElement('select'); twoColumnRightInput.id = headerInputId + '-' + k; twoColumnRightInput.name = headerInputId; var defaultOption = document.createElement('option'); defaultOption.innerHTML = 'None'; defaultOption.selected = !fieldGuesses[k]; defaultOption.value = -1; twoColumnRightInput.appendChild(defaultOption); for (var i = 0; i < header.length; i++) { var optionElement = document.createElement('option'); optionElement.innerHTML = header[i]; optionElement.value = i; if (fieldGuesses[k] && fieldGuesses[k].index == i) optionElement.selected = true; twoColumnRightInput.appendChild(optionElement); } twoColumnRight.appendChild(twoColumnRightInput); twoColumnContainer.appendChild(twoColumnRight); twoColumnContainer.appendChild(document.createElement('br')); }); contentContainer.appendChild(twoColumnContainer); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); var nextButton = createNextButton(); nextButton.onclick = function() { if (this.classList.contains('disabled')) return; Object.keys(relevantFields).forEach(k => { inputInfo[k] = document.getElementById(headerInputId + '-' + k).value; }); step5(); }; contentContainer.appendChild(nextButton); } } async function step5() { await clearContentContainer(); var capitalizationInputId = 'input-capitalization'; var trimInputId = 'input-trim'; var hashSaltInputId = 'input-hashsalt'; var tagFormatInputId = 'input-tagformat'; var tagLengthInputId = 'input-taglength'; var hashLengthInputId = 'input-hashlength'; var hashEncodingInputId = 'input-hashencoding'; var title = document.createElement('div'); title.className = 'content-container-title'; title.innerHTML = 'Options'; contentContainer.appendChild(title); contentContainer.appendChild(createSpacer()); var content = document.createElement('div'); content.innerHTML += 'Are there any options you\'d like to select for your output file?'; contentContainer.appendChild(content); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); var optionsDiv = document.createElement('div'); optionsDiv.className = 'center-left'; var capitalizationDiv = document.createElement('div'); capitalizationDiv.innerHTML = 'Edit capitalization:'; capitalizationDiv.appendChild(document.createElement('br')); var radioLowerCase = document.createElement('input'); radioLowerCase.checked = true; radioLowerCase.id = capitalizationInputId + '-lower'; radioLowerCase.name = capitalizationInputId; radioLowerCase.type = 'radio'; radioLowerCase.value = 1; var radioLowerCaseLabel = document.createElement('label'); radioLowerCaseLabel.htmlFor = capitalizationInputId + '-lower'; radioLowerCaseLabel.innerHTML = 'Lowercase'; capitalizationDiv.appendChild(radioLowerCase); capitalizationDiv.appendChild(radioLowerCaseLabel); capitalizationDiv.appendChild(document.createElement('br')); var radioUpperCase = document.createElement('input'); radioUpperCase.id = capitalizationInputId + '-upper'; radioUpperCase.name = capitalizationInputId; radioUpperCase.type = 'radio'; radioUpperCase.value = 2; var radioUpperCaseLabel = document.createElement('label'); radioUpperCaseLabel.htmlFor = capitalizationInputId + '-upper'; radioUpperCaseLabel.innerHTML = 'Uppercase'; capitalizationDiv.appendChild(radioUpperCase); capitalizationDiv.appendChild(radioUpperCaseLabel); capitalizationDiv.appendChild(document.createElement('br')); var radioNoCapitalization = document.createElement('input'); radioNoCapitalization.id = capitalizationInputId + '-none'; radioNoCapitalization.name = capitalizationInputId; radioNoCapitalization.type = 'radio'; radioNoCapitalization.value = 0; var radioNoCapitalizationLabel = document.createElement('label'); radioNoCapitalizationLabel.htmlFor = capitalizationInputId + '-none'; radioNoCapitalizationLabel.innerHTML = 'Maintain capitalization'; capitalizationDiv.appendChild(radioNoCapitalization); capitalizationDiv.appendChild(radioNoCapitalizationLabel); optionsDiv.appendChild(capitalizationDiv); optionsDiv.appendChild(createSpacer()); optionsDiv.appendChild(createSpacer()); var trimDiv = document.createElement('div'); trimDiv.innerHTML = 'Trim fields:'; trimDiv.appendChild(document.createElement('br')); var radioTrim = document.createElement('input'); radioTrim.checked = true; radioTrim.id = trimInputId + '-trim'; radioTrim.name = trimInputId; radioTrim.type = 'radio'; radioTrim.value = 1; var radioTrimLabel = document.createElement('label'); radioTrimLabel.htmlFor = trimInputId + '-trim'; radioTrimLabel.innerHTML = 'Trim leading/trailing whitespace'; trimDiv.appendChild(radioTrim); trimDiv.appendChild(radioTrimLabel); trimDiv.appendChild(document.createElement('br')); var radioNoTrim = document.createElement('input'); radioNoTrim.id = trimInputId + '-notrim'; radioNoTrim.name = trimInputId; radioNoTrim.type = 'radio'; radioNoTrim.value = 0; var radioNoTrimLabel = document.createElement('label'); radioNoTrimLabel.htmlFor = trimInputId + '-notrim'; radioNoTrimLabel.innerHTML = 'Don\'t trim leading/trailing whitespace'; trimDiv.appendChild(radioNoTrim); trimDiv.appendChild(radioNoTrimLabel); optionsDiv.appendChild(trimDiv); contentContainer.appendChild(optionsDiv); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); var advancedOptionsButton = document.createElement('div'); advancedOptionsButton.className = 'accent clickable'; advancedOptionsButton.innerHTML = 'Show advanced options'; advancedOptionsButton.onclick = () => { var advancedOptionsDiv = document.getElementById('advanced-options'); if (advancedOptionsDiv.style.display == 'none') advancedOptionsDiv.style.display = 'inline-block'; else advancedOptionsDiv.style.display = 'none'; }; contentContainer.appendChild(advancedOptionsButton); var advancedOptionsDiv = document.createElement('div'); advancedOptionsDiv.className = 'center-left'; advancedOptionsDiv.id = 'advanced-options'; advancedOptionsDiv.style.display = 'none'; advancedOptionsDiv.appendChild(createSpacer()); advancedOptionsDiv.appendChild(createSpacer()); var hashSaltDiv = document.createElement('div'); hashSaltDiv.innerHTML = 'Hash salt (this doesn\'t apply to Versium SimpleTags)'; hashSaltDiv.appendChild(document.createElement('br')); var inputHashSalt = document.createElement('input'); inputHashSalt.id = hashSaltInputId; inputHashSalt.name = hashSaltInputId; inputHashSalt.type = 'text'; inputHashSalt.value = ''; inputHashSalt.placeholder = 'Salt to be prepended'; hashSaltDiv.appendChild(inputHashSalt); advancedOptionsDiv.appendChild(hashSaltDiv); advancedOptionsDiv.appendChild(createSpacer()); advancedOptionsDiv.appendChild(createSpacer()); var tagFormatDiv = document.createElement('div'); tagFormatDiv.innerHTML = 'Versium SimpleTag format:'; tagFormatDiv.appendChild(document.createElement('br')); var radioDecode = document.createElement('input'); radioDecode.checked = true; radioDecode.id = tagFormatInputId + '-decode'; radioDecode.name = tagFormatInputId; radioDecode.type = 'radio'; radioDecode.value = 1; var radioDecodeLabel = document.createElement('label'); radioDecodeLabel.htmlFor = tagFormatInputId + '-decode'; radioDecodeLabel.innerHTML = 'Decode numbers (default)'; tagFormatDiv.appendChild(radioDecode); tagFormatDiv.appendChild(radioDecodeLabel); tagFormatDiv.appendChild(document.createElement('br')); var radioNoDecode = document.createElement('input'); radioNoDecode.id = tagFormatInputId + '-nodecode'; radioNoDecode.name = tagFormatInputId; radioNoDecode.type = 'radio'; radioNoDecode.value = 0; var radioNoDecodeLabel = document.createElement('label'); radioNoDecodeLabel.htmlFor = tagFormatInputId + '-nodecode'; radioNoDecodeLabel.innerHTML = 'Do not decode numbers'; tagFormatDiv.appendChild(radioNoDecode); tagFormatDiv.appendChild(radioNoDecodeLabel); advancedOptionsDiv.appendChild(tagFormatDiv); advancedOptionsDiv.appendChild(createSpacer()); advancedOptionsDiv.appendChild(createSpacer()); var tagLengthDiv = document.createElement('div'); tagLengthDiv.innerHTML = 'Versium SimpleTag address length:'; tagLengthDiv.appendChild(document.createElement('br')); var radioEight = document.createElement('input'); radioEight.checked = true; radioEight.id = tagLengthInputId + '-eight'; radioEight.name = tagLengthInputId; radioEight.type = 'radio'; radioEight.value = 8; var radioEightLabel = document.createElement('label'); radioEightLabel.htmlFor = tagLengthInputId + '-eight'; radioEightLabel.innerHTML = 'Eight (default)'; tagLengthDiv.appendChild(radioEight); tagLengthDiv.appendChild(radioEightLabel); tagLengthDiv.appendChild(document.createElement('br')); var radioTen = document.createElement('input'); radioTen.id = tagLengthInputId + '-ten'; radioTen.name = tagLengthInputId; radioTen.type = 'radio'; radioTen.value = 10; var radioTenLabel = document.createElement('label'); radioTenLabel.htmlFor = tagLengthInputId + '-ten'; radioTenLabel.innerHTML = 'Ten'; tagLengthDiv.appendChild(radioTen); tagLengthDiv.appendChild(radioTenLabel); tagLengthDiv.appendChild(document.createElement('br')); var radioTwelve = document.createElement('input'); radioTwelve.id = tagLengthInputId + '-twelve'; radioTwelve.name = tagLengthInputId; radioTwelve.type = 'radio'; radioTwelve.value = 12; var radioTwelveLabel = document.createElement('label'); radioTwelveLabel.htmlFor = tagLengthInputId + '-twelve'; radioTwelveLabel.innerHTML = 'Twelve'; tagLengthDiv.appendChild(radioTwelve); tagLengthDiv.appendChild(radioTwelveLabel); advancedOptionsDiv.appendChild(tagLengthDiv); advancedOptionsDiv.appendChild(createSpacer()); advancedOptionsDiv.appendChild(createSpacer()); var hashLengthDiv = document.createElement('div'); hashLengthDiv.innerHTML = 'Versium SimpleTag hash length:'; hashLengthDiv.appendChild(document.createElement('br')); var radio64Bit = document.createElement('input'); radio64Bit.checked = true; radio64Bit.id = hashLengthInputId + '-64bit'; radio64Bit.name = hashLengthInputId; radio64Bit.type = 'radio'; radio64Bit.value = 64; var radio64BitLabel = document.createElement('label'); radio64BitLabel.htmlFor = hashLengthInputId + '-64bit'; radio64BitLabel.innerHTML = '64 Bit (default)'; hashLengthDiv.appendChild(radio64Bit); hashLengthDiv.appendChild(radio64BitLabel); hashLengthDiv.appendChild(document.createElement('br')); var radio32Bit = document.createElement('input'); radio32Bit.id = hashLengthInputId + '-32bit'; radio32Bit.name = hashLengthInputId; radio32Bit.type = 'radio'; radio32Bit.value = 32; var radio32BitLabel = document.createElement('label'); radio32BitLabel.htmlFor = hashLengthInputId + '-32bit'; radio32BitLabel.innerHTML = '32 Bit'; hashLengthDiv.appendChild(radio32Bit); hashLengthDiv.appendChild(radio32BitLabel); hashLengthDiv.appendChild(document.createElement('br')); var radio16Bit = document.createElement('input'); radio16Bit.id = hashLengthInputId + '-16bit'; radio16Bit.name = hashLengthInputId; radio16Bit.type = 'radio'; radio16Bit.value = 16; var radio16BitLabel = document.createElement('label'); radio16BitLabel.htmlFor = hashLengthInputId + '-16bit'; radio16BitLabel.innerHTML = '16 Bit'; hashLengthDiv.appendChild(radio16Bit); hashLengthDiv.appendChild(radio16BitLabel); hashLengthDiv.appendChild(document.createElement('br')); var radio8Bit = document.createElement('input'); radio8Bit.id = hashLengthInputId + '-8bit'; radio8Bit.name = hashLengthInputId; radio8Bit.type = 'radio'; radio8Bit.value = 8; var radio8BitLabel = document.createElement('label'); radio8BitLabel.htmlFor = hashLengthInputId + '-8bit'; radio8BitLabel.innerHTML = '8 Bit'; hashLengthDiv.appendChild(radio8Bit); hashLengthDiv.appendChild(radio8BitLabel); advancedOptionsDiv.appendChild(hashLengthDiv); advancedOptionsDiv.appendChild(createSpacer()); advancedOptionsDiv.appendChild(createSpacer()); var hashEncodingDiv = document.createElement('div'); hashEncodingDiv.innerHTML = 'Versium SimpleTag hash encoding:'; hashEncodingDiv.appendChild(document.createElement('br')); var radioDecimal = document.createElement('input'); radioDecimal.checked = true; radioDecimal.id = hashEncodingInputId + '-decimal'; radioDecimal.name = hashEncodingInputId; radioDecimal.type = 'radio'; radioDecimal.value = 0; var radioDecimalLabel = document.createElement('label'); radioDecimalLabel.htmlFor = hashEncodingInputId + '-decimal'; radioDecimalLabel.innerHTML = 'Decimal (default)'; hashEncodingDiv.appendChild(radioDecimal); hashEncodingDiv.appendChild(radioDecimalLabel); hashEncodingDiv.appendChild(document.createElement('br')); var radioHex = document.createElement('input'); radioHex.id = hashEncodingInputId + '-hex'; radioHex.name = hashEncodingInputId; radioHex.type = 'radio'; radioHex.value = 1; var radioHexLabel = document.createElement('label'); radioHexLabel.htmlFor = hashEncodingInputId + '-hex'; radioHexLabel.innerHTML = 'Hex'; hashEncodingDiv.appendChild(radioHex); hashEncodingDiv.appendChild(radioHexLabel); hashEncodingDiv.appendChild(document.createElement('br')); var radioBase64 = document.createElement('input'); radioBase64.id = hashEncodingInputId + '-base64'; radioBase64.name = hashEncodingInputId; radioBase64.type = 'radio'; radioBase64.value = 2; var radioBase64Label = document.createElement('label'); radioBase64Label.htmlFor = hashEncodingInputId + '-base64'; radioBase64Label.innerHTML = 'Base64'; hashEncodingDiv.appendChild(radioBase64); hashEncodingDiv.appendChild(radioBase64Label); advancedOptionsDiv.appendChild(hashEncodingDiv); contentContainer.appendChild(advancedOptionsDiv); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); var nextButton = createNextButton('Finish'); nextButton.onclick = async function() { outputInfo.capitalization = Number(getRadioButtonValue(capitalizationInputId)); outputInfo.trim = Number(getRadioButtonValue(trimInputId)); outputInfo.hashSalt = document.getElementById(hashSaltInputId).value; outputInfo.tagFormat = Number(getRadioButtonValue(tagFormatInputId)); outputInfo.tagLength = Number(getRadioButtonValue(tagLengthInputId)); outputInfo.hashLength = Number(getRadioButtonValue(hashLengthInputId)); outputInfo.hashEncoding = Number(getRadioButtonValue(hashEncodingInputId)); step6(); }; contentContainer.appendChild(nextButton); } async function step6() { await clearContentContainer(); var title = document.createElement('div'); title.className = 'content-container-title'; title.innerHTML = 'Hashing...'; contentContainer.appendChild(title); contentContainer.appendChild(createSpacer()); var content = document.createElement('div'); content.innerHTML += 'We\'re processing your file. It will download as soon as we\'re done with it! Depending on the size of your file and your selected hashing algorithm, this may take several minutes.'; contentContainer.appendChild(content); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); var loadingSpinnerDiv = document.createElement('div'); loadingSpinnerDiv.className = 'spinner'; var emptyDiv = document.createElement('div'); for (var i = 0; i < 4; i++) loadingSpinnerDiv.appendChild(emptyDiv.cloneNode(true)); contentContainer.appendChild(loadingSpinnerDiv); var reader = new FileReader(); reader.fileName = fileInput.files[0].name; reader.onload = async function (progressEvent) { var delimiter = inputInfo.delimiter; var additionalHeaderFields = []; if (inputInfo.address >= 0 && inputInfo.zip >= 0) { additionalHeaderFields = ['VERSIUM STADDR HASHED']; if (inputInfo.last >= 0) { additionalHeaderFields.push('VERSIUM STHHLD HASHED'); if (inputInfo.first >= 0) additionalHeaderFields.push('VERSIUM ST10 HASHED'); } } var outputHeader = header.concat(additionalHeaderFields); output = outputHeader.join(delimiter) + '\n'; var textEncoder = new TextEncoder('utf-8'); var inputLines = this.result.split('\n'); for (var i = 1; i < inputLines.length; i++) { if (inputLines[i].trim() == '') continue; if (outputInfo.capitalization == 1) inputLines[i] = inputLines[i].toLowerCase(); else if (outputInfo.capitalization == 2) inputLines[i] = inputLines[i].toUpperCase(); var parsedLine = parseLine(inputLines[i], delimiter); if (parsedLine.length != header.length) { output += 'Malformatted line\n'; errors++; continue; } var outputLine = parsedLine.slice(); for (var j = 0; j < outputLine.length; j++) { if (outputInfo.trim) outputLine[j] = outputLine[j].trim(); outputLine[j] = await hashingFunctions[hashingAlgorithm](outputInfo.hashSalt + outputLine[j], textEncoder); } var appendedData = await generateHashedSimpleTags(parsedLine[inputInfo.first], parsedLine[inputInfo.last], parsedLine[inputInfo.address], parsedLine[inputInfo.zip], outputInfo.tagFormat, outputInfo.tagLength, outputInfo.hashLength, outputInfo.hashEncoding); outputLine = outputLine.concat(appendedData); output += outputLine.join(delimiter) + '\n'; } download(appendToFilename(this.fileName, '_versium'), output); step7(); }; reader.readAsText(fileInput.files[0]); } async function step7() { await clearContentContainer(); var title = document.createElement('div'); title.className = 'content-container-title'; title.innerHTML = 'All done'; contentContainer.appendChild(title); contentContainer.appendChild(createSpacer()); var content = document.createElement('div'); content.innerHTML += 'Your file has been processed and is downloading now!'; if (errors) { content.appendChild(createSpacer()); content.innerHTML += 'There were ' + errors + ' errors or malformatted lines in your input file. These lines are noted in the output file as "Malformatted line".'; } contentContainer.appendChild(content); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); contentContainer.appendChild(createSpacer()); var nextButton = createNextButton('Start Over'); nextButton.onclick = function() { if (this.classList.contains('disabled')) return; step0(); }; contentContainer.appendChild(nextButton); } function appendToFilename(filename, string){ var dotIndex = filename.lastIndexOf("."); if (dotIndex == -1) return filename + string; else return filename.substring(0, dotIndex) + string + filename.substring(dotIndex); } async function clearContentContainer() { contentContainer.style.opacity = 0; await new Promise(r => setTimeout(r, 500)); contentContainer.innerHTML = ''; contentContainer.style.opacity = 1; } function createNextButton(innerText = 'Next') { var nextButton = document.createElement('div'); nextButton.className = 'content-container-button'; nextButton.innerHTML = innerText; var icon = document.createElement('i'); icon.className = 'icon-right-arrow'; nextButton.appendChild(icon); return nextButton; } function createSpacer() { var spacer = document.createElement('div'); spacer.className = 'spacer'; return spacer; } function download(filename, contents) { var e = document.createElement('a'); e.setAttribute('href', 'data:text/plain;charset=utf-8,' + encodeURIComponent(contents)); e.setAttribute('download', filename); e.style.display = 'none'; document.body.appendChild(e); e.click(); document.body.removeChild(e); } function getFileHeader() { var reader = new FileReader(); reader.onload = function (progressEvent) { header = reader.result.split('\n').shift(); } reader.readAsText(fileInput.files[0]); } function getRadioButtonValue(name) { return document.querySelector('input[name="' + name + '"]:checked').value; } function parseLine(line, delimiter = ',') { var insideQuote = false, entries = [], entry = []; line = line.trimEnd(); line.split('').forEach (c => { if (c === '"') { insideQuote = !insideQuote; } else { if (c == delimiter && !insideQuote) { entries.push(entry.join('')); entry = []; } else { entry.push(c); } } }); entries.push(entry.join('')); return entries; } </script> <script> step0(true); </script> </body>