diff --git a/generate-wallet.html b/generate-wallet.html index 10e7f1d..750c520 100644 --- a/generate-wallet.html +++ b/generate-wallet.html @@ -22,14 +22,14 @@ This software (HTML and JavaScript) is ©Copyright 2014 Canton Becker and bitaddress.org and licensed under the MIT license. - Special thanks to pointbiz/bitaddress.org and Artiom Chilaru/flexlabs.org for significant - contributions to this software. + Special thanks to pointbiz/bitaddress.org, Artiom Chilaru/flexlabs.org, and Martin Ankerl + for significant contributions to this software. - Portions of this code are copyrighted by their respective authors - see below. - **************************************************************************************** **************************************************************************************** * Contributions to support the bitcoinpaperwallet.com developers are extremely welcome. + * Donations are shared between Canton Becker, Artiom Chilaru (BIP38 implementation) + * Martin Ankerl (guilloche patterns), and 'pointbiz' (original bitaddress.org code) * Bitcoin Donation Address: 1Pjg628vjMLBvADrPHsthtzKiryM2y46DG * GitHub Repository: https://github.com/cantonbecker/bitcoinpaperwallet **************************************************************************************** @@ -39,6 +39,7 @@ **************************************************************************************** **************************************************************************************** + Portions of this code are copyrighted by their respective authors - see below. Notice of Copyrights and Licenses: *********************************** @@ -54,11 +55,11 @@ Array.prototype.map Public Domain window.Crypto BSD License window.SecureRandom BSD License - window.EllipticCurve BSD License + window.EllipticCurve BSD License window.BigInteger BSD License window.QRCode MIT License window.Bitcoin MIT License - window.Crypto_scrypt MIT License + window.Crypto_scrypt MIT License ninja.(*) MIT License // from bitaddress.org parser.setUA GPLv2 & MIT // Browser User Agent detection jsqrcode Apache License, 2.0 // QR Code scanning from webcam @@ -87,6 +88,7 @@ Wallet Generator Updated March 1, 2014 +
Open Source JavaScript Paper Wallet Generator Updated June 25, 2014
For help, security tips, or wallet making supplies visit bitcoinpaperwallet.com
@@ -10379,11 +10425,11 @@ Bitcoin.Util = {
Instructions
-
1. Calibrate Printer
+
1. Calibrate
2. Print Front
3. Print Back
-
4. Cut, Fold, & Seal
-
✓ Validate
+
4. Cut, Fold & Seal
+
✓ Validate or Decrypt
@@ -10401,11 +10447,11 @@ Bitcoin.Util = {
Skip »
-

You may skip this step if you do not plan to use the random key generator.

+

You may safely skip this step if you do not need to use the random wallet generator.

- Secure wallet generation requires that you provide some "entropy", or random data.
+ Secure random number generation requires that you provide some unpredictable data, also called "entropy".
Please move your mouse around and/or type random keystrokes into this box: @@ -10426,11 +10472,8 @@ Bitcoin.Util = {

Welcome! Let's print out a beautiful and secure wallet for your s.

- - Here's an overview of what will happen, step by step. - -

- + Here's an overview of what will happen, step by step.
+
@@ -10462,60 +10505,63 @@ Bitcoin.Util = { } -
  • ? Are you using a secure operating system guaranteed to be free of spyware and viruses, for example, an Ubuntu LiveCD?
  • +
  • ? Are you using a secure operating system installation guaranteed to be free of spyware and viruses, for example, the Ubuntu LiveCD?
  • More security tips and answers to frequently asked questions »

    - Order hologram stickers / zip bags / Ubuntu CD » -
    - - - -
    - Flags - -
    -

    Change your wallet design, language, or cryptocurrency? - +

    Change your wallet's design, language, or cryptocurrency here: + +

    - Step 1: Calibrate Printer
    - Before printing out a wallet you'll need to "calibrate" your output using the zoom and horizontal shift adjustments to accommodate your particular browser / printer combination. Without adjustment, your wallet may print out too small or with a misaligned reverse side.
    -

    - Step 2: Print Front
    - Next you will print out the front side of your wallet. A public address and private key will automatically be generated, or you can supply your own key by rolling dice, shuffling cards, etc. You may also duplicate an existing paper wallet or provide a "vanity" address if you have one. If you are printing several wallets, it's possible to print two per page.
    -

    - Step 3: Print Back
    - You will put the same page back in your printer (but flipped over) to print out the back side of your wallet. This isn't just for good looks: the back side design includes additional tamper-resistant safeguards.
    -

    - Step 4: Cut, Fold, Seal & Fund
    - Find your scissors! The final step is to cut out your wallet, fold it, and seal it with tamper-evident hologram stickers or at least opaque (light-blocking) tape. Now you're ready to transfer funds from your online holdings to your new wallet.
    -

    - 90 second tutorial video on YouTube » - -
    + Step 1: Calibrate Printer
    +Before printing out a wallet you may need to calibrate your output using the zoom and horizontal shift adjustments to account for your particular browser and printer combination. Otherwise, your wallet's back side may not line up with the front side.
    +
    +Step 2: Print Front
    +Next you will print out the front side of your wallet. The public address and private key will be random-generated, or you can supply your own key by rolling dice, shuffling cards, etc. You may also duplicate an existing paper wallet or provide a "vanity" address if you have one. Tip: If you are printing several wallets, it's possible to print two per page.

    +
    +Step 3: Print Back
    +Flip the front page over and put it back in your printer. This isn't just for good looks! The reverse design includes important tamper-resistant safeguards.

    +
    +Step 4: Cut, Fold, Seal & Fund
    +Find your scissors! The final step is to cut out your wallet, fold it, and seal it with tamper-evident hologram stickers or at least opaque (light-blocking) tape. Now you're ready to transfer funds from your online holdings to your new wallet.

    +
    +90 second tutorial video on YouTube » +

    + Order hologram stickers / zip bags / Ubuntu CD » + + + +
    @@ -10524,9 +10570,11 @@ Bitcoin.Util = {
    -

    The calibration step helps line up the front and back sides of your printout. Print out this page in LANDSCAPE (wide) format. Based on the results, adjust the "zoom" and "horizontal shift" until your printed ruler (1) approximates a real ruler, and (2) is horizontally centered, leaving approximately equal margins on the left and right sides of the page.
    +

    The calibration step helps line up the front and back sides of your printout. Print out this page in LANDSCAPE FORMAT (wide instead of tall.) Based on the results, adjust the "zoom" and "horizontal shift" in the upper-right corner of this page and keep printing tests until:

    + (1) Your printed ruler more or less measures like a real ruler. (REDUCE the zoom to increase the ruler size.)

    + (2) Your printed ruler is horizontally centered on the page, leaving approximately equal margins on the left and right sides of the page. (INCREASE the horizontal shift to push the ruler to the right.)

    - What settings worked best for you? Feedback form »

    + What settings worked best for you? Feedback form »

    Print Calibration Test @@ -10553,33 +10601,34 @@ Bitcoin.Util = {
    - + - +
    -
    -
    +
    + BIP38 Encrypt
    @@ -10611,17 +10662,17 @@ Bitcoin.Util = {

    Help / Instructions »

    About "BIP38" Encryption

    -

    - The advantage to encrypting your paper wallet is that if your paper wallet is stolen or otherwise exposed, the balance on the wallet is safe unless the passphrase used to encrypt the wallet is guessed. However, if you encrypt your private key with BIP38 and you lose your passphrase, it will be impossible for you to recover the funds you have sent to this wallet.

    -

    Also, note that not many wallet applications or web services are able to import BIP38 private keys. In this case, you will have to use the "Validate" feature on this webpage to extract the unencrypted Wallet Import Format (WIF) key as an intermediate step before sweeping the balance.

    +

    The advantage with BIP38 is that if your paper wallet is stolen or compromised, the private key cannot be recovered without your password. Even a very short password adds a strong degree of protection. However, if you encrypt your private key with BIP38 and you lose your password, it will be impossible for you to recover the funds you have sent to this wallet.

    +

    Also, note that not all wallet applications or web services are able to import or "sweep" BIP38 encrypted keys. In this case, you will have to use the Validate or Decrypt feature on this webpage to reveal the unencrypted Wallet Import Format (WIF) key as an intermediate step before sweeping the balance.

    WARNING: Before sending any funds to a BIP38-encrypted wallet, first do a test make sure you are able to decrypt the printed private key back to ordinary WIF format.

    -

    In short, if you do not have a strong understanding of the BIP38 encryption and decryption workflow, click cancel, print your paper wallet out without encryption, and keep it safe the same way you would jewels or cash. +

    Unless you have a strong understanding of the BIP38 encryption and decryption workflow, click cancel, print your paper wallet without BIP38, and just keep it safe and hidden like you would jewels or cash.

    - - + +

    @@ -10642,20 +10693,107 @@ Bitcoin.Util = {
    -

    Reload the front page into your printer first. Note: The back design is intentionally larger than the front side. (Use the front side as your guide when cutting to trim off the excess.)

    - - Print Wallet Back - -
    +

    Reload the front page into your printer first. Note: The back design is intentionally larger than the front side. Use the front side as your guide when cutting out your wallet.

    + + Print Wallet Back
    - + +
    +
    + Denomination: + + +
    + +
    + Message: + + +
    + +
    + + Include notes area +
    +
    +
    + +
    +

    1000

    mBTC

    + +
    +
    +
    +
    +

     

     

     

     

     

    + +
    + + +
    +
    +
      +
    • +
    • +
    • +
    • +
    +
    +
    + -
    + +
      @@ -10667,21 +10805,21 @@ Bitcoin.Util = {
    Folding Diagram

    Sample Wallet

    - Sealed Wallet + Sealed Wallet
    - Cut out your wallet using the dotted lines on the front as a guide, not the back! + Cut out your wallet using the front side as a guide, not the back! - Note that the design on the back side is intentionally larger than the front side so that back design will "bleed" to the edges even if your front and back sides are misaligned up to 1cm. + The design on the reverse side is intentionally larger than the front side so that back design will "bleed" to the edges even if your front and back sides are somewhat misaligned.

    - Now fold the more narrow private key area in half, and then over again as indicated by the dotted lines in this diagram. The unusual "butterfly" shape can be a bit of a pain to cut out, but it's essential to keep your private key safe.
    + Now fold the more narrow private key area in half, and then over again as indicated by the dotted lines in this diagram. The "butterfly" shape is time-consuming to cut out, but without all those cuts and angles, someone can reveal your private key without removing the tape!


    The final wallet will be a rectangle shape with the more narrow private key area folded over it.

    - Seal your wallet by placing two strips of sturdy light-blocking tape over the top and bottom edges of the private (folded) area. A zip-seal bag will keep it safe from moisture (especially important for inkjet prints.) + Seal your wallet by placing two strips of sturdy light-blocking tape over the top and bottom edges of the private (folded) area. A zip-seal bag will keep it safe from moisture, which is especially important when using an inkjet printer.

    Purchase hologram stickers and/or zip-sealing bags » @@ -10699,7 +10837,13 @@ Bitcoin.Util = {
    If you elected to BIP38-encrypt your wallet, you may need to recover your unencrypted WIF "Wallet Import Format" private key by using the "validate" tab on this web page.
    -
    Visit BitcoinPaperWallet.com for detailed information on adding to and spending from a paper wallet » +
    +

    Tips & FAQ's

    + • What kind of printer and paper to use »
    + • Adding to and spending from a paper wallet »
    + • How and when to use BIP38 encryption »
    + +
    @@ -10713,9 +10857,11 @@ Bitcoin.Util = {

    Validate or decrypt private key

    -

    Enter or scan any private key into this form to display all available details in various formats. If your private key decodes properly, then you have verified that you should be able to retrieve any funds sent to that wallet. This should work for private keys generated by BitcoinPaperWallet.com as well as any other Bitcoin service.

    -

    For example, if you have a wallet encrypted with a BIP38 passphrase, you can use this form to decrypt your private key and retrieve the standard Wallet Import Format (WIF) key suitable for wallet software and services that don't directly support BIP38 importing.

    -

    You may also enter a passphrase (or "brain wallet") to see the corresponding SHA256 hashed keys.

    +

    This feature works for private keys made here as well as those generated using any other Bitcoin service or software, e.g. bitaddress.org or brainwallet.org

    +

    Enter or scan any private key to verify that the key is valid and show its corresponding public key. If your private key validates, then you may be reassured that you will able to retrieve any funds sent to that wallet.

    +

    To duplicate or BIP38-encrypt any paper wallet, just validate its private key and click the "Use these details to print a paper wallet" button.

    +

    Is your wallet BIP38 encrypted? If so, you can use this form to decrypt your private key and recover the standard Wallet Import Format (WIF) key suitable for wallet software and services that don't directly support BIP38 importing.

    +

    You may also type in a "brain wallet" passphrase to see the corresponding SHA256 hashed keys.

    @@ -10858,8 +11004,8 @@ Bitcoin.Util = {
    Security & signatures - For secure wallet generation, do not print wallets using https://bitcoinpaperwallet.com.
    Instead, download the generator and run the HTML as a local file in your browser instead.
    - Download (GitHub ZIP file) + For secure wallet generation, do not print wallets "live" from bitcoinpaperwallet.com.
    Instead, download this generator from GitHub and run the HTML as a "local" file:
    + Secure Download (GitHub ZIP file) Validate PGP/GPG signature @@ -11321,7 +11467,7 @@ ninja.publicKey = { // let's make the entropy 'points' grow and change color! percentageComplete = ninja.seeder.seedCount / ninja.seeder.seedLimit; pointSize = 2 + Math.ceil(9*percentageComplete) + 'px' - pointColor = 255 - Math.ceil(140 * percentageComplete); + pointColor = 255 - Math.ceil(110 * percentageComplete); div.style.backgroundColor = '#' + pointColor.toString(16) + 'FF' + pointColor.toString(16); div.style.width = pointSize; div.style.height = pointSize; @@ -11561,7 +11707,7 @@ ninja.publicKey = { // paper wallet html "paperlabeladdressesperpage": "Direcciones por página:", "paperlabeladdressestogenerate": "Direcciones en total:", - "papergenerate": "Generar", + "papergenerate1": "Generar", "paperprint": "Imprimir" }, @@ -11592,7 +11738,7 @@ ninja.publicKey = { // paper wallet html "paperlabeladdressesperpage": "Adresses par page:", "paperlabeladdressestogenerate": "Nombre d'adresses à créer:", - "papergenerate": "Générer", + "papergenerate1": "Générer", "paperprint": "Imprimer" } } @@ -11841,6 +11987,33 @@ ninja.publicKey = { }, showArtisticWallet: function (idPostFix, bitcoinAddress, privateKey) { + + // BIP38 coloration + var colors = { + 'bip38': { + publicUpper: "#fff57c", + publicLower: "#f7931a", + privateLeft: "#78bad6", + privateRight: "#fff67d", + pointer: "#0084a8", + guilloche: "white", + text: "#1937a9", + textShadow: "white", + textPointer: "white", + }, + 'default': { + publicUpper: "#fff57c", + publicLower: "#f7931a", + privateLeft: "#8cd96f", + privateRight: "#fff67d", + pointer: "#02ab5c", + guilloche: "white", + text: "#1937a9", + textShadow: "white", + textPointer: "white", + } + }; + var keyValuePair = {}; keyValuePair["qrcode_public" + idPostFix] = bitcoinAddress; keyValuePair["qrcode_private" + idPostFix] = privateKey; @@ -11852,14 +12025,22 @@ ninja.publicKey = { var half = privateKey.length / 2; document.getElementById("btcencryptedkey" + idPostFix).innerHTML = privateKey; document.getElementById("dupbtcencryptedkey" + idPostFix).innerHTML = privateKey; - document.getElementById("wallettype" + idPostFix).innerHTML = 'BIP38 Encrypted'; + if (window.designChoice != 'default') document.getElementById("wallettype" + idPostFix).innerHTML = 'BIP38 ENCRYPTED'; // only add for special designs + drawOpts.text['walletImportFormat'] = 'BIP38 ENCRYPTED'; + drawOpts.color = colors['bip38']; } else { document.getElementById("btcprivwif" + idPostFix).innerHTML = privateKey; document.getElementById("dupbtcprivwif" + idPostFix).innerHTML = privateKey; document.getElementById("wallettype" + idPostFix).innerHTML = ''; + drawOpts.color = colors['default']; + if (window.designChoice != 'default') document.getElementById("wallettype" + idPostFix).innerHTML = ''; // special designs should have this burned into the JPG + drawOpts.text['walletImportFormat'] = 'WALLET IMPORT FORMAT'; } + if (window.designChoice == 'default') { // if we are not loading up a special JPG-based design, render the canvas + document.getElementById("papersvg1").src = PaperWallet.draw.frontImage(bitcoinAddress, drawOpts); + } // CODE to modify SVG DOM elements //var paperSvg = document.getElementById("papersvg" + idPostFix); //if (paperSvg) { @@ -11897,7 +12078,7 @@ ninja.publicKey = { } document.getElementById('paperbip38settings').className = show ? 'show' : ''; - document.getElementById('paperencryptpassphrase').innerText = + if (!cancelSave) document.getElementById('paperencryptpassphrase').innerText = document.getElementById('paperencryptpassphrase').textContent = document.getElementById('paperpassphrase').value; if (!show && !cancelSave) { @@ -12765,6 +12946,424 @@ ninja.publicKey = { Brand new JS functions added for bitcoinpaperwallet.com ********************************************************/ + +// Dynamic drawing of standard bitcoin design + if (typeof PaperWallet == "undefined") { + // global PaperWallet object + var PaperWallet = window.PaperWallet = {}; + } + if (!PaperWallet.draw) { + (function() { + // images minimized with https://tinypng.com/ and http://pnggauntlet.com/ and encoded with http://www.base64-image.de/ + var imgBitcoinLogo = new Image(); + imgBitcoinLogo.src = "images/front-logo.png"; + + var draw = PaperWallet.draw = { + imgBitcoinLogo: imgBitcoinLogo, + + /** + * Draw guilloche pattern with the given parameters into the canvas. + * Source: https://gist.github.com/3n/803329 + */ + guilloche: function(ctx, opts) { + ctx.save(); + + var opts = opts || {}; + var scale = opts.scale || 1, + majorR = opts.majorR || 49, + minorR = opts.minorR || 20.4324, + angleMultiplier = opts.angleMultiplier || 1, + radiusEffectConstant = opts.radiusEffectConstant || 33, + steps = opts.steps || 1000, + centerPoint = opts.centerPoint || { x: 0, y:0 }, + color = opts.color || 'rgb(0,0,255)', + width = opts.width || 0.5; + globalAlpha = opts.globalAlpha || 1.0; + + ctx.globalAlpha = globalAlpha; + + var diff = majorR - minorR, + s = diff / minorR, + theta = 0, + radiusEffect = radiusEffectConstant + minorR, + oldX, oldY; + + for (var i = steps; i>0; --i) { + var new_theta = angleMultiplier * theta, + x = (diff * Math.sin(new_theta) + radiusEffect * Math.sin(new_theta * s))*scale + (centerPoint.x), + y = (diff * Math.cos(new_theta) - radiusEffect * Math.cos(new_theta * s))*scale + (centerPoint.y); + + theta += Math.PI * 4 / steps; + + if (oldX) { + ctx.strokeStyle = color; + ctx.lineWidth = width; + ctx.beginPath(); + ctx.moveTo(oldX, oldY); + ctx.lineTo(x, y); + ctx.closePath(); + ctx.stroke(); + } + + oldX = x; + oldY = y; + } + + ctx.restore(); + }, + + + /** + * Calculate unique parameters for pattern drawing. + * @param {String} msg Message to extract parameters from. + */ + guillocheParams: function(msg) { + /** + * Extracts a float in range [0, 1[ from first 4 bytes of the given bytearray. + * This treats the first 4 bytes as a signed integer value, scales it to size 1, + * and adds 0.5 to move to the range [0, 1[. + * + * @param {bytearray} Byte array, at least 4 bytes + */ + function float01(bytearray) { + var v = bytearray[0] << 24; + v |= bytearray[1] << 16; + v |= bytearray[2] << 8; + v |= bytearray[3]; + v /= 4294967296; + + // since v is signed, just add 0.5 to remap to [0, 1[. + return v + 0.5; + } + + /** + * Linear interpolation between min and max based on v [0, 1[. + */ + function scaled_between(v, min, max) { + return v*(max - min) + min; + }; + + // this contains some hand tuned constant to make the generated patterns look pretty. + var msg = Crypto.SHA256(msg, {asBytes: true}); + var majorR = scaled_between(float01(msg), 55, 300); // overall radius. + + var msg = Crypto.SHA256(msg, {asBytes: true}); + var scale = scaled_between(float01(msg), 5, 15); + + msg = Crypto.SHA256(msg, {asBytes: true}); + var minorR = scaled_between(float01(msg), 0.01, 10); // a large minorR makes for a sparser design, bigger holes in the spiderweb. + + + msg = Crypto.SHA256(msg, {asBytes: true}); + var steps = scaled_between(float01(msg), 1500, 7000); + + var width = (5000/steps) * .5; + width = Math.min(width,1.2); // prevent overly chunky lines + + msg = Crypto.SHA256(msg, {asBytes: true}); + var radiusEffectConstant = scaled_between(float01(msg), majorR*0.9, majorR); + + msg = Crypto.SHA256(msg, {asBytes: true}); + var angleMultiplier = Math.ceil(scaled_between(float01(msg), 0, 3)); + + // majorR = 110; scale = 11; minorR = .5; steps = 5000; width = .3; radiusEffectConstant = 100; angleMultiplier = 2; // test suite + + return { + angleMultiplier: angleMultiplier, + majorR: majorR, + minorR: minorR, + radiusEffectConstant: radiusEffectConstant, + scale: scale, + steps: steps, + width: width + } + }, + + + /** + * Draws a rounded rectangle using the current state of the canvas. + * If you omit the last three params, it will draw a rectangle + * outline with a 5 pixel border radius + * + * Source: http://stackoverflow.com/a/3368118 + * + * @param {CanvasRenderingContext2D} ctx + * @param {Number} x The top left x coordinate + * @param {Number} y The top left y coordinate + * @param {Number} width The width of the rectangle + * @param {Number} height The height of the rectangle + * @param {Number} radius The corner radius. Defaults to 5; + * @param {Boolean} fill Whether to fill the rectangle. Defaults to false. + * @param {Boolean} stroke Whether to stroke the rectangle. Defaults to true. + */ + roundRect: function(ctx, x, y, w, h, opts) { + ctx.save(); + + ctx.beginPath(); + ctx.moveTo(x + opts.radius, y); + ctx.lineTo(x + w - opts.radius, y); + ctx.quadraticCurveTo(x + w, y, x + w, y + opts.radius); + ctx.lineTo(x + w, y + h - opts.radius); + ctx.quadraticCurveTo(x + w, y + h, x + w - opts.radius, y + h); + ctx.lineTo(x + opts.radius, y + h); + ctx.quadraticCurveTo(x, y + h, x, y + h - opts.radius); + ctx.lineTo(x, y + opts.radius); + ctx.quadraticCurveTo(x, y, x + opts.radius, y); + ctx.closePath(); + + if (typeof opts.lineWidth != "undefined") { + ctx.lineWidth = opts.lineWidth; + } + if (typeof opts.strokeStyle != "undefined") { + ctx.strokeStyle = opts.strokeStyle; + ctx.stroke(); + } + if (typeof opts.fillStyle != "undefined") { + ctx.fillStyle = opts.fillStyle; + ctx.fill(); + } + + ctx.restore(); + }, + + + /** + * Draw text with strong blur around it. + * This is quite a hack, but I couldn't find a better way. + */ + fillTextStrongBlur: function(ctx, msg, x, y, offset, opts) { + ctx.save(); + + ctx.shadowColor = opts.color.textShadow; + ctx.shadowBlur = 7; + ctx.shadowOffsetX = 4; + ctx.shadowOffsetY = 4; + + // run multiple times for stronger blur + for (var i=0; i<1; ++i) { + ctx.shadowOffsetX = offset; + ctx.shadowOffsetY = offset; + ctx.fillText(msg, x, y); + + ctx.shadowOffsetX = -offset; + ctx.shadowOffsetY = offset; + ctx.fillText(msg, x, y); + + ctx.shadowOffsetX = offset; + ctx.shadowOffsetY = -offset; + ctx.fillText(msg, x, y); + + ctx.shadowOffsetX = -offset; + ctx.shadowOffsetY = -offset; + ctx.fillText(msg, x, y); + } + + ctx.restore(); + }, + + text: function(ctx, opts) { + ctx.save(); + + ctx.font = opts.font; + ctx.textAlign = "center"; + ctx.fillStyle = opts.color.text; + draw.fillTextStrongBlur(ctx, opts.text.publicAddress, 268, 320, 4, opts); + draw.fillTextStrongBlur(ctx, opts.text.depositVerify, 268, 696, 4, opts); + + ctx.rotate(-Math.PI*2/4); + draw.fillTextStrongBlur(ctx, opts.text.privateKey, -482, 2396, 4, opts); + draw.fillTextStrongBlur(ctx, opts.text.walletImportFormat, -482, 2844, 4, opts); + + + // text for green pointer + ctx.fillStyle = opts.color.textPointer; + ctx.fillText(opts.text.withdraw, -482, 1015); + + ctx.restore(); + }, + + obfuscation: function(ctx, x, y, w, h) { + ctx.save(); + ctx.font = "18px Courier"; + ctx.textAlign = "center"; + var obfuscationColors = ["#000000","#222222","#333333","#666666"]; + + // create random text to black out private key pattern + var possible_letters = "฿#"; + // var possible_letters = "■•"; + for (var i=0; i<1500; ++i) { + ctx.fillStyle = obfuscationColors[Math.floor(Math.random() * obfuscationColors.length)]; + var letter = possible_letters[Math.floor(Math.random()*possible_letters.length)]; + ctx.fillText(letter, x + Math.random()*w, y + Math.random()*h); + } + ctx.restore(); + }, + + gradients: function(ctx, opts) { + ctx.save(); + + // draw background gradients + var grd = ctx.createLinearGradient(55, 82, 55, 807); + grd.addColorStop(0, opts.color.publicUpper); + grd.addColorStop(1, opts.color.publicLower); + ctx.fillStyle = grd; + ctx.fillRect(55, 82, 958, 807); + + grd = ctx.createLinearGradient(1013, 0, 2961, 0) + grd.addColorStop(0, opts.color.privateLeft); + grd.addColorStop(1, opts.color.privateRight); + ctx.fillStyle = grd; + ctx.fillRect(1013, 82, 2961, 807); + ctx.restore(); + }, + + logos: function(ctx, cp) { + var w = 90; + + /* draw circle for bitcoin logo + ctx.arc(cp.x, cp.y, w/2+5, 0, 2*Math.PI, false); + ctx.fillStyle = "white"; + ctx.fill(); + */ + + // logo + ctx.drawImage(PaperWallet.draw.imgBitcoinLogo, (cp.x-w/2)-4, (cp.y-w/2)-4, 305, 101); + }, + + pointer: function(ctx, colorPointer) { + ctx.save(); + ctx.beginPath(); + ctx.moveTo(980, 82); + ctx.lineTo(1030, 82); + ctx.lineTo(1030, 355); + ctx.quadraticCurveTo(1030, 466, 1070, 486); + ctx.quadraticCurveTo(1030, 506, 1030, 617); + ctx.lineTo(1030, 889); + ctx.lineTo(980, 889); + ctx.closePath(); + ctx.fillStyle = colorPointer; + ctx.shadowColor = "#555"; + ctx.shadowBlur = 5; + ctx.shadowOffsetX = 5; + ctx.shadowOffsetY = 0; + ctx.fill(); + ctx.restore(); + }, + + clearSurroundings: function(ctx) { + ctx.save(); + + ctx.beginPath(); + ctx.moveTo(55, 82); + ctx.lineTo(1708, 82); + ctx.lineTo(1708, 182); + ctx.lineTo(2308, 182); + ctx.lineTo(2809, 90); + ctx.lineTo(2906, 182); + + ctx.lineTo(2906, 971-182); + ctx.lineTo(2809, 971-90); + ctx.lineTo(2308, 971-182); + ctx.lineTo(1708, 971-182); + ctx.lineTo(1708, 971-82); + ctx.lineTo(55, 971-82); + + // surroundings, move back + ctx.lineTo(0, 971); + ctx.lineTo(2962, 971); + ctx.lineTo(2962, 0); + ctx.lineTo(0, 0); + ctx.lineTo(0, 971); + ctx.lineTo(55, 971); + ctx.closePath(); + + ctx.lineWidth = 0.2; + ctx.strokeStyle = "black"; + ctx.fillStyle = "white"; + + ctx.stroke(); + ctx.fill(); + + ctx.restore(); + }, + + // Creates an image based on the given public key. + frontImage: function(key, opts) { + // create the temporary draw canvas + var canvas = document.createElement('canvas'); + canvas.width = 2962; + canvas.height = 971; + + var key = key || ""; + + var ctx = canvas.getContext('2d'); + var cp = {x: 665, y: 389}; + + PaperWallet.draw.gradients(ctx, opts); + + // draw guilloche + guillocheParams = PaperWallet.draw.guillocheParams(key); + guillocheParams.color = opts.color.guilloche; + guillocheParams.centerPoint = cp; + PaperWallet.draw.guilloche(ctx, guillocheParams); + + PaperWallet.draw.logos(ctx, cp); + PaperWallet.draw.pointer(ctx, opts.color.pointer); + + PaperWallet.draw.clearSurroundings(ctx); + + // draw empty rectangle for public QR code + var qrOpts = { + lineWidth: 0.2, + strokeStyle: "black", + fillStyle: "white", + radius: 20, + }; + + PaperWallet.draw.roundRect(ctx, 113, 344, 310, 310, qrOpts); + PaperWallet.draw.roundRect(ctx, 1816, 290, 384, 384, qrOpts); + PaperWallet.draw.roundRect(ctx, 2416, 290, 384, 384, qrOpts); + + PaperWallet.draw.text(ctx, opts); + + PaperWallet.draw.obfuscation(ctx, 1831, 310, 350, 350); + + return canvas.toDataURL(); + } + }; + })(); + } + + // global drawing options, most of which will be replaced in setDesign() for translation + + var drawOpts = { + color: { + publicUpper: "#fff57c", + publicLower: "#f7931a", + privateLeft: "#8cd96f", + privateRight: "#fff67d", + pointer: "#03ab5d", + guilloche: "white", + + text: "#1937a9", + textShadow: "white", + textPointer: "white", + }, + + font: "bold 20pt sans-serif", + + text: { + publicAddress: "", + depositVerify: "", + privateKey: "", + walletImportFormat: "", + withdraw: "", + }, + }; + + + // Functions for printer calibration + var inlineMediaStyle = null; function printZoom (changeBy) { /* handle +/- buttons for print zoom */ @@ -12881,8 +13480,265 @@ ninja.publicKey = { updateCalibrationInfo(); } -function setDesign (whichDesign, isOnLoad) { +function setDesign (whichDesign, isOnLoad, whichLanguage) { + + whichLanguage = typeof whichLanguage !== 'undefined' ? whichLanguage : 'english'; + + // these translations are only applicable to the standard bitcoin wallet. + // alt-currencies and special holiday/theme designs use JPGs instead. + + var translations = { + + 'english': { + publicAddress: "PUBLIC ADDRESS", + depositVerify: "DEPOSIT / VERIFY", + privateKey: "PRIVATE KEY", + walletImportFormat: "WALLET IMPORT FORMAT", + withdraw: "PRIVATE KEY / WITHDRAW", + backLongTextFontSize: "9px", + backPaperWallet: "BITCOIN PAPER WALLET", + backAmount: "Amount Added", + backDate: "Date", + backNotes: "Notes:", + backInst1: "To deposit additional funds to this paper wallet, send bitcoins to its public address, anytime.", + backInst2: "Verify your balance by searching for the public address using a service such as blockchain.info", + backInst3: "Do not reveal the private key until you are ready to import the balance on this wallet to a bitcoin client, exchange, or online wallet.", + backInst4: "When withdrawing your funds from this wallet you should remove the ENTIRE BALANCE. If you attempt to spend only some of the funds you will likely lose the remaining bitcoins forever.
    For instructions visit bitcoinpaperwallet.com", + }, + + 'chinese': { + publicAddress: "公开地址", + depositVerify: "存款/查询余额", + privateKey: "钱包导入格式", + walletImportFormat: "WALLET IMPORT FORMAT", + withdraw: "私钥/取款", + backLongTextFontSize: "10px", + backPaperWallet: "BITCOIN 纸钱包", + backAmount: "添加数量", + backDate: "日期", + backNotes: "注释", + backInst1: "可以随时向公开地址发送比特币为该纸钱包添加资金。", + backInst2: "使用例如blockchain.info这样的服务,通过搜索公开地址来查询你的余额。", + backInst3: "在你准备将余额导入到比特币客户端、交易平台或者在线钱包之前,请勿泄露私钥。", + backInst4: "从此钱包取出资金时,请将全部余额一次性取走。如果你试图只取走一部分,那么你很有可能永远失去剩余的比特币。

    有疑问?请访问 : bitcoinpaperwallet.com", + }, + + 'traditional-chinese': { + publicAddress: "公开地址", + depositVerify: "存款/查询余额", + privateKey: "钱包导入格式", + walletImportFormat: "WALLET IMPORT FORMAT", + withdraw: "私钥/取款", + backLongTextFontSize: "10px", + backPaperWallet: "BITCOIN 纸钱包", + backAmount: "添加数量", + backDate: "日期", + backNotes: "注释", + backInst1: "可以随时向公开地址发送比特币为该纸钱包添加资金。", + backInst2: "使用例如blockchain.info这样的服务,通过搜索公开地址来查询你的余额。", + backInst3: "在你准备将余额导入到比特币客户端、交易平台或者在线钱包之前,请勿泄露私钥。", + backInst4: "从此钱包取出资金时,请将全部余额一次性取走。如果你试图只取走一部分,那么你很有可能永远失去剩余的比特币。

    有疑问?请访问 : bitcoinpaperwallet.com", + }, + + 'czech': { + publicAddress: "VEŘEJNÁ ADRESA", + depositVerify: "VLOŽIT / OVĚŘIT", + privateKey: "SOUKROMÝ KLÍČ", + walletImportFormat: "WALLET IMPORT FORMAT", + withdraw: "SOUKROMÝ KLÍČ / VYBRAT", + backLongTextFontSize: "9px", + backPaperWallet: "BITCOIN Papírová peněženka", + backAmount: "Částka", + backDate: "Datum", + backNotes: "Poznámky:", + backInst1: "Pokud chcete vložit více peněz do této peněženky, můžete kdykoli použít stejnou veřejnou adresu.", + backInst2: "Pokud chcete ověřit zůstatek v peněžence, můžete použít službu blockchain.info a zadat veřejnou adresu.", + backInst3: "NEUKAZUJTE NIKOMU SOUKROMÝ KLÍČ dokud se nerozhodnete opravdu převést zůstatek do klienta,na burzu nebo do online peněženky. ", + backInst4: "Při výběru z této peněženky byste měli převést CELÝ ZŮSTATEK. Pokud převedete jen část, je možné že příjdete o zbytek.
    Dotazy? Navštivte bitcoinpaperwallet.com", + }, + + 'dutch': { + publicAddress: "PUBLIEK ADRES", + depositVerify: "STORTING / VERIFICATIE", + privateKey: "PRIVÉ SLEUTEL", + walletImportFormat: "WALLET IMPORT FORMAT", + withdraw: "PRIVÉ SLEUTEL", + backLongTextFontSize: "8px", + backPaperWallet: "BITCOIN Papieren portemonnee", + backAmount: "Bedrag toegevoegd", + backDate: "Datum", + backNotes: "Notities:", + backInst1: "Om meer geld naar deze papieren portemonnee over te maken kun je altijd bitcoins naar het publieke adres sturen.", + backInst2: "Verifieer je saldo door het publieke adress op te zoeken met een service zoals blockchain.info", + backInst3: "LAAT JE PRIVÉ SLEUTEL NIET ZIEN totdat je het saldo van deze portemonnee over wilt maken naar een bitcoin client, exchange of online portemonnee.", + backInst4: "Als je het saldo van deze portemonnee uit wilt laten betalen, dan kun je het beste het HELE BEDRAG opnemen. Als je probeert een beetje op te nemen ben je waarschijnlijk de overige bitcoins kwijt.

    Vragen? Ga naar bitcoinpaperwallet.com", + }, + + 'french': { + publicAddress: "ADRESSE PUBLIQUE", + depositVerify: "Faire un Dépôt/Vérifier", + privateKey: "CLÉ SECRÈTE", + walletImportFormat: "WALLET IMPORT FORMAT", + withdraw: "CLÉ SECRÈTE / FAIRE UN RETRAIT", + backLongTextFontSize: "8px", + backPaperWallet: "BITCOIN Portefeuille papier", + backAmount: "Montant ajouté", + backDate: "Date", + backNotes: "Notes:", + backInst1: "Pour déposer plus de fonds dans ce portefeuille papier, envoyer des Bitcoins à son adresse publique n'importe quand.", + backInst2: "Vérifiez le solde en faisant une recherche de l'adresse publique en utilisant un service tel que blockchain.info", + backInst3: "NE RÉVÉLEZ PAS VOTRE CLÉ SECRÈTE jusqu'à ce que vous soyez prêt à transférer le solde de ce portefeuille à un client Bitcoin, une plate-forme d'échange ou un portefeuille en-ligne.", + backInst4: "Lorsque vous allez retirer des fonds de ce portefeuille, vous devriez transférer le SOLDE EN ENTIER. Si vous essayez de ne dépenser qu'une partie des fonds, vous risquez de perdre le reste des Bitcoins à tout jamais.

    Questions ? Visitez : bitcoinpaperwallet.com", + }, + + 'german': { + publicAddress: "ÖFFENTLICHE ADRESSE", + depositVerify: "EINZAHLEN / BESTÄTIGEN", + privateKey: "PRIVATER SCHLÜSSEL", + walletImportFormat: "WALLET IMPORT FORMAT", + withdraw: "PRIVATER SCHLÜSSEL / ABHEBEN", + backLongTextFontSize: "8px", + backPaperWallet: "BITCOIN PAPIERBRIEFTASCHE", + backAmount: "Gutschriften", + backDate: "Datum", + backNotes: "Notizen:", + backInst1: "Um das Guthaben auf dieser Papierbrieftasche zu erhöhen, sende bitcoins an ihre öffentliche Adresse, jederzeit.", + backInst2: "Um das Guthaben zu bestätigen suche nach der öffentlichen Adresse auf einem Service wie blockchain.info", + backInst3: "Mache den privaten schlüssel nie bekannt bis du bereit bist das Guthaben auf dieser Brieftasche abzuheben auf einen Bitcoin-Client, eine Wechselbörse oder eine Onlinebrieftasche.", + backInst4: "Beim Abheben des Guthabens sollte das Ganze Guthaben auf einmal abgehoben werden. Beim Versuch nur einen Teil des Guthaben auszugeben kann es zum dauerhaften Verlust des restlichen Guthabens kommen.

    Fragen? Geh auf bitcoinpaperwallet.com", + }, + + 'hebrew': { + publicAddress: "כתובת פומבית", + depositVerify: "הפקדה / וידוא מאזן", + privateKey: "מפתח פרטי", + walletImportFormat: "WALLET IMPORT FORMAT", + withdraw: "מפתח פרטי / משיכה", + backLongTextFontSize: "9px", + backPaperWallet: "BITCOIN ארנק נייר", + backAmount: "כמות שנטענה", + backDate: "תאריך", + backNotes: "הערות", + backInst1: "כדי להפקיד לארנק זה, שלחו ביטקוינים לכתובת הפומבית, בכל זמן", + backInst2: "ודאו את המאזן שלכם ע\"י חיפוש הכתובת הפומבית בעזרת שירות כמו blockchain.info", + backInst3: "אף פעם אל תגלו לאף אחד את המפתח הפרטי שלכם עד שאתם מוכנים לייצא את המטבעות לארנק ביטקוין ממוחשב, בורסאת ביטקוין או ארנק מקוון", + backInst4: "בעת משיכת המטבעות, יש למשוך את -כל המטבעות- שבארנק. אם תנסו להוציא רק חלק מהמטבעות, תאבדו את המטבעות הנשארים לנצח שאלות? פנו ל: bitcoinpaperwallet.com", + }, + + 'italian': { + publicAddress: "INDIRIZZO PUBBLICO", + depositVerify: "DEPOSITA / VERIFICA", + privateKey: "CHIAVE PRIVATA", + walletImportFormat: "WALLET IMPORT FORMAT", + withdraw: "CHIAVE PRIVATA / PRELEVA", + backLongTextFontSize: "9px", + backPaperWallet: "BITCOIN PORTAFOGLIO CARTACEO", + backAmount: "Importo depositato", + backDate: "Data", + backNotes: "Note:", + backInst1: "Per depositare altri fondi su questo portafoglio cartaceo, invia bitcoins al suo indirizzo pubblico, in qualsiasi momento.", + backInst2: "Puoi verificare il saldo cercando l'indirizzo pubblico su un servizio tipo blockchain.info", + backInst3: "NON ESPORRE LA CHIAVE PRIVATA se non per importare immediatamente il contenuto del portafoglio su un client bitcoin, un exchange o un portafoglio online.", + backInst4: "Se prelevi i fondi depositati su questo portafoglio, ti consigliamo di prelevare l'INTERO IMPORTO depositato. Se cerchi di spendere i fondi solo in parte, probabilmente perderai i bitcoins residui.
    Domande? Visita : bitcoinpaperwallet.com", + }, + + 'japanese': { + publicAddress: "公 開 ア ド レ ス", + depositVerify: "入 金 / 残 高 確 認", + privateKey: "秘 密 鍵", + walletImportFormat: "WALLET IMPORT FORMAT", + withdraw: "秘 密 鍵 / 出 金", + backLongTextFontSize: "8px", + backPaperWallet: "BITCOIN ペーパーウォレット", + backAmount: "入金金額", + backDate: "日付", + backNotes: "メモ:", + backInst1: "公開アドレス宛てに入金することで、このウォレットにいつでもビットコインを追加することができます。", + backInst2: "blockchain.infoのようなサービスで公開アドレスを検索することで、残高を確認することができます。", + backInst3: "ビットコインクライアントやオンラインウォレット、交換所にビットコインを移動する必要があるまで、秘密鍵を開封しないでください。", + backInst4: "このウォレットからビットコインを出金する際には、全ての残高を出金してください。一部を残して出金した場合、残ったビットコインが永遠に失われる恐れがあります。

    詳しい情報は、bitcoinpaperwallet.comを参照してください。", + }, + + 'portuguese': { + publicAddress: "ENDEREÇO PÚBLICO", + depositVerify: "DEPOSITE / VERIFIQUE", + privateKey: "CHAVE PRIVADA", + walletImportFormat: "WALLET IMPORT FORMAT", + withdraw: "CHAVE PRIVADA / SAQUE", + backLongTextFontSize: "8px", + backPaperWallet: "BITCOIN PAPER WALLET", + backAmount: "Saldo", + backDate: "Data", + backNotes: "Anotações:", + backInst1: "Para depositar fundos adicionais na sua paper wallet deve-se enviar os bitcoins para seu endereço público, a qualquer hora.", + backInst2: "Verifique o saldo através do seu endereço público usando um serviço como o blockchain.info", + backInst3: "NÃO REVELE A SUA CHAVE PRIVADA até que você esteja pronto a importar o saldo total para um cliente bitcoin ou uma carteira online ou, ainda, trocar sua paper wallet com alguém.", + backInst4: "TRANSFIRA TODO O SALDO dessa carteira ao importá-la. Se você tentar importar somente uma parcela provavelmente o risco de perder o restante dos bitcoins é alto.

    Para mais informações visite bitcoinpaperwallet.com", + }, + + 'russian': { + publicAddress: "ОТКРЫТЫЙ АДРЕС", + depositVerify: "ДЕПОЗИТ / ПРОВЕРКА", + privateKey: "ЗАКРЫТЫЙ КЛЮЧ", + walletImportFormat: "WALLET IMPORT FORMAT", + withdraw: "ЗАКРЫТЫЙ КЛЮЧ / ВЫВОД", + backLongTextFontSize: "8px", + backPaperWallet: "BITCOIN БУМАЖНЫЙ КОШЕЛЁК", + backAmount: "Внесено", + backDate: "Дата", + backNotes: "Заметки:", + backInst1: "Для внесения дополнительных финансов на этот бумажный кошелёк отправьте биткоины на его открытый адрес в любое время.", + backInst2: "Проверьте баланс найдя открытый адрес в таком сервисе, как blockchain.info", + backInst3: "НЕ ВСКРЫВАЙТЕ СВОЙ ЗАКРЫТЫЙ КЛЮЧ до тех пор, пока вы не будете готовы импортировать баланс этого кошелька в свой bitcoin-клиент, обменник, или онлайн-кошелёк.", + backInst4: "Вы должны снять ВСЕ свои средства с этого кошелька единовременно. Если вы попытаетесь истратить только некую часть своих финансов, то, скорее всего, потеряете оставшиеся биткоины навсегда.

    Вопросы? Посетите: bitcoinpaperwallet.com.", + }, + + 'spanish': { + publicAddress: "DIRECCIÓN PÚBLICA", + depositVerify: "DEPOSITAR / VERIFICAR", + privateKey: "CLAVE PRIVADA", + walletImportFormat: "WALLET IMPORT FORMAT", + withdraw: "CLAVE PRIVADA / RETIRAR", + backLongTextFontSize: "9px", + backPaperWallet: "BITCOIN Cartera de papel", + backAmount: "Cantidad añadida", + backDate: "Fecha", + backNotes: "Notas:", + backInst1: "Para depositar fondos adicionales en esta cartera de papel, envíe unos Bitcoins a su dirección pública.", + backInst2: "Para verificar su saldo, busque su dirección pública en un sitio web como blockchain.info", + backInst3: "NUNCA DEBE MOSTRAR SU CLAVE PRIVADA hasta que este preparado para importar el saldo de esta cartera a un cliente de bitcoin, un sitio de cambio, o una cartera de internet.", + backInst4: "Cuando este sacando sus fondos, debe sacar LA TOTALIDAD DE SU SALDO. Si no hace esto, es posible que pierda sus Bitcoin.

    ¿Dudas? Vaya a bitcoinpaperwallet.com", + }, + + 'swedish': { + publicAddress: "PUBLIK ADRESS", + depositVerify: "SÄTT IN / VERIFIERA", + privateKey: "PRIVAT NYCKEL", + walletImportFormat: "WALLET IMPORT FORMAT", + withdraw: "PRIVAT NYCKEL / TA UT", + backLongTextFontSize: "9px", + backPaperWallet: "BITCOIN PAPER WALLET", + backAmount: "Insatt belopp", + backDate: "Datum", + backNotes: "Anteckningar:", + backInst1: "För att sätta in bitcoin: använd den publika adressen.", + backInst2: "Kontrollera saldot genom att söka efter den publika adressen på till exempel blockchain.info", + backInst3: "ANVÄND ELLER VISA INTE DEN PRIVATA NYCKELN förrän det är dags att flytta saldot till en bitcoinklient, börs eller online-plånbok.", + backInst4: "När du flyttar bitcoin från en paperwallet bör du alltid TÖMMA DEN HELT. Om du inte gör det riskerar du att förlora kvarvarande bitcoin.

    Frågor? Besök bitcoinpaperwallet.com", + } + }; + + drawOpts.text = translations[whichLanguage]; + if (!whichDesign) whichDesign = 'default'; + + // show the language menu if we're using the default design + if (whichDesign == 'default') { + document.getElementById('langPicker').style.display = 'inline'; + } else { + document.getElementById('langPicker').style.display = 'none'; + } + // if we used the dropdown menu to select an alt-coin, we need to reload the page altogether. if (!isOnLoad && whichDesign.substring(0,4) == 'alt-') window.location='?design=' + whichDesign; // if we're already using an altcoin, reload the page no matter which design we choose. @@ -12893,20 +13749,68 @@ function setDesign (whichDesign, isOnLoad) { document.getElementById('logoback').style.backgroundImage = 'url(images/logo-' + whichDesign + '.png)'; } + // now deal with JPG-style backgrounds, and apply instructions to default design + if (whichDesign == 'default') { var myFront = './images/front-300dpi.jpg'; var myBack = './images/back-300dpi.jpg'; var myPreview = './images/finished-sample-sealed.jpg'; + // setup proper back text translation & controls + document.getElementById('backTextControl').style.display='block'; + document.getElementById('backText').style.display='block'; + document.getElementById("backPaperWallet").innerHTML = translations[whichLanguage]['backPaperWallet']; + document.getElementById("backAmount").innerHTML = translations[whichLanguage]['backAmount']; + document.getElementById("backDate").innerHTML = translations[whichLanguage]['backDate']; + document.getElementById("backNotes").innerHTML = translations[whichLanguage]['backNotes']; + document.getElementById("backLongText").style.fontSize = translations[whichLanguage]['backLongTextFontSize']; + document.getElementById("backInst1").innerHTML = translations[whichLanguage]['backInst1']; + document.getElementById("backInst2").innerHTML = translations[whichLanguage]['backInst2']; + document.getElementById("backInst3").innerHTML = translations[whichLanguage]['backInst3']; + document.getElementById("backInst4").innerHTML = translations[whichLanguage]['backInst4']; + } else { var myFront = './images/front-300dpi-' + whichDesign + '.jpg'; var myBack = './images/back-300dpi-' + whichDesign + '.jpg'; var myPreview = './images/finished-sample-' + whichDesign + '.jpg'; + document.getElementById('backTextControl').style.display='none'; + document.getElementById('backText').style.display='none'; } + window.frontJPG=myFront; - if (document.getElementById('papersvg1') != null) document.getElementById('papersvg1').src=myFront; + window.designChoice=whichDesign; + + // if the front artwork is ready, set it to either the correspondong JPG, or draw it + if (document.getElementById('papersvg1') != null) { + if (whichDesign == 'default') { + document.getElementById("papersvg1").src = PaperWallet.draw.frontImage(document.getElementById("btcaddress1").innerHTML, drawOpts); + } else { + document.getElementById('papersvg1').src=myFront; + } + } + + // set back document.getElementById('backsvg1').src=myBack; document.getElementById('designPreview').src=myPreview; document.getElementById('designPicker').value = whichDesign; // force menu option in case it was picked during onload. + document.getElementById('langPicker').value = whichLanguage; +} + +function setDenomination(valueAndUnit) { + if (!valueAndUnit) { + document.getElementById('backSeal').style.display = 'none'; + document.getElementById('backDenominationOther').style.display = 'none'; + } else if (valueAndUnit == 'other') { + document.getElementById('backDenominationOther').style.display = 'block'; + document.getElementById('backSeal').style.display = 'block'; + document.getElementById('backSealBig').innerHTML = document.getElementById('backDenominationOtherAmount').value; + document.getElementById('backSealSmall').innerHTML = document.getElementById('backDenominationOtherUnit').value; + } else { + denominationParts = valueAndUnit.split(' '); + document.getElementById('backSeal').style.display = 'block'; + document.getElementById('backSealBig').innerHTML = denominationParts[0]; + document.getElementById('backSealSmall').innerHTML = denominationParts[1]; + document.getElementById('backDenominationOther').style.display = 'none' + } } diff --git a/images/back-300dpi-chinese.jpg b/images/back-300dpi-chinese.jpg deleted file mode 100644 index e6a0823..0000000 Binary files a/images/back-300dpi-chinese.jpg and /dev/null differ diff --git a/images/back-300dpi-counterparty.jpg b/images/back-300dpi-counterparty.jpg new file mode 100644 index 0000000..2dece32 Binary files /dev/null and b/images/back-300dpi-counterparty.jpg differ diff --git a/images/back-300dpi-custom.jpg b/images/back-300dpi-custom.jpg new file mode 100644 index 0000000..863349f Binary files /dev/null and b/images/back-300dpi-custom.jpg differ diff --git a/images/back-300dpi-czech.jpg b/images/back-300dpi-czech.jpg deleted file mode 100644 index d2dcdf4..0000000 Binary files a/images/back-300dpi-czech.jpg and /dev/null differ diff --git a/images/back-300dpi-dutch.jpg b/images/back-300dpi-dutch.jpg deleted file mode 100644 index 9739e8c..0000000 Binary files a/images/back-300dpi-dutch.jpg and /dev/null differ diff --git a/images/back-300dpi-french.jpg b/images/back-300dpi-french.jpg deleted file mode 100644 index be32376..0000000 Binary files a/images/back-300dpi-french.jpg and /dev/null differ diff --git a/images/back-300dpi-german.jpg b/images/back-300dpi-german.jpg deleted file mode 100644 index 6ae342e..0000000 Binary files a/images/back-300dpi-german.jpg and /dev/null differ diff --git a/images/back-300dpi-italian.jpg b/images/back-300dpi-italian.jpg deleted file mode 100644 index 18121f9..0000000 Binary files a/images/back-300dpi-italian.jpg and /dev/null differ diff --git a/images/back-300dpi-portuguese.jpg b/images/back-300dpi-portuguese.jpg deleted file mode 100644 index e9b81ee..0000000 Binary files a/images/back-300dpi-portuguese.jpg and /dev/null differ diff --git a/images/back-300dpi-russian.jpg b/images/back-300dpi-russian.jpg deleted file mode 100644 index 5de03b8..0000000 Binary files a/images/back-300dpi-russian.jpg and /dev/null differ diff --git a/images/back-300dpi-spanish.jpg b/images/back-300dpi-spanish.jpg deleted file mode 100644 index 42d329c..0000000 Binary files a/images/back-300dpi-spanish.jpg and /dev/null differ diff --git a/images/back-300dpi-traditional-chinese.jpg b/images/back-300dpi-traditional-chinese.jpg deleted file mode 100644 index 4d34d41..0000000 Binary files a/images/back-300dpi-traditional-chinese.jpg and /dev/null differ diff --git a/images/back-300dpi.jpg b/images/back-300dpi.jpg index b485465..3c1760e 100644 Binary files a/images/back-300dpi.jpg and b/images/back-300dpi.jpg differ diff --git a/images/finished-sample-bagged.jpg b/images/finished-sample-bagged.jpg new file mode 100644 index 0000000..cfffa40 Binary files /dev/null and b/images/finished-sample-bagged.jpg differ diff --git a/images/finished-sample-chinese.jpg b/images/finished-sample-chinese.jpg deleted file mode 100644 index c605dc6..0000000 Binary files a/images/finished-sample-chinese.jpg and /dev/null differ diff --git a/images/finished-sample-counterparty.jpg b/images/finished-sample-counterparty.jpg new file mode 100644 index 0000000..369998a Binary files /dev/null and b/images/finished-sample-counterparty.jpg differ diff --git a/images/finished-sample-custom.jpg b/images/finished-sample-custom.jpg new file mode 100644 index 0000000..15cdaa3 Binary files /dev/null and b/images/finished-sample-custom.jpg differ diff --git a/images/finished-sample-czech.jpg b/images/finished-sample-czech.jpg deleted file mode 100644 index 6c4ff9c..0000000 Binary files a/images/finished-sample-czech.jpg and /dev/null differ diff --git a/images/finished-sample-dutch.jpg b/images/finished-sample-dutch.jpg deleted file mode 100644 index f5dcb41..0000000 Binary files a/images/finished-sample-dutch.jpg and /dev/null differ diff --git a/images/finished-sample-french.jpg b/images/finished-sample-french.jpg deleted file mode 100644 index 83a02d0..0000000 Binary files a/images/finished-sample-french.jpg and /dev/null differ diff --git a/images/finished-sample-german.jpg b/images/finished-sample-german.jpg deleted file mode 100644 index 6cb5405..0000000 Binary files a/images/finished-sample-german.jpg and /dev/null differ diff --git a/images/finished-sample-italian.jpg b/images/finished-sample-italian.jpg deleted file mode 100644 index 5711131..0000000 Binary files a/images/finished-sample-italian.jpg and /dev/null differ diff --git a/images/finished-sample-portuguese.jpg b/images/finished-sample-portuguese.jpg deleted file mode 100644 index 36af582..0000000 Binary files a/images/finished-sample-portuguese.jpg and /dev/null differ diff --git a/images/finished-sample-russian.jpg b/images/finished-sample-russian.jpg deleted file mode 100644 index cb1a00b..0000000 Binary files a/images/finished-sample-russian.jpg and /dev/null differ diff --git a/images/finished-sample-sealed.jpg b/images/finished-sample-sealed.jpg index 0dc69e1..6144ad0 100644 Binary files a/images/finished-sample-sealed.jpg and b/images/finished-sample-sealed.jpg differ diff --git a/images/finished-sample-spanish.jpg b/images/finished-sample-spanish.jpg deleted file mode 100644 index 3904ba1..0000000 Binary files a/images/finished-sample-spanish.jpg and /dev/null differ diff --git a/images/finished-sample-traditional-chinese.jpg b/images/finished-sample-traditional-chinese.jpg deleted file mode 100644 index 5c1d28c..0000000 Binary files a/images/finished-sample-traditional-chinese.jpg and /dev/null differ diff --git a/images/finished-sample.jpg b/images/finished-sample.jpg index 44be908..220862b 100644 Binary files a/images/finished-sample.jpg and b/images/finished-sample.jpg differ diff --git a/images/flags.png b/images/flags.png deleted file mode 100644 index f9cd950..0000000 Binary files a/images/flags.png and /dev/null differ diff --git a/images/front-300dpi-chinese.jpg b/images/front-300dpi-chinese.jpg deleted file mode 100644 index 1428e05..0000000 Binary files a/images/front-300dpi-chinese.jpg and /dev/null differ diff --git a/images/front-300dpi-counterparty.jpg b/images/front-300dpi-counterparty.jpg new file mode 100644 index 0000000..7febf46 Binary files /dev/null and b/images/front-300dpi-counterparty.jpg differ diff --git a/images/front-300dpi-custom.jpg b/images/front-300dpi-custom.jpg new file mode 100644 index 0000000..0d73879 Binary files /dev/null and b/images/front-300dpi-custom.jpg differ diff --git a/images/front-300dpi-czech.jpg b/images/front-300dpi-czech.jpg deleted file mode 100644 index 0df4c4c..0000000 Binary files a/images/front-300dpi-czech.jpg and /dev/null differ diff --git a/images/front-300dpi-dutch.jpg b/images/front-300dpi-dutch.jpg deleted file mode 100644 index 062f227..0000000 Binary files a/images/front-300dpi-dutch.jpg and /dev/null differ diff --git a/images/front-300dpi-french.jpg b/images/front-300dpi-french.jpg deleted file mode 100644 index e318a5c..0000000 Binary files a/images/front-300dpi-french.jpg and /dev/null differ diff --git a/images/front-300dpi-german.jpg b/images/front-300dpi-german.jpg deleted file mode 100644 index 9043267..0000000 Binary files a/images/front-300dpi-german.jpg and /dev/null differ diff --git a/images/front-300dpi-italian.jpg b/images/front-300dpi-italian.jpg deleted file mode 100644 index 803c233..0000000 Binary files a/images/front-300dpi-italian.jpg and /dev/null differ diff --git a/images/front-300dpi-portuguese.jpg b/images/front-300dpi-portuguese.jpg deleted file mode 100644 index 8ed9f62..0000000 Binary files a/images/front-300dpi-portuguese.jpg and /dev/null differ diff --git a/images/front-300dpi-russian.jpg b/images/front-300dpi-russian.jpg deleted file mode 100644 index 1f7df7f..0000000 Binary files a/images/front-300dpi-russian.jpg and /dev/null differ diff --git a/images/front-300dpi-spanish.jpg b/images/front-300dpi-spanish.jpg deleted file mode 100644 index 45f383f..0000000 Binary files a/images/front-300dpi-spanish.jpg and /dev/null differ diff --git a/images/front-300dpi-traditional-chinese.jpg b/images/front-300dpi-traditional-chinese.jpg deleted file mode 100644 index df57eec..0000000 Binary files a/images/front-300dpi-traditional-chinese.jpg and /dev/null differ diff --git a/images/front-300dpi.jpg b/images/front-300dpi.jpg index d47d08d..2f9347f 100644 Binary files a/images/front-300dpi.jpg and b/images/front-300dpi.jpg differ diff --git a/images/front-logo.png b/images/front-logo.png new file mode 100644 index 0000000..9883d37 Binary files /dev/null and b/images/front-logo.png differ diff --git a/images/front-seal.png b/images/front-seal.png new file mode 100644 index 0000000..1aab150 Binary files /dev/null and b/images/front-seal.png differ diff --git a/images/logo.png b/images/logo.png index 5db963e..35747bb 100644 Binary files a/images/logo.png and b/images/logo.png differ diff --git a/images/webpage-background.png b/images/webpage-background.png index 377f40c..5c3dc08 100644 Binary files a/images/webpage-background.png and b/images/webpage-background.png differ