diff --git a/apps/browser-extension-wallet/src/features/dapp/components/confirm-transaction/DappTransactionContainer.tsx b/apps/browser-extension-wallet/src/features/dapp/components/confirm-transaction/DappTransactionContainer.tsx index 8f7c5a175..84ac10179 100644 --- a/apps/browser-extension-wallet/src/features/dapp/components/confirm-transaction/DappTransactionContainer.tsx +++ b/apps/browser-extension-wallet/src/features/dapp/components/confirm-transaction/DappTransactionContainer.tsx @@ -23,6 +23,7 @@ import { useCurrencyStore, useAppSettingsContext } from '@providers'; import { logger, walletRepository } from '@lib/wallet-api-ui'; import { useComputeTxCollateral } from '@hooks/useComputeTxCollateral'; import { utxoAndBackendChainHistoryResolver } from '@src/utils/utxo-chain-history-resolver'; +import { eraSlotDateTime } from '@src/utils/era-slot-datetime'; import { AddressBookSchema, useDbStateValue } from '@lib/storage'; import { getAllWalletsAddresses } from '@src/utils/get-all-wallets-addresses'; @@ -83,6 +84,7 @@ export const DappTransactionContainer = withAddressBookContext( const userRewardAccounts = useObservable(inMemoryWallet.delegation.rewardAccounts$); const rewardAccountsAddresses = useMemo(() => userRewardAccounts?.map((key) => key.address), [userRewardAccounts]); const protocolParameters = useObservable(inMemoryWallet?.protocolParameters$); + const eraSummaries = useObservable(inMemoryWallet?.eraSummaries$); const allWalletsAddresses = getAllWalletsAddresses(useObservable(walletRepository.wallets$)); useEffect(() => { @@ -153,6 +155,7 @@ export const DappTransactionContainer = withAddressBookContext( errorMessage={errorMessage} toAddress={toAddressTokens} collateral={txCollateral} + expiresBy={eraSlotDateTime(eraSummaries, tx.body.validityInterval?.invalidHereafter)} ownAddresses={allWalletsAddresses.length > 0 ? allWalletsAddresses : ownAddresses} addressToNameMap={addressToNameMap} /> diff --git a/apps/browser-extension-wallet/src/hooks/useInitializeTx.ts b/apps/browser-extension-wallet/src/hooks/useInitializeTx.ts index ac9ee0c9b..ad3143036 100644 --- a/apps/browser-extension-wallet/src/hooks/useInitializeTx.ts +++ b/apps/browser-extension-wallet/src/hooks/useInitializeTx.ts @@ -86,7 +86,8 @@ export const useInitializeTx = ( fee: inspection.inputSelection.fee, hash: inspection.hash, outputs: inspection.inputSelection.outputs, - handleResolutions: inspection.handleResolutions + handleResolutions: inspection.handleResolutions, + validityInterval: inspection.body.validityInterval }, tx, totalMinimumCoins, diff --git a/apps/browser-extension-wallet/src/lib/translations/en.json b/apps/browser-extension-wallet/src/lib/translations/en.json index 8af585036..4554c0ff9 100644 --- a/apps/browser-extension-wallet/src/lib/translations/en.json +++ b/apps/browser-extension-wallet/src/lib/translations/en.json @@ -838,6 +838,10 @@ "core.outputSummaryList.metaData": "Metadata", "core.outputSummaryList.deposit": "Deposit", "core.outputSummaryList.output": "Bundle", + "core.outputSummaryList.expiresBy": "Expires by", + "core.outputSummaryList.expiresByTooltip": "This transaction can be added to the blockchain until the specified time. After this, it will automatically expire and no longer be valid.", + "core.outputSummaryList.noLimit": "No limit", + "core.outputSummaryList.utc": "UTC", "core.sendReceive.send": "Send", "core.sendReceive.receive": "Receive", "core.coinInputSelection.assetSelection": "Select tokens or NFTs", diff --git a/apps/browser-extension-wallet/src/utils/__tests__/era-slot-datetime.test.ts b/apps/browser-extension-wallet/src/utils/__tests__/era-slot-datetime.test.ts new file mode 100644 index 000000000..2d799cf7a --- /dev/null +++ b/apps/browser-extension-wallet/src/utils/__tests__/era-slot-datetime.test.ts @@ -0,0 +1,33 @@ +/* eslint-disable unicorn/no-useless-undefined */ +/* eslint-disable no-magic-numbers */ +import { Cardano, EraSummary } from '@cardano-sdk/core'; +import { eraSlotDateTime } from '../era-slot-datetime'; +import { fromSerializableObject } from '@cardano-sdk/util'; + +export const eraSummaries = fromSerializableObject([ + { + parameters: { epochLength: 4320, slotLength: 20_000 }, + start: { slot: 0, time: { __type: 'Date', value: 1_666_656_000_000 } } + }, + { + parameters: { epochLength: 86_400, slotLength: 1000 }, + start: { slot: 0, time: { __type: 'Date', value: 1_666_656_000_000 } } + } +]); + +const testSlot = Cardano.Slot(36_201_583); + +describe('Testing eraSlotDateTime', () => { + test('should return undefined', async () => { + expect(eraSlotDateTime(undefined, testSlot)).toEqual(undefined); + }); + test('should return undefined', async () => { + expect(eraSlotDateTime(eraSummaries, undefined)).toEqual(undefined); + }); + test('should return formatted time', async () => { + expect(eraSlotDateTime(eraSummaries, testSlot)).toEqual({ + utcDate: '12/17/2023', + utcTime: '23:59:43' + }); + }); +}); diff --git a/apps/browser-extension-wallet/src/utils/era-slot-datetime.ts b/apps/browser-extension-wallet/src/utils/era-slot-datetime.ts new file mode 100644 index 000000000..3258dc43c --- /dev/null +++ b/apps/browser-extension-wallet/src/utils/era-slot-datetime.ts @@ -0,0 +1,19 @@ +/* eslint-disable consistent-return */ +import { EraSummary } from '@cardano-sdk/core'; +import { Wallet } from '@lace/cardano'; +import { formatDate, formatTime } from './format-date'; + +export const eraSlotDateTime = ( + eraSummaries: EraSummary[] | undefined, + slot: Wallet.Cardano.Slot | undefined +): { utcDate: string; utcTime: string } | undefined => { + if (!eraSummaries || !slot) { + return undefined; + } + const slotTimeCalc = Wallet.createSlotTimeCalc(eraSummaries); + const date = slotTimeCalc(slot); + return { + utcDate: formatDate({ date, format: 'MM/DD/YYYY', type: 'utc' }), + utcTime: formatTime({ date, type: 'utc' }) + }; +}; diff --git a/apps/browser-extension-wallet/src/views/browser-view/features/send-transaction/components/SendTransactionSummary.tsx b/apps/browser-extension-wallet/src/views/browser-view/features/send-transaction/components/SendTransactionSummary.tsx index 15acca482..73f41422c 100644 --- a/apps/browser-extension-wallet/src/views/browser-view/features/send-transaction/components/SendTransactionSummary.tsx +++ b/apps/browser-extension-wallet/src/views/browser-view/features/send-transaction/components/SendTransactionSummary.tsx @@ -17,6 +17,7 @@ import { getTokenAmountInFiat, parseFiat } from '@src/utils/assets-transformers' import { useObservable, Banner } from '@lace/common'; import ExclamationIcon from '../../../../../assets/icons/exclamation-triangle-red.component.svg'; import { WalletType } from '@cardano-sdk/web-extension'; +import { eraSlotDateTime } from '@src/utils/era-slot-datetime'; import { getAllWalletsAddresses } from '@src/utils/get-all-wallets-addresses'; import { walletRepository } from '@lib/wallet-api-ui'; @@ -100,7 +101,8 @@ interface SendTransactionSummaryProps { export const SendTransactionSummary = withAddressBookContext( ({ isPopupView = false }: SendTransactionSummaryProps): React.ReactElement => { const { t } = useTranslation(); - const { builtTxData: { uiTx: { fee, outputs, handleResolutions } = {} } = {} } = useBuiltTxState(); + const { builtTxData: { uiTx: { fee, outputs, handleResolutions, validityInterval } = {} } = {} } = + useBuiltTxState(); const [metadata] = useMetadata(); const { inMemoryWallet, @@ -115,6 +117,7 @@ export const SendTransactionSummary = withAddressBookContext( const { fiatCurrency } = useCurrencyStore(); const assetsInfo = useObservable(inMemoryWallet.assetInfo$); + const eraSummaries = useObservable(inMemoryWallet.eraSummaries$); const outputSummaryListTranslation = { recipientAddress: t('core.outputSummaryList.recipientAddress'), @@ -122,7 +125,11 @@ export const SendTransactionSummary = withAddressBookContext( output: t('core.outputSummaryList.output'), metadata: t('core.outputSummaryList.metaData'), deposit: t('core.outputSummaryList.deposit'), - txFee: t('core.outputSummaryList.txFee') + txFee: t('core.outputSummaryList.txFee'), + expiresBy: t('core.outputSummaryList.expiresBy'), + expiresByTooltip: t('core.outputSummaryList.expiresByTooltip'), + noLimit: t('core.outputSummaryList.noLimit'), + utc: t('core.outputSummaryList.utc') }; const addressToNameMap = useMemo( @@ -149,6 +156,7 @@ export const SendTransactionSummary = withAddressBookContext( <> ; fee: Wallet.Cardano.Lovelace; handleResolutions?: HandleResolution[]; + validityInterval?: Wallet.Cardano.ValidityInterval; }; error?: string; reachedMaxAmountList?: (string | Wallet.Cardano.AssetId)[]; diff --git a/packages/core/src/ui/components/DappTransaction/DappTransaction.tsx b/packages/core/src/ui/components/DappTransaction/DappTransaction.tsx index 09f2e94ab..420ff5d86 100644 --- a/packages/core/src/ui/components/DappTransaction/DappTransaction.tsx +++ b/packages/core/src/ui/components/DappTransaction/DappTransaction.tsx @@ -10,7 +10,16 @@ import styles from './DappTransaction.module.scss'; import { useTranslate } from '@src/ui/hooks'; import { TransactionFee, Collateral } from '@ui/components/ActivityDetail'; -import { TransactionType, DappTransactionSummary, TransactionAssets, Text, Box, Divider } from '@lace/ui'; +import { + TransactionType, + DappTransactionSummary, + TransactionAssets, + DappTransactionTextField, + Flex, + Text, + Box, + Divider +} from '@lace/ui'; import { DappAddressSections } from '../DappAddressSections/DappAddressSections'; const amountTransformer = (fiat: { price: number; code: string }) => (ada: string) => @@ -26,6 +35,7 @@ export interface DappTransactionProps { fiatCurrencyCode?: string; fiatCurrencyPrice?: number; coinSymbol?: string; + expiresBy?: { utcDate: string; utcTime: string }; /** tokens send to being sent to or from the user */ fromAddress: Map; toAddress: Map; @@ -113,6 +123,7 @@ export const DappTransaction = ({ fiatCurrencyPrice, coinSymbol, dappInfo, + expiresBy, ownAddresses = [], addressToNameMap = new Map() }: DappTransactionProps): React.ReactElement => { @@ -124,6 +135,17 @@ export const DappTransaction = ({ const isFromAddressesEnabled = groupedFromAddresses.size > 0; const isToAddressesEnabled = groupedToAddresses.size > 0; + const expireByText = expiresBy ? ( + + {expiresBy.utcDate} + + {expiresBy.utcTime} {t('core.outputSummaryList.utc')} + + + ) : ( + t('core.outputSummaryList.noLimit') + ); + return (
{errorMessage && } @@ -184,6 +206,14 @@ export const DappTransaction = ({ /> )} +
+ +
+ {returnedDeposit !== BigInt(0) && ( ; + translations: TranslationsFor< + | 'recipientAddress' + | 'sending' + | 'txFee' + | 'deposit' + | 'metadata' + | 'output' + | 'expiresBy' + | 'expiresByTooltip' + | 'noLimit' + | 'utc' + >; ownAddresses?: string[]; onFeeTooltipHover?: () => unknown; onDepositTooltipHover?: () => unknown; @@ -33,6 +46,7 @@ export const OutputSummaryList = ({ metadata, deposit, translations, + expiresBy, ownAddresses, onFeeTooltipHover, onDepositTooltipHover @@ -42,6 +56,17 @@ export const OutputSummaryList = ({ sending: translations.sending }; + const expireByText = expiresBy ? ( + + {expiresBy.utcDate} + + {expiresBy.utcTime} {translations.utc} + + + ) : ( + translations.noLimit + ); + return (
{rows.map((row, idx) => ( @@ -54,7 +79,6 @@ export const OutputSummaryList = ({
))} - {metadata && (
@@ -67,6 +91,18 @@ export const OutputSummaryList = ({ )}
+ + {renderLabel({ + label: translations.expiresBy, + tooltipContent: translations.expiresByTooltip, + dataTestId: 'validity-interval-expires-by-label' + })} + + + {expireByText} + + + {deposit && ( {renderLabel({ diff --git a/packages/ui/src/design-system/dapp-transaction-summary/dapp-transaction-summary.css.ts b/packages/ui/src/design-system/dapp-transaction-summary/dapp-transaction-summary.css.ts index cd7d2cab3..244129c25 100644 --- a/packages/ui/src/design-system/dapp-transaction-summary/dapp-transaction-summary.css.ts +++ b/packages/ui/src/design-system/dapp-transaction-summary/dapp-transaction-summary.css.ts @@ -7,11 +7,9 @@ export const transactionTypeContainer = style({ export const cardanoIcon = style([ sx({ display: 'flex', + w: '$32', + height: '$32', }), - { - width: '32px', - height: '32px', - }, ]); export const txAmountContainer = style({ diff --git a/packages/ui/src/design-system/dapp-transaction-summary/dapp-transaction-summary.stories.tsx b/packages/ui/src/design-system/dapp-transaction-summary/dapp-transaction-summary.stories.tsx index e6a3100d7..0407a9562 100644 --- a/packages/ui/src/design-system/dapp-transaction-summary/dapp-transaction-summary.stories.tsx +++ b/packages/ui/src/design-system/dapp-transaction-summary/dapp-transaction-summary.stories.tsx @@ -13,6 +13,7 @@ import { Grid, Cell } from '../grid'; import { TransactionAssets } from './dapp-transaction-assets.component'; import { TransactionSummary } from './dapp-transaction-summary.component'; +import { TransactionTextField } from './dapp-transaction-text-field.component'; import { TransactionType } from './dapp-transaction-type.component'; const subtitle = `Control that displays data items in rows.`; @@ -102,6 +103,13 @@ const Example = (): JSX.Element => ( tokenName={value.tokenName} /> ))} + + + ); @@ -128,6 +136,9 @@ const MainComponents = (): JSX.Element => ( /> ))} + + + diff --git a/packages/ui/src/design-system/dapp-transaction-summary/dapp-transaction-text-field.component.tsx b/packages/ui/src/design-system/dapp-transaction-summary/dapp-transaction-text-field.component.tsx new file mode 100644 index 000000000..d0e80c484 --- /dev/null +++ b/packages/ui/src/design-system/dapp-transaction-summary/dapp-transaction-text-field.component.tsx @@ -0,0 +1,51 @@ +import type { ReactNode } from 'react'; +import React from 'react'; + +import { ReactComponent as InfoIcon } from '@lace/icons/dist/InfoComponent'; + +import { Box } from '../box'; +import { Flex } from '../flex'; +import { Grid, Cell } from '../grid'; +import { Text } from '../text'; +import { Tooltip } from '../tooltip'; + +import * as cx from './dapp-transaction-text-field.css'; + +import type { OmitClassName } from '../../types'; + +type Props = OmitClassName<'div'> & { + label: string; + text: ReactNode | string; + tooltip?: string; +}; + +export const TransactionTextField = ({ + label, + text, + tooltip, + ...props +}: Readonly): JSX.Element => { + return ( + + + + {label} + {tooltip !== undefined && ( + + + + + + )} + + + + + + {text} + + + + + ); +}; diff --git a/packages/ui/src/design-system/dapp-transaction-summary/dapp-transaction-text-field.css.ts b/packages/ui/src/design-system/dapp-transaction-summary/dapp-transaction-text-field.css.ts new file mode 100644 index 000000000..baf771cf0 --- /dev/null +++ b/packages/ui/src/design-system/dapp-transaction-summary/dapp-transaction-text-field.css.ts @@ -0,0 +1,14 @@ +import { sx, style } from '../../design-tokens'; + +export const text = style({ + wordBreak: 'break-all', +}); + +export const tooltipIcon = style([ + sx({ + color: '$text_primary', + width: '$24', + height: '$24', + fontSize: '$25', + }), +]); diff --git a/packages/ui/src/design-system/dapp-transaction-summary/index.ts b/packages/ui/src/design-system/dapp-transaction-summary/index.ts index 24037ad5d..fb69cd8b3 100644 --- a/packages/ui/src/design-system/dapp-transaction-summary/index.ts +++ b/packages/ui/src/design-system/dapp-transaction-summary/index.ts @@ -1,3 +1,4 @@ export { TransactionType } from './dapp-transaction-type.component'; export { TransactionSummary } from './dapp-transaction-summary.component'; export { TransactionAssets } from './dapp-transaction-assets.component'; +export { TransactionTextField } from './dapp-transaction-text-field.component'; diff --git a/packages/ui/src/design-system/index.ts b/packages/ui/src/design-system/index.ts index 8705e0de9..86495ed5a 100644 --- a/packages/ui/src/design-system/index.ts +++ b/packages/ui/src/design-system/index.ts @@ -47,7 +47,10 @@ export { SelectGroup } from './select'; export { ActionCard } from './action-card'; export { Loader } from './loader'; export { TransactionType } from './dapp-transaction-summary'; -export { TransactionSummary as DappTransactionSummary } from './dapp-transaction-summary'; +export { + TransactionSummary as DappTransactionSummary, + TransactionTextField as DappTransactionTextField, +} from './dapp-transaction-summary'; export { TransactionAssets } from './dapp-transaction-summary'; export { SummaryExpander } from './summary-expander'; export * from './auto-suggest-box';