Skip to content

Commit

Permalink
Refactor: map props to hooks
Browse files Browse the repository at this point in the history
  • Loading branch information
katspaugh committed Sep 29, 2023
1 parent e807c40 commit 4a0429c
Show file tree
Hide file tree
Showing 6 changed files with 81 additions and 17 deletions.
16 changes: 12 additions & 4 deletions src/components/address-book/AddressBookHeader/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -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,
Expand All @@ -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)
const AddressBookHeader = ({
allAddressBooks,
handleOpenModal,
searchQuery,
onSearchQueryChange,
}: Props): ReactElement => {
const canExport = Object.values(allAddressBooks).some((addressBook) => Object.keys(addressBook || {}).length > 0)

return (
Expand Down Expand Up @@ -96,4 +102,6 @@ const AddressBookHeader = ({ handleOpenModal, searchQuery, onSearchQueryChange }
)
}

export default AddressBookHeader
export default mapProps(AddressBookHeader, {
allAddressBooks: () => useAppSelector(selectAllAddressBooks),

Check failure on line 106 in src/components/address-book/AddressBookHeader/index.tsx

View workflow job for this annotation

GitHub Actions / ESLint Results

react-hooks/rules-of-hooks

React Hook "useAppSelector" is called in function "allAddressBooks" that is neither a React function component nor a custom React Hook function. React component names must start with an uppercase letter. React Hook names must start with the word "use".
})
18 changes: 13 additions & 5 deletions src/components/address-book/AddressBookTable/index.tsx
Original file line number Diff line number Diff line change
@@ -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'
Expand All @@ -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' },
Expand All @@ -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']
}

const AddressBookTable = ({ chain, setTxFlow }: AddressBookTableProps) => {
const [open, setOpen] = useState<typeof defaultOpen>(defaultOpen)
const [searchQuery, setSearchQuery] = useState('')
const [defaultValues, setDefaultValues] = useState<AddressEntry | undefined>(undefined)
Expand Down Expand Up @@ -170,4 +175,7 @@ const AddressBookTable = () => {
)
}

export default AddressBookTable
export default madProps(AddressBookTable, {
chain: useCurrentChain,
setTxFlow: () => useContext(TxModalContext).setTxFlow,

Check failure on line 180 in src/components/address-book/AddressBookTable/index.tsx

View workflow job for this annotation

GitHub Actions / ESLint Results

react-hooks/rules-of-hooks

React Hook "useContext" is called in function "setTxFlow" that is neither a React function component nor a custom React Hook function. React component names must start with an uppercase letter. React Hook names must start with the word "use".
})
8 changes: 6 additions & 2 deletions src/components/address-book/EntryDialog/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ 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
Expand All @@ -22,14 +23,15 @@ const EntryDialog = ({
},
disableAddressInput = false,
chainId,
currentChainId,
}: {
handleClose: () => void
defaultValues?: AddressEntry
disableAddressInput?: boolean
chainId?: string
currentChainId: string
}): ReactElement => {
const dispatch = useAppDispatch()
const currentChainId = useChainId()

const methods = useForm<AddressEntry>({
defaultValues,
Expand Down Expand Up @@ -81,4 +83,6 @@ const EntryDialog = ({
)
}

export default EntryDialog
export default madProps(EntryDialog, {
currentChainId: useChainId,
})
18 changes: 13 additions & 5 deletions src/components/address-book/ExportDialog/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -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'
Expand All @@ -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<number>((acc, entries) => acc + Object.keys(entries).length, 0)
const ExportDialog = ({
allAddressBooks,
handleClose,
}: {
allAddressBooks: AddressBookState
handleClose: () => void
}): ReactElement => {
const length = Object.values(allAddressBooks).reduce<number>((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()
Expand Down Expand Up @@ -87,4 +93,6 @@ const ExportDialog = ({ handleClose }: { handleClose: () => void }): ReactElemen
)
}

export default ExportDialog
export default madProps(ExportDialog, {
allAddressBooks: () => useAppSelector(selectAllAddressBooks),

Check failure on line 97 in src/components/address-book/ExportDialog/index.tsx

View workflow job for this annotation

GitHub Actions / ESLint Results

react-hooks/rules-of-hooks

React Hook "useAppSelector" is called in function "allAddressBooks" that is neither a React function component nor a custom React Hook function. React component names must start with an uppercase letter. React Hook names must start with the word "use".
})
2 changes: 1 addition & 1 deletion src/components/tx-flow/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -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
Expand Down
36 changes: 36 additions & 0 deletions src/utils/mad-props.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
import React, { ComponentType, FC, useEffect, useState } from 'react'

Check failure on line 1 in src/utils/mad-props.tsx

View workflow job for this annotation

GitHub Actions / ESLint Results

@typescript-eslint/consistent-type-imports

Imports "ComponentType" and "FC" are only used as types.

type SinglePropHook<T> = () => T
type HookMap<P> = {
[K in keyof P]?: SinglePropHook<P[K]>
}

const madProps = <P extends Record<string, unknown>, H extends keyof P>(
Component: ComponentType<P>,
useHook: SinglePropHook<Pick<P, H>> | HookMap<Pick<P, H>>,
): FC<Omit<P, H>> => {
return (externalProps: Omit<P, H>) => {

Check failure on line 12 in src/utils/mad-props.tsx

View workflow job for this annotation

GitHub Actions / ESLint Results

react/display-name

Component definition is missing display name
const [computedProps, setComputedProps] = useState<P>({ ...(externalProps as P) })

useEffect(() => {
let newProps: P = { ...computedProps }

if (typeof useHook === 'function') {
newProps = { ...newProps, ...useHook() } as P
} else {
for (const key in useHook) {
const hook = useHook[key]
if (hook !== undefined) {
newProps[key as H] = hook()
}
}
}

setComputedProps(newProps)
}, [externalProps, useHook])

Check warning on line 30 in src/utils/mad-props.tsx

View workflow job for this annotation

GitHub Actions / ESLint Results

react-hooks/exhaustive-deps

React Hook useEffect has a missing dependency: 'computedProps'. Either include it or remove the dependency array. Outer scope values like 'useHook' aren't valid dependencies because mutating them doesn't re-render the component. * [SUGGESTION] Update the dependencies array to be: [computedProps, externalProps]

return <Component {...computedProps} />
}
}

export default madProps

0 comments on commit 4a0429c

Please sign in to comment.