Browse Source

Merge pull request #1589 from LedgerHQ/develop

Prepare 1.2.1
master
Meriadec Pillet 6 years ago
committed by GitHub
parent
commit
e34f35f139
No known key found for this signature in database GPG Key ID: 4AEE18F83AFDEB23
  1. 11
      .circleci/config.yml
  2. 2
      .github/ISSUE_TEMPLATE/feature_request.md
  3. 82
      babel.config.js
  4. 48
      package.json
  5. 50
      scripts/check-wordings.js
  6. 9
      scripts/cli/cli.sh
  7. 21
      scripts/cli/getDevice.js
  8. 106
      scripts/cli/txBetweenAccounts.js
  9. 49
      scripts/patch-appimage.sh
  10. 50
      scripts/release.sh
  11. 5
      scripts/shasums/patch-appimage-sums.txt
  12. 61
      scripts/upload-github-release-asset.sh
  13. 2
      src/__mocks__/storybook-state.js
  14. 2
      src/actions/general.js
  15. 2
      src/api/Ripple.js
  16. 46
      src/bridge/EthereumJSBridge.js
  17. 10
      src/bridge/LibcoreBridge.js
  18. 39
      src/bridge/RippleJSBridge.js
  19. 4
      src/bridge/makeMockBridge.js
  20. 2
      src/commands/debugAppInfosForCurrency.js
  21. 2
      src/commands/getAddress.js
  22. 3
      src/commands/index.js
  23. 29
      src/commands/libcoreGetFees.js
  24. 8
      src/commands/libcoreScanFromXPUB.js
  25. 66
      src/commands/libcoreSignAndBroadcast.js
  26. 16
      src/commands/libcoreSyncAccount.js
  27. 8
      src/components/AccountPage/AccountHeaderActions.js
  28. 6
      src/components/AccountPage/EmptyStateAccount.js
  29. 4
      src/components/AccountPage/index.js
  30. 8
      src/components/AdvancedOptions/BitcoinKind.js
  31. 4
      src/components/AdvancedOptions/EthereumKind.js
  32. 2
      src/components/AdvancedOptions/RippleKind.js
  33. 14
      src/components/BalanceSummary/BalanceInfos.js
  34. 2
      src/components/BalanceSummary/index.js
  35. 2
      src/components/BalanceSummary/stories.js
  36. 2
      src/components/CalculateBalance.js
  37. 2
      src/components/CounterValue/stories.js
  38. 20
      src/components/CurrentAddress/index.js
  39. 2
      src/components/DashboardPage/AccountCardListHeader.js
  40. 4
      src/components/DashboardPage/AccountCardPlaceholder.js
  41. 8
      src/components/DashboardPage/AccountsOrder.js
  42. 6
      src/components/DashboardPage/CurrentGreetings.js
  43. 8
      src/components/DashboardPage/EmptyState.js
  44. 2
      src/components/DashboardPage/SummaryDesc.js
  45. 2
      src/components/DashboardPage/index.js
  46. 14
      src/components/DevToolsPage/AccountImporter.js
  47. 4
      src/components/DeviceInteraction/components.js
  48. 15
      src/components/EnsureDeviceApp.js
  49. 4
      src/components/ExchangePage/ExchangeCard.js
  50. 4
      src/components/ExchangePage/index.js
  51. 2
      src/components/ExportLogsBtn.js
  52. 4
      src/components/FeesField/BitcoinKind.js
  53. 2
      src/components/FeesField/GenericContainer.js
  54. 2
      src/components/GenuineCheck.js
  55. 2
      src/components/GenuineCheckModal.js
  56. 2
      src/components/GlobalSearch.js
  57. 16
      src/components/IsUnlocked.js
  58. 23
      src/components/MainSideBar/index.js
  59. 12
      src/components/ManagerPage/AppsList.js
  60. 6
      src/components/ManagerPage/Dashboard.js
  61. 8
      src/components/ManagerPage/FirmwareUpdate.js
  62. 2
      src/components/ManagerPage/ManagerApp.js
  63. 4
      src/components/ManagerPage/ManagerGenuineCheck.js
  64. 6
      src/components/ManagerPage/PlugYourDevice.js
  65. 4
      src/components/ManagerPage/UpdateFirmwareButton.js
  66. 4
      src/components/Onboarding/OnboardingFooter.js
  67. 22
      src/components/Onboarding/steps/Analytics.js
  68. 6
      src/components/Onboarding/steps/Finish.js
  69. 16
      src/components/Onboarding/steps/GenuineCheck/GenuineCheckErrorPage.js
  70. 8
      src/components/Onboarding/steps/GenuineCheck/GenuineCheckUnavailable.js
  71. 20
      src/components/Onboarding/steps/GenuineCheck/index.js
  72. 10
      src/components/Onboarding/steps/Init.js
  73. 10
      src/components/Onboarding/steps/NoDevice.js
  74. 6
      src/components/Onboarding/steps/SelectDevice.js
  75. 12
      src/components/Onboarding/steps/SelectPIN/SelectPINblue.js
  76. 14
      src/components/Onboarding/steps/SelectPIN/SelectPINnano.js
  77. 12
      src/components/Onboarding/steps/SelectPIN/SelectPINrestoreBlue.js
  78. 14
      src/components/Onboarding/steps/SelectPIN/SelectPINrestoreNano.js
  79. 4
      src/components/Onboarding/steps/SelectPIN/index.js
  80. 16
      src/components/Onboarding/steps/SetPassword.js
  81. 4
      src/components/Onboarding/steps/Start.js
  82. 18
      src/components/Onboarding/steps/WriteSeed/WriteSeedBlue.js
  83. 18
      src/components/Onboarding/steps/WriteSeed/WriteSeedNano.js
  84. 26
      src/components/Onboarding/steps/WriteSeed/WriteSeedRestore.js
  85. 2
      src/components/OpenUserDataDirectoryBtn.js
  86. 2
      src/components/OperationsList/AmountCell.js
  87. 2
      src/components/OperationsList/ConfirmationCheck.js
  88. 2
      src/components/OperationsList/DateCell.js
  89. 6
      src/components/OperationsList/index.js
  90. 2
      src/components/PillsDaysCount.js
  91. 44
      src/components/QRCodeExporter.js
  92. 2
      src/components/RecipientAddress/index.js
  93. 23
      src/components/RenderError.js
  94. 2
      src/components/RequestAmount/index.js
  95. 4
      src/components/SelectAccount/index.js
  96. 5
      src/components/SelectCurrency/index.js
  97. 6
      src/components/SelectExchange.js
  98. 22
      src/components/SettingsPage/CleanButton.js
  99. 2
      src/components/SettingsPage/CounterValueSelect.js
  100. 10
      src/components/SettingsPage/DisablePasswordModal.js

11
.circleci/config.yml

@ -10,13 +10,20 @@ jobs:
<<: *defaults <<: *defaults
steps: steps:
- run: sudo apt-get install -y libudev-dev - run: sudo apt-get install -y libudev-dev
- run:
name: Install latest yarn
command: |
curl -sS http://dl.yarnpkg.com/debian/pubkey.gpg | sudo apt-key add - ;
echo "deb http://dl.yarnpkg.com/debian/ stable main" | sudo tee /etc/apt/sources.list.d/yarn.list ;
sudo apt-get update && sudo apt-get install yarn
sudo rm /usr/local/bin/yarn # remove docker yarn
- checkout - checkout
- restore_cache: - restore_cache:
keys: keys:
- v10-yarn-packages-{{ checksum "yarn.lock" }} - v11-yarn-packages-{{ checksum "yarn.lock" }}
- run: yarn install - run: yarn install
- save_cache: - save_cache:
key: v10-yarn-packages-{{ checksum "yarn.lock" }} key: v11-yarn-packages-{{ checksum "yarn.lock" }}
paths: paths:
- node_modules - node_modules
- run: yarn lint - run: yarn lint

2
.github/ISSUE_TEMPLATE/feature_request.md

@ -1,6 +1,6 @@
--- ---
name: ✨ Feature Request name: ✨ Feature Request
about: Any feature you find missing in Ledger Live? Discuss to suggest feature requests. about: Any feature you find missing in Ledger Live? Discuss to suggest feature requests. For crypto asset support, please read Issue 1650.
--- ---
- [ ] I have checked this feature was not yet requested. - [ ] I have checked this feature was not yet requested.

82
babel.config.js

@ -1,32 +1,62 @@
const { NODE_ENV } = process.env const { NODE_ENV, CLI } = process.env
const __TEST__ = NODE_ENV === 'test' const __TEST__ = NODE_ENV === 'test'
const __CLI__ = !!CLI
module.exports = () => ({ module.exports = (api) => {
presets: [
[ if (api) {
require('@babel/preset-env'), api.cache(true);
{ }
loose: true,
modules: __TEST__ ? 'commonjs' : false, return {
targets: { presets: [
electron: '1.8', [
node: 'current', require('@babel/preset-env'),
{
loose: true,
modules: __TEST__ || __CLI__ ? 'commonjs' : false,
targets: {
electron: '1.8',
node: 'current',
},
}, },
}, ],
require('@babel/preset-flow'),
require('@babel/preset-react'),
], ],
require('@babel/preset-flow'), plugins: [
require('@babel/preset-react'), [require('babel-plugin-module-resolver'), { root: ['src'] }],
require('@babel/preset-stage-0'), [
], require('babel-plugin-styled-components'),
plugins: [ {
[require('babel-plugin-module-resolver'), { root: ['src'] }], displayName: true,
[ ssr: __TEST__,
require('babel-plugin-styled-components'), },
{ ],
displayName: true, // Stage 0
ssr: __TEST__, "@babel/plugin-proposal-function-bind",
},
// Stage 1
"@babel/plugin-proposal-export-default-from",
"@babel/plugin-proposal-logical-assignment-operators",
["@babel/plugin-proposal-optional-chaining", { "loose": false }],
["@babel/plugin-proposal-pipeline-operator", { "proposal": "minimal" }],
["@babel/plugin-proposal-nullish-coalescing-operator", { "loose": false }],
"@babel/plugin-proposal-do-expressions",
// Stage 2
["@babel/plugin-proposal-decorators", { "legacy": true }],
"@babel/plugin-proposal-function-sent",
"@babel/plugin-proposal-export-namespace-from",
"@babel/plugin-proposal-numeric-separator",
"@babel/plugin-proposal-throw-expressions",
// Stage 3
"@babel/plugin-syntax-dynamic-import",
"@babel/plugin-syntax-import-meta",
["@babel/plugin-proposal-class-properties", { "loose": false }],
"@babel/plugin-proposal-json-strings"
], ],
], }
}) }

48
package.json

@ -13,6 +13,7 @@
"dist": "bash ./scripts/dist.sh", "dist": "bash ./scripts/dist.sh",
"dist:dir": "bash ./scripts/dist-dir.sh", "dist:dir": "bash ./scripts/dist-dir.sh",
"compile": "bash ./scripts/compile.sh", "compile": "bash ./scripts/compile.sh",
"cli": "bash ./scripts/cli/cli.sh",
"lint": "eslint src webpack .storybook test-e2e", "lint": "eslint src webpack .storybook test-e2e",
"flow": "flow", "flow": "flow",
"test": "jest src", "test": "jest src",
@ -34,13 +35,13 @@
} }
}, },
"dependencies": { "dependencies": {
"@ledgerhq/hw-app-btc": "4.21.0", "@ledgerhq/hw-app-btc": "4.24.0",
"@ledgerhq/hw-app-eth": "^4.14.0", "@ledgerhq/hw-app-eth": "^4.24.0",
"@ledgerhq/hw-app-xrp": "^4.13.0", "@ledgerhq/hw-app-xrp": "^4.24.0",
"@ledgerhq/hw-transport": "^4.13.0", "@ledgerhq/hw-transport": "^4.24.0",
"@ledgerhq/hw-transport-node-hid": "4.22.0", "@ledgerhq/hw-transport-node-hid": "4.24.0",
"@ledgerhq/ledger-core": "2.0.0-rc.7", "@ledgerhq/ledger-core": "2.0.0-rc.8",
"@ledgerhq/live-common": "^3.7.1", "@ledgerhq/live-common": "4.0.0-beta.1",
"animated": "^0.2.2", "animated": "^0.2.2",
"async": "^2.6.1", "async": "^2.6.1",
"axios": "^0.18.0", "axios": "^0.18.0",
@ -69,6 +70,7 @@
"measure-scrollbar": "^1.1.0", "measure-scrollbar": "^1.1.0",
"moment": "^2.22.2", "moment": "^2.22.2",
"qrcode": "^1.2.0", "qrcode": "^1.2.0",
"qrloop": "^0.6.1",
"qs": "^6.5.1", "qs": "^6.5.1",
"raven": "^2.5.0", "raven": "^2.5.0",
"raven-js": "^3.24.2", "raven-js": "^3.24.2",
@ -114,12 +116,28 @@
"zxcvbn": "^4.4.2" "zxcvbn": "^4.4.2"
}, },
"devDependencies": { "devDependencies": {
"@babel/core": "7.0.0-beta.42", "@babel/core": "7.1.2",
"@babel/polyfill": "7.0.0-beta.42", "@babel/plugin-proposal-class-properties": "7.1.0",
"@babel/preset-env": "7.0.0-beta.42", "@babel/plugin-proposal-decorators": "7.1.2",
"@babel/preset-flow": "7.0.0-beta.42", "@babel/plugin-proposal-do-expressions": "7.0.0",
"@babel/preset-react": "7.0.0-beta.42", "@babel/plugin-proposal-export-default-from": "7.0.0",
"@babel/preset-stage-0": "7.0.0-beta.42", "@babel/plugin-proposal-export-namespace-from": "7.0.0",
"@babel/plugin-proposal-function-bind": "7.0.0",
"@babel/plugin-proposal-function-sent": "7.1.0",
"@babel/plugin-proposal-json-strings": "7.0.0",
"@babel/plugin-proposal-logical-assignment-operators": "7.0.0",
"@babel/plugin-proposal-nullish-coalescing-operator": "7.0.0",
"@babel/plugin-proposal-numeric-separator": "7.0.0",
"@babel/plugin-proposal-optional-chaining": "7.0.0",
"@babel/plugin-proposal-pipeline-operator": "7.0.0",
"@babel/plugin-proposal-throw-expressions": "7.0.0",
"@babel/plugin-syntax-dynamic-import": "7.0.0",
"@babel/plugin-syntax-import-meta": "7.0.0",
"@babel/polyfill": "7.0.0",
"@babel/preset-env": "7.1.0",
"@babel/preset-flow": "7.0.0",
"@babel/preset-react": "7.0.0",
"@babel/register": "7.0.0",
"@octokit/rest": "^15.10.0", "@octokit/rest": "^15.10.0",
"@storybook/addon-actions": "^3.4.7", "@storybook/addon-actions": "^3.4.7",
"@storybook/addon-knobs": "^3.4.7", "@storybook/addon-knobs": "^3.4.7",
@ -167,5 +185,9 @@
"webpack-cli": "^2.0.14", "webpack-cli": "^2.0.14",
"yaml-loader": "^0.5.0" "yaml-loader": "^0.5.0"
}, },
"engines": {
"node": ">=8.9.0 <=8.12.0",
"yarn": "^1.10.1"
},
"private": true "private": true
} }

50
scripts/check-wordings.js

