Skip to content

Commit

Permalink
Add reveal mnemonic logic (#2217)
Browse files Browse the repository at this point in the history
  • Loading branch information
OKendigelyan authored Dec 10, 2024
1 parent 6c0b658 commit 188cfc7
Show file tree
Hide file tree
Showing 11 changed files with 213 additions and 65 deletions.
2 changes: 2 additions & 0 deletions apps/desktop-e2e/src/steps/onboarding.ts
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,8 @@ Then("I am on {string} onboarding page", async function (this: CustomWorld, moda
});

Then("I record generated seedphrase", async function (this: CustomWorld) {
await this.page.getByRole("button", { name: "Show seed phrase" }).click();

const words: string[] = [];
for (let i = 0; i < 24; i++) {
words.push(await this.page.getByTestId(`mnemonic-word-${i}`).innerText());
Expand Down
10 changes: 9 additions & 1 deletion apps/desktop/public/electron.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
// Module to control the application lifecycle and the native browser window.
const { app, BrowserWindow, shell, net, ipcMain, protocol } = require("electron");
const { app, BrowserWindow, shell, net, ipcMain, protocol, clipboard } = require("electron");
const path = require("path");
const url = require("url");
const process = require("process");
Expand Down Expand Up @@ -242,6 +242,14 @@ function start() {

// Listen to install-app-update event from UI, start update on getting the event.
ipcMain.on("install-app-update", () => autoUpdater.quitAndInstall());

ipcMain.on("clipboard-write", (_, text) => {
clipboard.writeText(text);
});

ipcMain.on("clipboard-clear", () => {
clipboard.clear();
});
}

start();
3 changes: 3 additions & 0 deletions apps/desktop/public/preload.js
Original file line number Diff line number Diff line change
Expand Up @@ -17,4 +17,7 @@ contextBridge.exposeInMainWorld("electronAPI", {

// Notify Electron that app update should be installed.
installAppUpdateAndQuit: () => ipcRenderer.send("install-app-update"),

clipboardWriteText: text => ipcRenderer.send("clipboard-write", text),
clipboardClear: () => ipcRenderer.send("clipboard-clear"),
});
2 changes: 1 addition & 1 deletion apps/desktop/src/components/Onboarding/notice/Notice.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ export const Notice = ({ goToStep }: { goToStep: (step: OnboardingStep) => void
content: "Make sure there is no one around you or looking over your shoulder.",
},
{
content: "Do not copy and paste the Seed Phrase or store it on your device.",
content: "Do not store Seed Phrase on your device.",
},
{
content: "Do not take a screenshot of your Seed Phrase.",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,8 @@ describe("<ShowSeedphrase />", () => {
const user = userEvent.setup();
render(fixture());

await act(() => user.click(screen.getByTestId("show-seedphrase-button")));

mnemonic1.split(" ").forEach(word => {
expect(screen.getByText(word)).toBeInTheDocument();
});
Expand Down
170 changes: 123 additions & 47 deletions apps/desktop/src/components/Onboarding/showSeedphrase/ShowSeedphrase.tsx
Original file line number Diff line number Diff line change
@@ -1,59 +1,135 @@
import { Button, Flex, Heading, SimpleGrid, Text, VStack } from "@chakra-ui/react";
import {
Button,
Flex,
Heading,
Popover,
PopoverArrow,
PopoverBody,
PopoverContent,
PopoverTrigger,
SimpleGrid,
Text,
VStack,
useDisclosure,
} from "@chakra-ui/react";
import { useState } from "react";

import { KeyIcon } from "../../../assets/icons";
import { EyeIcon, EyeSlashIcon, FileCopyIcon, KeyIcon } from "../../../assets/icons";
import colors from "../../../style/colors";
import { ModalContentWrapper } from "../ModalContentWrapper";
import { type OnboardingStep, type ShowSeedphraseStep } from "../OnboardingStep";

const COPY_TIMEOUT = 30_000;
const COPIED_POPUP_DURATION = 2000;

export const ShowSeedphrase = ({
goToStep,
account,
}: {
goToStep: (step: OnboardingStep) => void;
account: ShowSeedphraseStep["account"];
}) => (
<ModalContentWrapper
icon={<KeyIcon width="24px" height="24px" />}
subtitle="Please record the following 24 words in sequence in order to restore it in the future."
title="Record Seed Phrase"
>
<VStack>
<SimpleGrid columns={3} spacing={2}>
{account.mnemonic.split(" ").map((item, index) => (
<Flex
key={index}
width="126px"
padding="6px"
border="1px dashed"
borderColor={colors.gray[500]}
borderRadius="4px"
>
<Heading
width="18px"
marginRight="10px"
paddingTop="2px"
color={colors.gray[450]}
textAlign="right"
size="sm"
}) => {
const [isHidden, setIsHidden] = useState(true);
const {
isOpen: isPopoverOpen,
onOpen: setIsPopoverOpen,
onClose: setIsPopoverClose,
} = useDisclosure();

const handleCopy = () => {
window.electronAPI.clipboardWriteText(account.mnemonic);
setIsPopoverOpen();

setTimeout(() => {
setIsPopoverClose();
}, COPIED_POPUP_DURATION);

setTimeout(() => {
window.electronAPI.clipboardClear();
}, COPY_TIMEOUT);
};

return (
<ModalContentWrapper
icon={<KeyIcon width="24px" height="24px" />}
subtitle="Please record the following 24 words in sequence in order to restore it in the future."
title="Record Seed Phrase"
>
<VStack>
<SimpleGrid userSelect="none" columns={3} spacing={2}>
{account.mnemonic.split(" ").map((item, index) => (
<Flex
key={index}
width="126px"
padding="6px"
border="1px dashed"
borderColor={colors.gray[500]}
borderRadius="4px"
>
{index + 1}
</Heading>
<Text data-testid={`mnemonic-word-${index}`} size="sm">
{item}
</Text>
</Flex>
))}
</SimpleGrid>
<Button
width="100%"
marginTop="20px"
onClick={_ => {
goToStep({ type: "verifySeedphrase", account });
}}
size="lg"
>
OK, I've recorded it
</Button>
</VStack>
</ModalContentWrapper>
);
<Heading
width="18px"
marginRight="10px"
paddingTop="2px"
color={colors.gray[450]}
textAlign="right"
size="sm"
>
{index + 1}
</Heading>
<Text
sx={{
WebkitTextSecurity: isHidden ? "disc" : "none",
}}
data-testid={`mnemonic-word-${index}`}
size="sm"
>
{isHidden ? "********" : item}
</Text>
</Flex>
))}
</SimpleGrid>
<Flex justifyContent="space-between" gap="16px" width="100%" marginTop="20px">
<Button
width="100%"
data-testid="show-seedphrase-button"
leftIcon={isHidden ? <EyeSlashIcon /> : <EyeIcon />}
onClick={() => setIsHidden(!isHidden)}
variant="outline"
>
{isHidden ? "Show" : "Hide"} seed phrase
</Button>
<Popover autoFocus={false} closeOnBlur={false} isOpen={isPopoverOpen}>
<PopoverTrigger>
<Button
width="100%"
leftIcon={<FileCopyIcon stroke={colors.gray[450]} />}
onClick={handleCopy}
variant="outline"
>
Copy to clipboard
</Button>
</PopoverTrigger>
<PopoverContent maxWidth="max-content" background="white">
<PopoverArrow background="white !important" />
<PopoverBody padding="8px 12px">
<Text color="black" fontWeight="medium" size="sm">
Copied!
</Text>
</PopoverBody>
</PopoverContent>
</Popover>
</Flex>
<Button
width="100%"
marginTop="8px"
onClick={_ => {
goToStep({ type: "verifySeedphrase", account });
}}
size="lg"
>
OK, I've recorded it
</Button>
</VStack>
</ModalContentWrapper>
);
};
13 changes: 13 additions & 0 deletions apps/desktop/src/global.d.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
export {};

declare global {
interface Window {
electronAPI: {
clipboardWriteText: (text: string) => void;
clipboardClear: () => void;
onDeeplink: (callback: (url: string) => void) => void;
onAppUpdateDownloaded: (callback: () => void) => void;
installAppUpdateAndQuit: () => void;
};
}
}
24 changes: 22 additions & 2 deletions apps/web/src/components/CopyButton/CopyButton.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -13,12 +13,21 @@ import { type MouseEvent, type PropsWithChildren } from "react";

import { useColor } from "../../styles/useColor";

const COPY_TIMEOUT = 30_000;

type CopyButtonProps = {
value: string;
isCopyDisabled?: boolean;
isDisposable?: boolean;
} & ButtonProps;

export const CopyButton = ({
value,
children,
isCopyDisabled = false,
isDisposable = false,
...props
}: PropsWithChildren<{ value: string; isCopyDisabled?: boolean } & ButtonProps>) => {
}: PropsWithChildren<CopyButtonProps>) => {
const color = useColor();

const { isOpen, onOpen, onClose } = useDisclosure();
Expand All @@ -30,7 +39,18 @@ export const CopyButton = ({

event.stopPropagation();
setTimeout(onClose, 1000);
return navigator.clipboard.writeText(value);

return navigator.clipboard.writeText(value).then(() => {
if (isDisposable) {
setTimeout(() => {
try {
void navigator.clipboard.writeText("");
} catch (error: unknown) {
console.error("Failed to clear clipboard", error);
}
}, COPY_TIMEOUT);
}
});
};

return (
Expand Down
7 changes: 6 additions & 1 deletion apps/web/src/components/MnemonicWord/MnemonicWord.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,13 +7,15 @@ import { useColor } from "../../styles/useColor";
type MnemonicWordProps = {
index: number;
word?: string;
isHidden?: boolean;
indexProps?: TextProps;
autocompleteProps?: ComponentProps<typeof MnemonicAutocomplete>;
} & GridItemProps;

export const MnemonicWord = ({
index,
word,
isHidden,
autocompleteProps,
indexProps,
...props
Expand All @@ -39,11 +41,14 @@ export const MnemonicWord = ({
{autocompleteProps && <MnemonicAutocomplete {...autocompleteProps} />}
{word && (
<Text
sx={{
WebkitTextSecurity: isHidden ? "disc" : "none",
}}
paddingLeft={{ base: "22px", md: "26px" }}
fontSize={{ base: "12px", md: "14px" }}
fontWeight="medium"
>
{word}
{isHidden ? "*******" : word}
</Text>
)}
</GridItem>
Expand Down
Loading

1 comment on commit 188cfc7

@github-actions
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Title Lines Statements Branches Functions
apps/desktop Coverage: 83%
83.68% (1770/2115) 79.51% (846/1064) 78.08% (449/575)
apps/web Coverage: 83%
83.68% (1770/2115) 79.51% (846/1064) 78.08% (449/575)
packages/components Coverage: 97%
97.51% (196/201) 95.91% (94/98) 88.13% (52/59)
packages/core Coverage: 81%
82.47% (207/251) 72.72% (88/121) 81.35% (48/59)
packages/crypto Coverage: 100%
100% (43/43) 90.9% (10/11) 100% (7/7)
packages/data-polling Coverage: 97%
95.27% (141/148) 87.5% (21/24) 92.85% (39/42)
packages/multisig Coverage: 98%
98.47% (129/131) 85.71% (18/21) 100% (36/36)
packages/social-auth Coverage: 100%
100% (21/21) 100% (11/11) 100% (3/3)
packages/state Coverage: 85%
84.76% (818/965) 80.86% (186/230) 78.59% (301/383)
packages/tezos Coverage: 89%
88.72% (118/133) 94.59% (35/37) 86.84% (33/38)
packages/tzkt Coverage: 89%
87.32% (62/71) 87.5% (14/16) 80.48% (33/41)

Please sign in to comment.