* Add types for SafeProps Adds getSafePropsValuesFromQueryParams implementation Replaces window.location with useLocation hook * Replaces SafeProps with import in Layout.tsx Adds downlevelIteration to tsconfig.json to allow array.entries() * Type createSafe() * SafeDeployment Types * Type Paragraph and refactor to functional component * Fix validateQueryParams and types Co-authored-by: nicolas <nicosampler@users.noreply.github.com> Co-authored-by: Daniel Sanchez <daniel.sanchez@gnosis.pm>
This commit is contained in:
parent
d47bd1fe98
commit
0ea73c4eda
|
@ -1,23 +1,34 @@
|
||||||
import classNames from 'classnames/bind'
|
import classNames from 'classnames/bind'
|
||||||
import * as React from 'react'
|
import React, { MouseEventHandler, CSSProperties, ReactElement, ReactNode } from 'react'
|
||||||
|
|
||||||
import styles from './index.module.scss'
|
import styles from './index.module.scss'
|
||||||
|
|
||||||
const cx = classNames.bind(styles)
|
const cx = classNames.bind(styles)
|
||||||
|
|
||||||
class Paragraph extends React.PureComponent<any> {
|
interface Props {
|
||||||
render() {
|
align?: string
|
||||||
const { align, children, className, color, dot, noMargin, size, transform, weight, ...props } = this.props
|
children: ReactNode
|
||||||
|
className?: string
|
||||||
|
color?: string
|
||||||
|
dot?: string
|
||||||
|
noMargin?: boolean
|
||||||
|
size?: string
|
||||||
|
transform?: string
|
||||||
|
weight?: string
|
||||||
|
onClick?: MouseEventHandler<HTMLParagraphElement>
|
||||||
|
style?: CSSProperties
|
||||||
|
}
|
||||||
|
|
||||||
|
const Paragraph = (props: Props): ReactElement => {
|
||||||
|
const { align, children, className, color, dot, noMargin, size, transform, weight, ...restProps } = props
|
||||||
return (
|
return (
|
||||||
<p
|
<p
|
||||||
className={cx(styles.paragraph, className, weight, { noMargin, dot }, size, color, transform, align)}
|
className={cx(styles.paragraph, className, weight, { noMargin, dot }, size, color, transform, align)}
|
||||||
{...props}
|
{...restProps}
|
||||||
>
|
>
|
||||||
{children}
|
{children}
|
||||||
</p>
|
</p>
|
||||||
)
|
)
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export default Paragraph
|
export default Paragraph
|
||||||
|
|
|
@ -24,19 +24,13 @@ import { networkSelector, providerNameSelector, userAccountSelector } from 'src/
|
||||||
import { useSelector } from 'react-redux'
|
import { useSelector } from 'react-redux'
|
||||||
import { addressBookSelector } from 'src/logic/addressBook/store/selectors'
|
import { addressBookSelector } from 'src/logic/addressBook/store/selectors'
|
||||||
import { getNameFromAddressBook } from 'src/logic/addressBook/utils'
|
import { getNameFromAddressBook } from 'src/logic/addressBook/utils'
|
||||||
|
import { SafeProps } from 'src/routes/open/container/Open'
|
||||||
|
|
||||||
const { useEffect } = React
|
const { useEffect } = React
|
||||||
|
|
||||||
const getSteps = () => ['Name', 'Owners and confirmations', 'Review']
|
const getSteps = () => ['Name', 'Owners and confirmations', 'Review']
|
||||||
|
|
||||||
type SafeProps = {
|
export type InitialValuesForm = {
|
||||||
name: string
|
|
||||||
ownerAddresses: any
|
|
||||||
ownerNames: string
|
|
||||||
threshold: string
|
|
||||||
}
|
|
||||||
|
|
||||||
type InitialValuesForm = {
|
|
||||||
owner0Address?: string
|
owner0Address?: string
|
||||||
owner0Name?: string
|
owner0Name?: string
|
||||||
confirmations: string
|
confirmations: string
|
||||||
|
|
|
@ -3,8 +3,8 @@ import queryString from 'query-string'
|
||||||
import React, { useEffect, useState } from 'react'
|
import React, { useEffect, useState } from 'react'
|
||||||
import ReactGA from 'react-ga'
|
import ReactGA from 'react-ga'
|
||||||
import { useDispatch, useSelector } from 'react-redux'
|
import { useDispatch, useSelector } from 'react-redux'
|
||||||
import Opening from 'src/routes/opening'
|
import { SafeDeployment } from 'src/routes/opening'
|
||||||
import { Layout } from 'src/routes/open/components/Layout'
|
import { InitialValuesForm, Layout } from 'src/routes/open/components/Layout'
|
||||||
import Page from 'src/components/layout/Page'
|
import Page from 'src/components/layout/Page'
|
||||||
import { getSafeDeploymentTransaction } from 'src/logic/contracts/safeContracts'
|
import { getSafeDeploymentTransaction } from 'src/logic/contracts/safeContracts'
|
||||||
import { checkReceiptStatus } from 'src/logic/wallets/ethTransactions'
|
import { checkReceiptStatus } from 'src/logic/wallets/ethTransactions'
|
||||||
|
@ -24,21 +24,51 @@ import { userAccountSelector } from 'src/logic/wallets/store/selectors'
|
||||||
import { SafeRecordProps } from 'src/logic/safe/store/models/safe'
|
import { SafeRecordProps } from 'src/logic/safe/store/models/safe'
|
||||||
import { addOrUpdateSafe } from 'src/logic/safe/store/actions/addOrUpdateSafe'
|
import { addOrUpdateSafe } from 'src/logic/safe/store/actions/addOrUpdateSafe'
|
||||||
import { PromiEvent, TransactionReceipt } from 'web3-core'
|
import { PromiEvent, TransactionReceipt } from 'web3-core'
|
||||||
|
import { useLocation } from 'react-router-dom'
|
||||||
|
|
||||||
const SAFE_PENDING_CREATION_STORAGE_KEY = 'SAFE_PENDING_CREATION_STORAGE_KEY'
|
const SAFE_PENDING_CREATION_STORAGE_KEY = 'SAFE_PENDING_CREATION_STORAGE_KEY'
|
||||||
|
|
||||||
const validateQueryParams = (ownerAddresses, ownerNames, threshold, safeName) => {
|
interface SafeCreationQueryParams {
|
||||||
|
ownerAddresses: string | string[] | null
|
||||||
|
ownerNames: string | string[] | null
|
||||||
|
threshold: number | null
|
||||||
|
safeName: string | null
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SafeProps {
|
||||||
|
name: string
|
||||||
|
ownerAddresses: string[]
|
||||||
|
ownerNames: string[]
|
||||||
|
threshold: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const validateQueryParams = (queryParams: SafeCreationQueryParams): boolean => {
|
||||||
|
const { ownerAddresses, ownerNames, threshold, safeName } = queryParams
|
||||||
|
|
||||||
if (!ownerAddresses || !ownerNames || !threshold || !safeName) {
|
if (!ownerAddresses || !ownerNames || !threshold || !safeName) {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
if (!ownerAddresses.length || ownerNames.length === 0) {
|
|
||||||
|
if (Number.isNaN(threshold)) {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Number.isNaN(Number(threshold))) {
|
return threshold > 0 && threshold <= ownerAddresses.length
|
||||||
return false
|
}
|
||||||
|
|
||||||
|
const getSafePropsValuesFromQueryParams = (queryParams: SafeCreationQueryParams): SafeProps | undefined => {
|
||||||
|
if (!validateQueryParams(queryParams)) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const { threshold, safeName, ownerAddresses, ownerNames } = queryParams
|
||||||
|
|
||||||
|
return {
|
||||||
|
name: safeName as string,
|
||||||
|
threshold: (threshold as number).toString(),
|
||||||
|
ownerAddresses: Array.isArray(ownerAddresses) ? ownerAddresses : [ownerAddresses as string],
|
||||||
|
ownerNames: Array.isArray(ownerNames) ? ownerNames : [ownerNames as string],
|
||||||
}
|
}
|
||||||
return threshold <= ownerAddresses.length
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const getSafeProps = async (
|
export const getSafeProps = async (
|
||||||
|
@ -54,7 +84,7 @@ export const getSafeProps = async (
|
||||||
return safeProps
|
return safeProps
|
||||||
}
|
}
|
||||||
|
|
||||||
export const createSafe = (values, userAccount) => {
|
export const createSafe = (values: InitialValuesForm, userAccount: string): PromiEvent<TransactionReceipt> => {
|
||||||
const confirmations = getThresholdFrom(values)
|
const confirmations = getThresholdFrom(values)
|
||||||
const name = getSafeNameFrom(values)
|
const name = getSafeNameFrom(values)
|
||||||
const ownersNames = getNamesFrom(values)
|
const ownersNames = getNamesFrom(values)
|
||||||
|
@ -86,24 +116,26 @@ const Open = (): React.ReactElement => {
|
||||||
const [loading, setLoading] = useState(false)
|
const [loading, setLoading] = useState(false)
|
||||||
const [showProgress, setShowProgress] = useState(false)
|
const [showProgress, setShowProgress] = useState(false)
|
||||||
const [creationTxPromise, setCreationTxPromise] = useState<PromiEvent<TransactionReceipt>>()
|
const [creationTxPromise, setCreationTxPromise] = useState<PromiEvent<TransactionReceipt>>()
|
||||||
const [safeCreationPendingInfo, setSafeCreationPendingInfo] = useState<any>()
|
const [safeCreationPendingInfo, setSafeCreationPendingInfo] = useState<{ txHash?: string } | undefined>()
|
||||||
const [safePropsFromUrl, setSafePropsFromUrl] = useState()
|
const [safePropsFromUrl, setSafePropsFromUrl] = useState<SafeProps | undefined>()
|
||||||
const userAccount = useSelector(userAccountSelector)
|
const userAccount = useSelector(userAccountSelector)
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
|
const location = useLocation()
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
// #122: Allow to migrate an old Multisig by passing the parameters to the URL.
|
// #122: Allow to migrate an old Multisig by passing the parameters to the URL.
|
||||||
const query = queryString.parse(window.location.search, { arrayFormat: 'comma' })
|
const query = queryString.parse(location.search, { arrayFormat: 'comma' })
|
||||||
const { name, owneraddresses, ownernames, threshold } = query
|
const { name, owneraddresses, ownernames, threshold } = query
|
||||||
if (validateQueryParams(owneraddresses, ownernames, threshold, name)) {
|
|
||||||
setSafePropsFromUrl({
|
const safeProps = getSafePropsValuesFromQueryParams({
|
||||||
name,
|
|
||||||
ownerAddresses: owneraddresses,
|
ownerAddresses: owneraddresses,
|
||||||
ownerNames: ownernames,
|
ownerNames: ownernames,
|
||||||
threshold,
|
threshold: Number(threshold),
|
||||||
} as any)
|
safeName: name as string | null,
|
||||||
}
|
})
|
||||||
}, [])
|
|
||||||
|
setSafePropsFromUrl(safeProps)
|
||||||
|
}, [location])
|
||||||
|
|
||||||
// check if there is a safe being created
|
// check if there is a safe being created
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
@ -121,7 +153,7 @@ const Open = (): React.ReactElement => {
|
||||||
load()
|
load()
|
||||||
}, [])
|
}, [])
|
||||||
|
|
||||||
const createSafeProxy = async (formValues?: any) => {
|
const createSafeProxy = async (formValues?: InitialValuesForm) => {
|
||||||
let values = formValues
|
let values = formValues
|
||||||
|
|
||||||
// save form values, used when the user rejects the TX and wants to retry
|
// save form values, used when the user rejects the TX and wants to retry
|
||||||
|
@ -132,7 +164,7 @@ const Open = (): React.ReactElement => {
|
||||||
values = await loadFromStorage(SAFE_PENDING_CREATION_STORAGE_KEY)
|
values = await loadFromStorage(SAFE_PENDING_CREATION_STORAGE_KEY)
|
||||||
}
|
}
|
||||||
|
|
||||||
const promiEvent = createSafe(values, userAccount)
|
const promiEvent = createSafe(values as InitialValuesForm, userAccount)
|
||||||
setCreationTxPromise(promiEvent)
|
setCreationTxPromise(promiEvent)
|
||||||
setShowProgress(true)
|
setShowProgress(true)
|
||||||
}
|
}
|
||||||
|
@ -186,7 +218,7 @@ const Open = (): React.ReactElement => {
|
||||||
return (
|
return (
|
||||||
<Page>
|
<Page>
|
||||||
{showProgress ? (
|
{showProgress ? (
|
||||||
<Opening
|
<SafeDeployment
|
||||||
creationTxHash={safeCreationPendingInfo?.txHash}
|
creationTxHash={safeCreationPendingInfo?.txHash}
|
||||||
onCancel={onCancel}
|
onCancel={onCancel}
|
||||||
onRetry={onRetry}
|
onRetry={onRetry}
|
||||||
|
|
|
@ -20,6 +20,7 @@ import LoaderDotsSvg from './assets/loader-dots.svg'
|
||||||
import SuccessSvg from './assets/success.svg'
|
import SuccessSvg from './assets/success.svg'
|
||||||
import VaultErrorSvg from './assets/vault-error.svg'
|
import VaultErrorSvg from './assets/vault-error.svg'
|
||||||
import VaultSvg from './assets/vault.svg'
|
import VaultSvg from './assets/vault.svg'
|
||||||
|
import { PromiEvent, TransactionReceipt } from 'web3-core'
|
||||||
|
|
||||||
const Wrapper = styled.div`
|
const Wrapper = styled.div`
|
||||||
display: grid;
|
display: grid;
|
||||||
|
@ -56,13 +57,17 @@ const Body = styled.div`
|
||||||
const CardTitle = styled.div`
|
const CardTitle = styled.div`
|
||||||
font-size: 20px;
|
font-size: 20px;
|
||||||
`
|
`
|
||||||
const FullParagraph = styled(Paragraph)`
|
|
||||||
background-color: ${(p) => (p.inverseColors ? connected : background)};
|
interface FullParagraphProps {
|
||||||
color: ${(p) => (p.inverseColors ? background : connected)};
|
inversecolors: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const FullParagraph = styled(Paragraph)<FullParagraphProps>`
|
||||||
|
background-color: ${(p) => (p.inversecolors ? connected : background)};
|
||||||
|
color: ${(p) => (p.inversecolors ? background : connected)};
|
||||||
padding: 24px;
|
padding: 24px;
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
margin-bottom: 16px;
|
margin-bottom: 16px;
|
||||||
|
|
||||||
transition: color 0.3s ease-in-out, background-color 0.3s ease-in-out;
|
transition: color 0.3s ease-in-out, background-color 0.3s ease-in-out;
|
||||||
`
|
`
|
||||||
|
|
||||||
|
@ -95,16 +100,21 @@ const BackButton = styled(Button)`
|
||||||
margin: 20px auto 0;
|
margin: 20px auto 0;
|
||||||
`
|
`
|
||||||
|
|
||||||
// type Props = {
|
type Props = {
|
||||||
// provider: string
|
creationTxHash?: string
|
||||||
// creationTxHash: Promise<any>
|
submittedPromise?: PromiEvent<TransactionReceipt>
|
||||||
// submittedPromise: Promise<any>
|
onRetry: () => void
|
||||||
// onRetry: () => void
|
onSuccess: (createdSafeAddress: string) => void
|
||||||
// onSuccess: () => void
|
onCancel: () => void
|
||||||
// onCancel: () => void
|
}
|
||||||
// }
|
|
||||||
|
|
||||||
const SafeDeployment = ({ creationTxHash, onCancel, onRetry, onSuccess, submittedPromise }): React.ReactElement => {
|
export const SafeDeployment = ({
|
||||||
|
creationTxHash,
|
||||||
|
onCancel,
|
||||||
|
onRetry,
|
||||||
|
onSuccess,
|
||||||
|
submittedPromise,
|
||||||
|
}: Props): React.ReactElement => {
|
||||||
const [loading, setLoading] = useState(true)
|
const [loading, setLoading] = useState(true)
|
||||||
const [stepIndex, setStepIndex] = useState(0)
|
const [stepIndex, setStepIndex] = useState(0)
|
||||||
const [safeCreationTxHash, setSafeCreationTxHash] = useState('')
|
const [safeCreationTxHash, setSafeCreationTxHash] = useState('')
|
||||||
|
@ -326,7 +336,7 @@ const SafeDeployment = ({ creationTxHash, onCancel, onRetry, onSuccess, submitte
|
||||||
<BodyLoader>{!error && stepIndex <= 4 && <Img alt="Loader dots" src={LoaderDotsSvg} />}</BodyLoader>
|
<BodyLoader>{!error && stepIndex <= 4 && <Img alt="Loader dots" src={LoaderDotsSvg} />}</BodyLoader>
|
||||||
|
|
||||||
<BodyInstruction>
|
<BodyInstruction>
|
||||||
<FullParagraph color="primary" inverseColors={confirmationStep} noMargin size="md">
|
<FullParagraph color="primary" inversecolors={confirmationStep.toString()} noMargin size="md">
|
||||||
{error ? 'You can Cancel or Retry the Safe creation process.' : steps[stepIndex].instruction}
|
{error ? 'You can Cancel or Retry the Safe creation process.' : steps[stepIndex].instruction}
|
||||||
</FullParagraph>
|
</FullParagraph>
|
||||||
</BodyInstruction>
|
</BodyInstruction>
|
||||||
|
@ -350,5 +360,3 @@ const SafeDeployment = ({ creationTxHash, onCancel, onRetry, onSuccess, submitte
|
||||||
</Wrapper>
|
</Wrapper>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
export default SafeDeployment
|
|
||||||
|
|
|
@ -82,6 +82,7 @@ export const aMinedSafe = async (
|
||||||
[FIELD_NAME]: name,
|
[FIELD_NAME]: name,
|
||||||
[FIELD_CONFIRMATIONS]: `${threshold}`,
|
[FIELD_CONFIRMATIONS]: `${threshold}`,
|
||||||
[FIELD_OWNERS]: `${owners}`,
|
[FIELD_OWNERS]: `${owners}`,
|
||||||
|
safeCreationSalt: 0
|
||||||
}
|
}
|
||||||
|
|
||||||
for (let i = 0; i < owners; i += 1) {
|
for (let i = 0; i < owners; i += 1) {
|
||||||
|
|
|
@ -21,7 +21,8 @@
|
||||||
"isolatedModules": true,
|
"isolatedModules": true,
|
||||||
"noEmit": true,
|
"noEmit": true,
|
||||||
"jsx": "react",
|
"jsx": "react",
|
||||||
"noFallthroughCasesInSwitch": true
|
"noFallthroughCasesInSwitch": true,
|
||||||
|
"downlevelIteration": true
|
||||||
},
|
},
|
||||||
"paths": {
|
"paths": {
|
||||||
"src/*": [
|
"src/*": [
|
||||||
|
|
Loading…
Reference in New Issue