@ -5,34 +5,34 @@ const { spawn } = require('child_process')
// those wordings are dynamically created, so they are detected // those wordings are dynamically created, so they are detected
// as false positive // as false positive
const WHITELIST = [ const WHITELIST = [
'app:operation.type.IN', 'operation.type.IN',
'app:operation.type.OUT', 'operation.type.OUT',
'app:exchange.coinhouse', 'exchange.coinhouse',
'app:exchange.changelly', 'exchange.changelly',
'app:exchange.coinmama', 'exchange.coinmama',
'app:exchange.simplex', 'exchange.simplex',
'app:exchange.paybis', 'exchange.paybis',
'app:addAccounts.accountToImportSubtitle_plural', 'addAccounts.accountToImportSubtitle_plural',
'app:dashboard.summary_plural', 'dashboard.summary_plural',
'app:addAccounts.success_plural', 'addAccounts.success_plural',
'app:addAccounts.successDescription_plural', 'addAccounts.successDescription_plural',
'app:time.since.day', 'time.since.day',
'app:time.since.week', 'time.since.week',
'app:time.since.month', 'time.since.month',
'app:time.since.year', 'time.since.year',
'app:time.day', 'time.day',
'app:time.week', 'time.week',
'app:time.month', 'time.month',
'app:time.year', 'time.year',
'app:addAccounts.cta.add_plural', 'addAccounts.cta.add_plural',
'app:manager.apps.installing', 'manager.apps.installing',
'app:manager.apps.uninstalling', 'manager.apps.uninstalling',
'app:manager.apps.installSuccess', 'manager.apps.installSuccess',
'app:manager.apps.uninstallSuccess', 'manager.apps.uninstallSuccess',
] ]
const WORDINGS = { const WORDINGS = {
app: require('../static/i18n/en/app.json'), require('../static/i18n/en/app.json'),
onboarding: require('../static/i18n/en/onboarding.json'), onboarding: require('../static/i18n/en/onboarding.json'),
// errors: require('../static/i18n/en/errors.json'), // errors: require('../static/i18n/en/errors.json'),
// language: require('../static/i18n/en/language.json'), // language: require('../static/i18n/en/language.json'),

9
scripts/cli/cli.sh

@ -0,0 +1,9 @@
#!/bin/bash
# TODO: os specific
export LEDGER_DATA_DIR="$HOME/.config/Electron"
export LEDGER_LOGS_DIRECTORY="$LEDGER_DATA_DIR/logs"
export LEDGER_LIVE_SQLITE_PATH="$LEDGER_DATA_DIR/sqlite"
export CLI=1
node -r @babel/register scripts/cli/txBetweenAccounts.js

21
scripts/cli/getDevice.js

@ -0,0 +1,21 @@
import CommNodeHid from '@ledgerhq/hw-transport-node-hid'
export default function getDevice() {
return new Promise((resolve, reject) => {
const sub = CommNodeHid.listen({
error: err => {
sub.unsubscribe()
reject(err)
},
next: async e => {
if (!e.device) {
return
}
if (e.type === 'add') {
sub.unsubscribe()
resolve(e.device)
}
},
})
})
}

106
scripts/cli/txBetweenAccounts.js

@ -0,0 +1,106 @@
/* eslint-disable no-console */
import chalk from 'chalk'
import path from 'path'
import fs from 'fs'
import inquirer from 'inquirer'
import { createAccountModel } from '@ledgerhq/live-common/lib/models/account'
import { formatCurrencyUnit } from '@ledgerhq/live-common/lib/helpers/currencies'
import 'globals'
import withLibcore from 'helpers/withLibcore'
import { doSignAndBroadcast } from 'commands/libcoreSignAndBroadcast'
import getDevice from './getDevice'
const accountModel = createAccountModel()
async function main() {
try {
// GET ACCOUNTS
const app = await parseAppFile()
const accounts = app.accounts.map(accountModel.decode)
// GET SENDER ACCOUNT
const senderAccount = await chooseAccount(accounts, 'Choose sender account')
// GET RECIPIENT ACCOUNT
const recipientAccount = await chooseAccount(accounts, 'Choose recipient account')
// GET AMOUNT & FEE
const { amount, feePerByte } = await inquirer.prompt([
{
type: 'input',
name: 'amount',
message: 'Amount',
default: 0,
},
{
type: 'input',
name: 'feePerByte',
message: 'Fee per byte',
default: 0,
},
])
// GET DEVICE
console.log(chalk.blue(`Waiting for device...`))
const device = await getDevice()
console.log(chalk.blue(`Using device with path [${device.path}]`))
await withLibcore(async core =>
doSignAndBroadcast({
accountId: senderAccount.id,
currencyId: senderAccount.currency.id,
xpub: senderAccount.xpub,
freshAddress: senderAccount.freshAddress,
freshAddressPath: senderAccount.freshAddressPath,
index: senderAccount.index,
transaction: {
amount,
feePerByte,
recipient: recipientAccount.freshAddress,
},
deviceId: device.path,
core,
isCancelled: () => false,
onSigned: () => {
console.log(`>> signed`)
},
onOperationBroadcasted: operation => {
console.log(`>> broadcasted`, operation)
},
}),
)
} catch (err) {
console.log(`[ERROR]`, err)
}
}
async function parseAppFile() {
const appFilePath = path.resolve(process.env.LEDGER_DATA_DIR, 'app.json')
const appFileContent = fs.readFileSync(appFilePath, 'utf-8')
const parsedApp = JSON.parse(appFileContent)
return parsedApp.data
}
async function chooseAccount(accounts, msg) {
const { account } = await inquirer.prompt([
{
type: 'list',
choices: accounts.map(account => ({
name: `${account.name} | ${chalk.green(
formatCurrencyUnit(account.unit, account.balance, {
showCode: true,
}),
)}`,
value: account,
})),
name: 'account',
message: msg,
},
])
return account
}
main()

49
scripts/patch-appimage.sh

@ -0,0 +1,49 @@
#!/bin/bash
# Patch .AppImage to address libcore crash on some
# distributions, due to loading system libraries
# instead of embedded ones.
#
# see https://github.com/LedgerHQ/ledger-live-desktop/issues/1010
set -e
BASE_URL=http://mirrors.kernel.org/ubuntu/pool/main/k/krb5
PACKAGE_SUFFIX=-2build1_amd64.deb
TMP_DIR=$(mktemp -d)
LEDGER_LIVE_VERSION=$(grep version package.json | sed -E 's/.*: "(.*)",/\1/g')
cp "dist/ledger-live-desktop-$LEDGER_LIVE_VERSION-linux-x86_64.AppImage" "$TMP_DIR"
pushd "$TMP_DIR"
declare -a LIBRARIES=(
"libgssapi-krb5-2_1.16"
"libk5crypto3_1.16"
"libkrb5-3_1.16"
"libkrb5support0_1.16"
)
for PACKAGE in "${LIBRARIES[@]}"; do
curl -fOL "$BASE_URL/$PACKAGE$PACKAGE_SUFFIX"
ar p "$PACKAGE$PACKAGE_SUFFIX" data.tar.xz | tar xvJf >/dev/null - ./usr/lib/x86_64-linux-gnu/
rm "$PACKAGE$PACKAGE_SUFFIX"
done
curl -fOL "https://s3-eu-west-1.amazonaws.com/ledger-ledgerlive-resources-dev/public_resources/appimagetool-x86_64.AppImage"
cp "$OLDPWD/scripts/shasums/patch-appimage-sums.txt" .
sha512sum --quiet --check patch-appimage-sums.txt || exit 1
./ledger-live-desktop-"$LEDGER_LIVE_VERSION"-linux-x86_64.AppImage --appimage-extract
cp -a usr/lib/x86_64-linux-gnu/*.so.* squashfs-root/usr/lib
chmod +x appimagetool-x86_64.AppImage
./appimagetool-x86_64.AppImage squashfs-root "$OLDPWD/dist/ledger-live-desktop-$LEDGER_LIVE_VERSION-linux-x86_64.AppImage"
popd
MD5_SUM=$(sha512sum "dist/ledger-live-desktop-$LEDGER_LIVE_VERSION-linux-x86_64.AppImage" | cut -f1 -d\ | xxd -r -p | base64 | paste -sd "")
sed -i "s|sha512: .*|sha512: ${MD5_SUM}|g" dist/latest-linux.yml
SIZE=$(stat --printf="%s" "dist/ledger-live-desktop-$LEDGER_LIVE_VERSION-linux-x86_64.AppImage")
sed -i "s|size: .*|size: ${SIZE}|g" dist/latest-linux.yml

50
scripts/release.sh

@ -13,7 +13,9 @@ if [ "$(git rev-parse --abbrev-ref HEAD)" != "master" ]; then
exit 0 exit 0
fi fi
if ! git describe --exact-match --tags 2>/dev/null >/dev/null; then GH_TAG=$(git describe --exact-match --tags 2>/dev/null || echo '')
if [[ $GH_TAG == "" ]]; then
echo "You are not on a tag. Exiting properly. (CI)" echo "You are not on a tag. Exiting properly. (CI)"
exit 0 exit 0
fi fi
@ -57,7 +59,6 @@ fi
# exit 1 # exit 1
# fi # fi
if [[ $(uname) == 'Linux' ]]; then # only run it on one target, to prevent race conditions if [[ $(uname) == 'Linux' ]]; then # only run it on one target, to prevent race conditions
runJob \ runJob \
"node scripts/create-draft-release.js" \ "node scripts/create-draft-release.js" \
@ -68,9 +69,48 @@ fi
runJob "yarn compile" "compiling..." "compiled" "failed to compile" "verbose" runJob "yarn compile" "compiling..." "compiled" "failed to compile" "verbose"
# --------------------------------------------------------------------
# Linux: Internal process error (null)
#
# context: https://github.com/LedgerHQ/ledger-live-desktop/issues/1010
# Linux: Internal process error (null)
#
# The "fix" is not optimal, as it doesn't really solve the problem
# (electron loading system openssl before we can load our embedded one)
# Quick summary:
#
# - build without publishing
# - unpack the .AppImage
# - download reported working libs from ubuntu mirrors, put it inside
# - re-pack the .AppImage
# - checksum stuff
# - upload to gh
runJob \ runJob \
"DEBUG=electron-builder electron-builder build --publish always" \ "DEBUG=electron-builder electron-builder build --publish never" \
"building, packaging and publishing app..." \ "building and packaging app..." \
"app built, packaged and published successfully" \ "app built and packaged successfully" \
"failed to build app" \ "failed to build app" \
"verbose" "verbose"
runJob \
"scripts/patch-appimage.sh" \
"patching AppImage..." \
"AppImage patched successfully" \
"failed to patch AppImage"
LEDGER_LIVE_VERSION=$(grep version package.json | sed -E 's/.*: "(.*)",/\1/g')
scripts/upload-github-release-asset.sh \
github_api_token="$GH_TOKEN" \
owner=LedgerHQ \
repo=ledger-live-desktop \
tag="$GH_TAG" \
filename="dist/ledger-live-desktop-$LEDGER_LIVE_VERSION-linux-x86_64.AppImage"
scripts/upload-github-release-asset.sh \
github_api_token="$GH_TOKEN" \
owner=LedgerHQ \
repo=ledger-live-desktop \
tag="$GH_TAG" \
filename="dist/latest-linux.yml"

5
scripts/shasums/patch-appimage-sums.txt

@ -0,0 +1,5 @@
bebb42401a43971cfe3e31f2c9ee4efee352ce0d29a8ccc95ca1356a58463afd4876b133d9f4295697f96b76eb21b50c1909a073db753569e8969065eb40b306 usr/lib/x86_64-linux-gnu/libgssapi_krb5.so.2.2
d7d2b38a46d65a06560241b226f61d81c4df28d56c6841dd34bb428802ace0fc80cf94de1e5117f0b85b2c69b550df61ac999184d5cfe8ecd3bea4d8394d1d21 usr/lib/x86_64-linux-gnu/libkrb5.so.3.3
b025b755eb9a64f0d03a8e92c9e4b4f95c2c506bf070cf037841ef8cdb9013e16390d0e17330f2ce8c98c3b1f05b917a3018109acfde7aab50bc9d9fa70ea12b usr/lib/x86_64-linux-gnu/libk5crypto.so.3.1
f181e41f306819c10054ff8ceebf4943858f2cd34dea5206b51141877e2f651be3c6435bb02538cbde2cc0415f38e476423a9fd6a428ca9d425e9c662483b9af usr/lib/x86_64-linux-gnu/libkrb5support.so.0.1
dd8d81d4c1485209a65a1446225428a1b919478a74fd5698aff64cb8a67992544e62455f849ad73392505707cb94739de00af5ab340a22a87bb752c3808a55d2 appimagetool-x86_64.AppImage

61
scripts/upload-github-release-asset.sh

@ -0,0 +1,61 @@
#!/usr/bin/env bash
#
# Author: Stefan Buck
# License: MIT
# https://gist.github.com/stefanbuck/ce788fee19ab6eb0b4447a85fc99f447
#
#
# This script accepts the following parameters:
#
# * owner
# * repo
# * filename
# * github_api_token
#
# Script to upload a release asset using the GitHub API v3.
#
# Example:
#
# upload-github-release-asset.sh github_api_token=TOKEN owner=stefanbuck repo=playground filename=./build.zip
#
# Check dependencies.
set -e
# Validate settings.
[ "$TRACE" ] && set -x
# shellcheck disable=SC2124
CONFIG=$@
for line in $CONFIG; do
eval "$line"
done
# Define variables.
GH_API="https://api.github.com"
# shellcheck disable=SC2154
GH_REPO="$GH_API/repos/$owner/$repo"
# shellcheck disable=SC2154
AUTH="Authorization: token $github_api_token"
# github_api_token=$GH_TOKEN owner=LedgerHQ repo=ledger-live-desktop tag=v1.2.2 filename=./dist/electron-builder-debug.yml
LATEST_RELEASE_ID=$(curl -sH "$AUTH" "$GH_API/repos/LedgerHQ/ledger-live-desktop/releases" | grep '"id":' | head -n 1 | sed -E 's/.*: (.*),/\1/')
# Validate token.
curl -o /dev/null -sH "$AUTH" "$GH_REPO" || { echo "Error: Invalid repo, token or network issue!"; exit 1; }
# Get ID of the asset based on given filename.
# shellcheck disable=SC2154
[ "$LATEST_RELEASE_ID" ] || { echo "Error: Failed to get release id"; exit 1; }
# Upload asset
echo "Uploading asset... "
# Construct url
# shellcheck disable=SC2154
GH_ASSET="https://uploads.github.com/repos/$owner/$repo/releases/$LATEST_RELEASE_ID/assets?name=$(basename "$filename")"
curl --data-binary @"$filename" -H "Authorization: token $github_api_token" -H "Content-Type: application/octet-stream" "$GH_ASSET"

2
src/__mocks__/storybook-state.js

@ -2,7 +2,7 @@ import { genStoreState } from '@ledgerhq/live-common/lib/countervalues/mock'
import { import {
getCryptoCurrencyById, getCryptoCurrencyById,
getFiatCurrencyByTicker, getFiatCurrencyByTicker,
} from '@ledgerhq/live-common/lib/helpers/currencies' } from '@ledgerhq/live-common/lib/currencies'
export default { export default {
countervalues: genStoreState([ countervalues: genStoreState([

2
src/actions/general.js

@ -8,7 +8,7 @@ import {
getOrderAccounts, getOrderAccounts,
} from 'reducers/settings' } from 'reducers/settings'
import { accountsSelector } from 'reducers/accounts' import { accountsSelector } from 'reducers/accounts'
import { sortAccounts } from 'helpers/accountOrdering' import { sortAccounts } from '@ledgerhq/live-common/lib/account'
const accountsBtcBalanceSelector = createSelector( const accountsBtcBalanceSelector = createSelector(
accountsSelector, accountsSelector,

2
src/api/Ripple.js

@ -6,7 +6,7 @@ import {
parseCurrencyUnit, parseCurrencyUnit,
getCryptoCurrencyById, getCryptoCurrencyById,
formatCurrencyUnit, formatCurrencyUnit,
} from '@ledgerhq/live-common/lib/helpers/currencies' } from '@ledgerhq/live-common/lib/currencies'
const rippleUnit = getCryptoCurrencyById('ripple').units[0] const rippleUnit = getCryptoCurrencyById('ripple').units[0]

46
src/bridge/EthereumJSBridge.js

@ -8,14 +8,22 @@ import AdvancedOptions from 'components/AdvancedOptions/EthereumKind'
import throttle from 'lodash/throttle' import throttle from 'lodash/throttle'
import flatMap from 'lodash/flatMap' import flatMap from 'lodash/flatMap'
import uniqBy from 'lodash/uniqBy' import uniqBy from 'lodash/uniqBy'
import {
getDerivationModesForCurrency,
getDerivationScheme,
runDerivationScheme,
getMandatoryEmptyAccountSkip,
} from '@ledgerhq/live-common/lib/derivation'
import {
getAccountPlaceholderName,
getNewAccountPlaceholderName,
} from '@ledgerhq/live-common/lib/account'
import type { Account, Operation } from '@ledgerhq/live-common/lib/types' import type { Account, Operation } from '@ledgerhq/live-common/lib/types'
import eip55 from 'eip55' import eip55 from 'eip55'
import { apiForCurrency } from 'api/Ethereum' import { apiForCurrency } from 'api/Ethereum'
import type { Tx } from 'api/Ethereum' import type { Tx } from 'api/Ethereum'
import { getDerivations } from 'helpers/derivations'
import getAddressCommand from 'commands/getAddress' import getAddressCommand from 'commands/getAddress'
import signTransactionCommand from 'commands/signTransaction' import signTransactionCommand from 'commands/signTransaction'
import { getAccountPlaceholderName, getNewAccountPlaceholderName } from 'helpers/accountName'
import { NotEnoughBalance, FeeNotLoaded, ETHAddressNonEIP } from 'config/errors' import { NotEnoughBalance, FeeNotLoaded, ETHAddressNonEIP } from 'config/errors'
import type { EditProps, WalletBridge } from './types' import type { EditProps, WalletBridge } from './types'
@ -210,8 +218,8 @@ const EthereumBridge: WalletBridge<Transaction> = {
async function stepAddress( async function stepAddress(
index, index,
{ address, path: freshAddressPath, publicKey }, { address, path: freshAddressPath },
isStandard, derivationMode,
shouldSkipEmpty, shouldSkipEmpty,
): { account?: Account, complete?: boolean } { ): { account?: Account, complete?: boolean } {
const balance = await api.getAccountBalance(address) const balance = await api.getAccountBalance(address)
@ -222,19 +230,21 @@ const EthereumBridge: WalletBridge<Transaction> = {
if (finished) return { complete: true } if (finished) return { complete: true }
const freshAddress = address const freshAddress = address
const accountId = `ethereumjs:${currency.id}:${address}:${publicKey}` const accountId = `ethereumjs:2:${currency.id}:${address}:${derivationMode}`
if (txs.length === 0 && balance.isZero()) { if (txs.length === 0 && balance.isZero()) {
// this is an empty account // this is an empty account
if (isStandard) { if (derivationMode === '') {
// is standard derivation
if (newAccountCount === 0) { if (newAccountCount === 0) {
// first zero account will emit one account as opportunity to create a new account.. // first zero account will emit one account as opportunity to create a new account..
const account: $Exact<Account> = { const account: $Exact<Account> = {
id: accountId, id: accountId,
xpub: '', seedIdentifier: freshAddress,
freshAddress, freshAddress,
freshAddressPath, freshAddressPath,
name: getNewAccountPlaceholderName(currency, index), derivationMode,
name: getNewAccountPlaceholderName({ currency, index, derivationMode }),
balance, balance,
blockHeight: currentBlock.height, blockHeight: currentBlock.height,
index, index,
@ -258,10 +268,11 @@ const EthereumBridge: WalletBridge<Transaction> = {
const account: $Exact<Account> = { const account: $Exact<Account> = {
id: accountId, id: accountId,
xpub: '', seedIdentifier: freshAddress,
freshAddress, freshAddress,
freshAddressPath, freshAddressPath,
name: getAccountPlaceholderName(currency, index, !isStandard), derivationMode,
name: getAccountPlaceholderName({ currency, index, derivationMode }),
balance, balance,
blockHeight: currentBlock.height, blockHeight: currentBlock.height,
index, index,
@ -288,21 +299,22 @@ const EthereumBridge: WalletBridge<Transaction> = {
async function main() { async function main() {
try { try {
const derivations = getDerivations(currency) const derivationModes = getDerivationModesForCurrency(currency)
const last = derivations[derivations.length - 1] for (const derivationMode of derivationModes) {
for (const derivation of derivations) {
const isStandard = last === derivation
let emptyCount = 0 let emptyCount = 0
const mandatoryEmptyAccountSkip = derivation.mandatoryEmptyAccountSkip || 0 const mandatoryEmptyAccountSkip = getMandatoryEmptyAccountSkip(derivationMode)
const derivationScheme = getDerivationScheme({ derivationMode, currency })
for (let index = 0; index < 255; index++) { for (let index = 0; index < 255; index++) {
const freshAddressPath = derivation({ currency, x: index, segwit: false }) const freshAddressPath = runDerivationScheme(derivationScheme, currency, {
account: index,
})
const res = await getAddressCommand const res = await getAddressCommand
.send({ currencyId: currency.id, devicePath: deviceId, path: freshAddressPath }) .send({ currencyId: currency.id, devicePath: deviceId, path: freshAddressPath })
.toPromise() .toPromise()
const r = await stepAddress( const r = await stepAddress(
index, index,
res, res,
isStandard, derivationMode,
emptyCount < mandatoryEmptyAccountSkip, emptyCount < mandatoryEmptyAccountSkip,
) )
logger.log( logger.log(

10
src/bridge/LibcoreBridge.js

@ -129,7 +129,9 @@ const LibcoreBridge: WalletBridge<Transaction> = {
libcoreSyncAccount libcoreSyncAccount
.send({ .send({
accountId: account.id, accountId: account.id,
freshAddressPath: account.freshAddressPath, derivationMode: account.derivationMode,
xpub: account.xpub || '',
seedIdentifier: account.seedIdentifier,
index: account.index, index: account.index,
currencyId: account.currency.id, currencyId: account.currency.id,
}) })
@ -218,9 +220,9 @@ const LibcoreBridge: WalletBridge<Transaction> = {
.send({ .send({
accountId: account.id, accountId: account.id,
currencyId: account.currency.id, currencyId: account.currency.id,
xpub: account.xpub, xpub: account.xpub || '', // FIXME only reason is to build the op id. we need to consider another id for making op id.
freshAddress: account.freshAddress, derivationMode: account.derivationMode,
freshAddressPath: account.freshAddressPath, seedIdentifier: account.seedIdentifier,
index: account.index, index: account.index,
transaction: serializeTransaction(transaction), transaction: serializeTransaction(transaction),
deviceId, deviceId,

39
src/bridge/RippleJSBridge.js

@ -7,7 +7,15 @@ import bs58check from 'ripple-bs58check'
import { computeBinaryTransactionHash } from 'ripple-hashes' import { computeBinaryTransactionHash } from 'ripple-hashes'
import throttle from 'lodash/throttle' import throttle from 'lodash/throttle'
import type { Account, Operation } from '@ledgerhq/live-common/lib/types' import type { Account, Operation } from '@ledgerhq/live-common/lib/types'
import { getDerivations } from 'helpers/derivations' import {
getDerivationModesForCurrency,
getDerivationScheme,
runDerivationScheme,
} from '@ledgerhq/live-common/lib/derivation'
import {
getAccountPlaceholderName,
getNewAccountPlaceholderName,
} from '@ledgerhq/live-common/lib/account'
import getAddress from 'commands/getAddress' import getAddress from 'commands/getAddress'
import signTransaction from 'commands/signTransaction' import signTransaction from 'commands/signTransaction'
import { import {
@ -19,7 +27,6 @@ import {
} from 'api/Ripple' } from 'api/Ripple'
import FeesRippleKind from 'components/FeesField/RippleKind' import FeesRippleKind from 'components/FeesField/RippleKind'
import AdvancedOptionsRippleKind from 'components/AdvancedOptions/RippleKind' import AdvancedOptionsRippleKind from 'components/AdvancedOptions/RippleKind'
import { getAccountPlaceholderName, getNewAccountPlaceholderName } from 'helpers/accountName'
import { import {
NotEnoughBalance, NotEnoughBalance,
FeeNotLoaded, FeeNotLoaded,
@ -211,7 +218,7 @@ const txToOperation = (account: Account) => ({
} }
const op: $Exact<Operation> = { const op: $Exact<Operation> = {
id, id: `${account.id}-${id}-${type}`,
hash: id, hash: id,
accountId: account.id, accountId: account.id,
type, type,
@ -289,17 +296,19 @@ const RippleJSBridge: WalletBridge<Transaction> = {
const minLedgerVersion = Number(ledgers[0]) const minLedgerVersion = Number(ledgers[0])
const maxLedgerVersion = Number(ledgers[1]) const maxLedgerVersion = Number(ledgers[1])
const derivations = getDerivations(currency) const derivationModes = getDerivationModesForCurrency(currency)
for (const derivation of derivations) { for (const derivationMode of derivationModes) {
const legacy = derivation !== derivations[derivations.length - 1] const derivationScheme = getDerivationScheme({ derivationMode, currency })
for (let index = 0; index < 255; index++) { for (let index = 0; index < 255; index++) {
const freshAddressPath = derivation({ currency, x: index, segwit: false }) const freshAddressPath = runDerivationScheme(derivationScheme, currency, {
const { address, publicKey } = await await getAddress account: index,
})
const { address } = await await getAddress
.send({ currencyId: currency.id, devicePath: deviceId, path: freshAddressPath }) .send({ currencyId: currency.id, devicePath: deviceId, path: freshAddressPath })
.toPromise() .toPromise()
if (finished) return if (finished) return
const accountId = `ripplejs:${currency.id}:${address}:${publicKey}` const accountId = `ripplejs:2:${currency.id}:${address}:${derivationMode}`
let info let info
try { try {
@ -316,11 +325,12 @@ const RippleJSBridge: WalletBridge<Transaction> = {
if (!info) { if (!info) {
// account does not exist in Ripple server // account does not exist in Ripple server
// we are generating a new account locally // we are generating a new account locally
if (!legacy) { if (derivationMode === '') {
o.next({ o.next({
id: accountId, id: accountId,
xpub: '', seedIdentifier: freshAddress,
name: getNewAccountPlaceholderName(currency, index), derivationMode,
name: getNewAccountPlaceholderName({ currency, index, derivationMode }),
freshAddress, freshAddress,
freshAddressPath, freshAddressPath,
balance: BigNumber(0), balance: BigNumber(0),
@ -353,8 +363,9 @@ const RippleJSBridge: WalletBridge<Transaction> = {
const account: $Exact<Account> = { const account: $Exact<Account> = {
id: accountId, id: accountId,
xpub: '', seedIdentifier: freshAddress,
name: getAccountPlaceholderName(currency, index, legacy), derivationMode,
name: getAccountPlaceholderName({ currency, index, derivationMode }),
freshAddress, freshAddress,
freshAddressPath, freshAddressPath,
balance, balance,

4
src/bridge/makeMockBridge.js

@ -6,11 +6,11 @@ import {
genAddingOperationsInAccount, genAddingOperationsInAccount,
genOperation, genOperation,
} from '@ledgerhq/live-common/lib/mock/account' } from '@ledgerhq/live-common/lib/mock/account'
import { getOperationAmountNumber } from '@ledgerhq/live-common/lib/helpers/operation' import { getOperationAmountNumber } from '@ledgerhq/live-common/lib/operation'
import Prando from 'prando' import Prando from 'prando'
import { BigNumber } from 'bignumber.js' import { BigNumber } from 'bignumber.js'
import type { Operation } from '@ledgerhq/live-common/lib/types' import type { Operation } from '@ledgerhq/live-common/lib/types'
import { validateNameEdition } from 'helpers/accountName' import { validateNameEdition } from '@ledgerhq/live-common/lib/account'
import { MOCK_DATA_SEED } from 'config/constants' import { MOCK_DATA_SEED } from 'config/constants'
import type { WalletBridge } from './types' import type { WalletBridge } from './types'

2
src/commands/debugAppInfosForCurrency.js

@ -1,6 +1,6 @@
// @flow // @flow
import { getCryptoCurrencyById } from '@ledgerhq/live-common/lib/helpers/currencies' import { getCryptoCurrencyById } from '@ledgerhq/live-common/lib/currencies'
import { createCommand, Command } from 'helpers/ipc' import { createCommand, Command } from 'helpers/ipc'
import { fromPromise } from 'rxjs/observable/fromPromise' import { fromPromise } from 'rxjs/observable/fromPromise'
import { withDevice } from 'helpers/deviceAccess' import { withDevice } from 'helpers/deviceAccess'

2
src/commands/getAddress.js

@ -1,6 +1,6 @@
// @flow // @flow
import { getCryptoCurrencyById } from '@ledgerhq/live-common/lib/helpers/currencies' import { getCryptoCurrencyById } from '@ledgerhq/live-common/lib/currencies'
import { createCommand, Command } from 'helpers/ipc' import { createCommand, Command } from 'helpers/ipc'
import { fromPromise } from 'rxjs/observable/fromPromise' import { fromPromise } from 'rxjs/observable/fromPromise'
import { withDevice } from 'helpers/deviceAccess' import { withDevice } from 'helpers/deviceAccess'

3
src/commands/index.js

@ -69,8 +69,11 @@ const all: Array<Command<any, any>> = [
uninstallApp, uninstallApp,
] ]
export const commandsById = {}
all.forEach(cmd => { all.forEach(cmd => {
invariant(!all.some(c => c !== cmd && c.id === cmd.id), `duplicate command '${cmd.id}'`) invariant(!all.some(c => c !== cmd && c.id === cmd.id), `duplicate command '${cmd.id}'`)
commandsById[cmd.id] = cmd
}) })
export default all export default all

29
src/commands/libcoreGetFees.js

@ -4,17 +4,16 @@ import { Observable } from 'rxjs'
import { BigNumber } from 'bignumber.js' import { BigNumber } from 'bignumber.js'
import withLibcore from 'helpers/withLibcore' import withLibcore from 'helpers/withLibcore'
import { createCommand, Command } from 'helpers/ipc' import { createCommand, Command } from 'helpers/ipc'
import { getCryptoCurrencyById } from '@ledgerhq/live-common/lib/currencies'
import { getWalletName } from '@ledgerhq/live-common/lib/account'
import type { Account } from '@ledgerhq/live-common/lib/types' import type { Account } from '@ledgerhq/live-common/lib/types'
import * as accountIdHelper from 'helpers/accountId'
import { import {
isValidAddress, isValidAddress,
libcoreAmountToBigNumber, libcoreAmountToBigNumber,
bigNumberToLibcoreAmount, bigNumberToLibcoreAmount,
getOrCreateWallet, getOrCreateWallet,
} from 'helpers/libcore' } from 'helpers/libcore'
import { isSegwitPath, isUnsplitPath } from 'helpers/bip32'
import { InvalidAddress } from 'config/errors' import { InvalidAddress } from 'config/errors'
import { splittedCurrencies } from 'config/cryptocurrencies'
type BitcoinLikeTransaction = { type BitcoinLikeTransaction = {
// TODO we rename this Transaction concept into transactionInput // TODO we rename this Transaction concept into transactionInput
@ -24,22 +23,20 @@ type BitcoinLikeTransaction = {
} }
type Input = { type Input = {
accountId: string,
accountIndex: number, accountIndex: number,
transaction: BitcoinLikeTransaction, transaction: BitcoinLikeTransaction,
currencyId: string, currencyId: string,
isSegwit: boolean, derivationMode: string,
isUnsplit: boolean, seedIdentifier: string,
} }
export const extractGetFeesInputFromAccount = (a: Account) => { export const extractGetFeesInputFromAccount = (a: Account) => {
const currencyId = a.currency.id const currencyId = a.currency.id
return { return {
accountId: a.id,
accountIndex: a.index, accountIndex: a.index,
currencyId, currencyId,
isSegwit: isSegwitPath(a.freshAddressPath), derivationMode: a.derivationMode,
isUnsplit: isUnsplitPath(a.freshAddressPath, splittedCurrencies[currencyId]), seedIdentifier: a.seedIdentifier,
} }
} }
@ -47,17 +44,21 @@ type Result = { totalFees: string }
const cmd: Command<Input, Result> = createCommand( const cmd: Command<Input, Result> = createCommand(
'libcoreGetFees', 'libcoreGetFees',
({ accountId, currencyId, isSegwit, isUnsplit, accountIndex, transaction }) => ({ currencyId, derivationMode, seedIdentifier, accountIndex, transaction }) =>
Observable.create(o => { Observable.create(o => {
let unsubscribed = false let unsubscribed = false
const isCancelled = () => unsubscribed const isCancelled = () => unsubscribed
const currency = getCryptoCurrencyById(currencyId)
withLibcore(async core => { withLibcore(async core => {
const { walletName } = accountIdHelper.decode(accountId) const walletName = getWalletName({
currency,
derivationMode,
seedIdentifier,
})
const njsWallet = await getOrCreateWallet(core, walletName, { const njsWallet = await getOrCreateWallet(core, walletName, {
currencyId, currency,
isSegwit, derivationMode,
isUnsplit,
}) })
if (isCancelled()) return if (isCancelled()) return
const njsAccount = await njsWallet.getAccount(accountIndex) const njsAccount = await njsWallet.getAccount(accountIndex)

8
src/commands/libcoreScanFromXPUB.js

@ -10,18 +10,18 @@ import { scanAccountsFromXPUB } from 'helpers/libcore'
type Input = { type Input = {
currencyId: string, currencyId: string,
xpub: string, xpub: string,
isSegwit: boolean, derivationMode: string,
isUnsplit: boolean, seedIdentifier: string,
} }
type Result = AccountRaw type Result = AccountRaw
const cmd: Command<Input, Result> = createCommand( const cmd: Command<Input, Result> = createCommand(
'libcoreScanFromXPUB', 'libcoreScanFromXPUB',
({ currencyId, xpub, isSegwit, isUnsplit }) => ({ currencyId, xpub, derivationMode, seedIdentifier }) =>
fromPromise( fromPromise(
withLibcore(async core => withLibcore(async core =>
scanAccountsFromXPUB({ core, currencyId, xpub, isSegwit, isUnsplit }), scanAccountsFromXPUB({ core, currencyId, xpub, derivationMode, seedIdentifier }),
), ),
), ),
) )

66
src/commands/libcoreSignAndBroadcast.js

@ -2,22 +2,21 @@
import logger from 'logger' import logger from 'logger'
import { BigNumber } from 'bignumber.js' import { BigNumber } from 'bignumber.js'
import type { OperationRaw } from '@ledgerhq/live-common/lib/types'
import Btc from '@ledgerhq/hw-app-btc' import Btc from '@ledgerhq/hw-app-btc'
import { Observable } from 'rxjs' import { Observable } from 'rxjs'
import { getCryptoCurrencyById } from '@ledgerhq/live-common/lib/helpers/currencies' import { isSegwitDerivationMode } from '@ledgerhq/live-common/lib/derivation'
import { isSegwitPath, isUnsplitPath } from 'helpers/bip32' import { getCryptoCurrencyById } from '@ledgerhq/live-common/lib/currencies'
import type { OperationRaw, CryptoCurrency } from '@ledgerhq/live-common/lib/types'
import { getWalletName } from '@ledgerhq/live-common/lib/account'
import { import {
libcoreAmountToBigNumber, libcoreAmountToBigNumber,
bigNumberToLibcoreAmount, bigNumberToLibcoreAmount,
getOrCreateWallet, getOrCreateWallet,
} from 'helpers/libcore' } from 'helpers/libcore'
import { splittedCurrencies } from 'config/cryptocurrencies'
import withLibcore from 'helpers/withLibcore' import withLibcore from 'helpers/withLibcore'
import { createCommand, Command } from 'helpers/ipc' import { createCommand, Command } from 'helpers/ipc'
import { withDevice } from 'helpers/deviceAccess' import { withDevice } from 'helpers/deviceAccess'
import * as accountIdHelper from 'helpers/accountId'
type BitcoinLikeTransaction = { type BitcoinLikeTransaction = {
amount: string, amount: string,
@ -28,9 +27,9 @@ type BitcoinLikeTransaction = {
type Input = { type Input = {
accountId: string, accountId: string,
currencyId: string, currencyId: string,
derivationMode: string,
seedIdentifier: string,
xpub: string, xpub: string,
freshAddress: string,
freshAddressPath: string,
index: number, index: number,
transaction: BitcoinLikeTransaction, transaction: BitcoinLikeTransaction,
deviceId: string, deviceId: string,
@ -42,17 +41,18 @@ type Result = { type: 'signed' } | { type: 'broadcasted', operation: OperationRa
const cmd: Command<Input, Result> = createCommand( const cmd: Command<Input, Result> = createCommand(
'libcoreSignAndBroadcast', 'libcoreSignAndBroadcast',
({ accountId, currencyId, xpub, freshAddress, freshAddressPath, index, transaction, deviceId }) => ({ accountId, currencyId, derivationMode, seedIdentifier, xpub, index, transaction, deviceId }) =>
Observable.create(o => { Observable.create(o => {
let unsubscribed = false let unsubscribed = false
const currency = getCryptoCurrencyById(currencyId)
const isCancelled = () => unsubscribed const isCancelled = () => unsubscribed
withLibcore(core => withLibcore(core =>
doSignAndBroadcast({ doSignAndBroadcast({
accountId, accountId,
currencyId, currency,
derivationMode,
seedIdentifier,
xpub, xpub,
freshAddress,
freshAddressPath,
index, index,
transaction, transaction,
deviceId, deviceId,
@ -78,28 +78,26 @@ const cmd: Command<Input, Result> = createCommand(
async function signTransaction({ async function signTransaction({
hwApp, hwApp,
currencyId, currency,
transaction, transaction,
derivationMode,
sigHashType, sigHashType,
supportsSegwit,
isSegwit,
hasTimestamp, hasTimestamp,
}: { }: {
hwApp: Btc, hwApp: Btc,
currencyId: string, currency: CryptoCurrency,
transaction: *, transaction: *,
derivationMode: string,
sigHashType: number, sigHashType: number,
supportsSegwit: boolean,
isSegwit: boolean,
hasTimestamp: boolean, hasTimestamp: boolean,
}) { }) {
const additionals = [] const additionals = []
let expiryHeight let expiryHeight
if (currencyId === 'bitcoin_cash' || currencyId === 'bitcoin_gold') additionals.push('bip143') if (currency.id === 'bitcoin_cash' || currency.id === 'bitcoin_gold') additionals.push('bip143')
if (currencyId === 'zcash') expiryHeight = Buffer.from([0x00, 0x00, 0x00, 0x00]) if (currency.id === 'zcash') expiryHeight = Buffer.from([0x00, 0x00, 0x00, 0x00])
const rawInputs = transaction.getInputs() const rawInputs = transaction.getInputs()
const hasExtraData = currencyId === 'zcash' const hasExtraData = currency.id === 'zcash'
const inputs = await Promise.all( const inputs = await Promise.all(
rawInputs.map(async input => { rawInputs.map(async input => {
@ -107,7 +105,7 @@ async function signTransaction({
const hexPreviousTransaction = Buffer.from(rawPreviousTransaction).toString('hex') const hexPreviousTransaction = Buffer.from(rawPreviousTransaction).toString('hex')
const previousTransaction = hwApp.splitTransaction( const previousTransaction = hwApp.splitTransaction(
hexPreviousTransaction, hexPreviousTransaction,
supportsSegwit, currency.supportsSegwit,
hasTimestamp, hasTimestamp,
hasExtraData, hasExtraData,
) )
@ -156,7 +154,7 @@ async function signTransaction({
outputScriptHex, outputScriptHex,
lockTime, lockTime,
sigHashType, sigHashType,
isSegwit, isSegwitDerivationMode(derivationMode),
initialTimestamp, initialTimestamp,
additionals, additionals,
expiryHeight, expiryHeight,
@ -167,9 +165,10 @@ async function signTransaction({
export async function doSignAndBroadcast({ export async function doSignAndBroadcast({
accountId, accountId,
currencyId, derivationMode,
seedIdentifier,
currency,
xpub, xpub,
freshAddressPath,
index, index,
transaction, transaction,
deviceId, deviceId,
@ -179,10 +178,10 @@ export async function doSignAndBroadcast({
onOperationBroadcasted, onOperationBroadcasted,
}: { }: {
accountId: string, accountId: string,
currencyId: string, derivationMode: string,
seedIdentifier: string,
currency: CryptoCurrency,
xpub: string, xpub: string,
freshAddress: string,
freshAddressPath: string,
index: number, index: number,
transaction: BitcoinLikeTransaction, transaction: BitcoinLikeTransaction,
deviceId: string, deviceId: string,
@ -191,11 +190,9 @@ export async function doSignAndBroadcast({
onSigned: () => void, onSigned: () => void,
onOperationBroadcasted: (optimisticOp: $Exact<OperationRaw>) => void, onOperationBroadcasted: (optimisticOp: $Exact<OperationRaw>) => void,
}): Promise<void> { }): Promise<void> {
const { walletName } = accountIdHelper.decode(accountId) const walletName = getWalletName({ currency, seedIdentifier, derivationMode })
const isSegwit = isSegwitPath(freshAddressPath) const njsWallet = await getOrCreateWallet(core, walletName, { currency, derivationMode })
const isUnsplit = isUnsplitPath(freshAddressPath, splittedCurrencies[currencyId])
const njsWallet = await getOrCreateWallet(core, walletName, { currencyId, isSegwit, isUnsplit })
if (isCancelled()) return if (isCancelled()) return
const njsAccount = await njsWallet.getAccount(index) const njsAccount = await njsWallet.getAccount(index)
if (isCancelled()) return if (isCancelled()) return
@ -221,17 +218,14 @@ export async function doSignAndBroadcast({
const hasTimestamp = !!njsWalletCurrency.bitcoinLikeNetworkParameters.UsesTimestampedTransaction const hasTimestamp = !!njsWalletCurrency.bitcoinLikeNetworkParameters.UsesTimestampedTransaction
// TODO: const timestampDelay = njsWalletCurrency.bitcoinLikeNetworkParameters.TimestampDelay // TODO: const timestampDelay = njsWalletCurrency.bitcoinLikeNetworkParameters.TimestampDelay
const currency = getCryptoCurrencyById(currencyId)
const signedTransaction = await withDevice(deviceId)(async transport => const signedTransaction = await withDevice(deviceId)(async transport =>
signTransaction({ signTransaction({
hwApp: new Btc(transport), hwApp: new Btc(transport),
currencyId, currency,
transaction: builded, transaction: builded,
sigHashType: parseInt(sigHashType, 16), sigHashType: parseInt(sigHashType, 16),
supportsSegwit: !!currency.supportsSegwit,
isSegwit: isSegwitPath(freshAddressPath),
hasTimestamp, hasTimestamp,
derivationMode,
}), }),
) )

16
src/commands/libcoreSyncAccount.js

@ -1,6 +1,7 @@
// @flow // @flow
import type { AccountRaw } from '@ledgerhq/live-common/lib/types' import type { AccountRaw } from '@ledgerhq/live-common/lib/types'
import { getCryptoCurrencyById } from '@ledgerhq/live-common/lib/currencies'
import { fromPromise } from 'rxjs/observable/fromPromise' import { fromPromise } from 'rxjs/observable/fromPromise'
import { createCommand, Command } from 'helpers/ipc' import { createCommand, Command } from 'helpers/ipc'
@ -9,15 +10,24 @@ import withLibcore from 'helpers/withLibcore'
type Input = { type Input = {
accountId: string, accountId: string,
freshAddressPath: string,
currencyId: string, currencyId: string,
xpub: string,
derivationMode: string,
seedIdentifier: string,
index: number, index: number,
} }
type Result = { rawAccount: AccountRaw, requiresCacheFlush: boolean } type Result = { rawAccount: AccountRaw, requiresCacheFlush: boolean }
const cmd: Command<Input, Result> = createCommand('libcoreSyncAccount', accountInfos => const cmd: Command<Input, Result> = createCommand(
fromPromise(withLibcore(core => syncAccount({ ...accountInfos, core }))), 'libcoreSyncAccount',
({ currencyId, ...accountInfos }) =>
fromPromise(
withLibcore(core => {
const currency = getCryptoCurrencyById(currencyId)
return syncAccount({ ...accountInfos, currency, core })
}),
),
) )
export default cmd export default cmd

8
src/components/AccountPage/AccountHeaderActions.js

@ -7,7 +7,7 @@ import { translate } from 'react-i18next'
import styled from 'styled-components' import styled from 'styled-components'
import type { Account } from '@ledgerhq/live-common/lib/types' import type { Account } from '@ledgerhq/live-common/lib/types'
import Tooltip from 'components/base/Tooltip' import Tooltip from 'components/base/Tooltip'
import isAccountEmpty from 'helpers/isAccountEmpty' import { isAccountEmpty } from '@ledgerhq/live-common/lib/account'
import { MODAL_SEND, MODAL_RECEIVE, MODAL_SETTINGS_ACCOUNT } from 'config/constants' import { MODAL_SEND, MODAL_RECEIVE, MODAL_SETTINGS_ACCOUNT } from 'config/constants'
@ -67,19 +67,19 @@ class AccountHeaderActions extends PureComponent<Props> {
<Button small primary onClick={() => openModal(MODAL_SEND, { account })}> <Button small primary onClick={() => openModal(MODAL_SEND, { account })}>
<Box horizontal flow={1} alignItems="center"> <Box horizontal flow={1} alignItems="center">
<IconSend size={12} /> <IconSend size={12} />
<Box>{t('app:send.title')}</Box> <Box>{t('send.title')}</Box>
</Box> </Box>
</Button> </Button>
<Button small primary onClick={() => openModal(MODAL_RECEIVE, { account })}> <Button small primary onClick={() => openModal(MODAL_RECEIVE, { account })}>
<Box horizontal flow={1} alignItems="center"> <Box horizontal flow={1} alignItems="center">
<IconReceive size={12} /> <IconReceive size={12} />
<Box>{t('app:receive.title')}</Box> <Box>{t('receive.title')}</Box>
</Box> </Box>
</Button> </Button>
</Fragment> </Fragment>
) : null} ) : null}
<Tooltip render={() => t('app:account.settings.title')}> <Tooltip render={() => t('account.settings.title')}>
<ButtonSettings onClick={() => openModal(MODAL_SETTINGS_ACCOUNT, { account })}> <ButtonSettings onClick={() => openModal(MODAL_SETTINGS_ACCOUNT, { account })}>
<Box justifyContent="center"> <Box justifyContent="center">
<IconAccountSettings size={16} /> <IconAccountSettings size={16} />

6
src/components/AccountPage/EmptyStateAccount.js

@ -40,9 +40,9 @@ class EmptyStateAccount extends PureComponent<Props, *> {
height="89" height="89"
/> />
<Box mt={5} alignItems="center"> <Box mt={5} alignItems="center">
<Title>{t('app:account.emptyState.title')}</Title> <Title>{t('account.emptyState.title')}</Title>
<Description mt={3} style={{ display: 'block' }}> <Description mt={3} style={{ display: 'block' }}>
<Trans i18nKey="app:account.emptyState.desc"> <Trans i18nKey="account.emptyState.desc">
{'Make sure the'} {'Make sure the'}
<Text ff="Open Sans|SemiBold" color="dark"> <Text ff="Open Sans|SemiBold" color="dark">
{account.currency.managerAppName} {account.currency.managerAppName}
@ -53,7 +53,7 @@ class EmptyStateAccount extends PureComponent<Props, *> {
<Button mt={5} primary onClick={() => openModal(MODAL_RECEIVE, { account })}> <Button mt={5} primary onClick={() => openModal(MODAL_RECEIVE, { account })}>
<Box horizontal flow={1} alignItems="center"> <Box horizontal flow={1} alignItems="center">
<IconReceive size={12} /> <IconReceive size={12} />
<Box>{t('app:account.emptyState.buttons.receiveFunds')}</Box> <Box>{t('account.emptyState.buttons.receiveFunds')}</Box>
</Box> </Box>
</Button> </Button>
</Box> </Box>

4
src/components/AccountPage/index.js

@ -8,7 +8,7 @@ import { Redirect } from 'react-router'
import type { Currency, Account } from '@ledgerhq/live-common/lib/types' import type { Currency, Account } from '@ledgerhq/live-common/lib/types'
import type { T } from 'types/common' import type { T } from 'types/common'
import { accountSelector } from 'reducers/accounts' import { accountSelector } from 'reducers/accounts'
import isAccountEmpty from 'helpers/isAccountEmpty' import { isAccountEmpty } from '@ledgerhq/live-common/lib/account'
import { import {
counterValueCurrencySelector, counterValueCurrencySelector,
localeSelector, localeSelector,
@ -97,7 +97,7 @@ class AccountPage extends PureComponent<Props> {
/> />
</Box> </Box>
<OperationsList account={account} title={t('app:account.lastOperations')} /> <OperationsList account={account} title={t('account.lastOperations')} />
<StickyBackToTop /> <StickyBackToTop />
</Fragment> </Fragment>

8
src/components/AdvancedOptions/BitcoinKind.js

@ -15,12 +15,12 @@ type Props = {
} }
export default translate()(({ isRBF, onChangeRBF, t }: Props) => ( export default translate()(({ isRBF, onChangeRBF, t }: Props) => (
<Spoiler title={t('app:send.steps.amount.advancedOptions')}> <Spoiler title={t('send.steps.amount.advancedOptions')}>
<Box horizontal align="center" flow={5}> <Box horizontal align="center" flow={5}>
<Box style={{ width: 200 }}> <Box style={{ width: 200 }}>
<Label> <Label>
<span>{t('app:send.steps.amount.useRBF')}</span> <span>{t('send.steps.amount.useRBF')}</span>
<LabelInfoTooltip ml={1} text={t('app:send.steps.amount.useRBF')} /> <LabelInfoTooltip ml={1} text={t('send.steps.amount.useRBF')} />
</Label> </Label>
</Box> </Box>
<Box grow> <Box grow>
@ -32,7 +32,7 @@ export default translate()(({ isRBF, onChangeRBF, t }: Props) => (
<Box horizontal align="flex-start" flow={5}> <Box horizontal align="flex-start" flow={5}>
<Box style={{ width: 200 }}> <Box style={{ width: 200 }}>
<Label> <Label>
<span>{t('app:send.steps.amount.message')}</span> <span>{t('send.steps.amount.message')}</span>
</Label> </Label>
</Box> </Box>
<Box grow> <Box grow>

4
src/components/AdvancedOptions/EthereumKind.js

@ -15,11 +15,11 @@ type Props = {
} }
export default translate()(({ gasLimit, onChangeGasLimit, t }: Props) => ( export default translate()(({ gasLimit, onChangeGasLimit, t }: Props) => (
<Spoiler title={t('app:send.steps.amount.advancedOptions')}> <Spoiler title={t('send.steps.amount.advancedOptions')}>
<Box horizontal align="center" flow={5}> <Box horizontal align="center" flow={5}>
<Box style={{ width: 200 }}> <Box style={{ width: 200 }}>
<Label> <Label>
<span>{t('app:send.steps.amount.ethereumGasLimit')}</span> <span>{t('send.steps.amount.ethereumGasLimit')}</span>
</Label> </Label>
</Box> </Box>
<Box grow> <Box grow>

2
src/components/AdvancedOptions/RippleKind.js

@ -33,7 +33,7 @@ class RippleKind extends Component<Props> {
<Box vertical flow={5}> <Box vertical flow={5}>
<Box grow> <Box grow>
<Label> <Label>
<span>{t('app:send.steps.amount.rippleTag')}</span> <span>{t('send.steps.amount.rippleTag')}</span>
</Label> </Label>
<Input value={String(tag || '')} onChange={this.onChange} /> <Input value={String(tag || '')} onChange={this.onChange} />
</Box> </Box>

14
src/components/BalanceSummary/BalanceInfos.js

@ -56,11 +56,7 @@ export function BalanceSincePercent(props: BalanceSinceProps) {
withIcon withIcon
/> />
)} )}
{!isAvailable ? ( {!isAvailable ? <PlaceholderLine dark width={60} /> : <Sub>{t(`time.since.${since}`)}</Sub>}
<PlaceholderLine dark width={60} />
) : (
<Sub>{t(`app:time.since.${since}`)}</Sub>
)}
</Box> </Box>
) )
} }
@ -82,11 +78,7 @@ export function BalanceSinceDiff(props: Props) {
withIcon withIcon
/> />
)} )}
{!isAvailable ? ( {!isAvailable ? <PlaceholderLine dark width={60} /> : <Sub>{t(`time.since.${since}`)}</Sub>}
<PlaceholderLine dark width={60} />
) : (
<Sub>{t(`app:time.since.${since}`)}</Sub>
)}
</Box> </Box>
) )
} }
@ -127,7 +119,7 @@ function BalanceInfos(props: Props) {
isAvailable={isAvailable} isAvailable={isAvailable}
totalBalance={totalBalance} totalBalance={totalBalance}
> >
<Sub>{t('app:dashboard.totalBalance')}</Sub> <Sub>{t('dashboard.totalBalance')}</Sub>
</BalanceTotal> </BalanceTotal>
<BalanceSincePercent <BalanceSincePercent
alignItems="flex-end" alignItems="flex-end"

2
src/components/BalanceSummary/index.js

@ -3,7 +3,7 @@
import React, { Fragment } from 'react' import React, { Fragment } from 'react'
import { BigNumber } from 'bignumber.js' import { BigNumber } from 'bignumber.js'
import moment from 'moment' import moment from 'moment'
import { formatShort } from '@ledgerhq/live-common/lib/helpers/currencies' import { formatShort } from '@ledgerhq/live-common/lib/currencies'
import type { Currency, Account } from '@ledgerhq/live-common/lib/types' import type { Currency, Account } from '@ledgerhq/live-common/lib/types'
import Chart from 'components/base/Chart' import Chart from 'components/base/Chart'

2
src/components/BalanceSummary/stories.js

@ -4,7 +4,7 @@ import React from 'react'
import { storiesOf } from '@storybook/react' import { storiesOf } from '@storybook/react'
import { number } from '@storybook/addon-knobs' import { number } from '@storybook/addon-knobs'
import { translate } from 'react-i18next' import { translate } from 'react-i18next'
import { getFiatCurrencyByTicker } from '@ledgerhq/live-common/lib/helpers/currencies' import { getFiatCurrencyByTicker } from '@ledgerhq/live-common/lib/currencies'
import BalanceInfos from './BalanceInfos' import BalanceInfos from './BalanceInfos'

2
src/components/CalculateBalance.js

@ -6,7 +6,7 @@ import { connect } from 'react-redux'
import { BigNumber } from 'bignumber.js' import { BigNumber } from 'bignumber.js'
import type { Account } from '@ledgerhq/live-common/lib/types' import type { Account } from '@ledgerhq/live-common/lib/types'
import { getBalanceHistorySum } from '@ledgerhq/live-common/lib/helpers/account' import { getBalanceHistorySum } from '@ledgerhq/live-common/lib/account'
import CounterValues from 'helpers/countervalues' import CounterValues from 'helpers/countervalues'
import { import {
exchangeSettingsForAccountSelector, exchangeSettingsForAccountSelector,

2
src/components/CounterValue/stories.js

@ -1,7 +1,7 @@
// @flow // @flow
import React from 'react' import React from 'react'
import { getCryptoCurrencyById } from '@ledgerhq/live-common/lib/helpers/currencies' import { getCryptoCurrencyById } from '@ledgerhq/live-common/lib/currencies'
import { storiesOf } from '@storybook/react' import { storiesOf } from '@storybook/react'
import { number } from '@storybook/addon-knobs' import { number } from '@storybook/addon-knobs'

20
src/components/CurrentAddress/index.js

@ -3,7 +3,7 @@
import React, { PureComponent } from 'react' import React, { PureComponent } from 'react'
import { Trans, translate } from 'react-i18next' import { Trans, translate } from 'react-i18next'
import styled from 'styled-components' import styled from 'styled-components'
import { encodeURIScheme } from '@ledgerhq/live-common/lib/helpers/currencies' import { encodeURIScheme } from '@ledgerhq/live-common/lib/currencies'
import type { Account } from '@ledgerhq/live-common/lib/types' import type { Account } from '@ledgerhq/live-common/lib/types'
import noop from 'lodash/noop' import noop from 'lodash/noop'
@ -152,7 +152,7 @@ class CurrentAddress extends PureComponent<Props, { copyFeedback: boolean }> {
return ( return (
<FooterButton <FooterButton
icon={<IconCopy size={16} />} icon={<IconCopy size={16} />}
label={t('app:common.copyAddress')} label={t('common.copyAddress')}
onClick={() => { onClick={() => {
this.setState({ copyFeedback: true }) this.setState({ copyFeedback: true })
this._timeout = setTimeout(() => this.setState({ copyFeedback: false }), 1e3) this._timeout = setTimeout(() => this.setState({ copyFeedback: false }), 1e3)
@ -193,17 +193,17 @@ class CurrentAddress extends PureComponent<Props, { copyFeedback: boolean }> {
<Label> <Label>
<Box> <Box>
{accountName ? ( {accountName ? (
<Trans i18nKey="app:currentAddress.for" parent="div"> <Trans i18nKey="currentAddress.for" parent="div">
{'Address for '} {'Address for '}
<strong>{accountName}</strong> <strong>{accountName}</strong>
</Trans> </Trans>
) : ( ) : (
t('app:currentAddress.title') t('currentAddress.title')
)} )}
</Box> </Box>
</Label> </Label>
<Address> <Address>
{copyFeedback && <CopyFeedback>{t('app:common.addressCopied')}</CopyFeedback>} {copyFeedback && <CopyFeedback>{t('common.addressCopied')}</CopyFeedback>}
{address} {address}
</Address> </Address>
<Box horizontal flow={2} mt={2} alignItems="center" style={{ maxWidth: 320 }}> <Box horizontal flow={2} mt={2} alignItems="center" style={{ maxWidth: 320 }}>
@ -217,19 +217,17 @@ class CurrentAddress extends PureComponent<Props, { copyFeedback: boolean }> {
ff="Open Sans" ff="Open Sans"
> >
{isAddressVerified === null {isAddressVerified === null
? t('app:currentAddress.messageIfUnverified', { currencyName }) ? t('currentAddress.messageIfUnverified', { currencyName })
: isAddressVerified : isAddressVerified
? t('app:currentAddress.messageIfAccepted', { currencyName }) ? t('currentAddress.messageIfAccepted', { currencyName })
: t('app:currentAddress.messageIfSkipped', { currencyName })} : t('currentAddress.messageIfSkipped', { currencyName })}
</Box> </Box>
</Box> </Box>
<Footer> <Footer>
{isAddressVerified !== null ? ( {isAddressVerified !== null ? (
<FooterButton <FooterButton
icon={<IconRecheck size={16} />} icon={<IconRecheck size={16} />}
label={ label={isAddressVerified === false ? t('common.verify') : t('common.reverify')}
isAddressVerified === false ? t('app:common.verify') : t('app:common.reverify')
}
onClick={onVerify} onClick={onVerify}
/> />
) : null} ) : null}

2
src/components/DashboardPage/AccountCardListHeader.js

@ -20,7 +20,7 @@ class AccountCardListHeader extends PureComponent<Props> {
return ( return (
<Box horizontal alignItems="flex-end"> <Box horizontal alignItems="flex-end">
<Text color="dark" ff="Museo Sans" fontSize={6} data-e2e="dashboard_AccountCount"> <Text color="dark" ff="Museo Sans" fontSize={6} data-e2e="dashboard_AccountCount">
{t('app:dashboard.accounts.title', { count: accountsLength })} {t('dashboard.accounts.title', { count: accountsLength })}
</Text> </Text>
<Box ml="auto" horizontal flow={1}> <Box ml="auto" horizontal flow={1}>
<AccountsOrder /> <AccountsOrder />

4
src/components/DashboardPage/AccountCardPlaceholder.js

@ -44,10 +44,10 @@ class AccountCardPlaceholder extends PureComponent<{
textAlign="center" textAlign="center"
style={{ maxWidth: 150 }} style={{ maxWidth: 150 }}
> >
{t('app:dashboard.emptyAccountTile.desc')} {t('dashboard.emptyAccountTile.desc')}
</Box> </Box>
<Button primary onClick={this.onAddAccounts}> <Button primary onClick={this.onAddAccounts}>
{t('app:dashboard.emptyAccountTile.createAccount')} {t('dashboard.emptyAccountTile.createAccount')}
</Button> </Button>
</Wrapper> </Wrapper>
) )

8
src/components/DashboardPage/AccountsOrder.js

@ -91,11 +91,11 @@ class AccountsOrder extends Component<Props> {
return [ return [
{ {
key: 'name', key: 'name',
label: t('app:dashboard.accountsOrder.name'), label: t('dashboard.accountsOrder.name'),
}, },
{ {
key: 'balance', key: 'balance',
label: t('app:dashboard.accountsOrder.balance'), label: t('dashboard.accountsOrder.balance'),
}, },
].map(item => ({ ].map(item => ({
...item, ...item,
@ -141,7 +141,7 @@ class AccountsOrder extends Component<Props> {
> >
<Track onUpdate event="ChangeSort" orderAccounts={orderAccounts} /> <Track onUpdate event="ChangeSort" orderAccounts={orderAccounts} />
<Text ff="Open Sans|SemiBold" fontSize={4}> <Text ff="Open Sans|SemiBold" fontSize={4}>
{t('app:common.sortBy')} {t('common.sortBy')}
</Text> </Text>
<Box <Box
alignItems="center" alignItems="center"
@ -152,7 +152,7 @@ class AccountsOrder extends Component<Props> {
horizontal horizontal
> >
<Text color="dark"> <Text color="dark">
{t(`app:dashboard.accountsOrder.${this.getCurrentValue() || 'balance'}`)} {t(`dashboard.accountsOrder.${this.getCurrentValue() || 'balance'}`)}
</Text> </Text>
<IconAngleDown size={16} /> <IconAngleDown size={16} />
</Box> </Box>

6
src/components/DashboardPage/CurrentGreetings.js

@ -10,11 +10,11 @@ const getCurrentGreetings = () => {
const afternoon_breakpoint = 12 const afternoon_breakpoint = 12
const evening_breakpoint = 17 const evening_breakpoint = 17
if (localTimeHour >= afternoon_breakpoint && localTimeHour < evening_breakpoint) { if (localTimeHour >= afternoon_breakpoint && localTimeHour < evening_breakpoint) {
return 'app:dashboard.greeting.afternoon' return 'dashboard.greeting.afternoon'
} else if (localTimeHour >= evening_breakpoint) { } else if (localTimeHour >= evening_breakpoint) {
return 'app:dashboard.greeting.evening' return 'dashboard.greeting.evening'
} }
return 'app:dashboard.greeting.morning' return 'dashboard.greeting.morning'
} }
class CurrentGettings extends PureComponent<{ t: T }> { class CurrentGettings extends PureComponent<{ t: T }> {

8
src/components/DashboardPage/EmptyState.js

@ -45,9 +45,9 @@ class EmptyState extends PureComponent<Props, *> {
height="157" height="157"
/> />
<Box mt={5} alignItems="center"> <Box mt={5} alignItems="center">
<Title data-e2e="dashboard_empty_title">{t('app:emptyState.dashboard.title')}</Title> <Title data-e2e="dashboard_empty_title">{t('emptyState.dashboard.title')}</Title>
<Description mt={3} style={{ maxWidth: 600 }}> <Description mt={3} style={{ maxWidth: 600 }}>
{t('app:emptyState.dashboard.desc')} {t('emptyState.dashboard.desc')}
</Description> </Description>
<Box mt={5} horizontal style={{ width: 300 }} flow={3} justify="center"> <Box mt={5} horizontal style={{ width: 300 }} flow={3} justify="center">
<Button <Button
@ -56,7 +56,7 @@ class EmptyState extends PureComponent<Props, *> {
onClick={this.handleInstallApp} onClick={this.handleInstallApp}
data-e2e="dashboard_empty_OpenManager" data-e2e="dashboard_empty_OpenManager"
> >
{t('app:emptyState.dashboard.buttons.installApp')} {t('emptyState.dashboard.buttons.installApp')}
</Button> </Button>
<Button <Button
outline outline
@ -64,7 +64,7 @@ class EmptyState extends PureComponent<Props, *> {
onClick={() => openModal(MODAL_ADD_ACCOUNTS)} onClick={() => openModal(MODAL_ADD_ACCOUNTS)}
data-e2e="dashboard_empty_AddAccounts" data-e2e="dashboard_empty_AddAccounts"
> >
{t('app:emptyState.dashboard.buttons.addAccount')} {t('emptyState.dashboard.buttons.addAccount')}
</Button> </Button>
</Box> </Box>
</Box> </Box>

2
src/components/DashboardPage/SummaryDesc.js

@ -18,7 +18,7 @@ class SummaryDesc extends PureComponent<{
ff="Museo Sans|Light" ff="Museo Sans|Light"
data-e2e="dashboard_accountsSummaryDesc" data-e2e="dashboard_accountsSummaryDesc"
> >
{t('app:dashboard.summary', { count: totalAccounts })} {t('dashboard.summary', { count: totalAccounts })}
</Text> </Text>
) )
} }

2
src/components/DashboardPage/index.js

@ -129,7 +129,7 @@ class DashboardPage extends PureComponent<Props> {
<OperationsList <OperationsList
onAccountClick={this.onAccountClick} onAccountClick={this.onAccountClick}
accounts={accounts} accounts={accounts}
title={t('app:dashboard.recentActivity')} title={t('dashboard.recentActivity')}
withAccount withAccount
/> />
)} )}

14
src/components/DevToolsPage/AccountImporter.js

@ -21,7 +21,6 @@ import SelectCurrency from 'components/SelectCurrency'
import { CurrencyCircleIcon } from 'components/base/CurrencyBadge' import { CurrencyCircleIcon } from 'components/base/CurrencyBadge'
import { idleCallback } from 'helpers/promise' import { idleCallback } from 'helpers/promise'
import { splittedCurrencies } from 'config/cryptocurrencies'
import scanFromXPUB from 'commands/libcoreScanFromXPUB' import scanFromXPUB from 'commands/libcoreScanFromXPUB'
@ -80,12 +79,19 @@ class AccountImporter extends PureComponent<Props, State> {
try { try {
const { currency, xpub, isSegwit, isUnsplit } = this.state const { currency, xpub, isSegwit, isUnsplit } = this.state
invariant(currency, 'no currency') invariant(currency, 'no currency')
const derivationMode = isSegwit
? isUnsplit
? 'segwit_unsplit'
: 'segwit'
: isUnsplit
? 'unsplit'
: ''
const rawAccount = await scanFromXPUB const rawAccount = await scanFromXPUB
.send({ .send({
seedIdentifier: 'dev_tool',
currencyId: currency.id, currencyId: currency.id,
xpub, xpub,
isSegwit, derivationMode,
isUnsplit,
}) })
.toPromise() .toPromise()
const account = decodeAccount(rawAccount) const account = decodeAccount(rawAccount)
@ -107,7 +113,7 @@ class AccountImporter extends PureComponent<Props, State> {
render() { render() {
const { currency, xpub, isSegwit, isUnsplit, status, account, error } = this.state const { currency, xpub, isSegwit, isUnsplit, status, account, error } = this.state
const supportsSplit = !!currency && !!splittedCurrencies[currency.id] const supportsSplit = !!currency && !!currency.forkedFrom
return ( return (
<Card title="Import from xpub" flow={3}> <Card title="Import from xpub" flow={3}>
{status === 'idle' ? ( {status === 'idle' ? (

4
src/components/DeviceInteraction/components.js

@ -144,11 +144,11 @@ export const ErrorDescContainer = translate()(
<Box ml="auto" horizontal flow={2}> <Box ml="auto" horizontal flow={2}>
{!!errorHelpURL && ( {!!errorHelpURL && (
<FakeLink underline color="alertRed" onClick={() => openURL(errorHelpURL)}> <FakeLink underline color="alertRed" onClick={() => openURL(errorHelpURL)}>
{t('app:common.help')} {t('common.help')}
</FakeLink> </FakeLink>
)} )}
<FakeLink underline color="alertRed" onClick={onRetry}> <FakeLink underline color="alertRed" onClick={onRetry}>
{t('app:common.retry')} {t('common.retry')}
</FakeLink> </FakeLink>
</Box> </Box>
</Box> </Box>

15
src/components/EnsureDeviceApp.js

@ -10,8 +10,11 @@ import { getCryptoCurrencyIcon } from '@ledgerhq/live-common/lib/react'
import logger from 'logger' import logger from 'logger'
import getAddress from 'commands/getAddress' import getAddress from 'commands/getAddress'
import { createCancelablePolling } from 'helpers/promise' import { createCancelablePolling } from 'helpers/promise'
import { standardDerivation } from 'helpers/derivations' import {
import { isSegwitPath } from 'helpers/bip32' isSegwitDerivationMode,
getDerivationScheme,
runDerivationScheme,
} from '@ledgerhq/live-common/lib/derivation'
import DeviceInteraction from 'components/DeviceInteraction' import DeviceInteraction from 'components/DeviceInteraction'
import Text from 'components/base/Text' import Text from 'components/base/Text'
@ -74,7 +77,7 @@ class EnsureDeviceApp extends Component<{
const cur = account ? account.currency : currency const cur = account ? account.currency : currency
invariant(cur, 'No currency given') invariant(cur, 'No currency given')
return ( return (
<Trans i18nKey="app:deviceConnect.step2.open" parent="div"> <Trans i18nKey="deviceConnect.step2.open" parent="div">
{'Open the '} {'Open the '}
<Bold>{cur.managerAppName}</Bold> <Bold>{cur.managerAppName}</Bold>
{' app on your device'} {' app on your device'}
@ -94,7 +97,7 @@ class EnsureDeviceApp extends Component<{
{ {
id: 'device', id: 'device',
title: ( title: (
<Trans i18nKey="app:deviceConnect.step1.connect" parent="div"> <Trans i18nKey="deviceConnect.step1.connect" parent="div">
{'Connect and unlock your '} {'Connect and unlock your '}
<Bold>{'Ledger device'}</Bold> <Bold>{'Ledger device'}</Bold>
</Trans> </Trans>
@ -122,8 +125,8 @@ async function getAddressFromAccountOrCurrency(device, account, currency) {
currencyId: currency.id, currencyId: currency.id,
path: account path: account
? account.freshAddressPath ? account.freshAddressPath
: standardDerivation({ currency, segwit: false, x: 0 }), : runDerivationScheme(getDerivationScheme({ currency, derivationMode: '' }), currency),
segwit: account ? isSegwitPath(account.freshAddressPath) : false, segwit: account ? isSegwitDerivationMode(account.derivationMode) : false,
}) })
.toPromise() .toPromise()
return address return address

4
src/components/ExchangePage/ExchangeCard.js

@ -31,9 +31,9 @@ export default class ExchangeCard extends PureComponent<{ t: T, card: CardType }
{logo} {logo}
</Box> </Box>
<Box shrink ff="Open Sans|Regular" fontSize={4} flow={3}> <Box shrink ff="Open Sans|Regular" fontSize={4} flow={3}>
<Box>{t(`app:exchange.${id}`)}</Box> <Box>{t(`exchange.${id}`)}</Box>
<Box horizontal align="center" color="wallet" flow={1}> <Box horizontal align="center" color="wallet" flow={1}>
<FakeLink onClick={this.onClick}>{t('app:exchange.visitWebsite')}</FakeLink> <FakeLink onClick={this.onClick}>{t('exchange.visitWebsite')}</FakeLink>
<ExternalLinkIcon size={14} /> <ExternalLinkIcon size={14} />
</Box> </Box>
</Box> </Box>

4
src/components/ExchangePage/index.js

@ -86,10 +86,10 @@ class ExchangePage extends PureComponent<Props> {
<Box pb={6} selectable> <Box pb={6} selectable>
<TrackPage category="Exchange" /> <TrackPage category="Exchange" />
<Box ff="Museo Sans|Regular" fontSize={7} color="dark"> <Box ff="Museo Sans|Regular" fontSize={7} color="dark">
{t('app:exchange.title')} {t('exchange.title')}
</Box> </Box>
<Box ff="Museo Sans|Light" fontSize={5} mb={5}> <Box ff="Museo Sans|Light" fontSize={5} mb={5}>
{t('app:exchange.desc')} {t('exchange.desc')}
</Box> </Box>
<Box flow={3}>{cards.map(card => <ExchangeCard key={card.key} t={t} card={card} />)}</Box> <Box flow={3}>{cards.map(card => <ExchangeCard key={card.key} t={t} card={card} />)}</Box>
</Box> </Box>

2
src/components/ExportLogsBtn.js

@ -78,7 +78,7 @@ class ExportLogsBtn extends Component<{
<KeyHandler keyValue="e" onKeyHandle={this.onKeyHandle} /> <KeyHandler keyValue="e" onKeyHandle={this.onKeyHandle} />
) : ( ) : (
<Button small primary event="ExportLogs" onClick={this.handleExportLogs}> <Button small primary event="ExportLogs" onClick={this.handleExportLogs}>
{t('app:settings.exportLogs.btn')} {t('settings.exportLogs.btn')}
</Button> </Button>
) )
} }

4
src/components/FeesField/BitcoinKind.js

@ -148,9 +148,7 @@ class FeesField extends Component<OwnProps, State> {
loading={!feePerByte && !error} loading={!feePerByte && !error}
error={!feePerByte && error ? new FeeNotLoaded() : null} error={!feePerByte && error ? new FeeNotLoaded() : null}
renderRight={ renderRight={
<InputRight> <InputRight>{t('send.steps.amount.unitPerByte', { unit: satoshi.code })}</InputRight>
{t('app:send.steps.amount.unitPerByte', { unit: satoshi.code })}
</InputRight>
} }
allowZero allowZero
/> />

2
src/components/FeesField/GenericContainer.js

@ -16,7 +16,7 @@ export default translate()(({ children, t }: { children: React$Node, t: * }) =>
openURL(urls.feesMoreInfo) openURL(urls.feesMoreInfo)
track('Send Flow Fees Help Requested') track('Send Flow Fees Help Requested')
}} }}
label={t('app:send.steps.amount.fees')} label={t('send.steps.amount.fees')}
/> />
<Box horizontal flow={5}> <Box horizontal flow={5}>
{children} {children}

2
src/components/GenuineCheck.js

@ -124,7 +124,7 @@ class GenuineCheck extends PureComponent<Props> {
{ {
id: 'device', id: 'device',
title: ( title: (
<Trans i18nKey="app:deviceConnect.step1.connect" parent="div"> <Trans i18nKey="deviceConnect.step1.connect" parent="div">
{'Connect and unlock your '} {'Connect and unlock your '}
<Bold>{'Ledger device'}</Bold> <Bold>{'Ledger device'}</Bold>
</Trans> </Trans>

2
src/components/GenuineCheckModal.js

@ -20,7 +20,7 @@ class GenuineCheckModal extends PureComponent<Props> {
const { t, onSuccess, onFail, onUnavailable } = this.props const { t, onSuccess, onFail, onUnavailable } = this.props
return ( return (
<ModalBody onClose={onClose}> <ModalBody onClose={onClose}>
<ModalTitle>{t('app:genuinecheck.modal.title')}</ModalTitle> <ModalTitle>{t('genuinecheck.modal.title')}</ModalTitle>
<ModalContent> <ModalContent>
<GenuineCheck onSuccess={onSuccess} onFail={onFail} onUnavailable={onUnavailable} /> <GenuineCheck onSuccess={onSuccess} onFail={onFail} onUnavailable={onUnavailable} />
</ModalContent> </ModalContent>

2
src/components/GlobalSearch.js

@ -66,7 +66,7 @@ class GlobalSearch extends PureComponent<Props, State> {
<IconSearch size={16} /> <IconSearch size={16} />
</Box> </Box>
<Input <Input
placeholder={t('app:common.search')} placeholder={t('common.search')}
innerRef={input => (this._input = input)} innerRef={input => (this._input = input)}
onBlur={this.handleBlur} onBlur={this.handleBlur}
onFocus={this.handleFocus} onFocus={this.handleFocus}

16
src/components/IsUnlocked.js

@ -161,17 +161,17 @@ class IsUnlocked extends Component<Props, State> {
/> />
} }
/> />
<PageTitle>{t('app:common.lockScreen.title')}</PageTitle> <PageTitle>{t('common.lockScreen.title')}</PageTitle>
<LockScreenDesc> <LockScreenDesc>
{t('app:common.lockScreen.subTitle')} {t('common.lockScreen.subTitle')}
<br /> <br />
{t('app:common.lockScreen.description')} {t('common.lockScreen.description')}
</LockScreenDesc> </LockScreenDesc>
<Box horizontal align="center"> <Box horizontal align="center">
<Box style={{ width: 280 }}> <Box style={{ width: 280 }}>
<InputPassword <InputPassword
autoFocus autoFocus
placeholder={t('app:common.lockScreen.inputPlaceholder')} placeholder={t('common.lockScreen.inputPlaceholder')}
type="password" type="password"
onChange={this.handleChangeInput('password')} onChange={this.handleChangeInput('password')}
value={inputValue.password} value={inputValue.password}
@ -187,7 +187,7 @@ class IsUnlocked extends Component<Props, State> {
</Box> </Box>
</Box> </Box>
<Button type="button" mt={3} small onClick={this.handleOpenHardResetModal}> <Button type="button" mt={3} small onClick={this.handleOpenHardResetModal}>
{t('app:common.lockScreen.lostPassword')} {t('common.lockScreen.lostPassword')}
</Button> </Button>
</Box> </Box>
</form> </form>
@ -199,9 +199,9 @@ class IsUnlocked extends Component<Props, State> {
onClose={this.handleCloseHardResetModal} onClose={this.handleCloseHardResetModal}
onReject={this.handleCloseHardResetModal} onReject={this.handleCloseHardResetModal}
onConfirm={this.handleHardReset} onConfirm={this.handleHardReset}
confirmText={t('app:common.reset')} confirmText={t('common.reset')}
title={t('app:settings.hardResetModal.title')} title={t('settings.hardResetModal.title')}
desc={t('app:settings.hardResetModal.desc')} desc={t('settings.hardResetModal.desc')}
renderIcon={this.hardResetIconRender} renderIcon={this.hardResetIconRender}
/> />
</Box> </Box>

23
src/components/MainSideBar/index.js

@ -90,7 +90,7 @@ class MainSideBar extends PureComponent<Props> {
ADD_ACCOUNT_EMPTY_STATE = ( ADD_ACCOUNT_EMPTY_STATE = (
<Box relative pr={3}> <Box relative pr={3}>
<img style={{ position: 'absolute', top: -10, right: 5 }} alt="" src={i('arrow-add.svg')} /> <img style={{ position: 'absolute', top: -10, right: 5 }} alt="" src={i('arrow-add.svg')} />
{this.props.t('app:emptyState.sidebar.text')} {this.props.t('emptyState.sidebar.text')}
</Box> </Box>
) )
@ -107,10 +107,7 @@ class MainSideBar extends PureComponent<Props> {
const { pathname } = location const { pathname } = location
const addAccountButton = ( const addAccountButton = (
<AddAccountButton <AddAccountButton tooltipText={t('addAccounts.title')} onClick={this.handleOpenImportModal} />
tooltipText={t('app:addAccounts.title')}
onClick={this.handleOpenImportModal}
/>
) )
return ( return (
@ -118,9 +115,9 @@ class MainSideBar extends PureComponent<Props> {
<TopGradient /> <TopGradient />
<GrowScroll> <GrowScroll>
<Space of={70} /> <Space of={70} />
<SideBarList title={t('app:sidebar.menu')}> <SideBarList title={t('sidebar.menu')}>
<SideBarListItem <SideBarListItem
label={t('app:dashboard.title')} label={t('dashboard.title')}
icon={IconPieChart} icon={IconPieChart}
iconActiveColor="wallet" iconActiveColor="wallet"
onClick={this.handleClickDashboard} onClick={this.handleClickDashboard}
@ -128,28 +125,28 @@ class MainSideBar extends PureComponent<Props> {
hasNotif={updateStatus === 'downloaded'} hasNotif={updateStatus === 'downloaded'}
/> />
<SideBarListItem <SideBarListItem
label={t('app:send.title')} label={t('send.title')}
icon={IconSend} icon={IconSend}
iconActiveColor="wallet" iconActiveColor="wallet"
onClick={this.handleOpenSendModal} onClick={this.handleOpenSendModal}
disabled={accounts.length === 0} disabled={accounts.length === 0}
/> />
<SideBarListItem <SideBarListItem
label={t('app:receive.title')} label={t('receive.title')}
icon={IconReceive} icon={IconReceive}
iconActiveColor="wallet" iconActiveColor="wallet"
onClick={this.handleOpenReceiveModal} onClick={this.handleOpenReceiveModal}
disabled={accounts.length === 0} disabled={accounts.length === 0}
/> />
<SideBarListItem <SideBarListItem
label={t('app:sidebar.manager')} label={t('sidebar.manager')}
icon={IconManager} icon={IconManager}
iconActiveColor="wallet" iconActiveColor="wallet"
onClick={this.handleClickManager} onClick={this.handleClickManager}
isActive={pathname === '/manager'} isActive={pathname === '/manager'}
/> />
<SideBarListItem <SideBarListItem
label={t('app:sidebar.exchange')} label={t('sidebar.exchange')}
icon={IconExchange} icon={IconExchange}
iconActiveColor="wallet" iconActiveColor="wallet"
onClick={this.handleClickExchange} onClick={this.handleClickExchange}
@ -158,7 +155,7 @@ class MainSideBar extends PureComponent<Props> {
{developerMode && ( {developerMode && (
<KeyboardContent sequence="DEVTOOLS"> <KeyboardContent sequence="DEVTOOLS">
<SideBarListItem <SideBarListItem
label={t('app:sidebar.developer')} label={t('sidebar.developer')}
icon={IconDev} icon={IconDev}
iconActiveColor="wallet" iconActiveColor="wallet"
onClick={this.handleClickDev} onClick={this.handleClickDev}
@ -169,7 +166,7 @@ class MainSideBar extends PureComponent<Props> {
</SideBarList> </SideBarList>
<Space of={40} /> <Space of={40} />
<SideBarList <SideBarList
title={t('app:sidebar.accounts', { count: accounts.length })} title={t('sidebar.accounts', { count: accounts.length })}
titleRight={addAccountButton} titleRight={addAccountButton}
emptyState={this.ADD_ACCOUNT_EMPTY_STATE} emptyState={this.ADD_ACCOUNT_EMPTY_STATE}
> >

12
src/components/ManagerPage/AppsList.js

@ -214,7 +214,7 @@ class AppsList extends PureComponent<Props, State> {
</ModalTitle> </ModalTitle>
<ModalContent> <ModalContent>
<Text ff="Museo Sans|Regular" fontSize={6} color="dark"> <Text ff="Museo Sans|Regular" fontSize={6} color="dark">
{t(`app:manager.apps.${mode}`, { app })} {t(`manager.apps.${mode}`, { app })}
</Text> </Text>
<Box mt={6}> <Box mt={6}>
<Progress style={{ width: '100%' }} infinite /> <Progress style={{ width: '100%' }} infinite />
@ -256,7 +256,7 @@ class AppsList extends PureComponent<Props, State> {
</ModalContent> </ModalContent>
<ModalFooter horizontal justifyContent="flex-end" style={{ width: '100%' }}> <ModalFooter horizontal justifyContent="flex-end" style={{ width: '100%' }}>
<Button primary onClick={this.handleCloseModal}> <Button primary onClick={this.handleCloseModal}>
{t('app:common.close')} {t('common.close')}
</Button> </Button>
</ModalFooter> </ModalFooter>
</Fragment> </Fragment>
@ -275,7 +275,7 @@ class AppsList extends PureComponent<Props, State> {
style={{ maxWidth: 350 }} style={{ maxWidth: 350 }}
> >
{t( {t(
`app:manager.apps.${ `manager.apps.${
mode === 'installing' ? 'installSuccess' : 'uninstallSuccess' mode === 'installing' ? 'installSuccess' : 'uninstallSuccess'
}`, }`,
{ app }, { app },
@ -284,7 +284,7 @@ class AppsList extends PureComponent<Props, State> {
</ModalContent> </ModalContent>
<ModalFooter horizontal justifyContent="flex-end" style={{ width: '100%' }}> <ModalFooter horizontal justifyContent="flex-end" style={{ width: '100%' }}>
<Button primary onClick={this.handleCloseModal}> <Button primary onClick={this.handleCloseModal}>
{t('app:common.close')} {t('common.close')}
</Button> </Button>
</ModalFooter> </ModalFooter>
</Fragment> </Fragment>
@ -342,11 +342,11 @@ class AppsList extends PureComponent<Props, State> {
<Box flow={6}> <Box flow={6}>
<Box> <Box>
<Box mb={4} color="dark" ff="Museo Sans" fontSize={5} flow={2} horizontal align="center"> <Box mb={4} color="dark" ff="Museo Sans" fontSize={5} flow={2} horizontal align="center">
<span style={{ lineHeight: 1 }}>{t('app:manager.apps.all')}</span> <span style={{ lineHeight: 1 }}>{t('manager.apps.all')}</span>
<Tooltip <Tooltip
render={() => ( render={() => (
<Box ff="Open Sans|SemiBold" fontSize={2}> <Box ff="Open Sans|SemiBold" fontSize={2}>
{t('app:manager.apps.help')} {t('manager.apps.help')}
</Box> </Box>
)} )}
> >

6
src/components/ManagerPage/Dashboard.js

@ -27,14 +27,14 @@ const Dashboard = ({ device, deviceInfo, t, handleHelpRequest }: Props) => (
<TrackPage category="Manager" name="Dashboard" /> <TrackPage category="Manager" name="Dashboard" />
<Box> <Box>
<Text ff="Museo Sans|Regular" fontSize={7} color="dark"> <Text ff="Museo Sans|Regular" fontSize={7} color="dark">
{t('app:manager.title')} {t('manager.title')}
</Text> </Text>
<Box horizontal> <Box horizontal>
<Text ff="Museo Sans|Light" fontSize={5}> <Text ff="Museo Sans|Light" fontSize={5}>
{t('app:manager.subtitle')} {t('manager.subtitle')}
</Text> </Text>
<HelpLink onClick={handleHelpRequest}> <HelpLink onClick={handleHelpRequest}>
<div style={{ textDecoration: 'underline' }}>{t('app:common.needHelp')}</div> <div style={{ textDecoration: 'underline' }}>{t('common.needHelp')}</div>
<IconExternalLink size={14} /> <IconExternalLink size={14} />
</HelpLink> </HelpLink>
</Box> </Box>

8
src/components/ManagerPage/FirmwareUpdate.js

@ -136,17 +136,17 @@ class FirmwareUpdate extends PureComponent<Props, State> {
<Box horizontal align="center"> <Box horizontal align="center">
<Text ff="Open Sans|SemiBold" fontSize={4} color="dark"> <Text ff="Open Sans|SemiBold" fontSize={4} color="dark">
{device.product === 'Blue' {device.product === 'Blue'
? t('app:manager.firmware.titleBlue') ? t('manager.firmware.titleBlue')
: t('app:manager.firmware.titleNano')} : t('manager.firmware.titleNano')}
</Text> </Text>
<Box color="wallet" ml={2}> <Box color="wallet" ml={2}>
<Tooltip render={() => t('app:manager.yourDeviceIsGenuine')}> <Tooltip render={() => t('manager.yourDeviceIsGenuine')}>
<CheckFull size={13} color="wallet" /> <CheckFull size={13} color="wallet" />
</Tooltip> </Tooltip>
</Box> </Box>
</Box> </Box>
<Text ff="Open Sans|SemiBold" fontSize={2}> <Text ff="Open Sans|SemiBold" fontSize={2}>
{t('app:manager.firmware.installed', { {t('manager.firmware.installed', {
version: deviceInfo.fullVersion, version: deviceInfo.fullVersion,
})} })}
</Text> </Text>

2
src/components/ManagerPage/ManagerApp.js

@ -74,7 +74,7 @@ function ManagerApp({ name, version, icon, onInstall, onUninstall, t }: Props) {
appVersion: version, appVersion: version,
}} }}
> >
{t('app:manager.apps.install')} {t('manager.apps.install')}
</Button> </Button>
) : null} ) : null}
<Button <Button

4
src/components/ManagerPage/ManagerGenuineCheck.js

@ -31,10 +31,10 @@ class ManagerGenuineCheck extends PureComponent<Props> {
style={{ marginBottom: 30, maxWidth: 362, width: '100%' }} style={{ marginBottom: 30, maxWidth: 362, width: '100%' }}
/> />
<Text ff="Museo Sans|Regular" fontSize={7} color="dark" style={{ marginBottom: 10 }}> <Text ff="Museo Sans|Regular" fontSize={7} color="dark" style={{ marginBottom: 10 }}>
{t('app:manager.device.title')} {t('manager.device.title')}
</Text> </Text>
<Text ff="Museo Sans|Light" fontSize={5} color="grey" align="center"> <Text ff="Museo Sans|Light" fontSize={5} color="grey" align="center">
{t('app:manager.device.desc')} {t('manager.device.desc')}
</Text> </Text>
</Box> </Box>
<Space of={40} /> <Space of={40} />

6
src/components/ManagerPage/PlugYourDevice.js

@ -22,12 +22,12 @@ function PlugYourDevice(props: Props) {
<Box align="center" style={{ width: 365 }}> <Box align="center" style={{ width: 365 }}>
<Box mb={5}>hey</Box> <Box mb={5}>hey</Box>
<Box textAlign="center" mb={1} ff="Museo Sans|Regular" color="dark" fontSize={6}> <Box textAlign="center" mb={1} ff="Museo Sans|Regular" color="dark" fontSize={6}>
{t('app:manager.device.title')} {t('manager.device.title')}
</Box> </Box>
<Box textAlign="center" mb={5} ff="Open Sans|Regular" color="smoke" fontSize={4}> <Box textAlign="center" mb={5} ff="Open Sans|Regular" color="smoke" fontSize={4}>
{t('app:manager.device.desc')} {t('manager.device.desc')}
</Box> </Box>
<Button primary>{t('app:manager.device.cta')}</Button> <Button primary>{t('manager.device.cta')}</Button>
</Box> </Box>
</Card> </Card>
) )

4
src/components/ManagerPage/UpdateFirmwareButton.js

@ -23,7 +23,7 @@ const UpdateFirmwareButton = ({ t, firmware, onClick }: Props) =>
firmware ? ( firmware ? (
<Fragment> <Fragment>
<Text ff="Open Sans|Regular" fontSize={4} style={{ marginLeft: 'auto', marginRight: 15 }}> <Text ff="Open Sans|Regular" fontSize={4} style={{ marginLeft: 'auto', marginRight: 15 }}>
{t('app:manager.firmware.latest', { version: getCleanVersion(firmware.name) })} {t('manager.firmware.latest', { version: getCleanVersion(firmware.name) })}
</Text> </Text>
<Button <Button
primary primary
@ -33,7 +33,7 @@ const UpdateFirmwareButton = ({ t, firmware, onClick }: Props) =>
firmwareName: firmware.name, firmwareName: firmware.name,
}} }}
> >
{t('app:manager.firmware.update')} {t('manager.firmware.update')}
</Button> </Button>
</Fragment> </Fragment>
) : null ) : null

4
src/components/Onboarding/OnboardingFooter.js

@ -24,7 +24,7 @@ const OnboardingFooter = ({
}: Props) => ( }: Props) => (
<OnboardingFooterWrapper {...props}> <OnboardingFooterWrapper {...props}>
<Button outlineGrey onClick={() => prevStep()}> <Button outlineGrey onClick={() => prevStep()}>
{t('app:common.back')} {t('common.back')}
</Button> </Button>
<Button <Button
data-e2e="continue_button" data-e2e="continue_button"
@ -33,7 +33,7 @@ const OnboardingFooter = ({
onClick={() => nextStep()} onClick={() => nextStep()}
ml="auto" ml="auto"
> >
{t('app:common.continue')} {t('common.continue')}
</Button> </Button>
</OnboardingFooterWrapper> </OnboardingFooterWrapper>
) )

22
src/components/Onboarding/steps/Analytics.js

@ -70,14 +70,14 @@ class Analytics extends PureComponent<StepProps, State> {
deviceType={onboarding.isLedgerNano ? 'Nano S' : 'Blue'} deviceType={onboarding.isLedgerNano ? 'Nano S' : 'Blue'}
/> />
<StepContainerInner> <StepContainerInner>
<Title data-e2e="onboarding_title">{t('onboarding:analytics.title')}</Title> <Title data-e2e="onboarding_title">{t('onboarding.analytics.title')}</Title>
<Description>{t('onboarding:analytics.desc')}</Description> <Description>{t('onboarding.analytics.desc')}</Description>
<Box mt={5}> <Box mt={5}>
<Container> <Container>
<Box> <Box>
<Box horizontal mb={1}> <Box horizontal mb={1}>
<AnalyticsTitle data-e2e="analytics_techData"> <AnalyticsTitle data-e2e="analytics_techData">
{t('onboarding:analytics.technicalData.title')} {t('onboarding.analytics.technicalData.title')}
</AnalyticsTitle> </AnalyticsTitle>
<LearnMoreWrapper> <LearnMoreWrapper>
<FakeLink <FakeLink
@ -88,14 +88,14 @@ class Analytics extends PureComponent<StepProps, State> {
onClick={this.handleTechnicalDataModal} onClick={this.handleTechnicalDataModal}
data-e2e="analytics_techData_Link" data-e2e="analytics_techData_Link"
> >
{t('app:common.learnMore')} {t('common.learnMore')}
</FakeLink> </FakeLink>
</LearnMoreWrapper> </LearnMoreWrapper>
</Box> </Box>
<TechnicalData /> <TechnicalData />
<AnalyticsText>{t('onboarding:analytics.technicalData.desc')}</AnalyticsText> <AnalyticsText>{t('onboarding.analytics.technicalData.desc')}</AnalyticsText>
<MandatoryText> <MandatoryText>
{t('onboarding:analytics.technicalData.mandatoryText')} {t('onboarding.analytics.technicalData.mandatoryText')}
</MandatoryText> </MandatoryText>
</Box> </Box>
<Box justifyContent="center"> <Box justifyContent="center">
@ -106,7 +106,7 @@ class Analytics extends PureComponent<StepProps, State> {
<Box> <Box>
<Box horizontal mb={1}> <Box horizontal mb={1}>
<AnalyticsTitle data-e2e="analytics_shareAnalytics"> <AnalyticsTitle data-e2e="analytics_shareAnalytics">
{t('onboarding:analytics.shareAnalytics.title')} {t('onboarding.analytics.shareAnalytics.title')}
</AnalyticsTitle> </AnalyticsTitle>
<LearnMoreWrapper> <LearnMoreWrapper>
<FakeLink <FakeLink
@ -117,12 +117,12 @@ class Analytics extends PureComponent<StepProps, State> {
onClick={this.handleShareAnalyticsModal} onClick={this.handleShareAnalyticsModal}
data-e2e="analytics_shareAnalytics_Link" data-e2e="analytics_shareAnalytics_Link"
> >
{t('app:common.learnMore')} {t('common.learnMore')}
</FakeLink> </FakeLink>
</LearnMoreWrapper> </LearnMoreWrapper>
<ShareAnalytics /> <ShareAnalytics />
</Box> </Box>
<AnalyticsText>{t('onboarding:analytics.shareAnalytics.desc')}</AnalyticsText> <AnalyticsText>{t('onboarding.analytics.shareAnalytics.desc')}</AnalyticsText>
</Box> </Box>
<Box justifyContent="center"> <Box justifyContent="center">
<Track <Track
@ -140,10 +140,10 @@ class Analytics extends PureComponent<StepProps, State> {
<Box> <Box>
<Box mb={1}> <Box mb={1}>
<AnalyticsTitle data-e2e="analytics_reportBugs"> <AnalyticsTitle data-e2e="analytics_reportBugs">
{t('onboarding:analytics.sentryLogs.title')} {t('onboarding.analytics.sentryLogs.title')}
</AnalyticsTitle> </AnalyticsTitle>
</Box> </Box>
<AnalyticsText>{t('onboarding:analytics.sentryLogs.desc')}</AnalyticsText> <AnalyticsText>{t('onboarding.analytics.sentryLogs.desc')}</AnalyticsText>
</Box> </Box>
<Box justifyContent="center"> <Box justifyContent="center">
<Track <Track

6
src/components/Onboarding/steps/Finish.js

@ -103,12 +103,12 @@ export default class Finish extends Component<StepProps, *> {
</Box> </Box>
<Box pt={5} align="center"> <Box pt={5} align="center">
<Title data-e2e="finish_title">{t('onboarding:finish.title')}</Title> <Title data-e2e="finish_title">{t('onboarding.finish.title')}</Title>
<Description>{t('onboarding:finish.desc')}</Description> <Description>{t('onboarding.finish.desc')}</Description>
</Box> </Box>
<Box p={5}> <Box p={5}>
<Button primary onClick={() => finish()} data-e2e="continue_button"> <Button primary onClick={() => finish()} data-e2e="continue_button">
{t('onboarding:finish.openAppButton')} {t('onboarding.finish.openAppButton')}
</Button> </Button>
</Box> </Box>
<Box horizontal mt={3} flow={5} color="grey"> <Box horizontal mt={3} flow={5} color="grey">

16
src/components/Onboarding/steps/GenuineCheck/GenuineCheckErrorPage.js

@ -40,21 +40,21 @@ class GenuineCheckErrorPage extends PureComponent<Props, *> {
{onboarding.genuine.isGenuineFail ? ( {onboarding.genuine.isGenuineFail ? (
<Fragment> <Fragment>
{this.trackErrorPage('Not Genuine')} {this.trackErrorPage('Not Genuine')}
<Title>{t('onboarding:genuineCheck.errorPage.title.isGenuineFail')}</Title> <Title>{t('onboarding.genuineCheck.errorPage.title.isGenuineFail')}</Title>
<Description>{t('onboarding:genuineCheck.errorPage.desc.isGenuineFail')}</Description> <Description>{t('onboarding.genuineCheck.errorPage.desc.isGenuineFail')}</Description>
</Fragment> </Fragment>
) : !onboarding.genuine.pinStepPass ? ( ) : !onboarding.genuine.pinStepPass ? (
<Fragment> <Fragment>
{this.trackErrorPage('PIN Step')} {this.trackErrorPage('PIN Step')}
<Title>{t('onboarding:genuineCheck.errorPage.title.pinFailed')}</Title> <Title>{t('onboarding.genuineCheck.errorPage.title.pinFailed')}</Title>
<Description>{t('onboarding:genuineCheck.errorPage.desc.pinFailed')}</Description> <Description>{t('onboarding.genuineCheck.errorPage.desc.pinFailed')}</Description>
</Fragment> </Fragment>
) : ( ) : (
<Fragment> <Fragment>
{this.trackErrorPage('Recovery Phase Step')} {this.trackErrorPage('Recovery Phase Step')}
<Title>{t('onboarding:genuineCheck.errorPage.title.recoveryPhraseFailed')}</Title> <Title>{t('onboarding.genuineCheck.errorPage.title.recoveryPhraseFailed')}</Title>
<Description> <Description>
{t('onboarding:genuineCheck.errorPage.desc.recoveryPhraseFailed')} {t('onboarding.genuineCheck.errorPage.desc.recoveryPhraseFailed')}
</Description> </Description>
</Fragment> </Fragment>
)} )}
@ -78,12 +78,12 @@ class GenuineCheckErrorPage extends PureComponent<Props, *> {
</Box> </Box>
<OnboardingFooterWrapper> <OnboardingFooterWrapper>
<Button outlineGrey onClick={() => redoGenuineCheck()}> <Button outlineGrey onClick={() => redoGenuineCheck()}>
{t('app:common.back')} {t('common.back')}
</Button> </Button>
<ExternalLinkButton <ExternalLinkButton
danger danger
ml="auto" ml="auto"
label={t('onboarding:genuineCheck.buttons.contactSupport')} label={t('onboarding.genuineCheck.buttons.contactSupport')}
url={urls.contactSupport} url={urls.contactSupport}
/> />
</OnboardingFooterWrapper> </OnboardingFooterWrapper>

8
src/components/Onboarding/steps/GenuineCheck/GenuineCheckUnavailable.js

@ -27,7 +27,7 @@ export function GenuineCheckUnavailableFooter({
return ( return (
<OnboardingFooterWrapper> <OnboardingFooterWrapper>
<Button outlineGrey onClick={() => prevStep()}> <Button outlineGrey onClick={() => prevStep()}>
{t('app:common.back')} {t('common.back')}
</Button> </Button>
<Box horizontal ml="auto"> <Box horizontal ml="auto">
<Button <Button
@ -36,10 +36,10 @@ export function GenuineCheckUnavailableFooter({
onClick={() => nextStep()} onClick={() => nextStep()}
mx={2} mx={2}
> >
{t('app:common.skipThisStep')} {t('common.skipThisStep')}
</Button> </Button>
<Button onClick={nextStep} disabled primary> <Button onClick={nextStep} disabled primary>
{t('app:common.continue')} {t('common.continue')}
</Button> </Button>
</Box> </Box>
</OnboardingFooterWrapper> </OnboardingFooterWrapper>
@ -81,7 +81,7 @@ export function GenuineCheckUnavailableMessage({
}) })
}} }}
> >
{t('app:common.retry')} {t('common.retry')}
</FakeLink> </FakeLink>
</Box> </Box>
) )

20
src/components/Onboarding/steps/GenuineCheck/index.js

@ -59,12 +59,12 @@ class GenuineCheck extends PureComponent<StepProps, State> {
const { t } = this.props const { t } = this.props
return [ return [
{ {
label: t('app:common.labelYes'), label: t('common.labelYes'),
key: 'yes', key: 'yes',
pass: true, pass: true,
}, },
{ {
label: t('app:common.labelNo'), label: t('common.labelNo'),
key: 'no', key: 'no',
pass: false, pass: false,
}, },
@ -174,18 +174,18 @@ class GenuineCheck extends PureComponent<StepProps, State> {
deviceType={onboarding.isLedgerNano ? 'Nano S' : 'Blue'} deviceType={onboarding.isLedgerNano ? 'Nano S' : 'Blue'}
/> />
<StepContainerInner> <StepContainerInner>
<Title>{t('onboarding:genuineCheck.title')}</Title> <Title>{t('onboarding.genuineCheck.title')}</Title>
{onboarding.flowType === 'restoreDevice' ? ( {onboarding.flowType === 'restoreDevice' ? (
<Description>{t('onboarding:genuineCheck.descRestore')}</Description> <Description>{t('onboarding.genuineCheck.descRestore')}</Description>
) : ( ) : (
<Description>{t('onboarding:genuineCheck.descGeneric')}</Description> <Description>{t('onboarding.genuineCheck.descGeneric')}</Description>
)} )}
<Box mt={5}> <Box mt={5}>
<GenuineCheckCardWrapper> <GenuineCheckCardWrapper>
<Box justify="center"> <Box justify="center">
<Box horizontal> <Box horizontal>
<IconOptionRow>{'1.'}</IconOptionRow> <IconOptionRow>{'1.'}</IconOptionRow>
<CardTitle>{t('onboarding:genuineCheck.step1.title')}</CardTitle> <CardTitle>{t('onboarding.genuineCheck.step1.title')}</CardTitle>
</Box> </Box>
</Box> </Box>
<Box justify="center"> <Box justify="center">
@ -204,7 +204,7 @@ class GenuineCheck extends PureComponent<StepProps, State> {
<IconOptionRow color={!genuine.pinStepPass ? 'grey' : 'wallet'}> <IconOptionRow color={!genuine.pinStepPass ? 'grey' : 'wallet'}>
{'2.'} {'2.'}
</IconOptionRow> </IconOptionRow>
<CardTitle>{t('onboarding:genuineCheck.step2.title')}</CardTitle> <CardTitle>{t('onboarding.genuineCheck.step2.title')}</CardTitle>
</Box> </Box>
</Box> </Box>
<Box justify="center"> <Box justify="center">
@ -228,7 +228,7 @@ class GenuineCheck extends PureComponent<StepProps, State> {
<IconOptionRow color={!genuine.recoveryStepPass ? 'grey' : 'wallet'}> <IconOptionRow color={!genuine.recoveryStepPass ? 'grey' : 'wallet'}>
{'3.'} {'3.'}
</IconOptionRow> </IconOptionRow>
<CardTitle>{t('onboarding:genuineCheck.step3.title')}</CardTitle> <CardTitle>{t('onboarding.genuineCheck.step3.title')}</CardTitle>
</Box> </Box>
</Box> </Box>
{genuine.recoveryStepPass && ( {genuine.recoveryStepPass && (
@ -237,7 +237,7 @@ class GenuineCheck extends PureComponent<StepProps, State> {
<Box horizontal align="center" flow={1} color={colors.wallet}> <Box horizontal align="center" flow={1} color={colors.wallet}>
<IconCheck size={16} /> <IconCheck size={16} />
<Box ff="Open Sans|SemiBold" fontSize={4}> <Box ff="Open Sans|SemiBold" fontSize={4}>
{t('onboarding:genuineCheck.isGenuinePassed')} {t('onboarding.genuineCheck.isGenuinePassed')}
</Box> </Box>
</Box> </Box>
) : genuine.genuineCheckUnavailable ? ( ) : genuine.genuineCheckUnavailable ? (
@ -250,7 +250,7 @@ class GenuineCheck extends PureComponent<StepProps, State> {
disabled={!genuine.recoveryStepPass} disabled={!genuine.recoveryStepPass}
onClick={this.handleOpenGenuineCheckModal} onClick={this.handleOpenGenuineCheckModal}
> >
{t('onboarding:genuineCheck.buttons.genuineCheck')} {t('onboarding.genuineCheck.buttons.genuineCheck')}
</Button> </Button>
)} )}
</Box> </Box>

10
src/components/Onboarding/steps/Init.js

@ -30,7 +30,7 @@ class Init extends PureComponent<StepProps, *> {
{ {
key: 'newDevice', key: 'newDevice',
icon: <IconPlus size={20} />, icon: <IconPlus size={20} />,
title: t('onboarding:init.newDevice.title'), title: t('onboarding.init.newDevice.title'),
onClick: () => { onClick: () => {
jumpStep('selectDevice') jumpStep('selectDevice')
flowType('newDevice') flowType('newDevice')
@ -39,7 +39,7 @@ class Init extends PureComponent<StepProps, *> {
{ {
key: 'restoreDevice', key: 'restoreDevice',
icon: <IconRecover size={20} />, icon: <IconRecover size={20} />,
title: t('onboarding:init.restoreDevice.title'), title: t('onboarding.init.restoreDevice.title'),
onClick: () => { onClick: () => {
jumpStep('selectDevice') jumpStep('selectDevice')
flowType('restoreDevice') flowType('restoreDevice')
@ -48,7 +48,7 @@ class Init extends PureComponent<StepProps, *> {
{ {
key: 'initializedDevice', key: 'initializedDevice',
icon: <IconCheck size={20} />, icon: <IconCheck size={20} />,
title: t('onboarding:init.initializedDevice.title'), title: t('onboarding.init.initializedDevice.title'),
onClick: () => { onClick: () => {
jumpStep('selectDevice') jumpStep('selectDevice')
flowType('initializedDevice') flowType('initializedDevice')
@ -57,7 +57,7 @@ class Init extends PureComponent<StepProps, *> {
{ {
key: 'noDevice', key: 'noDevice',
icon: <IconExternalLink size={20} />, icon: <IconExternalLink size={20} />,
title: t('onboarding:noDevice.title'), title: t('onboarding.noDevice.title'),
onClick: () => { onClick: () => {
jumpStep('noDevice') jumpStep('noDevice')
flowType('noDevice') flowType('noDevice')
@ -77,7 +77,7 @@ class Init extends PureComponent<StepProps, *> {
} }
/> />
<Box m={5} style={{ maxWidth: 480 }}> <Box m={5} style={{ maxWidth: 480 }}>
<Title>{t('onboarding:init.title')}</Title> <Title>{t('onboarding.init.title')}</Title>
</Box> </Box>
<Box pt={4} flow={4}> <Box pt={4} flow={4}>
{optionCards.map(card => <OptionFlowCard key={card.key} card={card} />)} {optionCards.map(card => <OptionFlowCard key={card.key} card={card} />)}

10
src/components/Onboarding/steps/NoDevice.js

@ -26,7 +26,7 @@ class NoDevice extends PureComponent<StepProps, *> {
{ {
key: 'buyNew', key: 'buyNew',
icon: <IconCart size={20} />, icon: <IconCart size={20} />,
title: t('onboarding:noDevice.buyNew.title'), title: t('onboarding.noDevice.buyNew.title'),
onClick: () => { onClick: () => {
openURL(urls.noDeviceBuyNew) openURL(urls.noDeviceBuyNew)
}, },
@ -34,7 +34,7 @@ class NoDevice extends PureComponent<StepProps, *> {
{ {
key: 'trackOrder', key: 'trackOrder',
icon: <IconTruck size={20} />, icon: <IconTruck size={20} />,
title: t('onboarding:noDevice.trackOrder.title'), title: t('onboarding.noDevice.trackOrder.title'),
onClick: () => { onClick: () => {
openURL(urls.noDeviceTrackOrder) openURL(urls.noDeviceTrackOrder)
}, },
@ -42,7 +42,7 @@ class NoDevice extends PureComponent<StepProps, *> {
{ {
key: 'learnMore', key: 'learnMore',
icon: <IconInfoCircle size={20} />, icon: <IconInfoCircle size={20} />,
title: t('onboarding:noDevice.learnMore.title'), title: t('onboarding.noDevice.learnMore.title'),
onClick: () => { onClick: () => {
openURL(urls.noDeviceLearnMore) openURL(urls.noDeviceLearnMore)
}, },
@ -68,7 +68,7 @@ class NoDevice extends PureComponent<StepProps, *> {
} }
/> />
<Box m={5} style={{ maxWidth: 480 }}> <Box m={5} style={{ maxWidth: 480 }}>
<Title>{t('onboarding:noDevice.title')}</Title> <Title>{t('onboarding.noDevice.title')}</Title>
</Box> </Box>
<Box pt={4} flow={4}> <Box pt={4} flow={4}>
{optionCards.map(card => <OptionFlowCard key={card.key} card={card} />)} {optionCards.map(card => <OptionFlowCard key={card.key} card={card} />)}
@ -77,7 +77,7 @@ class NoDevice extends PureComponent<StepProps, *> {
</GrowScroll> </GrowScroll>
<OnboardingFooterWrapper> <OnboardingFooterWrapper>
<Button outlineGrey onClick={() => prevStep()} mr="auto"> <Button outlineGrey onClick={() => prevStep()} mr="auto">
{t('app:common.back')} {t('common.back')}
</Button> </Button>
</OnboardingFooterWrapper> </OnboardingFooterWrapper>
</Box> </Box>

6
src/components/Onboarding/steps/SelectDevice.js

@ -41,7 +41,7 @@ class SelectDevice extends PureComponent<StepProps, {}> {
<TrackPage category="Onboarding" name="Select Device" flowType={onboarding.flowType} /> <TrackPage category="Onboarding" name="Select Device" flowType={onboarding.flowType} />
<StepContainerInner> <StepContainerInner>
<Box mb={5}> <Box mb={5}>
<Title>{t('onboarding:selectDevice.title')}</Title> <Title>{t('onboarding.selectDevice.title')}</Title>
</Box> </Box>
<Box pt={4}> <Box pt={4}>
<Inner> <Inner>
@ -53,7 +53,7 @@ class SelectDevice extends PureComponent<StepProps, {}> {
<DeviceIcon> <DeviceIcon>
<img alt="" src={i('ledger-nano-onb.svg')} /> <img alt="" src={i('ledger-nano-onb.svg')} />
</DeviceIcon> </DeviceIcon>
<BlockTitle>{t('onboarding:selectDevice.ledgerNanoCard.title')}</BlockTitle> <BlockTitle>{t('onboarding.selectDevice.ledgerNanoCard.title')}</BlockTitle>
</DeviceContainer> </DeviceContainer>
<DeviceContainer <DeviceContainer
isActive={!onboarding.isLedgerNano && onboarding.isLedgerNano !== null} isActive={!onboarding.isLedgerNano && onboarding.isLedgerNano !== null}
@ -63,7 +63,7 @@ class SelectDevice extends PureComponent<StepProps, {}> {
<DeviceIcon> <DeviceIcon>
<img alt="" src={i('ledger-blue-onb.svg')} /> <img alt="" src={i('ledger-blue-onb.svg')} />
</DeviceIcon> </DeviceIcon>
<BlockTitle>{t('onboarding:selectDevice.ledgerBlueCard.title')}</BlockTitle> <BlockTitle>{t('onboarding.selectDevice.ledgerBlueCard.title')}</BlockTitle>
</DeviceContainer> </DeviceContainer>
</Inner> </Inner>
</Box> </Box>

12
src/components/Onboarding/steps/SelectPIN/SelectPINblue.js

@ -25,14 +25,14 @@ class SelectPIN extends PureComponent<Props, *> {
{ {
key: 'step1', key: 'step1',
icon: <IconOptionRow>{'1.'}</IconOptionRow>, icon: <IconOptionRow>{'1.'}</IconOptionRow>,
desc: t('onboarding:selectPIN.initialize.instructions.blue.step1'), desc: t('onboarding.selectPIN.initialize.instructions.blue.step1'),
}, },
{ {
key: 'step2', key: 'step2',
icon: <IconOptionRow>{'2.'}</IconOptionRow>, icon: <IconOptionRow>{'2.'}</IconOptionRow>,
desc: ( desc: (
<Box style={{ display: 'block' }}> <Box style={{ display: 'block' }}>
<Trans i18nKey="onboarding:selectPIN.initialize.instructions.blue.step2"> <Trans i18nKey="onboarding.selectPIN.initialize.instructions.blue.step2">
{'Tap on'} {'Tap on'}
<Text ff="Open Sans|SemiBold" color="dark"> <Text ff="Open Sans|SemiBold" color="dark">
{'Configure as new device'} {'Configure as new device'}
@ -44,7 +44,7 @@ class SelectPIN extends PureComponent<Props, *> {
{ {
key: 'step3', key: 'step3',
icon: <IconOptionRow>{'3.'}</IconOptionRow>, icon: <IconOptionRow>{'3.'}</IconOptionRow>,
desc: t('onboarding:selectPIN.initialize.instructions.blue.step3'), desc: t('onboarding.selectPIN.initialize.instructions.blue.step3'),
}, },
] ]
@ -52,17 +52,17 @@ class SelectPIN extends PureComponent<Props, *> {
{ {
key: 'note1', key: 'note1',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:selectPIN.disclaimer.note1'), desc: t('onboarding.selectPIN.disclaimer.note1'),
}, },
{ {
key: 'note2', key: 'note2',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:selectPIN.disclaimer.note2'), desc: t('onboarding.selectPIN.disclaimer.note2'),
}, },
{ {
key: 'note3', key: 'note3',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:selectPIN.disclaimer.note3'), desc: t('onboarding.selectPIN.disclaimer.note3'),
}, },
] ]

14
src/components/Onboarding/steps/SelectPIN/SelectPINnano.js

@ -25,19 +25,19 @@ class SelectPINnano extends PureComponent<Props, *> {
{ {
key: 'step1', key: 'step1',
icon: <IconOptionRow>{'1.'}</IconOptionRow>, icon: <IconOptionRow>{'1.'}</IconOptionRow>,
desc: t('onboarding:selectPIN.initialize.instructions.nano.step1'), desc: t('onboarding.selectPIN.initialize.instructions.nano.step1'),
}, },
{ {
key: 'step2', key: 'step2',
icon: <IconOptionRow>{'2.'}</IconOptionRow>, icon: <IconOptionRow>{'2.'}</IconOptionRow>,
desc: t('onboarding:selectPIN.initialize.instructions.nano.step2'), desc: t('onboarding.selectPIN.initialize.instructions.nano.step2'),
}, },
{ {
key: 'step3', key: 'step3',
icon: <IconOptionRow>{'3.'}</IconOptionRow>, icon: <IconOptionRow>{'3.'}</IconOptionRow>,
desc: ( desc: (
<Box style={{ display: 'block' }}> <Box style={{ display: 'block' }}>
<Trans i18nKey="onboarding:selectPIN.initialize.instructions.nano.step3"> <Trans i18nKey="onboarding.selectPIN.initialize.instructions.nano.step3">
{'Press the right button to select'} {'Press the right button to select'}
<Text ff="Open Sans|SemiBold" color="dark"> <Text ff="Open Sans|SemiBold" color="dark">
{'Configure as new device'} {'Configure as new device'}
@ -49,24 +49,24 @@ class SelectPINnano extends PureComponent<Props, *> {
{ {
key: 'step4', key: 'step4',
icon: <IconOptionRow>{'4.'}</IconOptionRow>, icon: <IconOptionRow>{'4.'}</IconOptionRow>,
desc: t('onboarding:selectPIN.initialize.instructions.nano.step4'), desc: t('onboarding.selectPIN.initialize.instructions.nano.step4'),
}, },
] ]
const disclaimerNotes = [ const disclaimerNotes = [
{ {
key: 'note1', key: 'note1',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:selectPIN.disclaimer.note1'), desc: t('onboarding.selectPIN.disclaimer.note1'),
}, },
{ {
key: 'note2', key: 'note2',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:selectPIN.disclaimer.note2'), desc: t('onboarding.selectPIN.disclaimer.note2'),
}, },
{ {
key: 'note3', key: 'note3',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:selectPIN.disclaimer.note3'), desc: t('onboarding.selectPIN.disclaimer.note3'),
}, },
] ]

12
src/components/Onboarding/steps/SelectPIN/SelectPINrestoreBlue.js

@ -25,14 +25,14 @@ class SelectPINrestoreBlue extends PureComponent<Props, *> {
{ {
key: 'step1', key: 'step1',
icon: <IconOptionRow>{'1.'}</IconOptionRow>, icon: <IconOptionRow>{'1.'}</IconOptionRow>,
desc: t('onboarding:selectPIN.restore.instructions.blue.step1'), desc: t('onboarding.selectPIN.restore.instructions.blue.step1'),
}, },
{ {
key: 'step2', key: 'step2',
icon: <IconOptionRow>{'2.'}</IconOptionRow>, icon: <IconOptionRow>{'2.'}</IconOptionRow>,
desc: ( desc: (
<Box style={{ display: 'block' }}> <Box style={{ display: 'block' }}>
<Trans i18nKey="onboarding:selectPIN.restore.instructions.blue.step2"> <Trans i18nKey="onboarding.selectPIN.restore.instructions.blue.step2">
{'Tap on'} {'Tap on'}
<Text ff="Open Sans|SemiBold" color="dark"> <Text ff="Open Sans|SemiBold" color="dark">
{'Restore configuration'} {'Restore configuration'}
@ -44,7 +44,7 @@ class SelectPINrestoreBlue extends PureComponent<Props, *> {
{ {
key: 'step3', key: 'step3',
icon: <IconOptionRow>{'3.'}</IconOptionRow>, icon: <IconOptionRow>{'3.'}</IconOptionRow>,
desc: t('onboarding:selectPIN.restore.instructions.blue.step3'), desc: t('onboarding.selectPIN.restore.instructions.blue.step3'),
}, },
] ]
@ -52,17 +52,17 @@ class SelectPINrestoreBlue extends PureComponent<Props, *> {
{ {
key: 'note1', key: 'note1',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:selectPIN.disclaimer.note1'), desc: t('onboarding.selectPIN.disclaimer.note1'),
}, },
{ {
key: 'note2', key: 'note2',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:selectPIN.disclaimer.note2'), desc: t('onboarding.selectPIN.disclaimer.note2'),
}, },
{ {
key: 'note3', key: 'note3',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:selectPIN.disclaimer.note3'), desc: t('onboarding.selectPIN.disclaimer.note3'),
}, },
] ]

14
src/components/Onboarding/steps/SelectPIN/SelectPINrestoreNano.js

@ -25,19 +25,19 @@ class SelectPINrestoreNano extends PureComponent<Props, *> {
{ {
key: 'step1', key: 'step1',
icon: <IconOptionRow>{'1.'}</IconOptionRow>, icon: <IconOptionRow>{'1.'}</IconOptionRow>,
desc: t('onboarding:selectPIN.restore.instructions.nano.step1'), desc: t('onboarding.selectPIN.restore.instructions.nano.step1'),
}, },
{ {
key: 'step2', key: 'step2',
icon: <IconOptionRow>{'2.'}</IconOptionRow>, icon: <IconOptionRow>{'2.'}</IconOptionRow>,
desc: t('onboarding:selectPIN.restore.instructions.nano.step2'), desc: t('onboarding.selectPIN.restore.instructions.nano.step2'),
}, },
{ {
key: 'step3', key: 'step3',
icon: <IconOptionRow>{'3.'}</IconOptionRow>, icon: <IconOptionRow>{'3.'}</IconOptionRow>,
desc: ( desc: (
<Box style={{ display: 'block' }}> <Box style={{ display: 'block' }}>
<Trans i18nKey="onboarding:selectPIN.restore.instructions.nano.step3"> <Trans i18nKey="onboarding.selectPIN.restore.instructions.nano.step3">
{'Press the left button to cancel'} {'Press the left button to cancel'}
<Text ff="Open Sans|SemiBold" color="dark"> <Text ff="Open Sans|SemiBold" color="dark">
{'Initialize as new device?'} {'Initialize as new device?'}
@ -53,24 +53,24 @@ class SelectPINrestoreNano extends PureComponent<Props, *> {
{ {
key: 'step4', key: 'step4',
icon: <IconOptionRow>{'4.'}</IconOptionRow>, icon: <IconOptionRow>{'4.'}</IconOptionRow>,
desc: t('onboarding:selectPIN.restore.instructions.nano.step4'), desc: t('onboarding.selectPIN.restore.instructions.nano.step4'),
}, },
] ]
const disclaimerNotes = [ const disclaimerNotes = [
{ {
key: 'note1', key: 'note1',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:selectPIN.disclaimer.note1'), desc: t('onboarding.selectPIN.disclaimer.note1'),
}, },
{ {
key: 'note2', key: 'note2',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:selectPIN.disclaimer.note2'), desc: t('onboarding.selectPIN.disclaimer.note2'),
}, },
{ {
key: 'note3', key: 'note3',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:selectPIN.disclaimer.note3'), desc: t('onboarding.selectPIN.disclaimer.note3'),
}, },
] ]

4
src/components/Onboarding/steps/SelectPIN/index.js

@ -29,14 +29,14 @@ export default (props: StepProps) => {
/> />
{onboarding.flowType === 'restoreDevice' ? ( {onboarding.flowType === 'restoreDevice' ? (
<Box grow alignItems="center"> <Box grow alignItems="center">
<Title>{t('onboarding:selectPIN.restore.title')}</Title> <Title>{t('onboarding.selectPIN.restore.title')}</Title>
<Box align="center" mt={7}> <Box align="center" mt={7}>
{onboarding.isLedgerNano ? <SelectPINrestoreNano /> : <SelectPINrestoreBlue />} {onboarding.isLedgerNano ? <SelectPINrestoreNano /> : <SelectPINrestoreBlue />}
</Box> </Box>
</Box> </Box>
) : ( ) : (
<Box grow alignItems="center"> <Box grow alignItems="center">
<Title>{t('onboarding:selectPIN.initialize.title')}</Title> <Title>{t('onboarding.selectPIN.initialize.title')}</Title>
<Box align="center" mt={7}> <Box align="center" mt={7}>
{onboarding.isLedgerNano ? <SelectPINnano /> : <SelectPINblue />} {onboarding.isLedgerNano ? <SelectPINnano /> : <SelectPINblue />}
</Box> </Box>

16
src/components/Onboarding/steps/SetPassword.js

@ -85,17 +85,17 @@ class SetPassword extends PureComponent<Props, State> {
{ {
key: 'note1', key: 'note1',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:setPassword.disclaimer.note1'), desc: t('onboarding.setPassword.disclaimer.note1'),
}, },
{ {
key: 'note2', key: 'note2',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:setPassword.disclaimer.note2'), desc: t('onboarding.setPassword.disclaimer.note2'),
}, },
{ {
key: 'note3', key: 'note3',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:setPassword.disclaimer.note3'), desc: t('onboarding.setPassword.disclaimer.note3'),
}, },
] ]
@ -110,9 +110,9 @@ class SetPassword extends PureComponent<Props, State> {
<StepContainerInner> <StepContainerInner>
<Fragment> <Fragment>
<Box alignItems="center"> <Box alignItems="center">
<Title>{t('onboarding:setPassword.title')}</Title> <Title>{t('onboarding.setPassword.title')}</Title>
<Description style={{ maxWidth: 620 }}> <Description style={{ maxWidth: 620 }}>
{t('onboarding:setPassword.desc')} {t('onboarding.setPassword.desc')}
</Description> </Description>
</Box> </Box>
<Box align="center" mt={2}> <Box align="center" mt={2}>
@ -134,7 +134,7 @@ class SetPassword extends PureComponent<Props, State> {
<OnboardingFooterWrapper> <OnboardingFooterWrapper>
<Button outlineGrey onClick={() => prevStep()}> <Button outlineGrey onClick={() => prevStep()}>
{t('app:common.back')} {t('common.back')}
</Button> </Button>
<Box horizontal ml="auto"> <Box horizontal ml="auto">
<Button <Button
@ -143,14 +143,14 @@ class SetPassword extends PureComponent<Props, State> {
disabled={false} disabled={false}
mx={2} mx={2}
> >
{t('app:common.skipThisStep')} {t('common.skipThisStep')}
</Button> </Button>
<Button <Button
onClick={this.handleSave} onClick={this.handleSave}
disabled={!this.isValid() || !newPassword.length || !confirmPassword.length} disabled={!this.isValid() || !newPassword.length || !confirmPassword.length}
primary primary
> >
{t('app:common.continue')} {t('common.continue')}
</Button> </Button>
</Box> </Box>
</OnboardingFooterWrapper> </OnboardingFooterWrapper>

4
src/components/Onboarding/steps/Start.js

@ -21,10 +21,10 @@ export default (props: StepProps) => {
icon={<img src={i('ledgerlive-logo.svg')} alt="" width={50} height={50} />} icon={<img src={i('ledgerlive-logo.svg')} alt="" width={50} height={50} />}
/> />
<Box my={5}> <Box my={5}>
<Title>{t('onboarding:start.title')}</Title> <Title>{t('onboarding.start.title')}</Title>
</Box> </Box>
<Button primary onClick={() => jumpStep('init')}> <Button primary onClick={() => jumpStep('init')}>
{t('onboarding:start.startBtn')} {t('onboarding.start.startBtn')}
</Button> </Button>
</Box> </Box>
</Box> </Box>

18
src/components/Onboarding/steps/WriteSeed/WriteSeedBlue.js

@ -31,14 +31,14 @@ class WriteSeedBlue extends PureComponent<Props, *> {
{ {
key: 'step1', key: 'step1',
icon: <IconOptionRow>{'1.'}</IconOptionRow>, icon: <IconOptionRow>{'1.'}</IconOptionRow>,
desc: t('onboarding:writeSeed.initialize.blue.step1'), desc: t('onboarding.writeSeed.initialize.blue.step1'),
}, },
{ {
key: 'step2', key: 'step2',
icon: <IconOptionRow>{'2.'}</IconOptionRow>, icon: <IconOptionRow>{'2.'}</IconOptionRow>,
desc: ( desc: (
<Box style={{ display: 'block' }}> <Box style={{ display: 'block' }}>
<Trans i18nKey="onboarding:writeSeed.initialize.blue.step2"> <Trans i18nKey="onboarding.writeSeed.initialize.blue.step2">
{'Tap'} {'Tap'}
<Text ff="Open Sans|SemiBold" color="dark"> <Text ff="Open Sans|SemiBold" color="dark">
{'Next'} {'Next'}
@ -55,37 +55,37 @@ class WriteSeedBlue extends PureComponent<Props, *> {
{ {
key: 'step3', key: 'step3',
icon: <IconOptionRow>{'3.'}</IconOptionRow>, icon: <IconOptionRow>{'3.'}</IconOptionRow>,
desc: t('onboarding:writeSeed.initialize.blue.step3'), desc: t('onboarding.writeSeed.initialize.blue.step3'),
}, },
] ]
const disclaimerNotes = [ const disclaimerNotes = [
{ {
key: 'note1', key: 'note1',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:writeSeed.disclaimer.note1'), desc: t('onboarding.writeSeed.disclaimer.note1'),
}, },
{ {
key: 'note2', key: 'note2',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:writeSeed.disclaimer.note2'), desc: t('onboarding.writeSeed.disclaimer.note2'),
}, },
{ {
key: 'note3', key: 'note3',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:writeSeed.disclaimer.note3'), desc: t('onboarding.writeSeed.disclaimer.note3'),
}, },
{ {
key: 'note4', key: 'note4',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:writeSeed.disclaimer.note4'), desc: t('onboarding.writeSeed.disclaimer.note4'),
}, },
] ]
return ( return (
<Fragment> <Fragment>
<Box mb={3}> <Box mb={3}>
<Title>{t('onboarding:writeSeed.initialize.title')}</Title> <Title>{t('onboarding.writeSeed.initialize.title')}</Title>
<Description>{t('onboarding:writeSeed.initialize.desc')}</Description> <Description>{t('onboarding.writeSeed.initialize.desc')}</Description>
</Box> </Box>
<Box align="center"> <Box align="center">
<Inner style={{ width: 760 }}> <Inner style={{ width: 760 }}>

18
src/components/Onboarding/steps/WriteSeed/WriteSeedNano.js

@ -33,7 +33,7 @@ class WriteSeedNano extends PureComponent<Props, *> {
icon: <IconOptionRow>{'1.'}</IconOptionRow>, icon: <IconOptionRow>{'1.'}</IconOptionRow>,
desc: ( desc: (
<Box style={{ display: 'block' }}> <Box style={{ display: 'block' }}>
<Trans i18nKey="onboarding:writeSeed.initialize.nano.step1"> <Trans i18nKey="onboarding.writeSeed.initialize.nano.step1">
{'Copy the word displayed below'} {'Copy the word displayed below'}
<Text ff="Open Sans|SemiBold" color="dark"> <Text ff="Open Sans|SemiBold" color="dark">
{'Word #1'} {'Word #1'}
@ -48,7 +48,7 @@ class WriteSeedNano extends PureComponent<Props, *> {
icon: <IconOptionRow>{'2.'}</IconOptionRow>, icon: <IconOptionRow>{'2.'}</IconOptionRow>,
desc: ( desc: (
<Box style={{ display: 'block' }}> <Box style={{ display: 'block' }}>
<Trans i18nKey="onboarding:writeSeed.initialize.nano.step2"> <Trans i18nKey="onboarding.writeSeed.initialize.nano.step2">
{'Press the right button to display'} {'Press the right button to display'}
<Text ff="Open Sans|SemiBold" color="dark"> <Text ff="Open Sans|SemiBold" color="dark">
{'Word #2'} {'Word #2'}
@ -61,37 +61,37 @@ class WriteSeedNano extends PureComponent<Props, *> {
{ {
key: 'step3', key: 'step3',
icon: <IconOptionRow>{'3.'}</IconOptionRow>, icon: <IconOptionRow>{'3.'}</IconOptionRow>,
desc: t('onboarding:writeSeed.initialize.nano.step3'), desc: t('onboarding.writeSeed.initialize.nano.step3'),
}, },
] ]
const disclaimerNotes = [ const disclaimerNotes = [
{ {
key: 'note1', key: 'note1',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:writeSeed.disclaimer.note1'), desc: t('onboarding.writeSeed.disclaimer.note1'),
}, },
{ {
key: 'note2', key: 'note2',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:writeSeed.disclaimer.note2'), desc: t('onboarding.writeSeed.disclaimer.note2'),
}, },
{ {
key: 'note3', key: 'note3',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:writeSeed.disclaimer.note3'), desc: t('onboarding.writeSeed.disclaimer.note3'),
}, },
{ {
key: 'note4', key: 'note4',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:writeSeed.disclaimer.note4'), desc: t('onboarding.writeSeed.disclaimer.note4'),
}, },
] ]
return ( return (
<Fragment> <Fragment>
<Box mb={3}> <Box mb={3}>
<Title>{t('onboarding:writeSeed.initialize.title')}</Title> <Title>{t('onboarding.writeSeed.initialize.title')}</Title>
<Description>{t('onboarding:writeSeed.initialize.desc')}</Description> <Description>{t('onboarding.writeSeed.initialize.desc')}</Description>
</Box> </Box>
<Box align="center" mt={3}> <Box align="center" mt={3}>
<Inner style={{ width: 700 }}> <Inner style={{ width: 700 }}>

26
src/components/Onboarding/steps/WriteSeed/WriteSeedRestore.js

@ -34,14 +34,14 @@ class WriteSeedRestore extends PureComponent<Props, *> {
{ {
key: 'step1', key: 'step1',
icon: <IconOptionRow>{'1.'}</IconOptionRow>, icon: <IconOptionRow>{'1.'}</IconOptionRow>,
desc: t('onboarding:writeSeed.restore.nano.step1'), desc: t('onboarding.writeSeed.restore.nano.step1'),
}, },
{ {
key: 'step2', key: 'step2',
icon: <IconOptionRow>{'2.'}</IconOptionRow>, icon: <IconOptionRow>{'2.'}</IconOptionRow>,
desc: ( desc: (
<Box style={{ display: 'block' }}> <Box style={{ display: 'block' }}>
<Trans i18nKey="onboarding:writeSeed.restore.nano.step2"> <Trans i18nKey="onboarding.writeSeed.restore.nano.step2">
{'Select the first letters of'} {'Select the first letters of'}
<Text ff="Open Sans|SemiBold" color="dark"> <Text ff="Open Sans|SemiBold" color="dark">
{'Word #1'} {'Word #1'}
@ -56,7 +56,7 @@ class WriteSeedRestore extends PureComponent<Props, *> {
icon: <IconOptionRow>{'3.'}</IconOptionRow>, icon: <IconOptionRow>{'3.'}</IconOptionRow>,
desc: ( desc: (
<Box style={{ display: 'block' }}> <Box style={{ display: 'block' }}>
<Trans i18nKey="onboarding:writeSeed.restore.nano.step3"> <Trans i18nKey="onboarding.writeSeed.restore.nano.step3">
{'Select'} {'Select'}
<Text ff="Open Sans|SemiBold" color="dark"> <Text ff="Open Sans|SemiBold" color="dark">
{'Word #1'} {'Word #1'}
@ -69,54 +69,54 @@ class WriteSeedRestore extends PureComponent<Props, *> {
{ {
key: 'step4', key: 'step4',
icon: <IconOptionRow>{'4.'}</IconOptionRow>, icon: <IconOptionRow>{'4.'}</IconOptionRow>,
desc: t('onboarding:writeSeed.restore.nano.step4'), desc: t('onboarding.writeSeed.restore.nano.step4'),
}, },
] ]
const stepsBlue = [ const stepsBlue = [
{ {
key: 'step1', key: 'step1',
icon: <IconOptionRow>{'1.'}</IconOptionRow>, icon: <IconOptionRow>{'1.'}</IconOptionRow>,
desc: t('onboarding:writeSeed.restore.blue.step1'), desc: t('onboarding.writeSeed.restore.blue.step1'),
}, },
{ {
key: 'step2', key: 'step2',
icon: <IconOptionRow>{'2.'}</IconOptionRow>, icon: <IconOptionRow>{'2.'}</IconOptionRow>,
desc: t('onboarding:writeSeed.restore.blue.step2'), desc: t('onboarding.writeSeed.restore.blue.step2'),
}, },
{ {
key: 'step3', key: 'step3',
icon: <IconOptionRow>{'3.'}</IconOptionRow>, icon: <IconOptionRow>{'3.'}</IconOptionRow>,
desc: t('onboarding:writeSeed.restore.blue.step3'), desc: t('onboarding.writeSeed.restore.blue.step3'),
}, },
] ]
const disclaimerNotes = [ const disclaimerNotes = [
{ {
key: 'note1', key: 'note1',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:writeSeed.disclaimer.note1'), desc: t('onboarding.writeSeed.disclaimer.note1'),
}, },
{ {
key: 'note2', key: 'note2',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:writeSeed.disclaimer.note2'), desc: t('onboarding.writeSeed.disclaimer.note2'),
}, },
{ {
key: 'note3', key: 'note3',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:writeSeed.disclaimer.note3'), desc: t('onboarding.writeSeed.disclaimer.note3'),
}, },
{ {
key: 'note4', key: 'note4',
icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />, icon: <IconChevronRight size={12} style={{ color: colors.smoke }} />,
desc: t('onboarding:writeSeed.disclaimer.note4'), desc: t('onboarding.writeSeed.disclaimer.note4'),
}, },
] ]
return ( return (
<Fragment> <Fragment>
<Box mb={3}> <Box mb={3}>
<Title>{t('onboarding:writeSeed.restore.title')}</Title> <Title>{t('onboarding.writeSeed.restore.title')}</Title>
<Description>{t('onboarding:writeSeed.restore.desc')}</Description> <Description>{t('onboarding.writeSeed.restore.desc')}</Description>
</Box> </Box>
<Box align="center"> <Box align="center">
<Inner style={{ width: 760 }}> <Inner style={{ width: 760 }}>

2
src/components/OpenUserDataDirectoryBtn.js

@ -21,7 +21,7 @@ class OpenUserDataDirectoryBtn extends Component<{
const { t } = this.props const { t } = this.props
return ( return (
<Button primary small onClick={this.handleOpenUserDataDirectory}> <Button primary small onClick={this.handleOpenUserDataDirectory}>
{t('app:settings.openUserDataDirectory.btn')} {t('settings.openUserDataDirectory.btn')}
</Button> </Button>
) )
} }

2
src/components/OperationsList/AmountCell.js

@ -2,7 +2,7 @@
import React, { PureComponent } from 'react' import React, { PureComponent } from 'react'
import styled from 'styled-components' import styled from 'styled-components'
import { getOperationAmountNumber } from '@ledgerhq/live-common/lib/helpers/operation' import { getOperationAmountNumber } from '@ledgerhq/live-common/lib/operation'
import type { Currency, Unit, Operation } from '@ledgerhq/live-common/lib/types' import type { Currency, Unit, Operation } from '@ledgerhq/live-common/lib/types'
import Box from 'components/base/Box' import Box from 'components/base/Box'
import CounterValue from 'components/CounterValue' import CounterValue from 'components/CounterValue'

2
src/components/OperationsList/ConfirmationCheck.js

@ -59,7 +59,7 @@ class ConfirmationCheck extends PureComponent<{
renderTooltip = () => { renderTooltip = () => {
const { t, isConfirmed } = this.props const { t, isConfirmed } = this.props
return t(isConfirmed ? 'app:operationDetails.confirmed' : 'app:operationDetails.notConfirmed') return t(isConfirmed ? 'operationDetails.confirmed' : 'operationDetails.notConfirmed')
} }
render() { render() {

2
src/components/OperationsList/DateCell.js

@ -29,7 +29,7 @@ class DateCell extends PureComponent<Props> {
return ( return (
<Cell> <Cell>
<Box ff="Open Sans|SemiBold" fontSize={3} color="smoke"> <Box ff="Open Sans|SemiBold" fontSize={3} color="smoke">
{t(`app:operation.type.${operation.type}`)} {t(`operation.type.${operation.type}`)}
</Box> </Box>
<OperationDate date={operation.date} /> <OperationDate date={operation.date} />
</Cell> </Cell>

6
src/components/OperationsList/index.js

@ -8,7 +8,7 @@ import { translate } from 'react-i18next'
import { import {
groupAccountOperationsByDay, groupAccountOperationsByDay,
groupAccountsOperationsByDay, groupAccountsOperationsByDay,
} from '@ledgerhq/live-common/lib/helpers/account' } from '@ledgerhq/live-common/lib/account'
import type { Operation, Account } from '@ledgerhq/live-common/lib/types' import type { Operation, Account } from '@ledgerhq/live-common/lib/types'
@ -139,13 +139,13 @@ export class OperationsList extends PureComponent<Props, State> {
) : null} ) : null}
{!groupedOperations.completed ? ( {!groupedOperations.completed ? (
<ShowMore onClick={this.fetchMoreOperations}> <ShowMore onClick={this.fetchMoreOperations}>
<span>{t('app:common.showMore')}</span> <span>{t('common.showMore')}</span>
<IconAngleDown size={12} /> <IconAngleDown size={12} />
</ShowMore> </ShowMore>
) : ( ) : (
<Box p={6} align="center"> <Box p={6} align="center">
<Text ff="Open Sans" fontSize={3}> <Text ff="Open Sans" fontSize={3}>
{t('app:operationList.noMoreOperations')} {t('operationList.noMoreOperations')}
</Text> </Text>
</Box> </Box>
)} )}

2
src/components/PillsDaysCount.js

@ -24,7 +24,7 @@ class PillsDaysCount extends PureComponent<Props> {
items={Object.keys(timeRangeDaysByKey).map((key: TimeRange) => ({ items={Object.keys(timeRangeDaysByKey).map((key: TimeRange) => ({
key, key,
value: timeRangeDaysByKey[key], value: timeRangeDaysByKey[key],
label: t(`app:time.${key}`), label: t(`time.${key}`),
}))} }))}
activeKey={selected} activeKey={selected}
onChange={onChange} onChange={onChange}

44
src/components/QRCodeExporter.js

@ -1,24 +1,32 @@
// @flow // @flow
import React, { PureComponent } from 'react' import React, { PureComponent } from 'react'
import { Buffer } from 'buffer'
import { createSelector } from 'reselect' import { createSelector } from 'reselect'
import { connect } from 'react-redux' import { connect } from 'react-redux'
import { accountsSelector } from 'reducers/accounts' import { accountsSelector } from 'reducers/accounts'
import { makeChunks } from '@ledgerhq/live-common/lib/bridgestream/exporter' import { exportSettingsSelector } from 'reducers/settings'
import { encode } from '@ledgerhq/live-common/lib/cross'
import { dataToFrames } from 'qrloop/exporter'
import QRCode from './base/QRCode' import QRCode from './base/QRCode'
const mapStateToProps = createSelector(accountsSelector, accounts => ({ const mapStateToProps = createSelector(
chunks: makeChunks({ accountsSelector,
accounts, exportSettingsSelector,
exporterName: 'desktop', (accounts, settings) => ({
exporterVersion: __APP_VERSION__, chunks: dataToFrames(
pad: true, encode({
accounts,
settings,
exporterName: 'desktop',
exporterVersion: __APP_VERSION__,
}),
200,
4,
),
}), }),
})) )
const LOW_FPS = 2
const HIGH_FPS = 8
class QRCodeExporter extends PureComponent< class QRCodeExporter extends PureComponent<
{ {
@ -31,23 +39,21 @@ class QRCodeExporter extends PureComponent<
}, },
> { > {
static defaultProps = { static defaultProps = {
size: 440, size: 460,
} }
state = { state = {
frame: 0, frame: 0,
fps: HIGH_FPS, fps: 3,
} }
componentDidMount() { componentDidMount() {
console.log(`BRIDGESTREAM_DATA=${btoa(JSON.stringify(this.props.chunks))}`) // eslint-disable-line const BRIDGESTREAM_DATA = Buffer.from(JSON.stringify(this.props.chunks)).toString('base64')
console.log(`BRIDGESTREAM_DATA=${BRIDGESTREAM_DATA}`) // eslint-disable-line
const nextFrame = ({ frame, fps }, { chunks }) => { const nextFrame = ({ frame }, { chunks }) => {
frame = (frame + 1) % chunks.length frame = (frame + 1) % chunks.length
return { return { frame }
frame,
fps: frame === 0 ? (fps === LOW_FPS ? HIGH_FPS : LOW_FPS) : fps,
}
} }
let lastT let lastT

2
src/components/RecipientAddress/index.js

@ -4,7 +4,7 @@ import type { BigNumber } from 'bignumber.js'
import React, { PureComponent, Fragment } from 'react' import React, { PureComponent, Fragment } from 'react'
import styled from 'styled-components' import styled from 'styled-components'
import noop from 'lodash/noop' import noop from 'lodash/noop'
import { decodeURIScheme } from '@ledgerhq/live-common/lib/helpers/currencies' import { decodeURIScheme } from '@ledgerhq/live-common/lib/currencies'
import type { CryptoCurrency } from '@ledgerhq/live-common/lib/types' import type { CryptoCurrency } from '@ledgerhq/live-common/lib/types'
import { radii } from 'styles/theme' import { radii } from 'styles/theme'

23
src/components/RenderError.js

@ -75,7 +75,7 @@ class RenderError extends PureComponent<
<img alt="" src={i('crash-screen.svg')} width={380} /> <img alt="" src={i('crash-screen.svg')} width={380} />
<Space of={40} /> <Space of={40} />
<Box ff="Museo Sans|Regular" fontSize={7} color="dark"> <Box ff="Museo Sans|Regular" fontSize={7} color="dark">
{t('app:crash.oops')} {t('crash.oops')}
</Box> </Box>
<Space of={15} /> <Space of={15} />
<Box <Box
@ -85,25 +85,20 @@ class RenderError extends PureComponent<
color="smoke" color="smoke"
fontSize={4} fontSize={4}
> >
{t('app:crash.uselessText')} {t('crash.uselessText')}
</Box> </Box>
<Space of={30} /> <Space of={30} />
<Box horizontal flow={2}> <Box horizontal flow={2}>
<Button small primary onClick={this.handleRestart}> <Button small primary onClick={this.handleRestart}>
{t('app:crash.restart')} {t('crash.restart')}
</Button> </Button>
<ExportLogsBtn withoutAppData={withoutAppData} /> <ExportLogsBtn withoutAppData={withoutAppData} />
<ExternalLinkButton <ExternalLinkButton small primary label={t('crash.support')} url={urls.contactSupport} />
small
primary
label={t('app:crash.support')}
url={urls.contactSupport}
/>
<Button small primary onClick={this.github}> <Button small primary onClick={this.github}>
{t('app:crash.github')} {t('crash.github')}
</Button> </Button>
<Button small danger onClick={this.handleOpenHardResetModal}> <Button small danger onClick={this.handleOpenHardResetModal}>
{t('app:common.reset')} {t('common.reset')}
</Button> </Button>
</Box> </Box>
<ConfirmModal <ConfirmModal
@ -114,9 +109,9 @@ class RenderError extends PureComponent<
onClose={this.handleCloseHardResetModal} onClose={this.handleCloseHardResetModal}
onReject={this.handleCloseHardResetModal} onReject={this.handleCloseHardResetModal}
onConfirm={this.handleHardReset} onConfirm={this.handleHardReset}
confirmText={t('app:common.reset')} confirmText={t('common.reset')}
title={t('app:settings.hardResetModal.title')} title={t('settings.hardResetModal.title')}
desc={t('app:settings.hardResetModal.desc')} desc={t('settings.hardResetModal.desc')}
renderIcon={this.hardResetIconRender} renderIcon={this.hardResetIconRender}
/> />
<Box my={6}> <Box my={6}>

2
src/components/RequestAmount/index.js

@ -180,7 +180,7 @@ export class RequestAmount extends PureComponent<Props> {
{withMax && ( {withMax && (
<Box grow justify="flex-end"> <Box grow justify="flex-end">
<Button primary onClick={this.handleClickMax}> <Button primary onClick={this.handleClickMax}>
{t('app:common.max')} {t('common.max')}
</Button> </Button>
</Box> </Box>
)} )}

4
src/components/SelectAccount/index.js

@ -74,9 +74,9 @@ const RawSelectAccount = ({ accounts, onChange, value, t, ...props }: Props) =>
getOptionValue={getOptionValue} getOptionValue={getOptionValue}
renderValue={renderOption} renderValue={renderOption}
renderOption={renderOption} renderOption={renderOption}
placeholder={t('app:common.selectAccount')} placeholder={t('common.selectAccount')}
noOptionsMessage={({ inputValue }) => noOptionsMessage={({ inputValue }) =>
t('app:common.selectAccountNoOption', { accountName: inputValue }) t('common.selectAccountNoOption', { accountName: inputValue })
} }
onChange={onChange} onChange={onChange}
/> />

5
src/components/SelectCurrency/index.js

@ -39,10 +39,9 @@ const SelectCurrency = ({ onChange, value, t, placeholder, currencies, ...props
renderOption={renderOption} renderOption={renderOption}
renderValue={renderOption} renderValue={renderOption}
options={options} options={options}
placeholder={placeholder || t('app:common.selectCurrency')} placeholder={placeholder || t('common.selectCurrency')}
data-e2e="test"
noOptionsMessage={({ inputValue }: { inputValue: string }) => noOptionsMessage={({ inputValue }: { inputValue: string }) =>
t('app:common.selectCurrencyNoOption', { currencyName: inputValue }) t('common.selectCurrencyNoOption', { currencyName: inputValue })
} }
onChange={item => onChange(item ? item.currency : null)} onChange={item => onChange(item ? item.currency : null)}
{...props} {...props}

6
src/components/SelectExchange.js

@ -127,11 +127,11 @@ class SelectExchange extends Component<
options={options} options={options}
onChange={onChange} onChange={onChange}
isLoading={isLoading} isLoading={isLoading}
placeholder={t('app:common.selectExchange')} placeholder={t('common.selectExchange')}
noOptionsMessage={({ inputValue }) => noOptionsMessage={({ inputValue }) =>
inputValue inputValue
? t('app:common.selectExchangeNoOption', { exchangeName: inputValue }) ? t('common.selectExchangeNoOption', { exchangeName: inputValue })
: t('app:common.selectExchangeNoOptionAtAll') : t('common.selectExchangeNoOptionAtAll')
} }
{...props} {...props}
/> />

22
src/components/SettingsPage/CleanButton.js

@ -3,11 +3,13 @@
import React, { Fragment, PureComponent } from 'react' import React, { Fragment, PureComponent } from 'react'
import { connect } from 'react-redux' import { connect } from 'react-redux'
import { translate } from 'react-i18next' import { translate } from 'react-i18next'
import logger from 'logger'
import type { T } from 'types/common' import type { T } from 'types/common'
import { cleanAccountsCache } from 'actions/accounts' import { cleanAccountsCache } from 'actions/accounts'
import Button from 'components/base/Button' import Button from 'components/base/Button'
import { ConfirmModal } from 'components/base/Modal' import { ConfirmModal } from 'components/base/Modal'
import { softReset } from 'helpers/reset' import { softReset } from 'helpers/reset'
import ResetFallbackModal from './ResetFallbackModal'
const mapDispatchToProps = { const mapDispatchToProps = {
cleanAccountsCache, cleanAccountsCache,
@ -20,36 +22,40 @@ type Props = {
type State = { type State = {
opened: boolean, opened: boolean,
fallbackOpened: boolean,
isLoading: boolean, isLoading: boolean,
} }
class CleanButton extends PureComponent<Props, State> { class CleanButton extends PureComponent<Props, State> {
state = { state = {
opened: false, opened: false,
fallbackOpened: false,
isLoading: false, isLoading: false,
} }
open = () => this.setState({ opened: true }) open = () => this.setState({ opened: true })
close = () => this.setState({ opened: false }) close = () => this.setState({ opened: false })
closeFallback = () => this.setState({ fallbackOpened: false })
action = async () => { action = async () => {
if (this.state.isLoading) return if (this.state.isLoading) return
try { try {
this.setState({ isLoading: true }) this.setState({ isLoading: true })
await softReset({ cleanAccountsCache: this.props.cleanAccountsCache }) await softReset({ cleanAccountsCache: this.props.cleanAccountsCache })
} finally { } catch (err) {
this.setState({ isLoading: false }) logger.error(err)
this.setState({ isLoading: false, fallbackOpened: true })
} }
} }
render() { render() {
const { t } = this.props const { t } = this.props
const { opened, isLoading } = this.state const { opened, isLoading, fallbackOpened } = this.state
return ( return (
<Fragment> <Fragment>
<Button small primary onClick={this.open} event="ClearCacheIntent"> <Button small primary onClick={this.open} event="ClearCacheIntent">
{t('app:settings.profile.softReset')} {t('settings.profile.softReset')}
</Button> </Button>
<ConfirmModal <ConfirmModal
@ -59,10 +65,12 @@ class CleanButton extends PureComponent<Props, State> {
onReject={this.close} onReject={this.close}
onConfirm={this.action} onConfirm={this.action}
isLoading={isLoading} isLoading={isLoading}
title={t('app:settings.softResetModal.title')} title={t('settings.softResetModal.title')}
subTitle={t('app:common.areYouSure')} subTitle={t('common.areYouSure')}
desc={t('app:settings.softResetModal.desc')} desc={t('settings.softResetModal.desc')}
/> />
<ResetFallbackModal isOpened={fallbackOpened} onClose={this.closeFallback} />
</Fragment> </Fragment>
) )
} }

2
src/components/SettingsPage/CounterValueSelect.js

@ -3,7 +3,7 @@
import React, { Fragment, PureComponent } from 'react' import React, { Fragment, PureComponent } from 'react'
import { connect } from 'react-redux' import { connect } from 'react-redux'
import { createStructuredSelector } from 'reselect' import { createStructuredSelector } from 'reselect'
import { listFiatCurrencies } from '@ledgerhq/live-common/lib/helpers/currencies' import { listFiatCurrencies } from '@ledgerhq/live-common/lib/currencies'
import type { Currency } from '@ledgerhq/live-common/lib/types' import type { Currency } from '@ledgerhq/live-common/lib/types'
import { setCounterValue } from 'actions/settings' import { setCounterValue } from 'actions/settings'
import { counterValueCurrencySelector } from 'reducers/settings' import { counterValueCurrencySelector } from 'reducers/settings'

10
src/components/SettingsPage/DisablePasswordModal.js

@ -69,15 +69,15 @@ class DisablePasswordModal extends PureComponent<Props, State> {
<form onSubmit={this.disablePassword}> <form onSubmit={this.disablePassword}>
<ModalBody onClose={onClose}> <ModalBody onClose={onClose}>
<ModalTitle data-e2e="disablePassword_modalTitle"> <ModalTitle data-e2e="disablePassword_modalTitle">
{t('app:password.disablePassword.title')} {t('password.disablePassword.title')}
</ModalTitle> </ModalTitle>
<ModalContent> <ModalContent>
<Box ff="Open Sans" color="smoke" fontSize={4} textAlign="center" px={4}> <Box ff="Open Sans" color="smoke" fontSize={4} textAlign="center" px={4}>
{t('app:password.disablePassword.desc')} {t('password.disablePassword.desc')}
<Box px={7} mt={4} flow={3}> <Box px={7} mt={4} flow={3}>
<Box flow={1}> <Box flow={1}>
<Label htmlFor="password"> <Label htmlFor="password">
{t('app:password.inputFields.currentPassword.label')} {t('password.inputFields.currentPassword.label')}
</Label> </Label>
<InputPassword <InputPassword
autoFocus autoFocus
@ -93,7 +93,7 @@ class DisablePasswordModal extends PureComponent<Props, State> {
</ModalContent> </ModalContent>
<ModalFooter horizontal align="center" justify="flex-end" flow={2}> <ModalFooter horizontal align="center" justify="flex-end" flow={2}>
<Button small type="button" onClick={onClose}> <Button small type="button" onClick={onClose}>
{t('app:common.cancel')} {t('common.cancel')}
</Button> </Button>
<Button <Button
small small
@ -101,7 +101,7 @@ class DisablePasswordModal extends PureComponent<Props, State> {
onClick={this.disablePassword} onClick={this.disablePassword}
disabled={!currentPassword && !incorrectPassword} disabled={!currentPassword && !incorrectPassword}
> >
{t('app:common.save')} {t('common.save')}
</Button> </Button>
</ModalFooter> </ModalFooter>
</ModalBody> </ModalBody>

Some files were not shown because too many files changed in this diff

Loading…
Cancel
Save