Browse Source

Merge pull request #733 from dasilvarosa/wording

Wording updates: onboarding and minor changes in app.
master
Meriadec Pillet 7 years ago
committed by GitHub
parent
commit
ef4a81ec63
No known key found for this signature in database GPG Key ID: 4AEE18F83AFDEB23
  1. 321
      src/components/DeviceConnect/index.js
  2. 64
      src/components/DeviceConnect/stories.js
  3. 2
      src/components/GenuineCheck.js
  4. 6
      static/i18n/en/app.yml
  5. 24
      static/i18n/en/onboarding.yml

321
src/components/DeviceConnect/index.js

@ -1,321 +0,0 @@
// @flow
import React, { PureComponent } from 'react'
import styled from 'styled-components'
import { Trans, translate } from 'react-i18next'
import type { CryptoCurrency } from '@ledgerhq/live-common/lib/types'
import type { T, Device } from 'types/common'
import noop from 'lodash/noop'
import Box from 'components/base/Box'
import Spinner from 'components/base/Spinner'
import CryptoCurrencyIcon from 'components/CryptoCurrencyIcon'
import TranslatedError from 'components/TranslatedError'
import IconCheck from 'icons/Check'
import IconExclamationCircle from 'icons/ExclamationCircle'
import IconInfoCircle from 'icons/InfoCircle'
import IconUsb from 'icons/Usb'
import IconHome from 'icons/Home'
import * as IconDevice from 'icons/device'
// TODO: CHECK IF COMPONENT CAN BE REMOVED
const Step = styled(Box).attrs({
borderRadius: 1,
justifyContent: 'center',
fontSize: 4,
})`
border: 1px solid
${p =>
p.validated
? p.theme.colors.wallet
: p.hasErrors
? p.theme.colors.alertRed
: p.theme.colors.fog};
`
const StepIcon = styled(Box).attrs({
alignItems: 'center',
justifyContent: 'center',
})`
width: 64px;
`
const StepContent = styled(Box).attrs({
color: 'dark',
horizontal: true,
alignItems: 'center',
})`
height: 60px;
line-height: 1.2;
strong {
font-weight: 600;
}
`
const ListDevices = styled(Box).attrs({
p: 3,
pt: 1,
flow: 2,
})``
const DeviceItem = styled(Box).attrs({
bg: 'lightGrey',
borderRadius: 1,
alignItems: 'center',
color: 'dark',
ff: 'Open Sans|SemiBold',
fontSize: 4,
horizontal: true,
pr: 3,
pl: 0,
})`
cursor: pointer;
height: 54px;
`
const DeviceIcon = styled(Box).attrs({
alignItems: 'center',
justifyContent: 'center',
color: 'graphite',
})`
width: 55px;
`
const DeviceSelected = styled(Box).attrs({
alignItems: 'center',
bg: p => (p.selected ? 'wallet' : 'white'),
color: 'white',
justifyContent: 'center',
})`
border-radius: 50%;
border: 1px solid ${p => (p.selected ? p.theme.colors.wallet : p.theme.colors.fog)};
height: 18px;
width: 18px;
`
const WrapperIconCurrency = styled(Box).attrs({
alignItems: 'center',
justifyContent: 'center',
})`
border: 1px solid ${p => p.theme.colors[p.color]};
border-radius: 8px;
height: 24px;
width: 24px;
`
const Info = styled(Box).attrs({
alignItems: 'center',
color: p => (p.hasErrors ? 'alertRed' : 'grey'),
flow: 2,
fontSize: 3,
horizontal: true,
ml: 1,
})`
strong {
font-weight: 600;
}
`
const StepCheck = ({ checked, hasErrors }: { checked: boolean, hasErrors?: boolean }) => (
<Box pr={5}>
{checked ? (
<Box color="wallet">
<IconCheck size={16} />
</Box>
) : hasErrors ? (
<Box color="alertRed">
<IconExclamationCircle size={16} />
</Box>
) : (
<Spinner color="grey" size={16} />
)}
</Box>
)
StepCheck.defaultProps = {
hasErrors: false,
}
type Props = {
appOpened: null | 'success' | 'fail',
genuineCheckStatus: null | 'success' | 'fail',
withGenuineCheck: boolean,
currency: CryptoCurrency,
devices: Device[],
deviceSelected: ?Device,
onChangeDevice: Device => void,
t: T,
error: ?Error,
}
const emitChangeDevice = props => {
const { onChangeDevice, deviceSelected, devices } = props
if (deviceSelected === null && devices.length > 0) {
onChangeDevice(devices[0])
}
}
class DeviceConnect extends PureComponent<Props> {
static defaultProps = {
accountName: null,
appOpened: null,
devices: [],
deviceSelected: null,
onChangeDevice: noop,
withGenuineCheck: false,
}
componentDidMount() {
emitChangeDevice(this.props)
}
componentWillReceiveProps(nextProps) {
emitChangeDevice(nextProps)
}
getStepState = stepStatus => ({
success: stepStatus === 'success',
fail: stepStatus === 'fail',
})
render() {
const {
deviceSelected,
genuineCheckStatus,
withGenuineCheck,
appOpened,
error,
currency,
t,
onChangeDevice,
devices,
} = this.props
const appState = this.getStepState(appOpened)
const genuineCheckState = this.getStepState(genuineCheckStatus)
const hasDevice = devices.length > 0
const hasMultipleDevices = devices.length > 1
// TODO: place custom wording in trans tags into yml file
/* eslint-disable react/jsx-no-literals */
return (
<Box flow={4} ff="Open Sans">
<Step validated={hasDevice}>
<StepContent>
<StepIcon>
<IconUsb size={36} />
</StepIcon>
<Box grow shrink>
<Trans i18nKey="app:deviceConnect.step1.connect" parent="div">
Connect and unlock your <strong>Ledger device</strong>
</Trans>
</Box>
<StepCheck checked={hasDevice} />
</StepContent>
{hasMultipleDevices && (
<ListDevices>
<Box color="graphite" fontSize={3}>
{t('app:deviceConnect.step1.choose', { count: devices.length })}
</Box>
<Box flow={2}>
{devices.map(d => {
const Icon = IconDevice[d.product.replace(/\s/g, '')]
return (
<DeviceItem key={d.path} onClick={() => onChangeDevice(d)}>
<DeviceIcon>
<Icon size={28} />
</DeviceIcon>
<Box grow noShrink>
{`${d.manufacturer} ${d.product}`}
</Box>
<Box>
<DeviceSelected selected={d === deviceSelected}>
<IconCheck size={10} />
</DeviceSelected>
</Box>
</DeviceItem>
)
})}
</Box>
</ListDevices>
)}
</Step>
<Step validated={appState.success} hasErrors={appState.fail}>
{currency ? (
<StepContent>
<StepIcon>
<WrapperIconCurrency>
<CryptoCurrencyIcon currency={currency} size={12} />
</WrapperIconCurrency>
</StepIcon>
<Box grow shrink>
<Trans i18nKey="deviceConnect:step2.open" parent="div">
{'Open the '}
<strong>{currency.name}</strong>
{' app on your device'}
</Trans>
</Box>
<StepCheck checked={appState.success} hasErrors={appState.fail} />
</StepContent>
) : (
<StepContent>
<StepIcon>
<WrapperIconCurrency>
<IconHome size={12} />
</WrapperIconCurrency>
</StepIcon>
<Box grow shrink>
<Trans i18nKey="app:dashboard.open" parent="div">
{'Navigate to the '}
<strong>{'dashboard'}</strong>
{' on your device'}
</Trans>
</Box>
<StepCheck checked={appState.success} hasErrors={appState.fail} />
</StepContent>
)}
</Step>
{/* GENUINE CHECK */}
{/* ------------- */}
{withGenuineCheck && (
<Step validated={genuineCheckState.success} hasErrors={genuineCheckState.fail}>
<StepContent>
<StepIcon>
<WrapperIconCurrency>
<IconCheck size={12} />
</WrapperIconCurrency>
</StepIcon>
<Box grow shrink>
<Trans i18nKey="deviceConnect:stepGenuine.open" parent="div">
{'Allow the '}
<strong>{'Ledger Manager'}</strong>
{' on your device'}
</Trans>
</Box>
<StepCheck checked={genuineCheckState.success} hasErrors={genuineCheckState.fail} />
</StepContent>
</Step>
)}
{error ? (
<Info hasErrors>
<IconInfoCircle size={12} />
<Box shrink selectable>
<TranslatedError error={error} />
</Box>
</Info>
) : null}
</Box>
)
}
}
export default translate()(DeviceConnect)

