Compare commits

...

32 Commits

Author SHA1 Message Date
fmtabbara 1ac4755a03 update password strength test + use autofocus prop for password input 2022-04-01 09:26:14 +01:00
fmtabbara f9d5ef144b use login type selector 2022-03-31 23:14:22 +01:00
fmtabbara 8cc4a9e230 ui tweaks 2022-03-31 16:29:36 +01:00
fmtabbara bf87a4c5ce use clipboard lib directly 2022-03-31 16:28:46 +01:00
fmtabbara 60d0775785 reorder pages 2022-03-31 16:28:02 +01:00
fmtabbara 745243ebd2 use new sign in pages 2022-03-31 16:09:09 +01:00
fmtabbara abe5a5f5d3 create step component 2022-03-31 16:08:05 +01:00
fmtabbara ec8d659a46 fix workmark svg sizing issue 2022-03-31 16:07:19 +01:00
fmtabbara 824822c19a add a hook for clipboard copy 2022-03-31 16:06:48 +01:00
fmtabbara d594d85d1c add separate sign in pages for mnemonic and password 2022-03-31 16:06:18 +01:00
fmtabbara 770b78f21d merge develop 2022-03-28 11:10:45 +01:00
fmtabbara 068361b044 remove non-existent method 2022-03-25 13:22:52 +00:00
fmtabbara e1239663ef Merge branch 'feature/wire-up-wallet-storage' into feature/password-for-wallet 2022-03-25 12:19:38 +00:00
fmtabbara a1cd454dc2 update components 2022-03-21 10:58:28 +00:00
fmtabbara d22f990300 connect new rust methods with frontend 2022-03-21 10:58:08 +00:00
fmtabbara a562984bc5 update pages 2022-03-21 10:57:48 +00:00
fmtabbara 155f8401f1 move state to context 2022-03-21 10:57:13 +00:00
fmtabbara ce8d74fd69 update sign in functions 2022-03-21 10:55:18 +00:00
fmtabbara 53573d7c21 create sign-in context 2022-03-21 10:54:58 +00:00
fmtabbara 42a0bd0492 merge rust work for password 2022-03-21 10:53:09 +00:00
Jon Häggblad 1bdf8ab6c1 wallet: tweak some type names 2022-03-21 11:17:40 +01:00
Jon Häggblad cf7315f680 wallet: general wallet_storage tidy 2022-03-21 11:07:21 +01:00
Jon Häggblad 07101a9dc5 wallet: tweak error enum names 2022-03-21 10:36:41 +01:00
Jon Häggblad 35efd07eda wallet: inline encryption of wallet file 2022-03-18 15:25:30 +01:00
fmtabbara 522d67670e UI for existing mnemonic to be use 2022-03-16 17:34:28 +00:00
Jon Häggblad ee6fd8808f wallet: swap println to log 2022-03-16 12:30:07 +01:00
Jon Häggblad eec722744c wallet: platform_constants 2022-03-16 12:30:07 +01:00
Jon Häggblad 0c3e30ee2c wallets: provide placeholder functions for ui password 2022-03-16 12:30:07 +01:00
fmtabbara 9f104a70cf dont load account when creating mnemonic 2022-03-15 14:04:03 +00:00
fmtabbara 7aca9848dd fix linting 2022-03-14 22:50:57 +00:00
fmtabbara 7082a12eae update global error and load state from children 2022-03-14 22:50:32 +00:00
fmtabbara e5898c9053 new password flow 2022-03-14 22:49:28 +00:00
27 changed files with 651 additions and 261 deletions
+1 -1
View File
@@ -1,4 +1,4 @@
<svg width="210" height="56" viewBox="0 0 210 56" fill="none" xmlns="http://www.w3.org/2000/svg">
<svg viewBox="0 0 210 56" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M45.8829 0.142822H45.7169V0.28114V48.637L25.3289 0.225818L25.3012 0.142822H25.1905H13.6272H0.652966H0.514648V0.28114V55.7189V55.8572H0.652966H13.6272H13.7655V55.7189V7.28002L34.2365 55.7742L34.2642 55.8572H34.3748H45.8829H58.8294H58.9677V55.7189V0.28114V0.142822H58.8294H45.8829Z"/>
<path d="M209.347 0.142822H184.616H184.477L184.45 0.253483L171.78 48.8583L159.082 0.253483L159.054 0.142822H158.944H134.157H133.991V0.28114V55.7189V55.8572H134.157H147.104H147.242V55.7189V7.66731L159.774 55.7466L159.801 55.8572H159.94H183.564H183.675L183.703 55.7466L196.234 7.66731V55.7189V55.8572H196.373H209.347H209.485V55.7189V0.28114V0.142822H209.347Z"/>
<path d="M112.663 0.142822H112.58L112.552 0.198153L96.8116 27.5574L80.988 0.198153L80.9604 0.142822H80.8774H65.9114H65.6348L65.7731 0.364136L90.1447 42.5787V55.7189V55.8572H90.283H103.257H103.396V55.7189V42.5787L127.767 0.364136L127.905 0.142822H127.629H112.663Z"/>

Before

Width:  |  Height:  |  Size: 1.0 KiB

After

Width:  |  Height:  |  Size: 1011 B

