diff --git a/src/components/address-book/AddressBookHeader/index.tsx b/src/components/address-book/AddressBookHeader/index.tsx index ba2baed186..c1b861103a 100644 --- a/src/components/address-book/AddressBookHeader/index.tsx +++ b/src/components/address-book/AddressBookHeader/index.tsx @@ -9,10 +9,11 @@ import { ADDRESS_BOOK_EVENTS } from '@/services/analytics/events/addressBook' import PageHeader from '@/components/common/PageHeader' import { ModalType } from '../AddressBookTable' import { useAppSelector } from '@/store' -import { selectAllAddressBooks } from '@/store/addressBookSlice' +import { type AddressBookState, selectAllAddressBooks } from '@/store/addressBookSlice' import ImportIcon from '@/public/images/common/import.svg' import ExportIcon from '@/public/images/common/export.svg' import AddCircleIcon from '@/public/images/common/add-outlined.svg' +import mapProps from '@/utils/mad-props' const HeaderButton = ({ icon, @@ -35,13 +36,18 @@ const HeaderButton = ({ } type Props = { + allAddressBooks: AddressBookState handleOpenModal: (type: ModalType) => () => void searchQuery: string onSearchQueryChange: (searchQuery: string) => void } -const AddressBookHeader = ({ handleOpenModal, searchQuery, onSearchQueryChange }: Props): ReactElement => { - const allAddressBooks = useAppSelector(selectAllAddressBooks) +function AddressBookHeader({ + allAddressBooks, + handleOpenModal, + searchQuery, + onSearchQueryChange, +}: Props): ReactElement { const canExport = Object.values(allAddressBooks).some((addressBook) => Object.keys(addressBook || {}).length > 0) return ( @@ -96,4 +102,8 @@ const AddressBookHeader = ({ handleOpenModal, searchQuery, onSearchQueryChange } ) } -export default AddressBookHeader +const useAllAddressBooks = () => useAppSelector(selectAllAddressBooks) + +export default mapProps(AddressBookHeader, { + allAddressBooks: useAllAddressBooks, +}) diff --git a/src/components/address-book/AddressBookTable/index.tsx b/src/components/address-book/AddressBookTable/index.tsx index 743f74c806..7e57bb39d2 100644 --- a/src/components/address-book/AddressBookTable/index.tsx +++ b/src/components/address-book/AddressBookTable/index.tsx @@ -1,5 +1,7 @@ import { useContext, useMemo, useState } from 'react' import { Box } from '@mui/material' +import type { ChainInfo } from '@safe-global/safe-gateway-typescript-sdk' + import EnhancedTable from '@/components/common/EnhancedTable' import type { AddressEntry } from '@/components/address-book/EntryDialog' import EntryDialog from '@/components/address-book/EntryDialog' @@ -21,9 +23,10 @@ import PagePlaceholder from '@/components/common/PagePlaceholder' import NoEntriesIcon from '@/public/images/address-book/no-entries.svg' import { useCurrentChain } from '@/hooks/useChains' import tableCss from '@/components/common/EnhancedTable/styles.module.css' -import { TxModalContext } from '@/components/tx-flow' +import { TxModalContext, type TxModalContextType } from '@/components/tx-flow' import TokenTransferFlow from '@/components/tx-flow/flows/TokenTransfer' import CheckWallet from '@/components/common/CheckWallet' +import madProps from '@/utils/mad-props' const headCells = [ { id: 'name', label: 'Name' }, @@ -45,10 +48,12 @@ const defaultOpen = { [ModalType.REMOVE]: false, } -const AddressBookTable = () => { - const chain = useCurrentChain() - const { setTxFlow } = useContext(TxModalContext) +type AddressBookTableProps = { + chain?: ChainInfo + setTxFlow: TxModalContextType['setTxFlow'] +} +function AddressBookTable({ chain, setTxFlow }: AddressBookTableProps) { const [open, setOpen] = useState(defaultOpen) const [searchQuery, setSearchQuery] = useState('') const [defaultValues, setDefaultValues] = useState(undefined) @@ -170,4 +175,9 @@ const AddressBookTable = () => { ) } -export default AddressBookTable +const useSetTxFlow = () => useContext(TxModalContext).setTxFlow + +export default madProps(AddressBookTable, { + chain: useCurrentChain, + setTxFlow: useSetTxFlow, +}) diff --git a/src/components/address-book/EntryDialog/index.tsx b/src/components/address-book/EntryDialog/index.tsx index fe63ae7b08..eaae59e9d5 100644 --- a/src/components/address-book/EntryDialog/index.tsx +++ b/src/components/address-book/EntryDialog/index.tsx @@ -8,13 +8,14 @@ import NameInput from '@/components/common/NameInput' import useChainId from '@/hooks/useChainId' import { useAppDispatch } from '@/store' import { upsertAddressBookEntry } from '@/store/addressBookSlice' +import madProps from '@/utils/mad-props' export type AddressEntry = { name: string address: string } -const EntryDialog = ({ +function EntryDialog({ handleClose, defaultValues = { name: '', @@ -22,14 +23,15 @@ const EntryDialog = ({ }, disableAddressInput = false, chainId, + currentChainId, }: { handleClose: () => void defaultValues?: AddressEntry disableAddressInput?: boolean chainId?: string -}): ReactElement => { + currentChainId: string +}): ReactElement { const dispatch = useAppDispatch() - const currentChainId = useChainId() const methods = useForm({ defaultValues, @@ -81,4 +83,6 @@ const EntryDialog = ({ ) } -export default EntryDialog +export default madProps(EntryDialog, { + currentChainId: useChainId, +}) diff --git a/src/components/address-book/ExportDialog/index.test.ts b/src/components/address-book/ExportDialog/index.test.tsx similarity index 71% rename from src/components/address-book/ExportDialog/index.test.ts rename to src/components/address-book/ExportDialog/index.test.tsx index 928363b9a3..ed839f4aaf 100644 --- a/src/components/address-book/ExportDialog/index.test.ts +++ b/src/components/address-book/ExportDialog/index.test.tsx @@ -1,4 +1,6 @@ import { _getCsvData } from '.' +import ExportDialog from '.' +import { render, screen } from '@/tests/test-utils' describe('ExportDialog', () => { describe('getCsvData', () => { @@ -22,4 +24,10 @@ describe('ExportDialog', () => { ]) }) }) + + it('should render the export dialog', () => { + const onClose = jest.fn() + render() + expect(screen.getByText('Export address book')).toBeInTheDocument() + }) }) diff --git a/src/components/address-book/ExportDialog/index.tsx b/src/components/address-book/ExportDialog/index.tsx index 9407e32058..2f5e176d77 100644 --- a/src/components/address-book/ExportDialog/index.tsx +++ b/src/components/address-book/ExportDialog/index.tsx @@ -12,6 +12,7 @@ import { useAppSelector } from '@/store' import { trackEvent, ADDRESS_BOOK_EVENTS } from '@/services/analytics' import ExternalLink from '@/components/common/ExternalLink' import { HelpCenterArticle } from '@/config/constants' +import madProps from '@/utils/mad-props' const COL_1 = 'address' const COL_2 = 'name' @@ -35,14 +36,19 @@ export const _getCsvData = (addressBooks: AddressBookState): CsvData => { return csvData } -const ExportDialog = ({ handleClose }: { handleClose: () => void }): ReactElement => { - const addressBooks: AddressBookState = useAppSelector(selectAllAddressBooks) - const length = Object.values(addressBooks).reduce((acc, entries) => acc + Object.keys(entries).length, 0) +function ExportDialog({ + allAddressBooks, + handleClose, +}: { + allAddressBooks: AddressBookState + handleClose: () => void +}): ReactElement { + const length = Object.values(allAddressBooks).reduce((acc, entries) => acc + Object.keys(entries).length, 0) const { CSVDownloader } = useCSVDownloader() // safe-address-book-1970-01-01 const filename = `safe-address-book-${new Date().toISOString().slice(0, 10)}` - const csvData = useMemo(() => _getCsvData(addressBooks), [addressBooks]) + const csvData = useMemo(() => _getCsvData(allAddressBooks), [allAddressBooks]) const onSubmit = (e: SyntheticEvent) => { e.preventDefault() @@ -87,4 +93,8 @@ const ExportDialog = ({ handleClose }: { handleClose: () => void }): ReactElemen ) } -export default ExportDialog +const useAllAddressBooks = () => useAppSelector(selectAllAddressBooks) + +export default madProps(ExportDialog, { + allAddressBooks: useAllAddressBooks, +}) diff --git a/src/components/tx-flow/index.tsx b/src/components/tx-flow/index.tsx index eebcbb0338..0cf7a993aa 100644 --- a/src/components/tx-flow/index.tsx +++ b/src/components/tx-flow/index.tsx @@ -5,7 +5,7 @@ import useSafeInfo from '@/hooks/useSafeInfo' const noop = () => {} -type TxModalContextType = { +export type TxModalContextType = { txFlow: JSX.Element | undefined setTxFlow: (txFlow: TxModalContextType['txFlow'], onClose?: () => void, shouldWarn?: boolean) => void setFullWidth: (fullWidth: boolean) => void diff --git a/src/utils/__tests__/mad-props.test.tsx b/src/utils/__tests__/mad-props.test.tsx new file mode 100644 index 0000000000..02b35c331d --- /dev/null +++ b/src/utils/__tests__/mad-props.test.tsx @@ -0,0 +1,34 @@ +import { render, waitFor, screen } from '@/tests/test-utils' +import madProps from '../mad-props' + +describe('madProps', () => { + it('should map a set of props to hooks', async () => { + const Component = ({ foo, bar }: { foo: string; bar: string }) => <>{foo + bar} + + const MappedComponent = madProps(Component, { + foo: () => 'foo', + bar: () => 'bar', + }) + + render() + + await waitFor(() => { + expect(screen.getByText('foobar')).toBeInTheDocument() + }) + }) + + it('should accept additional props', async () => { + const Component = ({ foo, bar, baz }: { foo: string; bar: string; baz: string }) => <>{foo + bar + baz} + + const MappedComponent = madProps(Component, { + foo: () => 'foo', + bar: () => 'bar', + }) + + render() + + await waitFor(() => { + expect(screen.getByText('foobarbaz')).toBeInTheDocument() + }) + }) +}) diff --git a/src/utils/mad-props.tsx b/src/utils/mad-props.tsx new file mode 100644 index 0000000000..1aae182d7a --- /dev/null +++ b/src/utils/mad-props.tsx @@ -0,0 +1,40 @@ +import type { ComponentType, FC } from 'react' +import React, { memo } from 'react' + +type HookMap

= { + [K in keyof P]?: () => P[K] +} + +/** + * Injects props into a component using hooks. + * This allows to keep the original component pure and testable. + * + * @param Component The component to wrap + * @param hookMap A map of hooks to use, keys are the props to inject, values are the hooks + * @returns A new component with the props injected + */ +const madProps =

, H extends keyof P>( + Component: ComponentType

, + hookMap: HookMap>, +): FC> => { + const MadComponent = (externalProps: Omit) => { + let newProps: P = { ...externalProps } as P + + for (const key in hookMap) { + const hook = hookMap[key] + if (hook !== undefined) { + newProps[key as H] = hook() + } + } + + return + } + + MadComponent.displayName = Component.displayName || Component.name + + // Wrapping MadComponent with React.memo and casting to FC> + // The casting is only needed because of memo, the component itself satisfies the type + return memo(MadComponent) as unknown as FC> +} + +export default madProps