Implement functionality to import wallet from mnemonic (#13)
Part of #11 Co-authored-by: IshaVenikar <ishavenikar7@gmail.com> Reviewed-on: #13
This commit is contained in:
parent
c26bddec1a
commit
2bb92205ba
@ -1,5 +1,5 @@
|
|||||||
import { View } from 'react-native';
|
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
import { View } from 'react-native';
|
||||||
import { Button } from 'react-native-paper';
|
import { Button } from 'react-native-paper';
|
||||||
|
|
||||||
import { CreateWalletProps } from '../types';
|
import { CreateWalletProps } from '../types';
|
||||||
|
66
src/components/ImportWalletDialog.tsx
Normal file
66
src/components/ImportWalletDialog.tsx
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
import React, { useEffect, useState } from 'react';
|
||||||
|
|
||||||
|
import { Dialog, DialogActions, DialogContent, DialogTitle, TextField, Grid, Button, Typography } from "@mui/material";
|
||||||
|
|
||||||
|
const ImportWalletDialog = ({
|
||||||
|
visible,
|
||||||
|
hideDialog,
|
||||||
|
importWalletHandler
|
||||||
|
}: {
|
||||||
|
visible: boolean;
|
||||||
|
hideDialog: () => void;
|
||||||
|
importWalletHandler: (recoveryPhrase: string) => Promise<void>;
|
||||||
|
}) => {
|
||||||
|
const [words, setWords] = useState(Array(12).fill(''));
|
||||||
|
|
||||||
|
const handleWordChange = (index: number, value: string) => {
|
||||||
|
const newWords = [...words];
|
||||||
|
newWords[index] = value;
|
||||||
|
setWords(newWords);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handlePaste = (event: React.ClipboardEvent<HTMLDivElement>) => {
|
||||||
|
const pastedText = event.clipboardData.getData('Text');
|
||||||
|
const splitWords = pastedText.trim().split(/\s+/);
|
||||||
|
|
||||||
|
if (splitWords.length === 12) {
|
||||||
|
setWords(splitWords);
|
||||||
|
event.preventDefault();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setWords(Array(12).fill(''));
|
||||||
|
},[visible]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog open={visible} onClose={hideDialog}>
|
||||||
|
<DialogTitle>Import your wallet from your mnemonic</DialogTitle>
|
||||||
|
<DialogContent>
|
||||||
|
<Typography>
|
||||||
|
(You can paste your entire mnemonic into the first textbox)
|
||||||
|
</Typography>
|
||||||
|
<Grid container spacing={2}>
|
||||||
|
{words.map((word, index) => (
|
||||||
|
<Grid item xs={6} sm={4} key={index}>
|
||||||
|
<TextField
|
||||||
|
value={word}
|
||||||
|
onChange={(e) => handleWordChange(index, e.target.value)}
|
||||||
|
onPaste={index === 0 ? handlePaste : undefined}
|
||||||
|
placeholder={`Word ${index + 1}`}
|
||||||
|
fullWidth
|
||||||
|
margin="normal"
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
))}
|
||||||
|
</Grid>
|
||||||
|
</DialogContent>
|
||||||
|
<DialogActions>
|
||||||
|
<Button onClick={() => importWalletHandler(words.join(' '))}>Import Wallet</Button>
|
||||||
|
<Button onClick={hideDialog}>Cancel</Button>
|
||||||
|
</DialogActions>
|
||||||
|
</Dialog>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default ImportWalletDialog;
|
@ -5,7 +5,7 @@ import styles from '../styles/stylesheet';
|
|||||||
import GridView from './Grid';
|
import GridView from './Grid';
|
||||||
import { CustomDialogProps } from '../types';
|
import { CustomDialogProps } from '../types';
|
||||||
|
|
||||||
const DialogComponent = ({ visible, hideDialog, contentText }: CustomDialogProps) => {
|
const MnemonicDialog = ({ visible, hideDialog, contentText }: CustomDialogProps) => {
|
||||||
const words = contentText.split(' ');
|
const words = contentText.split(' ');
|
||||||
|
|
||||||
const copyMnemonic = () => {
|
const copyMnemonic = () => {
|
||||||
@ -26,10 +26,10 @@ const DialogComponent = ({ visible, hideDialog, contentText }: CustomDialogProps
|
|||||||
</DialogContent>
|
</DialogContent>
|
||||||
<DialogActions>
|
<DialogActions>
|
||||||
<Button onClick={copyMnemonic}>Copy</Button>
|
<Button onClick={copyMnemonic}>Copy</Button>
|
||||||
<Button onClick={hideDialog}>Done</Button>
|
<Button onClick={hideDialog}>Cancel</Button>
|
||||||
</DialogActions>
|
</DialogActions>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export { DialogComponent };
|
export { MnemonicDialog };
|
@ -5,9 +5,11 @@ import { Button, Text } from 'react-native-paper';
|
|||||||
import { NativeStackNavigationProp } from '@react-navigation/native-stack';
|
import { NativeStackNavigationProp } from '@react-navigation/native-stack';
|
||||||
import { useNavigation } from '@react-navigation/native';
|
import { useNavigation } from '@react-navigation/native';
|
||||||
import { getSdkError } from '@walletconnect/utils';
|
import { getSdkError } from '@walletconnect/utils';
|
||||||
|
import { Portal, Snackbar } from '@mui/material';
|
||||||
|
|
||||||
import { createWallet, resetWallet, retrieveAccounts } from '../utils/accounts';
|
import { createWallet, resetWallet, retrieveAccounts } from '../utils/accounts';
|
||||||
import { DialogComponent } from '../components/Dialog';
|
import { MnemonicDialog } from '../components/MnemonicDialog';
|
||||||
|
import ImportWalletDialog from '../components/ImportWalletDialog';
|
||||||
import { NetworkDropdown } from '../components/NetworkDropdown';
|
import { NetworkDropdown } from '../components/NetworkDropdown';
|
||||||
import Accounts from '../components/Accounts';
|
import Accounts from '../components/Accounts';
|
||||||
import CreateWallet from '../components/CreateWallet';
|
import CreateWallet from '../components/CreateWallet';
|
||||||
@ -55,12 +57,15 @@ const HomeScreen = () => {
|
|||||||
|
|
||||||
const [isWalletCreated, setIsWalletCreated] = useState<boolean>(false);
|
const [isWalletCreated, setIsWalletCreated] = useState<boolean>(false);
|
||||||
const [isWalletCreating, setIsWalletCreating] = useState<boolean>(false);
|
const [isWalletCreating, setIsWalletCreating] = useState<boolean>(false);
|
||||||
const [walletDialog, setWalletDialog] = useState<boolean>(false);
|
const [importWalletDialog, setImportWalletDialog] = useState<boolean>(false);
|
||||||
|
const [mnemonicDialog, setMnemonicDialog] = useState<boolean>(false);
|
||||||
const [resetWalletDialog, setResetWalletDialog] = useState<boolean>(false);
|
const [resetWalletDialog, setResetWalletDialog] = useState<boolean>(false);
|
||||||
|
const [toastVisible, setToastVisible] = useState(false);
|
||||||
|
const [invalidMnemonicError, setInvalidMnemonicError] = useState('');
|
||||||
const [isAccountsFetched, setIsAccountsFetched] = useState<boolean>(true);
|
const [isAccountsFetched, setIsAccountsFetched] = useState<boolean>(true);
|
||||||
const [phrase, setPhrase] = useState('');
|
const [phrase, setPhrase] = useState('');
|
||||||
|
|
||||||
const hideWalletDialog = () => setWalletDialog(false);
|
const hideMnemonicDialog = () => setMnemonicDialog(false);
|
||||||
const hideResetDialog = () => setResetWalletDialog(false);
|
const hideResetDialog = () => setResetWalletDialog(false);
|
||||||
|
|
||||||
const fetchAccounts = useCallback(async () => {
|
const fetchAccounts = useCallback(async () => {
|
||||||
@ -83,12 +88,27 @@ const HomeScreen = () => {
|
|||||||
const mnemonic = await createWallet(networksData);
|
const mnemonic = await createWallet(networksData);
|
||||||
if (mnemonic) {
|
if (mnemonic) {
|
||||||
fetchAccounts();
|
fetchAccounts();
|
||||||
setWalletDialog(true);
|
setMnemonicDialog(true);
|
||||||
setPhrase(mnemonic);
|
setPhrase(mnemonic);
|
||||||
setSelectedNetwork(networksData[0]);
|
setSelectedNetwork(networksData[0]);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const importWalletHandler = async (recoveryPhrase: string) => {
|
||||||
|
try{
|
||||||
|
const mnemonic = await createWallet(networksData, recoveryPhrase);
|
||||||
|
if (mnemonic) {
|
||||||
|
fetchAccounts();
|
||||||
|
setPhrase(mnemonic);
|
||||||
|
setSelectedNetwork(networksData[0]);
|
||||||
|
setImportWalletDialog(false);
|
||||||
|
}
|
||||||
|
} catch (error: any) {
|
||||||
|
setInvalidMnemonicError((error.message as string).toUpperCase())
|
||||||
|
setToastVisible(true);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const confirmResetWallet = useCallback(async () => {
|
const confirmResetWallet = useCallback(async () => {
|
||||||
setIsWalletCreated(false);
|
setIsWalletCreated(false);
|
||||||
setIsWalletCreating(false);
|
setIsWalletCreating(false);
|
||||||
@ -152,14 +172,28 @@ const HomeScreen = () => {
|
|||||||
</View>
|
</View>
|
||||||
</>
|
</>
|
||||||
) : (
|
) : (
|
||||||
<CreateWallet
|
<>
|
||||||
isWalletCreating={isWalletCreating}
|
<CreateWallet
|
||||||
createWalletHandler={createWalletHandler}
|
isWalletCreating={isWalletCreating}
|
||||||
/>
|
createWalletHandler={createWalletHandler}
|
||||||
|
/>
|
||||||
|
<View style={styles.createWalletContainer}>
|
||||||
|
<Button
|
||||||
|
mode="contained"
|
||||||
|
onPress={() => setImportWalletDialog(true)}>
|
||||||
|
Import Wallet
|
||||||
|
</Button>
|
||||||
|
</View>
|
||||||
|
</>
|
||||||
)}
|
)}
|
||||||
<DialogComponent
|
<ImportWalletDialog
|
||||||
visible={walletDialog}
|
visible={importWalletDialog}
|
||||||
hideDialog={hideWalletDialog}
|
hideDialog={() => setImportWalletDialog(false)}
|
||||||
|
importWalletHandler={importWalletHandler}
|
||||||
|
/>
|
||||||
|
<MnemonicDialog
|
||||||
|
visible={mnemonicDialog}
|
||||||
|
hideDialog={hideMnemonicDialog}
|
||||||
contentText={phrase}
|
contentText={phrase}
|
||||||
/>
|
/>
|
||||||
<ConfirmDialog
|
<ConfirmDialog
|
||||||
@ -168,6 +202,16 @@ const HomeScreen = () => {
|
|||||||
hideDialog={hideResetDialog}
|
hideDialog={hideResetDialog}
|
||||||
onConfirm={confirmResetWallet}
|
onConfirm={confirmResetWallet}
|
||||||
/>
|
/>
|
||||||
|
<Portal>
|
||||||
|
<Snackbar
|
||||||
|
open={toastVisible}
|
||||||
|
autoHideDuration={3000}
|
||||||
|
message={invalidMnemonicError}
|
||||||
|
onClose={() => setToastVisible(false)}
|
||||||
|
anchorOrigin={{ horizontal: 'center', vertical: 'bottom' }}
|
||||||
|
ContentProps={{ style: { backgroundColor: 'red', color: 'white'} }}
|
||||||
|
/>
|
||||||
|
</Portal>
|
||||||
</View>
|
</View>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@ -5,7 +5,7 @@ const styles = StyleSheet.create({
|
|||||||
marginTop: 20,
|
marginTop: 20,
|
||||||
width: 150,
|
width: 150,
|
||||||
alignSelf: 'center',
|
alignSelf: 'center',
|
||||||
marginBottom: 40
|
marginBottom: 30,
|
||||||
},
|
},
|
||||||
signLink: {
|
signLink: {
|
||||||
alignItems: 'flex-end',
|
alignItems: 'flex-end',
|
||||||
|
@ -27,12 +27,23 @@ import { COSMOS, EIP155 } from './constants';
|
|||||||
|
|
||||||
const createWallet = async (
|
const createWallet = async (
|
||||||
networksData: NetworksDataState[],
|
networksData: NetworksDataState[],
|
||||||
|
recoveryPhrase?: string,
|
||||||
): Promise<string> => {
|
): Promise<string> => {
|
||||||
const mnemonic = utils.entropyToMnemonic(utils.randomBytes(16));
|
const mnemonic = recoveryPhrase ? recoveryPhrase : utils.entropyToMnemonic(utils.randomBytes(16));
|
||||||
await setInternetCredentials('mnemonicServer', 'mnemonic', mnemonic);
|
|
||||||
|
|
||||||
const hdNode = HDNode.fromMnemonic(mnemonic);
|
const hdNode = HDNode.fromMnemonic(mnemonic);
|
||||||
|
await setInternetCredentials('mnemonicServer', 'mnemonic', mnemonic);
|
||||||
|
|
||||||
|
await createWalletFromMnemonic(networksData, hdNode, mnemonic);
|
||||||
|
|
||||||
|
return mnemonic;
|
||||||
|
};
|
||||||
|
|
||||||
|
const createWalletFromMnemonic = async (
|
||||||
|
networksData: NetworksDataState[],
|
||||||
|
hdNode: HDNode,
|
||||||
|
mnemonic: string
|
||||||
|
): Promise<void> => {
|
||||||
for (const network of networksData) {
|
for (const network of networksData) {
|
||||||
const hdPath = `m/44'/${network.coinType}'/0'/0/0`;
|
const hdPath = `m/44'/${network.coinType}'/0'/0/0`;
|
||||||
const node = hdNode.derivePath(hdPath);
|
const node = hdNode.derivePath(hdPath);
|
||||||
@ -73,8 +84,6 @@ const createWallet = async (
|
|||||||
),
|
),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return mnemonic;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const addAccount = async (
|
const addAccount = async (
|
||||||
|
Loading…
Reference in New Issue
Block a user