64
src/components/DeviceConnect/stories.js

@ -1,64 +0,0 @@
// @flow
import React from 'react'
import { storiesOf } from '@storybook/react'
import { select } from '@storybook/addon-knobs'
import { action } from '@storybook/addon-actions'
import { getCryptoCurrencyById } from '@ledgerhq/live-common/lib/helpers/currencies'
import { listCryptoCurrencies } from 'config/cryptocurrencies'
import type { Currency } from '@ledgerhq/live-common/lib/types'
import DeviceConnect from 'components/DeviceConnect'
const currencies = listCryptoCurrencies().map(c => c.id)
const stories = storiesOf('Components', module)
const devices = [
{
manufacturer: 'Ledger',
product: 'Nano S',
vendorId: '1',
productId: '11',
path: '111',
},
{
manufacturer: 'Ledger',
product: 'Blue',
vendorId: '2',
productId: '22',
path: '222',
},
{
manufacturer: 'Ledger',
product: 'Nano S',
vendorId: '3',
productId: '33',
path: '333',
},
]
stories.add('DeviceConnect', () => (
<Wrapper currencyId={select('currencyId', currencies, 'bitcoin_testnet')}>
{({ currency }) => (
<DeviceConnect
currency={currency}
appOpened={select('appOpened', ['', 'success', 'fail'], '')}
devices={devices.slice(0, Number(select('devices', [0, 1, 2, 3], '0')))}
deviceSelected={devices[select('deviceSelected', ['', 0, 1, 2], '')] || null}
onChangeDevice={action('onChangeDevice')}
/>
)}
</Wrapper>
))
function Wrapper({
currencyId,
children,
}: {
currencyId: string,
children: (props: { currency: Currency }) => any,
}) {
const currency = getCryptoCurrencyById(currencyId)
return children({ currency })
}

