// Copyright 2017-2020 @polkadot/react-query authors & contributors // This software may be modified and distributed under the terms // of the Apache-2.0 license. See the LICENSE file for details. import BN from 'bn.js'; import React, { useState } from 'react'; import styled from 'styled-components'; import { Compact } from '@polkadot/types'; import { formatBalance } from '@polkadot/util'; import { useTranslation } from './translate'; interface Props { children?: React.ReactNode; className?: string; isShort?: boolean; label?: React.ReactNode; labelPost?: string; value?: Compact | BN | string | null | 'all'; withCurrency?: boolean; withSi?: boolean; } // for million, 2 * 3-grouping + comma const M_LENGTH = 6 + 1; const K_LENGTH = 3 + 1; function format (value: Compact | BN | string, currency: string | null, withSi?: boolean, _isShort?: boolean, labelPost?: string): React.ReactNode { const [prefix, postfix] = formatBalance(value, { forceUnit: '-', withSi: false }).split('.'); const isShort = _isShort || (withSi && prefix.length >= K_LENGTH); if (prefix.length > M_LENGTH) { // TODO Format with balance-postfix return `${formatBalance(value, { withUnit: !!currency })}${labelPost || ''}`; } return <>{`${prefix}${isShort ? '' : '.'}`}{!isShort && (<>{`000${postfix || ''}`.slice(-3)})}{`${currency ? ` ${currency}` : ''}${labelPost || ''}`}; } function FormatBalance ({ children, className = '', isShort, label, labelPost, value, withCurrency = true, withSi }: Props): React.ReactElement { const { t } = useTranslation(); const [currency] = useState(withCurrency ? formatBalance.getDefaults().unit : null); // labelPost here looks messy, however we ensure we have one less text node return (
{label || ''}{ value ? value === 'all' ? t('everything{{labelPost}}', { replace: { labelPost } }) : format(value, currency, withSi, isShort, labelPost) : `-${labelPost || ''}` }{children}
); } export default React.memo(styled(FormatBalance)` display: inline-block; vertical-align: baseline; white-space: nowrap; * { vertical-align: baseline !important; } > label, > .label { display: inline-block; margin-right: 0.25rem; vertical-align: baseline; } .ui--FormatBalance-value { text-align: right; > .ui--FormatBalance-postfix { font-weight: 100; opacity: 0.75; vertical-align: baseline; } } .ui--Icon { margin-top: 0.25rem; margin-bottom: -0.25rem; } .ui--Icon+.ui--FormatBalance-value { margin-left: 0.375rem; } `);