+1
View File
@@ -38,6 +38,7 @@
"react-hook-form": "^7.14.2",
"react-router-dom": "^5.2.0",
"semver": "^6.3.0",
"use-clipboard-copy": "^0.2.0",
"yup": "^0.32.9"
},
"devDependencies": {
+20 -6
View File
@@ -4,9 +4,10 @@ import { useSnackbar } from 'notistack';
import { Account, Network, TCurrency, TMixnodeBondDetails } from '../types';
import { TUseuserBalance, useGetBalance } from '../hooks/useGetBalance';
import { config } from '../../config';
import { getMixnodeBondDetails, selectNetwork, signInWithMnemonic, signOut } from '../requests';
import { getMixnodeBondDetails, selectNetwork, signInWithMnemonic, signInWithPassword, signOut } from '../requests';
import { currencyMap } from '../utils';
import { Console } from '../utils/console';
import { TLoginType } from 'src/pages/welcome/types';
export const { ADMIN_ADDRESS, IS_DEV_MODE } = config;
@@ -32,11 +33,14 @@ type TClientContext = {
currency?: TCurrency;
isLoading: boolean;
error?: string;
setIsLoading: (isLoading: boolean) => void;
setError: (value?: string) => void;
switchNetwork: (network: Network) => void;
getBondDetails: () => Promise<void>;
handleShowSettings: () => void;
handleShowAdmin: () => void;
logIn: (mnemonic: string) => void;
logIn: (opts: { type: 'mnemonic' | 'password'; value: string }) => void;
signInWithPassword: (password: string) => void;
logOut: () => void;
};
@@ -90,16 +94,23 @@ export const ClientContextProvider = ({ children }: { children: React.ReactNode
refreshAccount();
}, [network]);
const logIn = async (mnemonic: string) => {
const logIn = async ({ type, value }: { type: TLoginType; value: string }) => {
if (value.length === 0) {
setError(`A ${type} must be provided`);
return;
}
try {
setIsLoading(true);
await signInWithMnemonic(mnemonic || '');
await getBondDetails();
if (type === 'mnemonic') {
await signInWithMnemonic(value);
} else {
await signInWithPassword(value);
}
setNetwork('MAINNET');
} catch (e) {
setIsLoading(false);
setError(e as string);
} finally {
setIsLoading(false);
history.push('/balance');
}
};
@@ -131,6 +142,9 @@ export const ClientContextProvider = ({ children }: { children: React.ReactNode
showSettings,
network,
currency,
setIsLoading,
setError,
signInWithPassword,
switchNetwork,
getBondDetails,
handleShowSettings,
+4 -1
View File
@@ -10,6 +10,7 @@ import { Admin, Welcome, Settings } from './pages';
import { ErrorFallback } from './components';
import { NymWalletTheme, WelcomeTheme } from './theme';
import { maximizeWindow } from './utils';
import { SignInProvider } from './pages/welcome/context';
const App = () => {
const { clientDetails } = useContext(ClientContext);
@@ -20,7 +21,9 @@ const App = () => {
return !clientDetails ? (
<WelcomeTheme>
<Welcome />
<SignInProvider>
<Welcome />
</SignInProvider>
</WelcomeTheme>
) : (
<NymWalletTheme>
@@ -16,6 +16,9 @@ export const SendConfirmation = ({
isLoading: boolean;
}) => {
const { userBalance, currency, network } = useContext(ClientContext);
if (!data && !error && !isLoading) return null;
return (
<Box
sx={{
+13 -8
View File
@@ -128,13 +128,9 @@ export const SendWizard = () => {
px: 3,
}}
>
{activeStep === 0 ? (
<SendForm />
) : activeStep === 1 ? (
<SendReview transferFee={transferFee} />
) : (
<SendConfirmation data={confirmedData} isLoading={isLoading} error={requestError} />
)}
{activeStep === 0 && <SendForm />}
{activeStep === 1 && <SendReview transferFee={transferFee} />}
<SendConfirmation data={confirmedData} isLoading={isLoading} error={requestError} />
</Box>
<Box
sx={{
@@ -154,7 +150,16 @@ export const SendWizard = () => {
color="primary"
disableElevation
data-testid="button"
onClick={activeStep === 0 ? handleNextStep : activeStep === 1 ? handleSend : handleFinish}
onClick={() => {
switch (activeStep) {
case 0:
return handleNextStep();
case 1:
return handleSend();
default:
return handleFinish();
}
}}
disabled={!!(methods.formState.errors.amount || methods.formState.errors.to || isLoading)}
size="large"
>
@@ -139,13 +139,13 @@ export const SystemVariables = ({
pt: 0,
}}
>
{nodeUpdateResponse === 'success' ? (
{nodeUpdateResponse === 'success' && (
<Typography sx={{ color: 'success.main', fontWeight: 600 }}>Node successfully updated</Typography>
) : nodeUpdateResponse === 'failed' ? (
<Typography sx={{ color: 'error.main', fontWeight: 600 }}>Node update failed</Typography>
) : (
<Fee feeType="UpdateMixnodeConfig" />
)}
{nodeUpdateResponse === 'failed' && (
<Typography sx={{ color: 'error.main', fontWeight: 600 }}>Node update failed</Typography>
)}
{!nodeUpdateResponse && <Fee feeType="UpdateMixnodeConfig" />}
<Button
variant="contained"
color="primary"
@@ -17,7 +17,7 @@ export const SignInContent: React.FC = () => {
setInputError(undefined);
try {
await logIn(mnemonic || '');
await logIn({ type: 'mnemonic', value: mnemonic });
setIsLoading(false);
} catch (error: any) {
setIsLoading(false);
@@ -0,0 +1,8 @@
import React from 'react';
import { Alert } from '@mui/material';
export const Error = ({ message }: { message: string }) => (
<Alert severity="error" variant="outlined" data-testid="error" sx={{ color: 'error.light', width: '100%' }}>
{message}
</Alert>
);
@@ -2,3 +2,6 @@ export * from './heading';
export * from './word-tiles';
export * from './render-page';
export * from './password-strength';
export * from './error';
export * from './textfields';
export * from './step';
@@ -5,8 +5,8 @@ import { LinearProgress, Stack, Typography, Box } from '@mui/material';
type TStrength = 'weak' | 'medium' | 'strong' | 'init';
const strong = /^(?=.*[a-z])(?=.*[A-Z])(?=.*[0-9])(?=.*[!@#$%^&*])(?=.{8,})/;
const medium = /^(((?=.*[a-z])(?=.*[A-Z]))|((?=.*[a-z])(?=.*[0-9]))|((?=.*[A-Z])(?=.*[0-9])))(?=.{6,})/;
const strong = /^(?=.*[a-z])(?=.*[0-9])(?=.*[!@#$%^&*])(?=.{8,})/;
const medium = /^(((?=.*[a-z]))|((?=.*[a-z])(?=.*[0-9]))|((?=.*[0-9])))(?=.{6,})/;
const colorMap = {
init: 'inherit' as 'inherit',
@@ -41,7 +41,24 @@ const getTextColor = (strength: TStrength) => {
}
};
export const PasswordStrength = ({ password }: { password: string }) => {
const getPasswordStrength = (strength: TStrength) => {
switch (strength) {
case 'strong':
return 100;
case 'medium':
return 50;
default:
return 0;
}
};
export const PasswordStrength = ({
password,
onChange,
}: {
password: string;
onChange: (isStrong: boolean) => void;
}) => {
const [strength, setStrength] = useState<TStrength>('init');
useEffect(() => {
@@ -62,13 +79,17 @@ export const PasswordStrength = ({ password }: { password: string }) => {
setStrength('weak');
}, [password]);
useEffect(() => {
if (strength === 'strong') {
onChange(true);
} else {
onChange(false);
}
}, [strength]);
return (
<Stack spacing={0.5}>
<LinearProgress
variant="determinate"
color={colorMap[strength]}
value={strength === 'strong' ? 100 : strength === 'medium' ? 50 : 0}
/>
<LinearProgress variant="determinate" color={colorMap[strength]} value={getPasswordStrength(strength)} />
<Box display="flex" alignItems="center">
<LockOutlined sx={{ fontSize: 15, color: getTextColor(strength) }} />
<Typography variant="caption" sx={{ ml: 0.5, color: getTextColor(strength) }}>
@@ -0,0 +1,27 @@
import React, { useCallback } from 'react';
import { Typography } from '@mui/material';
import { TPages } from '../types';
export const Step = ({ currentPage, totalSteps }: { currentPage: TPages; totalSteps: number }) => {
const mapPage = useCallback(() => {
switch (currentPage) {
case 'create mnemonic':
return 1;
case 'verify mnemonic':
return 2;
case 'create password':
return 3;
default:
return 0;
}
}, [currentPage]);
if (mapPage() === 0) {
return null;
}
return (
<Typography sx={{ color: 'grey.400' }}>
Create account. Step {mapPage()}/{totalSteps}
</Typography>
);
};
@@ -0,0 +1,78 @@
import React, { useState } from 'react';
import { Box, IconButton, Link, Stack, TextField, Typography } from '@mui/material';
import { Visibility, VisibilityOff } from '@mui/icons-material';
import { Error } from './error';
export const MnemonicInput: React.FC<{
mnemonic: string;
error?: string;
onUpdateMnemonic: (mnemonic: string) => void;
}> = ({ mnemonic, error, onUpdateMnemonic }) => {
const [showPassword, setShowPassword] = useState(false);
return (
<Stack spacing={2}>
<TextField
label="Mnemonic"
type={showPassword ? 'input' : 'password'}
value={mnemonic}
onChange={(e) => onUpdateMnemonic(e.target.value)}
multiline={!!showPassword}
rows={4}
autoFocus
fullWidth
InputProps={{
endAdornment: (
<IconButton onClick={() => setShowPassword((show) => !show)}>
{showPassword ? <VisibilityOff /> : <Visibility />}
</IconButton>
),
}}
/>
{error && <Error message={error} />}
</Stack>
);
};
export const PasswordInput: React.FC<{
password: string;
error?: string;
label: string;
showForgottenPassword?: boolean;
autoFocus?: boolean;
onUpdatePassword: (password: string) => void;
}> = ({ password, label, error, showForgottenPassword, autoFocus, onUpdatePassword }) => {
const [showPassword, setShowPassword] = useState(false);
return (
<Stack spacing={2}>
<Box>
<TextField
label={label}
fullWidth
value={password}
onChange={(e) => onUpdatePassword(e.target.value)}
type={showPassword ? 'input' : 'password'}
autoFocus={autoFocus}
InputProps={{
endAdornment: (
<IconButton onClick={() => setShowPassword((show) => !show)}>
{showPassword ? <VisibilityOff /> : <Visibility />}
</IconButton>
),
}}
/>
{showForgottenPassword && (
<Link
underline="none"
variant="body2"
component="div"
sx={{ mt: 1, textAlign: 'right', color: 'info.main', cursor: 'pointer' }}
>
Forgotten password?
</Link>
)}
</Box>
{error && <Error message={error} />}
</Stack>
);
};
@@ -7,17 +7,19 @@ export const WordTile = ({
index,
disabled,
onClick,
button,
}: {
mnemonicWord: string;
index?: number;
disabled?: boolean;
onClick?: boolean;
button?: boolean;
}) => (
<Card
variant="outlined"
sx={{
background: '#151A2C',
border: '1px solid #3A4053',
background: button ? '#151A2C' : 'transparent',
border: button ? '1px solid #3A4053' : 'none',
cursor: onClick ? 'pointer' : 'default',
opacity: disabled ? 0.2 : 1,
}}
@@ -40,10 +42,12 @@ export const WordTiles = ({
mnemonicWords,
showIndex,
onClick,
buttons,
}: {
mnemonicWords?: TMnemonicWords;
showIndex?: boolean;
onClick?: ({ name, index }: { name: string; index: number }) => void;
buttons?: boolean;
}) => {
if (mnemonicWords) {
return (
@@ -55,6 +59,7 @@ export const WordTiles = ({
index={showIndex ? index : undefined}
onClick={!!onClick}
disabled={disabled}
button={buttons}
/>
</Grid>
))}
@@ -0,0 +1,72 @@
import React, { createContext, useEffect, useMemo, useState } from 'react';
import { createMnemonic, signInWithMnemonic } from 'src/requests';
import { TMnemonicWords } from '../types';
export const SignInContext = createContext({} as TSignInContent);
export type TSignInContent = {
error?: string;
password: string;
mnemonic: string;
mnemonicWords: TMnemonicWords;
setError: (err?: string) => void;
setMnemonic: (mnc: string) => void;
generateMnemonic: () => Promise<void>;
validateMnemonic: () => Promise<void>;
setPassword: (paswd: string) => void;
};
const mnemonicToArray = (mnemonic: string): TMnemonicWords =>
mnemonic
.split(' ')
.reduce((a, c: string, index) => [...a, { name: c, index: index + 1, disabled: false }], [] as TMnemonicWords);
export const SignInProvider: React.FC = ({ children }) => {
const [password, setPassword] = useState('');
const [mnemonic, setMnemonic] = useState('');
const [mnemonicWords, setMnemonicWords] = useState<TMnemonicWords>([]);
const [error, setError] = useState<string>();
const generateMnemonic = async () => {
const mnemonicPhrase = await createMnemonic();
setMnemonic(mnemonicPhrase);
};
const validateMnemonic = async () => {
try {
await signInWithMnemonic(mnemonic);
} catch (e) {
setError(e as string);
}
};
useEffect(() => {
if (mnemonic.length > 0) {
const mnemonicArray = mnemonicToArray(mnemonic);
setMnemonicWords(mnemonicArray);
} else {
setMnemonicWords([]);
}
}, [mnemonic]);
return (
<SignInContext.Provider
value={useMemo(
() => ({
error,
password,
mnemonic,
mnemonicWords,
setError,
setMnemonic,
generateMnemonic,
validateMnemonic,
setPassword,
}),
[error, password, mnemonic, mnemonicWords],
)}
>
{children}
</SignInContext.Provider>
);
};
+42 -38
View File
@@ -1,31 +1,24 @@
import React, { useContext, useState } from 'react';
import { NymLogo } from '@nymproject/react';
import { CircularProgress, Stack, Box } from '@mui/material';
import { ExistingAccount, WelcomeContent } from './pages';
import { TPages } from './types';
import { RenderPage } from './components';
import { CreateAccountContent } from './_legacy_create-account';
import { Stack, Box, CircularProgress } from '@mui/material';
import { NymWordmark } from '@nymproject/react';
import {
CreatePassword,
ExistingAccount,
CreateMnemonic,
VerifyMnemonic,
WelcomeContent,
SignInMnemonic,
} from './pages';
import { TLoginType, TPages } from './types';
import { RenderPage, Step } from './components';
import { ClientContext } from '../../context/main';
// const testMnemonic =
// 'futuristic big receptive caption saw hug odd spoon internal dime bike rake helpless left distribution gusty eyes beg enormous word influence trashy pets curl';
//
// const mnemonicToArray = (mnemonic: string): TMnemonicWords =>
// mnemonic
// .split(' ')
// .reduce((a, c: string, index) => [...a, { name: c, index: index + 1, disabled: false }], [] as TMnemonicWords);
import { SignInPassword } from './pages/signin-password';
export const Welcome = () => {
const [page, setPage] = useState<TPages>('welcome');
// const [mnemonicWords, setMnemonicWords] = useState<TMnemonicWords>();
const [loginType, setLoginType] = useState<TLoginType>('mnemonic');
const { isLoading } = useContext(ClientContext);
// useEffect(() => {
// const mnemonicArray = mnemonicToArray(testMnemonic)
// setMnemonicWords(mnemonicArray)
// }, [])
return (
<Box
sx={{
@@ -50,28 +43,39 @@ export const Welcome = () => {
<CircularProgress size={72} />
) : (
<Stack spacing={3} alignItems="center" sx={{ width: 1080 }}>
<NymLogo width={75} />
<NymWordmark width={75} />
<Step currentPage={page} totalSteps={3} />
<RenderPage page={page}>
<WelcomeContent
onUseExisting={() => setPage('existing account')}
onCreateAccountComplete={() => setPage('legacy create account')}
onCreateAccount={() => setPage('create mnemonic')}
page="welcome"
/>
<CreateAccountContent page="legacy create account" showSignIn={() => setPage('existing account')} />
{/* <MnemonicWords
mnemonicWords={mnemonicWords}
onNext={() => setPage('verify mnemonic')}
onPrev={() => setPage('welcome')}
page="create account"
/>
<VerifyMnemonic
mnemonicWords={mnemonicWords}
onComplete={() => setPage('create password')}
page="verify mnemonic"
/>
<CreatePassword page="create password" /> */}
<ExistingAccount onPrev={() => setPage('welcome')} page="existing account" />
<CreateMnemonic
onNext={() => setPage('verify mnemonic')}
onPrev={() => setPage('create password')}
page="create mnemonic"
/>
<VerifyMnemonic onNext={() => setPage('create password')} onPrev={() => {}} page="verify mnemonic" />
<CreatePassword
onSkip={() => {
setLoginType('mnemonic');
setPage('existing account');
}}
onNext={() => {
setLoginType('password');
setPage('existing account');
}}
page="create password"
/>
<ExistingAccount
onPrev={() => setPage('welcome')}
page="existing account"
loginType={loginType}
setLoginType={(loginType) => setLoginType(loginType)}
/>
<SignInMnemonic onPrev={() => setPage('welcome')} page="sign in with mnemonic" />
<SignInPassword onPrev={() => setPage('welcome')} page="sign in with password" />
</RenderPage>
</Stack>
)}
@@ -0,0 +1,60 @@
import React, { useContext, useEffect } from 'react';
import { Alert, Button, Grid, Stack, Typography } from '@mui/material';
import { Check, ContentCopySharp } from '@mui/icons-material';
import { useClipboard } from 'use-clipboard-copy';
import { WordTiles } from '../components';
import { TPages } from '../types';
import { SignInContext } from '../context';
export const CreateMnemonic = ({ onNext }: { page: TPages; onNext: () => void; onPrev: () => void }) => {
const { mnemonic, mnemonicWords, generateMnemonic, validateMnemonic, setMnemonic, setError } =
useContext(SignInContext);
useEffect(() => {
generateMnemonic();
}, []);
const { copy, copied } = useClipboard({ copiedTimeout: 5000 });
return (
<Stack alignItems="center" spacing={3}>
<Typography sx={{ color: 'common.white', fontWeight: 600 }} textAlign="center">
Write down your mnemonic
</Typography>
<Alert variant="outlined" severity="warning" sx={{ textAlign: 'center' }}>
<Typography>Below is your 24 word mnemonic, please store the mnemonic in a safe place.</Typography>
<Typography>This is the only way to access your wallet!</Typography>
</Alert>
<WordTiles mnemonicWords={mnemonicWords} showIndex />
<Button
color="inherit"
disableElevation
size="large"
onClick={() => {
copy(mnemonic);
}}
sx={{
width: 250,
}}
endIcon={!copied ? <ContentCopySharp /> : <Check color="success" />}
>
Copy mnemonic
</Button>
<Button
variant="contained"
color="primary"
disableElevation
size="large"
onClick={onNext}
sx={{ width: 250 }}
disabled={!copied}
>
I saved my mnemonic
</Button>
</Stack>
);
};
@@ -1,60 +1,74 @@
import React, { useState } from 'react';
import { Button, FormControl, Grid, IconButton, Stack, TextField } from '@mui/material';
import { VisibilityOff, Visibility } from '@mui/icons-material';
import React, { useContext, useState } from 'react';
import { Alert, Button, FormControl, Stack } from '@mui/material';
import { useSnackbar } from 'notistack';
import { TPages } from '../types';
import { Subtitle, Title, PasswordStrength } from '../components';
import { PasswordInput } from '../components/textfields';
import { SignInContext } from '../context';
import { createPassword } from '../../../requests';
export const CreatePassword = () => {
const [password, setPassword] = useState<string>('');
const [confirmedPassword, setConfirmedPassword] = useState<string>();
const [showPassword, setShowPassword] = useState(false);
const [showConfirmedPassword, setShowConfirmedPassword] = useState(false);
export const CreatePassword = ({ onSkip, onNext }: { page: TPages; onNext: () => void; onSkip: () => void }) => {
const { password, setPassword } = useContext(SignInContext);
const [confirmedPassword, setConfirmedPassword] = useState<string>('');
const [isStrongPassword, setIsStrongPassword] = useState(false);
const [isLoading, setIsLoading] = useState(false);
const { mnemonic } = useContext(SignInContext);
const handleSkip = () => {
setPassword('');
onSkip();
};
const { enqueueSnackbar } = useSnackbar();
const storePassword = async () => {
try {
setIsLoading(true);
await createPassword({ mnemonic, password });
enqueueSnackbar('Password successfully created', { variant: 'success' });
setPassword('');
onNext();
} catch (e) {
enqueueSnackbar(e as string, { variant: 'error' });
} finally {
setIsLoading(false);
}
};
return (
<>
<Title title="Create password" />
<Subtitle subtitle="Create a strong password. Min 8 characters, at least one capital letter, number and special symbol" />
<Grid container justifyContent="center">
<Grid item xs={6}>
<FormControl fullWidth>
<Stack spacing={2}>
<TextField
label="Password"
value={password}
onChange={(e) => setPassword(e.target.value)}
type={showPassword ? 'input' : 'password'}
InputProps={{
endAdornment: (
<IconButton onClick={() => setShowPassword((show) => !show)}>
{showPassword ? <VisibilityOff /> : <Visibility />}
</IconButton>
),
}}
/>
<PasswordStrength password={password} />
<TextField
label="Confirm password"
value={confirmedPassword}
onChange={(e) => setConfirmedPassword(e.target.value)}
type={showConfirmedPassword ? 'input' : 'password'}
InputProps={{
endAdornment: (
<IconButton onClick={() => setShowConfirmedPassword((show) => !show)}>
{showConfirmedPassword ? <VisibilityOff /> : <Visibility />}
</IconButton>
),
}}
/>
<Button
size="large"
variant="contained"
disabled={password !== confirmedPassword || password.length === 0}
>
Next
</Button>
</Stack>
</FormControl>
</Grid>
</Grid>
</>
<Stack spacing={3} alignItems="center" minWidth="50%">
<Title title="Create optional password" />
<Subtitle subtitle="Password should be min 8 characters, at least one number and one symbol" />
<FormControl fullWidth>
<Stack spacing={2}>
<>
<PasswordInput
password={password}
onUpdatePassword={(pswd) => setPassword(pswd)}
label="Password"
autoFocus
/>
<PasswordStrength password={password} onChange={(isStrong) => setIsStrongPassword(isStrong)} />
</>
<PasswordInput
password={confirmedPassword}
onUpdatePassword={(pswd) => setConfirmedPassword(pswd)}
label="Confirm password"
/>
<Button
size="large"
variant="contained"
disabled={password !== confirmedPassword || password.length === 0 || !isStrongPassword || isLoading}
onClick={storePassword}
>
Next
</Button>
<Button size="large" color="info" onClick={handleSkip}>
Skip and sign in with mnemonic
</Button>
</Stack>
</FormControl>
</Stack>
);
};
@@ -1,55 +1,78 @@
/* eslint-disable react/no-unused-prop-types */
import React, { useContext, useState } from 'react';
import { Alert, Button, Stack, TextField } from '@mui/material';
import { Subtitle } from '../components';
import { ClientContext } from '../../../context/main';
import { TPages } from '../types';
import { Alert, Button, FormControl, Stack, ToggleButton, ToggleButtonGroup } from '@mui/material';
import { ClientContext } from 'src/context/main';
import { Subtitle, MnemonicInput, PasswordInput } from '../components';
import { TLoginType, TPages } from '../types';
export const ExistingAccount: React.FC<{ page: TPages; onPrev: () => void }> = ({ onPrev }) => {
const [mnemonic, setMnemonic] = useState<string>('');
const { logIn, error } = useContext(ClientContext);
const handleSignIn = async () => {
await logIn(mnemonic);
};
const handleSignInOnEnter = ({ key }: React.KeyboardEvent<HTMLDivElement>) => {
if (key.toLowerCase() === 'enter') {
logIn(mnemonic);
}
};
export const ExistingAccount: React.FC<{
page: TPages;
loginType: TLoginType;
setLoginType: (type: 'mnemonic' | 'password') => void;
onPrev: () => void;
}> = ({ loginType, setLoginType, onPrev }) => {
const [password, setPassword] = useState('');
const [mnemonic, setMnemonic] = useState('');
const { setError, logIn, error } = useContext(ClientContext);
return (
<Stack spacing={2} sx={{ width: 400 }} alignItems="center">
<Subtitle subtitle="Enter your mnemonic from existing wallet" />
<TextField
value={mnemonic}
onChange={(e) => setMnemonic(e.target.value)}
multiline
rows={5}
fullWidth
onKeyDown={handleSignInOnEnter}
/>
{error && (
<Alert severity="error" variant="outlined" data-testid="error" sx={{ color: 'error.light', width: '100%' }}>
{error}
</Alert>
)}
<>
<Subtitle subtitle={`Enter your ${loginType} for existing wallet`} />
<Alert variant="outlined" severity="info">
You can use either a mnemonic or a password to access your wallet
</Alert>
<Stack spacing={2} minWidth="50%">
<ToggleButtonGroup
fullWidth
value={loginType}
exclusive
onChange={(_: React.MouseEvent<HTMLElement>, value: TLoginType) => {
setError(undefined);
setLoginType(value);
}}
>
<ToggleButton value="mnemonic">Mnemonic</ToggleButton>
<ToggleButton value="password">Password</ToggleButton>
</ToggleButtonGroup>
<FormControl fullWidth>
<Stack spacing={2}>
{loginType === 'mnemonic' && (
<MnemonicInput mnemonic={mnemonic} onUpdateMnemonic={(mnc) => setMnemonic(mnc)} error={error} />
)}
{loginType === 'password' && (
<PasswordInput
password={password}
onUpdatePassword={(pswd) => setPassword(pswd)}
label="Password"
autoFocus
error={error}
/>
)}
<Button variant="contained" size="large" fullWidth onClick={handleSignIn}>
Sign in
</Button>
<Button
variant="outlined"
disableElevation
size="large"
onClick={onPrev}
fullWidth
sx={{ color: 'common.white', border: '1px solid white', '&:hover': { border: '1px solid white' } }}
>
Back
</Button>
</Stack>
<Button
variant="contained"
size="large"
fullWidth
onClick={() => logIn({ type: loginType, value: loginType === 'mnemonic' ? mnemonic : password })}
>
{`Sign in with ${loginType}`}
</Button>
<Button
variant="outlined"
disableElevation
size="large"
onClick={() => {
setError(undefined);
onPrev();
}}
fullWidth
sx={{ color: 'common.white', border: '1px solid white', '&:hover': { border: '1px solid white' } }}
>
Back
</Button>
</Stack>
</FormControl>
</Stack>
</>
);
};
+2 -1
View File
@@ -1,5 +1,6 @@
export * from './welcome-content';
export * from './mnemonic-words';
export * from './create-mnemonic';
export * from './verify-mnemonic';
export * from './create-password';
export * from './existing-account';
export * from './signin-mnemonic';
@@ -1,34 +0,0 @@
import React from 'react';
import { Alert, Button, Typography } from '@mui/material';
import { WordTiles } from '../components';
import { TMnemonicWords } from '../types';
export const MnemonicWords = ({
mnemonicWords,
onNext,
onPrev,
}: {
mnemonicWords?: TMnemonicWords;
onNext: () => void;
onPrev: () => void;
}) => (
<>
<Typography sx={{ color: 'common.white', fontWeight: 600 }}>Write down your mnemonic</Typography>
<Alert icon={false} severity="info" sx={{ bgcolor: '#18263B', color: '#50ABFF', width: 625 }}>
Please store your mnemonic in a safe place. This is the only way to access your wallet!
</Alert>
<WordTiles mnemonicWords={mnemonicWords} showIndex />
<Button variant="contained" color="primary" disableElevation size="large" onClick={onNext} sx={{ width: 250 }}>
Verify mnemonic
</Button>
<Button
variant="outlined"
disableElevation
size="large"
onClick={onPrev}
sx={{ color: 'common.white', border: '1px solid white', '&:hover': { border: '1px solid white' }, width: 250 }}
>
Back
</Button>
</>
);
@@ -0,0 +1,43 @@
import React, { useContext, useState } from 'react';
import { Button, FormControl, Stack } from '@mui/material';
import { MnemonicInput, Subtitle } from '../components';
import { ClientContext } from '../../../context/main';
import { TPages } from '../types';
export const SignInMnemonic = ({ page, onPrev }: { page: TPages; onPrev: () => void }) => {
const [mnemonic, setMnemonic] = useState('');
const { setError, logIn, error } = useContext(ClientContext);
return (
<Stack spacing={2} alignItems="center" minWidth="50%">
<Subtitle subtitle="Enter a mnemonic to sign in" />
<FormControl fullWidth>
<Stack spacing={2}>
<MnemonicInput mnemonic={mnemonic} onUpdateMnemonic={(mnc) => setMnemonic(mnc)} error={error} />
<Button
variant="contained"
size="large"
fullWidth
onClick={() => logIn({ type: 'mnemonic', value: mnemonic })}
>
{`Sign in with mnemonic`}
</Button>
<Button
variant="outlined"
disableElevation
size="large"
onClick={() => {
setError(undefined);
onPrev();
}}
fullWidth
sx={{ color: 'common.white', border: '1px solid white', '&:hover': { border: '1px solid white' } }}
>
Back
</Button>
</Stack>
</FormControl>
</Stack>
);
};
@@ -0,0 +1,50 @@
import React, { useContext, useState } from 'react';
import { Button, FormControl, Stack } from '@mui/material';
import { PasswordInput, Subtitle } from '../components';
import { ClientContext } from '../../../context/main';
import { TPages } from '../types';
export const SignInPassword = ({ onPrev }: { page: TPages; onPrev: () => void }) => {
const [password, setPassword] = useState('');
const { setError, logIn, error } = useContext(ClientContext);
return (
<>
<Stack spacing={2} alignItems="center" minWidth="50%">
<Subtitle subtitle="Enter a password to sign in" />
<FormControl fullWidth>
<Stack spacing={2}>
<PasswordInput
label="Enter password"
password={password}
onUpdatePassword={(pswd) => setPassword(pswd)}
error={error}
autoFocus
/>
<Button
variant="contained"
size="large"
fullWidth
onClick={() => logIn({ type: 'password', value: password })}
>
{`Sign in with password`}
</Button>
<Button
variant="outlined"
disableElevation
size="large"
onClick={() => {
setError(undefined);
onPrev();
}}
fullWidth
sx={{ color: 'common.white', border: '1px solid white', '&:hover': { border: '1px solid white' } }}
>
Back
</Button>
</Stack>
</FormControl>
</Stack>
</>
);
};
@@ -1,22 +1,19 @@
import React, { useEffect, useState } from 'react';
import { Button } from '@mui/material';
import React, { useContext, useEffect, useState } from 'react';
import { Button, Stack } from '@mui/material';
import { HiddenWords, Subtitle, Title, WordTiles } from '../components';
import { THiddenMnemonicWord, THiddenMnemonicWords, TMnemonicWord, TMnemonicWords } from '../types';
import { THiddenMnemonicWord, THiddenMnemonicWords, TMnemonicWord, TMnemonicWords, TPages } from '../types';
import { randomNumberBetween } from '../../../utils';
import { SignInContext } from '../context';
const numberOfRandomWords = 4;
const numberOfRandomWords = 6;
export const VerifyMnemonic = ({
mnemonicWords,
onComplete,
}: {
mnemonicWords?: TMnemonicWords;
onComplete: () => void;
}) => {
export const VerifyMnemonic = ({ onNext }: { page: TPages; onNext: () => void; onPrev: () => void }) => {
const [randomWords, setRandomWords] = useState<TMnemonicWords>();
const [hiddenRandomWords, setHiddenRandomWords] = useState<THiddenMnemonicWords>();
const [currentSelection, setCurrentSelection] = useState(0);
const { mnemonicWords } = useContext(SignInContext);
useEffect(() => {
if (mnemonicWords) {
const newRandomWords = getRandomEntriesFromArray<TMnemonicWord>(mnemonicWords, numberOfRandomWords);
@@ -48,16 +45,21 @@ export const VerifyMnemonic = ({
<WordTiles
mnemonicWords={randomWords}
onClick={currentSelection !== numberOfRandomWords ? revealWord : undefined}
buttons
/>
<Button
variant="contained"
sx={{ width: 300 }}
size="large"
disabled={currentSelection !== numberOfRandomWords}
onClick={onComplete}
>
Next
</Button>
<Stack spacing={3} sx={{ width: 300 }}>
<Button
variant="contained"
fullWidth
size="large"
disabled={currentSelection !== numberOfRandomWords}
onClick={() => {
onNext();
}}
>
Next
</Button>
</Stack>
</>
);
}
@@ -7,36 +7,18 @@ import { TPages } from '../types';
export const WelcomeContent: React.FC<{
page: TPages;
onUseExisting: () => void;
onCreateAccountComplete: () => void;
}> = ({ onUseExisting, onCreateAccountComplete }) => (
onCreateAccount: () => void;
}> = ({ onUseExisting, onCreateAccount }) => (
<>
<Title title="Welcome to NYM" />
<SubtitleSlick subtitle="Next generation of privacy" />
<Stack spacing={3} sx={{ width: 300 }}>
<Button
fullWidth
variant="contained"
color="primary"
disableElevation
size="large"
onClick={onCreateAccountComplete}
>
Create Account
</Button>
<Button
fullWidth
variant="outlined"
size="large"
sx={{
color: 'common.white',
border: '1px solid white',
'&:hover': { border: '1px solid white', '&:hover': { background: 'none' } },
}}
onClick={onUseExisting}
disableRipple
>
<Stack spacing={3} minWidth={300}>
<Button fullWidth color="primary" variant="contained" size="large" onClick={onUseExisting}>
Sign in
</Button>
<Button fullWidth color="inherit" disableElevation size="large" onClick={onCreateAccount}>
Create account
</Button>
</Stack>
</>
);
+6 -2
View File
@@ -1,11 +1,13 @@
export type TPages =
| 'welcome'
| 'create account'
| 'create mnemonic'
| 'verify mnemonic'
| 'create password'
| 'existing account'
| 'select network'
| 'legacy create account';
| 'legacy create account'
| 'sign in with mnemonic'
| 'sign in with password';
export type TMnemonicWord = {
name: string;
@@ -17,3 +19,5 @@ export type TMnemonicWords = TMnemonicWord[];
export type THiddenMnemonicWord = { hidden: boolean } & TMnemonicWord;
export type THiddenMnemonicWords = THiddenMnemonicWord[];
export type TLoginType = 'mnemonic' | 'password';
+9 -8
View File
@@ -1,14 +1,10 @@
import { invoke } from '@tauri-apps/api';
import { Account, TCreateAccount } from '../types';
import { Account } from '../types';
export const createAccount = async (): Promise<TCreateAccount> => {
const res: TCreateAccount = await invoke('create_new_account');
return res;
};
export const createMnemonic = async (): Promise<string> => invoke('create_new_mnemonic');
export const createMnemonic = async (): Promise<string> => {
const res: string = await invoke('create_new_mnemonic');
return res;
export const createPassword = async ({ mnemonic, password }: { mnemonic: string; password: string }): Promise<void> => {
await invoke('create_password', { mnemonic, password });
};
export const signInWithMnemonic = async (mnemonic: string): Promise<Account> => {
@@ -16,6 +12,11 @@ export const signInWithMnemonic = async (mnemonic: string): Promise<Account> =>
return res;
};
export const signInWithPassword = async (password: string): Promise<Account> => {
const res: Account = await invoke('sign_in_with_password', { password });
return res;
};
export const signOut = async (): Promise<void> => {
await invoke('logout');
};