2
src/components/GenuineCheck.js

@ -130,7 +130,7 @@ class GenuineCheck extends PureComponent<Props> {
id: 'isGenuine', id: 'isGenuine',
title: ( title: (
<Trans i18nKey="deviceConnect:stepGenuine.open" parent="div"> <Trans i18nKey="deviceConnect:stepGenuine.open" parent="div">
{'Allow the '} {'Allow '}
<Bold>{'Ledger Manager'}</Bold> <Bold>{'Ledger Manager'}</Bold>
{' on your device'} {' on your device'}
</Trans> </Trans>

6
static/i18n/en/app.yml

@ -206,12 +206,12 @@ manager:
subtitle: Select apps to use on your device subtitle: Select apps to use on your device
device: device:
title: Connect your device title: Connect your device
desc: Follow the steps below to use the Manager desc: 'Follow the steps below to use the device Manager'
cta: Connect my device cta: Connect my device
errors: errors:
noDevice: No device is connected (TEMPLATE NEEDED) noDevice: No device is connected (TEMPLATE NEEDED)
noDashboard: Navigate to the dashboard on your device (TEMPLATED NEEDED) noDashboard: Navigate to the dashboard on your device (TEMPLATED NEEDED)
noGenuine: Allow the Manager to continue (TEMPLATE NEEDED) noGenuine: Allow Manager to continue (TEMPLATE NEEDED)
receive: receive:
title: Receive title: Receive
steps: steps:
@ -282,7 +282,7 @@ settings: # Always ensure descriptions carry full stops (.)
profile: Profile profile: Profile
about: Help about: Help
display: display:
desc: Control Ledger Live's appearance. desc: Unneeded description # Remove this, it controls multiple settings tabs and is not required.
language: Display language language: Display language
languageDesc: Set the language displayed in Ledger Live. languageDesc: Set the language displayed in Ledger Live.
counterValue: Base currency counterValue: Base currency

24
static/i18n/en/onboarding.yml

@ -1,10 +1,10 @@
breadcrumb: breadcrumb:
selectDevice: Select device selectDevice: Device selection
selectPIN: PIN code selectPIN: PIN code
writeSeed: Recovery phrase writeSeed: Recovery phrase
genuineCheck: Security checklist genuineCheck: Security checklist
setPassword: Encrypt data setPassword: Password lock
analytics: Analytics analytics: Bugs & analytics
start: start:
title: Welcome to Ledger Live title: Welcome to Ledger Live
startBtn: Get started startBtn: Get started
@ -105,17 +105,23 @@ genuineCheck:
isGenuinePassed: Your device is genuine isGenuinePassed: Your device is genuine
buttons: buttons:
genuineCheck: Check now genuineCheck: Check now
contactSupport: Ledger Support contactSupport: Contact us
errorPage: errorPage:
ledgerNano: ledgerNano:
title: Oops, something went wrong... title: Oops, your device does not seem genuine...
desc: Go back to the security checklist or request Ledger Support assistance #PIN: Didn't choose your own PIN code?
#recoveryPhrase: Didn't save your own recovery phrase?
#Genuine: Oops, your device does not seem genuine...
desc: Your device did not pass the authenticity test required to connect to Ledger’s secure server. Please contact Ledger Support to get assistance.
#PIN: Never use a device supplied with a PIN code. If your device was provided with a PIN code, please contact us.
#recoveryPhrase: Only save a recovery phrase that is displayed on your device. Please contact us in case of doubt. Otherwise, go back to the security checklist.
#Genuine: Your device did not pass the authenticity test required to connect to Ledger’s secure server. Please contact Ledger Support to get assistance.
ledgerBlue: ledgerBlue:
title: Oops, something went wrong... title: Oops, something went wrong...
desc: Go back to the security checklist or request Ledger Support assistance desc: Go back to the security checklist or request Ledger Support assistance
setPassword: setPassword:
title: Encrypt Ledger Live data title: Password lock (optional)
desc: Set a password to encrypt Ledger Live data stored on your computer, including account names, balances, transactions and public addresses. desc: Set a password to prevent unauthorized access to Ledger Live data stored on your computer, including account names, balances, transactions and public addresses.
disclaimer: disclaimer:
note1: Make sure to remember your password. Do not share it. note1: Make sure to remember your password. Do not share it.
note2: Losing your password requires resetting Ledger Live and re-adding accounts. note2: Losing your password requires resetting Ledger Live and re-adding accounts.
@ -134,5 +140,5 @@ analytics:
desc: Automatically send reports to help Ledger fix bugs desc: Automatically send reports to help Ledger fix bugs
finish: finish:
title: Your device is ready! title: Your device is ready!
desc: Proceed to your porfolio and start adding your accounts... desc: Proceed to your portfolio and start adding your accounts...
openAppButton: Open Ledger Live openAppButton: Open Ledger Live

Loading…
Cancel
Save