From 9cbbddb92a7154096bfdd26cad4bb8f766c6949a Mon Sep 17 00:00:00 2001 From: nicolasito1411 <60229704+Marchand-Nicolas@users.noreply.github.com> Date: Sat, 14 Oct 2023 13:07:08 +0200 Subject: [PATCH 1/5] feat: new connect button --- components/UI/avatar.tsx | 47 ++++++ components/UI/changeWallet.tsx | 98 ++++++++++++ .../UI/iconsComponents/icons/profilIcon.tsx | 24 +++ components/UI/navbar.tsx | 53 +------ components/navbar/walletButton.tsx | 147 ++++++++++++++++++ styles/components/navbar.module.css | 93 +++++++++++ styles/theme.ts | 3 + 7 files changed, 419 insertions(+), 46 deletions(-) create mode 100644 components/UI/avatar.tsx create mode 100644 components/UI/changeWallet.tsx create mode 100644 components/UI/iconsComponents/icons/profilIcon.tsx create mode 100644 components/navbar/walletButton.tsx diff --git a/components/UI/avatar.tsx b/components/UI/avatar.tsx new file mode 100644 index 00000000..49851af1 --- /dev/null +++ b/components/UI/avatar.tsx @@ -0,0 +1,47 @@ +import React, { + FunctionComponent, + useContext, + useEffect, + useState, +} from "react"; +import { StarknetIdJsContext } from "../../context/StarknetIdJsProvider"; +import ProfilIcon from "./iconsComponents/icons/profilIcon"; +import theme from "../../styles/theme"; + +type AvatarProps = { + address: string; + width?: string; +}; + +const Avatar: FunctionComponent = ({ address, width = "32" }) => { + const { starknetIdNavigator } = useContext(StarknetIdJsContext); + const [starknetId, setStarknetId] = useState(""); + + useEffect(() => { + if (!address) return; + (async () => { + const domain = await starknetIdNavigator?.getStarkName(address); + if (!domain) return; + const starknetId = await starknetIdNavigator?.getStarknetId(domain); + if (!starknetId) return; + setStarknetId(starknetId.toString()); + })(); + }, [address]); + + return ( + <> + {starknetId ? ( + + ) : ( + + )} + + ); +}; + +export default Avatar; diff --git a/components/UI/changeWallet.tsx b/components/UI/changeWallet.tsx new file mode 100644 index 00000000..f8367aa8 --- /dev/null +++ b/components/UI/changeWallet.tsx @@ -0,0 +1,98 @@ +import React from "react"; +import styles from "../../styles/components/wallets.module.css"; +import { Connector, useConnectors } from "@starknet-react/core"; +import Button from "./button"; +import { FunctionComponent } from "react"; +import { Modal } from "@mui/material"; +import WalletIcons from "./iconsComponents/icons/walletIcons"; +import getDiscoveryWallets from "get-starknet-core"; +import useGetDiscoveryWallets from "../../hooks/useGetDiscoveryWallets"; + +type ChangeWalletProps = { + closeWallet: () => void; + hasWallet: boolean; +}; + +const ChangeWallet: FunctionComponent = ({ + closeWallet, + hasWallet, +}) => { + const { connect, connectors } = useConnectors(); + const downloadLinks = useGetDiscoveryWallets( + getDiscoveryWallets.getDiscoveryWallets() + ); + + function connectWallet(connector: Connector): void { + connect(connector); + closeWallet(); + } + + return ( + +
+ +

Change wallet

+ {connectors.map((connector) => { + if (connector.available()) { + return ( +
+ +
+ ); + } else { + if (connector.id === "braavos" || connector.id === "argentX") { + return ( +
+ +
+ ); + } + } + })} +
+
+ ); +}; +export default ChangeWallet; diff --git a/components/UI/iconsComponents/icons/profilIcon.tsx b/components/UI/iconsComponents/icons/profilIcon.tsx new file mode 100644 index 00000000..f2e15a05 --- /dev/null +++ b/components/UI/iconsComponents/icons/profilIcon.tsx @@ -0,0 +1,24 @@ +import React, { FunctionComponent } from "react"; + +const ProfilIcon: FunctionComponent = ({ color, width }) => { + return ( + + + + + ); +}; + +export default ProfilIcon; diff --git a/components/UI/navbar.tsx b/components/UI/navbar.tsx index 17960ada..400241db 100644 --- a/components/UI/navbar.tsx +++ b/components/UI/navbar.tsx @@ -12,7 +12,6 @@ import { useConnectors, useAccount, useProvider, - useTransactionManager, Connector, } from "@starknet-react/core"; import Wallets from "./wallets"; @@ -20,12 +19,10 @@ import ModalMessage from "./modalMessage"; import { useDisplayName } from "../../hooks/displayName.tsx"; import { useDomainFromAddress } from "../../hooks/naming"; import { constants } from "starknet"; -import ModalWallet from "./modalWallet"; -import { CircularProgress } from "@mui/material"; -import AccountCircleIcon from "@mui/icons-material/AccountCircle"; import { useRouter } from "next/router"; import theme from "../../styles/theme"; import { FaDiscord, FaTwitter } from "react-icons/fa"; +import WalletButton from "../navbar/walletButton"; const Navbar: FunctionComponent = () => { const [nav, setNav] = useState(false); @@ -33,7 +30,6 @@ const Navbar: FunctionComponent = () => { const { address } = useAccount(); const [isConnected, setIsConnected] = useState(false); const [isWrongNetwork, setIsWrongNetwork] = useState(false); - const [txLoading, setTxLoading] = useState(0); const { available, connect, disconnect, connectors, refresh } = useConnectors(); const { provider } = useProvider(); @@ -44,7 +40,6 @@ const Navbar: FunctionComponent = () => { const network = process.env.NEXT_PUBLIC_IS_TESTNET === "true" ? "testnet" : "mainnet"; const [navbarBg, setNavbarBg] = useState(false); - const { hashes } = useTransactionManager(); const [showWallet, setShowWallet] = useState(false); const router = useRouter(); @@ -189,38 +184,12 @@ const Navbar: FunctionComponent = () => { {/* Note: I'm not sure that our testnet will be public so we don't show any link */} {/* */} -
- -
+
{
} /> - setShowWallet(false)} - disconnectByClick={disconnectByClick} - hashes={hashes} - setTxLoading={setTxLoading} - /> setHasWallet(false)} hasWallet={Boolean(hasWallet && !isWrongNetwork)} diff --git a/components/navbar/walletButton.tsx b/components/navbar/walletButton.tsx new file mode 100644 index 00000000..17b789d7 --- /dev/null +++ b/components/navbar/walletButton.tsx @@ -0,0 +1,147 @@ +import React, { FunctionComponent, useMemo, useState, useEffect } from "react"; +import Button from "../UI/button"; +import { useDisplayName } from "../../hooks/displayName.tsx"; +import { + useAccount, + useTransactionManager, + useTransactions, +} from "@starknet-react/core"; +import styles from "../../styles/components/navbar.module.css"; +import ProfilIcon from "../UI/iconsComponents/icons/profilIcon"; +import theme from "../../styles/theme"; +import Avatar from "../UI/avatar"; +import CopyIcon from "../UI/iconsComponents/icons/copyIcon"; +import { Wallet } from "@mui/icons-material"; +import LogoutIcon from "@mui/icons-material/Logout"; +import VerifiedIcon from "../UI/iconsComponents/icons/verifiedIcon"; +import ChangeWallet from "../UI/changeWallet"; + +type WalletButtonProps = { + setShowWallet: (showWallet: boolean) => void; + showWallet: boolean; + refreshAndShowWallet: () => void; + disconnectByClick: () => void; +}; + +const WalletButton: FunctionComponent = ({ + setShowWallet, + showWallet, + refreshAndShowWallet, + disconnectByClick, +}) => { + const { address } = useAccount(); + const { hashes } = useTransactionManager(); + const transactions = useTransactions({ hashes, watch: true }); + const domainOrAddressMinified = useDisplayName(address ?? ""); + const [txLoading, setTxLoading] = useState(0); + const [copied, setCopied] = useState(false); + const [changeWallet, setChangeWallet] = useState(false); + const buttonName = useMemo( + () => + address + ? txLoading + ? `${txLoading} on hold` + : domainOrAddressMinified + : "connect", + [address, domainOrAddressMinified] + ); + + useEffect(() => { + const interval = setInterval(() => { + for (const tx of transactions) { + tx.refetch(); + } + }, 3_000); + return () => clearInterval(interval); + }, [transactions?.length]); + + useEffect(() => { + if (transactions) { + // Give the number of tx that are loading (I use any because there is a problem on Starknet React types) + setTxLoading( + transactions.filter((tx) => (tx?.data as any)?.status === "RECEIVED") + .length + ); + } + }, [transactions]); + + const copyAddress = (e: React.MouseEvent) => { + e.stopPropagation(); + setCopied(true); + navigator.clipboard.writeText(address ?? ""); + setTimeout(() => { + setCopied(false); + }, 1500); + }; + + const handleDisconnect = (e: React.MouseEvent) => { + e.stopPropagation(); + disconnectByClick(); + }; + + const handleWalletChange = (e: React.MouseEvent) => { + e.stopPropagation(); + setChangeWallet(true); + }; + + return ( + <> +
+ + + +
+ ) : null} + + + + setChangeWallet(false)} + hasWallet={changeWallet} + /> + + ); +}; + +export default WalletButton; diff --git a/styles/components/navbar.module.css b/styles/components/navbar.module.css index 632303b4..c274c8da 100644 --- a/styles/components/navbar.module.css +++ b/styles/components/navbar.module.css @@ -46,3 +46,96 @@ .navbarScrolled { background-color: rgba(0, 0, 0, 1); } + +.buttonContainer { + margin: 0 1.25rem; +} + +.buttonContainer > button { + position: relative; + margin: 0; + padding: 8px 16px; + height: unset; + color: var(--background); +} + +.buttonText { + margin-right: 32px; + text-transform: capitalize; + font-weight: bold; + font-family: Sora-ExtraBold; + min-width: 124px; + /* Body/middle/bold */ + font-size: 18px; + font-weight: 700; + line-height: 24px; /* 133.333% */ + letter-spacing: 0.18px; +} + +.buttonSeparator { + position: absolute; + top: 0; + bottom: 0; + right: 64px; + align-self: stretch; + width: 1px; + background: var(--background); +} + +.buttonIcon svg { + display: block; + width: 32px; + height: 32px; +} + +.buttonContainer[aria-connected="true"] > button { + background: var(--background600); + color: white; +} + +.buttonContainer[aria-connected="true"] .buttonSeparator { + background: #aab1b6; +} + +.buttonContainer[aria-connected="true"] > svg { + background: #aab1b6; +} + +.buttonContainer[aria-selected="true"] > button { + border-radius: 10px 10px 0px 0px; + border: 1px solid var(--secondary500); +} + +.walletMenu { + position: absolute; + right: -1px; + left: -1px; + bottom: 0; + display: flex; + flex-direction: column; + align-items: left; + justify-content: left; + transform: translateY(100%); + padding: 12px 16px; + background-color: var(--background600); + border-radius: 0px 0px 10px 10px; + border: 1px solid var(--secondary500); +} + +.walletMenu button { + display: flex; + align-items: center; + color: var(--secondary500); +} + +.walletMenu svg { + margin-right: 12px; + color: var(--secondary500); + fill: var(--secondary500); +} + +.walletMenu button:hover svg, +.walletMenu button:hover p { + color: white; + fill: white; +} diff --git a/styles/theme.ts b/styles/theme.ts index c6ee113a..babac9e5 100644 --- a/styles/theme.ts +++ b/styles/theme.ts @@ -12,6 +12,9 @@ const theme = createTheme({ main: "#f4faff", dark: "#f4faff", }, + background: { + default: "#101012", + }, }, }); From 0c4379c6f440b1408aed85c7717c3a8118374423 Mon Sep 17 00:00:00 2001 From: nicolasito1411 <60229704+Marchand-Nicolas@users.noreply.github.com> Date: Sat, 14 Oct 2023 13:15:00 +0200 Subject: [PATCH 2/5] fixing build errors --- components/navbar/walletButton.tsx | 2 +- styles/components/navbar.module.css | 6 +++--- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/components/navbar/walletButton.tsx b/components/navbar/walletButton.tsx index 17b789d7..150c5205 100644 --- a/components/navbar/walletButton.tsx +++ b/components/navbar/walletButton.tsx @@ -88,7 +88,7 @@ const WalletButton: FunctionComponent = ({ <>
-
-
- {connector && connector.id === "braavos" ? ( - braavos logo - ) : ( - - )} - -

Connected with  {domain} 

-
-
-
- } - title="Disconnect" - width="auto" - /> - - ) : ( - - ) - } - title="Copy Address" - width="auto" - /> - {isWebWallet && ( - - window.open( - network === "mainnet" - ? "https://web.argent.xyz" - : "https://web.hydrogen.argent47.net", - "_blank", - "noopener noreferrer" - ) - } - icon={} - title="Web wallet Dashboard" - width="auto" - /> - )} -
-
-
My transactions
-
- {transactions && transactions.length > 0 ? ( - transactions.map((tx) => { - return ( - - ); - }) - ) : ( -

No ongoing transactions

- )} -
-
-
- - ); -}; -export default ModalWallet; From 774089b953423a62fd94fc24e8257e41d3514fd5 Mon Sep 17 00:00:00 2001 From: nicolasito1411 <60229704+Marchand-Nicolas@users.noreply.github.com> Date: Wed, 18 Oct 2023 16:30:55 +0200 Subject: [PATCH 5/5] adding web wallet and improving the UX --- components/navbar/walletButton.tsx | 39 +++++++++++++++++++++++++++-- styles/components/navbar.module.css | 2 ++ 2 files changed, 39 insertions(+), 2 deletions(-) diff --git a/components/navbar/walletButton.tsx b/components/navbar/walletButton.tsx index 150c5205..80169325 100644 --- a/components/navbar/walletButton.tsx +++ b/components/navbar/walletButton.tsx @@ -15,6 +15,7 @@ import { Wallet } from "@mui/icons-material"; import LogoutIcon from "@mui/icons-material/Logout"; import VerifiedIcon from "../UI/iconsComponents/icons/verifiedIcon"; import ChangeWallet from "../UI/changeWallet"; +import ArgentIcon from "../UI/iconsComponents/icons/argentIcon"; type WalletButtonProps = { setShowWallet: (showWallet: boolean) => void; @@ -29,13 +30,20 @@ const WalletButton: FunctionComponent = ({ refreshAndShowWallet, disconnectByClick, }) => { - const { address } = useAccount(); + const { address, connector } = useAccount(); const { hashes } = useTransactionManager(); const transactions = useTransactions({ hashes, watch: true }); const domainOrAddressMinified = useDisplayName(address ?? ""); const [txLoading, setTxLoading] = useState(0); const [copied, setCopied] = useState(false); const [changeWallet, setChangeWallet] = useState(false); + const [hovering, setHovering] = useState(false); + const [unfocus, setUnfocus] = useState(false); + + const network = + process.env.NEXT_PUBLIC_IS_TESTNET === "true" ? "testnet" : "mainnet"; + const isWebWallet = (connector as any)?._wallet?.id === "argentWebWallet"; + const buttonName = useMemo( () => address @@ -81,15 +89,36 @@ const WalletButton: FunctionComponent = ({ const handleWalletChange = (e: React.MouseEvent) => { e.stopPropagation(); + setHovering(false); + setUnfocus(true); setChangeWallet(true); }; + const handleOpenWebWallet = (e: React.MouseEvent) => { + window.open( + network === "mainnet" + ? "https://web.argent.xyz" + : "https://web.hydrogen.argent47.net", + "_blank", + "noopener noreferrer" + ); + }; + + useEffect(() => { + if (!unfocus) return; + if (hovering) setUnfocus(false); + else setShowWallet(false); + }, [unfocus, hovering]); + return ( <>
setUnfocus(true)} + onMouseEnter={() => setHovering(true)} + onMouseLeave={() => setHovering(false)} > + {isWebWallet && ( + + )}