ninja.seeder = { init: (function () { document.getElementById("generatekeyinput").value = ""; })(), // number of mouse movements to wait for seedLimit: (function () { var num = Crypto.util.randomBytes(12)[11]; return 200 + Math.floor(num); })(), seedCount: 0, // counter lastInputTime: new Date().getTime(), seedPoints: [], isStillSeeding: true, seederDependentWallets: ["singlewallet", "paperwallet", "bulkwallet", "vanitywallet", "splitwallet"], // seed function exists to wait for mouse movement to add more entropy before generating an address seed: function (evt) { if (!evt) var evt = window.event; var timeStamp = new Date().getTime(); // seeding is over now we generate and display the address if (ninja.seeder.seedCount == ninja.seeder.seedLimit) { ninja.seeder.seedCount++; ninja.seeder.seedingOver(); } // seed mouse position X and Y when mouse movements are greater than 40ms apart. else if ((ninja.seeder.seedCount < ninja.seeder.seedLimit) && evt && (timeStamp - ninja.seeder.lastInputTime) > 40) { SecureRandom.seedTime(); SecureRandom.seedInt16((evt.clientX * evt.clientY)); ninja.seeder.showPoint(evt.clientX, evt.clientY); ninja.seeder.seedCount++; ninja.seeder.lastInputTime = new Date().getTime(); ninja.seeder.showPool(); } }, // seed function exists to wait for mouse movement to add more entropy before generating an address seedKeyPress: function (evt) { if (!evt) var evt = window.event; // seeding is over now we generate and display the address if (ninja.seeder.seedCount == ninja.seeder.seedLimit) { ninja.seeder.seedCount++; ninja.seeder.seedingOver(); } // seed key press character else if ((ninja.seeder.seedCount < ninja.seeder.seedLimit) && evt.which) { var timeStamp = new Date().getTime(); // seed a bunch (minimum seedLimit) of times SecureRandom.seedTime(); SecureRandom.seedInt8(evt.which); var keyPressTimeDiff = timeStamp - ninja.seeder.lastInputTime; SecureRandom.seedInt8(keyPressTimeDiff); ninja.seeder.seedCount++; ninja.seeder.lastInputTime = new Date().getTime(); ninja.seeder.showPool(); } }, showPool: function () { var poolHex; if (SecureRandom.poolCopyOnInit != null) { poolHex = Crypto.util.bytesToHex(SecureRandom.poolCopyOnInit); document.getElementById("seedpool").innerHTML = poolHex; document.getElementById("seedpooldisplay").innerHTML = poolHex; } else { poolHex = Crypto.util.bytesToHex(SecureRandom.pool); document.getElementById("seedpool").innerHTML = poolHex; document.getElementById("seedpooldisplay").innerHTML = poolHex; } var percentSeeded = Math.round((ninja.seeder.seedCount / ninja.seeder.seedLimit) * 100) + "%"; document.getElementById("mousemovelimit").innerHTML = percentSeeded; for (var wIndex in ninja.seeder.seederDependentWallets) { document.getElementById(ninja.seeder.seederDependentWallets[wIndex]).innerHTML = percentSeeded; } }, showPoint: function (x, y) { var div = document.createElement("div"); div.setAttribute("class", "seedpoint"); div.style.top = y + "px"; div.style.left = x + "px"; document.body.appendChild(div); ninja.seeder.seedPoints.push(div); }, removePoints: function () { for (var i = 0; i < ninja.seeder.seedPoints.length; i++) { document.body.removeChild(ninja.seeder.seedPoints[i]); } ninja.seeder.seedPoints = []; }, seedingOver: function () { ninja.seeder.isStillSeeding = false; var walletType = ninja.tab.whichIsOpen(); if (walletType == null) { ninja.tab.select("singlewallet"); } else { ninja.tab.select(walletType) } document.getElementById("generate").style.display = "none"; // update labels for dependent wallets var culture = (ninja.getQueryString()["culture"] == null ? "en" : ninja.getQueryString()["culture"]); ninja.translator.translate(culture); ninja.seeder.removePoints(); } }; ninja.qrCode = { // determine which type number is big enough for the input text length getTypeNumber: function (text) { var lengthCalculation = text.length * 8 + 12; // length as calculated by the QRCode if (lengthCalculation < 72) { return 1; } else if (lengthCalculation < 128) { return 2; } else if (lengthCalculation < 208) { return 3; } else if (lengthCalculation < 288) { return 4; } else if (lengthCalculation < 368) { return 5; } else if (lengthCalculation < 480) { return 6; } else if (lengthCalculation < 528) { return 7; } else if (lengthCalculation < 688) { return 8; } else if (lengthCalculation < 800) { return 9; } else if (lengthCalculation < 976) { return 10; } return null; }, createCanvas: function (text, sizeMultiplier) { sizeMultiplier = (sizeMultiplier == undefined) ? 2 : sizeMultiplier; // default 2 // create the qrcode itself var typeNumber = ninja.qrCode.getTypeNumber(text); var qrcode = new QRCode(typeNumber, QRCode.ErrorCorrectLevel.H); qrcode.addData(text); qrcode.make(); var width = qrcode.getModuleCount() * sizeMultiplier; var height = qrcode.getModuleCount() * sizeMultiplier; // create canvas element var canvas = document.createElement('canvas'); var scale = 10.0; canvas.width = width * scale; canvas.height = height * scale; canvas.style.width = width + 'px'; canvas.style.height = height + 'px'; var ctx = canvas.getContext('2d'); ctx.scale(scale, scale); // compute tileW/tileH based on width/height var tileW = width / qrcode.getModuleCount(); var tileH = height / qrcode.getModuleCount(); // draw in the canvas for (var row = 0; row < qrcode.getModuleCount(); row++) { for (var col = 0; col < qrcode.getModuleCount(); col++) { ctx.fillStyle = qrcode.isDark(row, col) ? "#000000" : "#ffffff"; ctx.fillRect(col * tileW, row * tileH, tileW, tileH); } } // return just built canvas return canvas; }, // generate a QRCode and return it's representation as an Html table createTableHtml: function (text) { var typeNumber = ninja.qrCode.getTypeNumber(text); var qr = new QRCode(typeNumber, QRCode.ErrorCorrectLevel.H); qr.addData(text); qr.make(); var tableHtml = "
"; } else { tableHtml += " | "; } } tableHtml += " |