Gaëtan Renaudeau 7 years ago
parent
commit
761c3f54f9
  1. 6
      src/actions/accounts.js
  2. 43
      src/actions/general.js
  3. 80
      src/components/DashboardPage/AccountsOrder.js
  4. 4
      src/components/DashboardPage/index.js
  5. 38
      src/components/RefreshAccountsOrdering.js
  6. 5
      src/components/modals/AddAccounts/steps/04-step-finish.js
  7. 37
      src/helpers/accountOrdering.js

6
src/actions/accounts.js

@ -16,12 +16,6 @@ export const removeAccount: RemoveAccount = payload => ({
payload, payload,
}) })
export type ReorderAccounts = (string[]) => { type: string, payload: string[] }
export const reorderAccounts: ReorderAccounts = payload => ({
type: 'DB:REORDER_ACCOUNTS',
payload,
})
export type FetchAccounts = () => * export type FetchAccounts = () => *
export const fetchAccounts: FetchAccounts = () => { export const fetchAccounts: FetchAccounts = () => {
db.init('accounts', []) // FIXME the "init" pattern to drop imo. a simple get()||[] is enough db.init('accounts', []) // FIXME the "init" pattern to drop imo. a simple get()||[] is enough

43
src/actions/general.js

@ -0,0 +1,43 @@
// @flow
import { createSelector, createStructuredSelector } from 'reselect'
import CounterValues from 'helpers/countervalues'
import {
intermediaryCurrency,
currencySettingsForAccountSelector,
getOrderAccounts,
} from 'reducers/settings'
import { accountsSelector } from 'reducers/accounts'
import { sortAccounts } from 'helpers/accountOrdering'
const accountsBtcBalanceSelector = createSelector(
accountsSelector,
state => state,
(accounts, state) =>
accounts.map(account => {
const { exchange } = currencySettingsForAccountSelector(state, { account })
return CounterValues.calculateSelector(state, {
from: account.currency,
to: intermediaryCurrency,
exchange,
value: account.balance,
})
}),
)
const selectAccountsBalanceAndOrder = createStructuredSelector({
accounts: accountsSelector,
accountsBtcBalance: accountsBtcBalanceSelector,
orderAccounts: getOrderAccounts,
})
export const refreshAccountsOrdering = () => (dispatch: *, getState: *) => {
const all = selectAccountsBalanceAndOrder(getState())
const allRatesAvailable = all.accountsBtcBalance.every(b => typeof b === 'number')
if (allRatesAvailable) {
dispatch({
type: 'DB:REORDER_ACCOUNTS',
payload: sortAccounts(all),
})
}
}

80
src/components/DashboardPage/AccountsOrder.js

@ -1,32 +1,21 @@
// @flow // @flow
import logger from 'logger'
import React, { Component } from 'react' import React, { Component } from 'react'
import styled from 'styled-components' import styled from 'styled-components'
import { compose } from 'redux' import { compose } from 'redux'
import { translate } from 'react-i18next' import { translate } from 'react-i18next'
import { connect } from 'react-redux' import { connect } from 'react-redux'
import type { Account } from '@ledgerhq/live-common/lib/types' import { createStructuredSelector } from 'reselect'
import CounterValues from 'helpers/countervalues'
import type { T } from 'types/common' import type { T } from 'types/common'
import { refreshAccountsOrdering } from 'actions/general'
import {
getOrderAccounts,
intermediaryCurrency,
currencySettingsForAccountSelector,
} from 'reducers/settings'
import { createStructuredSelector, createSelector } from 'reselect'
import { reorderAccounts } from 'actions/accounts'
import { accountsSelector } from 'reducers/accounts'
import { saveSettings } from 'actions/settings' import { saveSettings } from 'actions/settings'
import { getOrderAccounts } from 'reducers/settings'
import Track from 'analytics/Track' import Track from 'analytics/Track'
import BoldToggle from 'components/base/BoldToggle' import BoldToggle from 'components/base/BoldToggle'
import Box from 'components/base/Box' import Box from 'components/base/Box'
import DropDown, { DropDownItem } from 'components/base/DropDown' import DropDown, { DropDownItem } from 'components/base/DropDown'
import Text from 'components/base/Text' import Text from 'components/base/Text'
import IconAngleDown from 'icons/AngleDown' import IconAngleDown from 'icons/AngleDown'
import IconArrowDown from 'icons/ArrowDown' import IconArrowDown from 'icons/ArrowDown'
import IconArrowUp from 'icons/ArrowUp' import IconArrowUp from 'icons/ArrowUp'
@ -34,41 +23,10 @@ import IconArrowUp from 'icons/ArrowUp'
type Props = { type Props = {
t: T, t: T,
orderAccounts: string, orderAccounts: string,
accounts: Account[], refreshAccountsOrdering: () => *,
accountsBtcBalance: number[], // eslint-disable-line
reorderAccounts: (string[]) => *,
saveSettings: (*) => *, saveSettings: (*) => *,
} }
type SortMethod = 'name' | 'balance'
const sortMethod: { [_: SortMethod]: (Account[], Props) => string[] } = {
balance: (accounts, { accountsBtcBalance }: Props) =>
accounts
.map((a, i) => [a.id, accountsBtcBalance[i]])
.sort((a, b) => a[1] - b[1])
.map(o => o[0]),
name: accounts =>
accounts
.slice(0)
.sort((a, b) => a.name.localeCompare(b.name))
.map(a => a.id),
}
function sortAccounts(accounts: Account[], orderAccounts: string, props: Props) {
const [order, sort] = orderAccounts.split('|')
if (order === 'name' || order === 'balance') {
const ids = sortMethod[order](accounts, props)
if (sort === 'asc') {
ids.reverse()
}
return ids
}
logger.warn(`sortAccounts not implemented for ${orderAccounts}`)
return null
}
const OrderIcon = styled(Box).attrs({ const OrderIcon = styled(Box).attrs({
alignItems: 'center', alignItems: 'center',
justifyContent: 'center', justifyContent: 'center',
@ -77,31 +35,12 @@ const OrderIcon = styled(Box).attrs({
opacity: ${p => (p.isActive ? 1 : 0)}; opacity: ${p => (p.isActive ? 1 : 0)};
` `
const accountsBtcBalanceSelector = createSelector(
accountsSelector,
state => state,
(accounts, state) =>
accounts.map(account => {
const { exchange } = currencySettingsForAccountSelector(state, { account })
return (
CounterValues.calculateSelector(state, {
from: account.currency,
to: intermediaryCurrency,
exchange,
value: account.balance,
}) || 0
)
}),
)
const mapStateToProps = createStructuredSelector({ const mapStateToProps = createStructuredSelector({
orderAccounts: getOrderAccounts, orderAccounts: getOrderAccounts,
accounts: accountsSelector,
accountsBtcBalance: accountsBtcBalanceSelector,
}) })
const mapDispatchToProps = { const mapDispatchToProps = {
reorderAccounts, refreshAccountsOrdering,
saveSettings, saveSettings,
} }
@ -120,12 +59,9 @@ class AccountsOrder extends Component<Props> {
} }
setAccountOrder = order => { setAccountOrder = order => {
const { saveSettings, reorderAccounts } = this.props const { saveSettings, refreshAccountsOrdering } = this.props
const maybeIds = sortAccounts(this.props.accounts, order, this.props) saveSettings({ orderAccounts: order })
if (maybeIds) { refreshAccountsOrdering()
reorderAccounts(maybeIds)
saveSettings({ orderAccounts: order })
}
} }
getCurrentOrder = () => { getCurrentOrder = () => {

4
src/components/DashboardPage/index.js

@ -24,10 +24,10 @@ import {
} from 'reducers/settings' } from 'reducers/settings'
import type { TimeRange } from 'reducers/settings' import type { TimeRange } from 'reducers/settings'
import { reorderAccounts } from 'actions/accounts'
import { saveSettings } from 'actions/settings' import { saveSettings } from 'actions/settings'
import TrackPage from 'analytics/TrackPage' import TrackPage from 'analytics/TrackPage'
import RefreshAccountsOrdering from 'components/RefreshAccountsOrdering'
import UpdateNotifier from 'components/UpdateNotifier' import UpdateNotifier from 'components/UpdateNotifier'
import BalanceInfos from 'components/BalanceSummary/BalanceInfos' import BalanceInfos from 'components/BalanceSummary/BalanceInfos'
import BalanceSummary from 'components/BalanceSummary' import BalanceSummary from 'components/BalanceSummary'
@ -51,7 +51,6 @@ const mapStateToProps = createStructuredSelector({
const mapDispatchToProps = { const mapDispatchToProps = {
push, push,
reorderAccounts,
saveSettings, saveSettings,
openModal, openModal,
} }
@ -102,6 +101,7 @@ class DashboardPage extends PureComponent<Props> {
return ( return (
<Fragment> <Fragment>
<UpdateNotifier /> <UpdateNotifier />
<RefreshAccountsOrdering onMount />
<TrackPage <TrackPage
category="Portfolio" category="Portfolio"
totalAccounts={totalAccounts} totalAccounts={totalAccounts}

38
src/components/RefreshAccountsOrdering.js

@ -0,0 +1,38 @@
// @flow
import { Component } from 'react'
import { connect } from 'react-redux'
import { refreshAccountsOrdering } from 'actions/general'
const mapStateToProps = null
const mapDispatchToProps = {
refreshAccountsOrdering,
}
class RefreshAccountsOrdering extends Component<{
refreshAccountsOrdering: () => *,
onMount?: boolean,
onUnmount?: boolean,
}> {
componentDidMount() {
if (this.props.onMount) {
this.props.refreshAccountsOrdering()
}
}
componentWillUnmount() {
if (this.props.onUnmount) {
this.props.refreshAccountsOrdering()
}
}
render() {
return null
}
}
export default connect(
mapStateToProps,
mapDispatchToProps,
)(RefreshAccountsOrdering)

5
src/components/modals/AddAccounts/steps/04-step-finish.js

@ -6,6 +6,7 @@ import styled from 'styled-components'
import TrackPage from 'analytics/TrackPage' import TrackPage from 'analytics/TrackPage'
import Box from 'components/base/Box' import Box from 'components/base/Box'
import Button from 'components/base/Button' import Button from 'components/base/Button'
import RefreshAccountsOrdering from 'components/RefreshAccountsOrdering'
import IconCheckFull from 'icons/CheckFull' import IconCheckFull from 'icons/CheckFull'
import { CurrencyCircleIcon } from '../../../base/CurrencyBadge' import { CurrencyCircleIcon } from '../../../base/CurrencyBadge'
import type { StepProps } from '../index' import type { StepProps } from '../index'
@ -30,6 +31,10 @@ const Text = styled(Box).attrs({
function StepFinish({ currency, t, checkedAccountsIds }: StepProps) { function StepFinish({ currency, t, checkedAccountsIds }: StepProps) {
return ( return (
<Box align="center" py={6}> <Box align="center" py={6}>
<RefreshAccountsOrdering onMount onUnmount />
{/* onMount because if we already have the countervalues we want to sort it straightaway
onUnmount because if not, it is useful to trigger a second refresh to ensure it get sorted */}
<TrackPage category="AddAccounts" name="Step4" /> <TrackPage category="AddAccounts" name="Step4" />
{currency ? ( {currency ? (
<Box color="positiveGreen" style={{ position: 'relative' }}> <Box color="positiveGreen" style={{ position: 'relative' }}>

37
src/helpers/accountOrdering.js

@ -0,0 +1,37 @@
// @flow
import type { Account } from '@ledgerhq/live-common/lib/types'
type Param = {
accounts: Account[],
accountsBtcBalance: number[],
orderAccounts: string,
}
type SortMethod = 'name' | 'balance'
const sortMethod: { [_: SortMethod]: (Param) => string[] } = {
balance: ({ accounts, accountsBtcBalance }) =>
accounts
.map((a, i) => [a.id, accountsBtcBalance[i]])
.sort((a, b) => a[1] - b[1])
.map(o => o[0]),
name: ({ accounts }) =>
accounts
.slice(0)
.sort((a, b) => a.name.localeCompare(b.name))
.map(a => a.id),
}
export function sortAccounts(param: Param) {
const [order, sort] = param.orderAccounts.split('|')
if (order === 'name' || order === 'balance') {
const ids = sortMethod[order](param)
if (sort === 'asc') {
ids.reverse()
}
return ids
}
return null
}
Loading…
Cancel
Save