chore: improve wallet dropdown

This commit is contained in:
Danish Arora 2025-04-10 16:15:26 +05:30
parent 9bc665efb5
commit cde0441c23
No known key found for this signature in database
GPG Key ID: 1C6EF37CDAE1426E

View File

@ -1,28 +1,39 @@
"use client";
import React, { useState, ChangeEvent } from 'react';
import { useWallet } from '../contexts/wallet';
import { useWallet } from "@/contexts/wallet";
import {
DropdownMenu,
DropdownMenuTrigger,
DropdownMenuContent,
DropdownMenuItem,
DropdownMenuLabel,
DropdownMenuSeparator
} from '../components/ui/dropdown-menu';
import { Button } from '../components/ui/button';
import { ChevronDown, PlusCircle } from 'lucide-react';
} from "@/components/ui/dropdown-menu";
import { Button } from "@/components/ui/button";
import {
ChevronDown,
PlusCircle,
Wallet,
ExternalLink,
Copy,
LogOut,
Coins
} from 'lucide-react';
import {
Dialog,
DialogContent,
DialogDescription,
DialogFooter,
DialogHeader,
DialogTitle,
DialogTrigger,
} from "../components/ui/dialog";
import { Input } from "../components/ui/input";
import { Label } from "../components/ui/label";
} from "@/components/ui/dialog";
import { Input } from "@/components/ui/input";
import {
Tooltip,
TooltipContent,
TooltipProvider,
TooltipTrigger,
} from "@/components/ui/tooltip";
export function WalletDropdown() {
const { isConnected, address, chainId, connectWallet, disconnectWallet, balance, wttBalance, mintWTT } = useWallet();
@ -38,125 +49,164 @@ export function WalletDropdown() {
setMintAmount(e.target.value);
};
// Format address in the "0xabc...xyz" pattern
const formatAddress = (address: string) => {
if (!address) return '';
return `0x${address.slice(2, 5)}...${address.slice(-3)}`;
};
const copyToClipboard = (text: string) => {
navigator.clipboard.writeText(text);
};
const getNetworkColor = () => {
if (chainId === 59141) return "text-green-400";
return "text-yellow-400";
};
const getNetworkName = () => {
if (chainId === 59141) return "Linea Sepolia";
return `Unknown (${chainId})`;
};
if (!isConnected || !address) {
return (
<Button
onClick={connectWallet}
variant="terminal"
size="sm"
className="text-xs"
className="text-xs flex items-center gap-1.5"
>
<Wallet className="h-3.5 w-3.5" />
Connect Wallet
</Button>
);
}
return (
<>
<DropdownMenu>
<DropdownMenuTrigger asChild>
<Button
variant="outline"
size="sm"
className="font-mono text-xs border-terminal-border hover:border-primary hover:text-primary"
>
<span
className={`inline-block h-2 w-2 rounded-full mr-1.5 bg-success-DEFAULT`}
/>
<span className="cursor-blink">
{address.slice(0, 6)}...{address.slice(-4)}
</span>
<ChevronDown className="ml-1 h-4 w-4" />
</Button>
</DropdownMenuTrigger>
<DropdownMenuContent align="end" className="w-64 font-mono border-terminal-border bg-terminal-background/95">
<DropdownMenuLabel className="border-b border-terminal-border text-primary">Wallet</DropdownMenuLabel>
<div className="p-2 space-y-2">
{/* Address */}
<div className="grid grid-cols-12 gap-1 text-xs">
<span className="text-muted-foreground col-span-3">Address:</span>
<span className="text-primary col-span-9 truncate">{address}</span>
<DropdownMenu>
<DropdownMenuTrigger asChild>
<Button
variant="outline"
size="sm"
className="font-mono text-xs border-terminal-border hover:border-primary hover:text-primary flex items-center gap-1.5"
>
<span className="inline-block h-2 w-2 rounded-full bg-success-DEFAULT" />
<span className="cursor-blink font-medium">
{formatAddress(address)}
</span>
<ChevronDown className="h-3.5 w-3.5" />
</Button>
</DropdownMenuTrigger>
<DropdownMenuContent align="end" className="w-72 font-mono border-terminal-border bg-terminal-background/95 p-0">
<div className="p-3 border-b border-terminal-border bg-terminal-background/80">
<div className="flex items-center justify-between mb-2">
<div className="flex items-center gap-1.5">
<Wallet className="h-4 w-4 text-primary" />
<span className="text-primary text-sm font-medium">Wallet</span>
</div>
{/* Network */}
<div className="grid grid-cols-12 gap-1 text-xs">
<span className="text-muted-foreground col-span-3">Network:</span>
<span className="text-green-400 col-span-9">{chainId === 59141 ? "Linea Sepolia" : `Unknown (${chainId})`}</span>
<div className={`text-xs px-1.5 py-0.5 rounded-sm ${getNetworkColor()} bg-terminal-background/80 border border-terminal-border`}>
{getNetworkName()}
</div>
{/* ETH Balance */}
{balance && (
<div className="grid grid-cols-12 gap-1 text-xs">
<span className="text-muted-foreground col-span-3">Balance:</span>
<span className="text-primary col-span-9">{parseFloat(balance).toFixed(4)} ETH</span>
</div>
)}
{/* WTT Balance with Mint Button */}
{wttBalance && (
<div className="text-xs">
<div className="grid grid-cols-12 gap-1 mb-1">
<span className="text-muted-foreground col-span-3">WTT</span>
<span className="text-primary col-span-9">{parseFloat(wttBalance).toFixed(4)}</span>
</div>
<div className="grid grid-cols-12 gap-1">
<span className="text-muted-foreground col-span-3">Balance:</span>
<span className="text-green-400 col-span-7">WTT</span>
<div className="col-span-2 text-right">
<Dialog open={isDialogOpen} onOpenChange={setIsDialogOpen}>
<DialogTrigger asChild>
<Button variant="ghost" size="icon" className="h-5 w-5 text-primary hover:text-green-400">
<PlusCircle className="h-4 w-4" />
</Button>
</DialogTrigger>
<DialogContent className="sm:max-w-[425px] bg-terminal-background border-terminal-border">
<DialogHeader>
<DialogTitle className="text-primary">Mint WTT Tokens</DialogTitle>
<DialogDescription>
Enter the amount of WTT tokens you want to mint.
</DialogDescription>
</DialogHeader>
<div className="grid gap-4 py-4">
<div className="grid grid-cols-4 items-center gap-4">
<Label htmlFor="amount" className="text-right">
Amount
</Label>
<Input
id="amount"
type="number"
value={mintAmount}
onChange={handleAmountChange}
className="col-span-3 bg-terminal-background border-terminal-border text-primary"
/>
</div>
</div>
<DialogFooter>
<Button
variant="terminal"
onClick={handleMint}
className="text-xs"
>
Mint Tokens
</Button>
</DialogFooter>
</DialogContent>
</Dialog>
</div>
</div>
</div>
)}
</div>
<DropdownMenuSeparator className="bg-terminal-border" />
<DropdownMenuItem
onClick={disconnectWallet}
className="text-destructive focus:text-destructive cursor-pointer"
>
Disconnect
</DropdownMenuItem>
</DropdownMenuContent>
</DropdownMenu>
</>
<div className="flex items-center justify-between">
<div className="truncate text-primary text-xs font-medium">{address}</div>
<div className="flex gap-1">
<Button
variant="ghost"
size="icon"
className="h-6 w-6 text-muted-foreground hover:text-primary"
onClick={() => copyToClipboard(address || '')}
>
<Copy className="h-3.5 w-3.5" />
</Button>
<Button
variant="ghost"
size="icon"
className="h-6 w-6 text-muted-foreground hover:text-primary"
onClick={() => window.open(`https://explorer.linea.build/address/${address}`, '_blank')}
>
<ExternalLink className="h-3.5 w-3.5" />
</Button>
</div>
</div>
</div>
<div className="p-3 space-y-3">
{/* Balances section */}
<div className="space-y-2">
<div className="flex items-center justify-between text-xs">
<div className="flex items-center gap-1.5 text-muted-foreground">
<Coins className="h-3.5 w-3.5" />
<span>ETH Balance</span>
</div>
<span className="text-primary font-medium">{parseFloat(balance || '0').toFixed(4)}</span>
</div>
<div className="flex items-center justify-between text-xs">
<div className="flex items-center gap-1.5 text-muted-foreground">
<Coins className="h-3.5 w-3.5" />
<span>WTT Balance</span>
</div>
<div className="flex items-center gap-1.5">
<span className="text-primary font-medium">{parseFloat(wttBalance || '0').toFixed(4)}</span>
<Dialog open={isDialogOpen} onOpenChange={setIsDialogOpen}>
<TooltipProvider>
<Tooltip>
<TooltipTrigger asChild>
<DialogTrigger asChild>
<Button variant="ghost" size="icon" className="h-5 w-5 p-0 text-muted-foreground hover:text-green-400">
<PlusCircle className="h-3.5 w-3.5" />
</Button>
</DialogTrigger>
</TooltipTrigger>
<TooltipContent side="left">
<p className="font-mono text-xs">Mint WTT tokens for testing</p>
</TooltipContent>
</Tooltip>
</TooltipProvider>
<DialogContent className="sm:max-w-[350px] bg-terminal-background border-terminal-border">
<DialogHeader>
<DialogTitle className="text-primary">Mint WTT Tokens</DialogTitle>
<DialogDescription className="text-muted-foreground text-xs">
Specify how many WTT tokens you want to mint for testing.
</DialogDescription>
</DialogHeader>
<div className="py-3">
<div className="flex items-center gap-3">
<Input
id="amount"
type="number"
value={mintAmount}
onChange={handleAmountChange}
className="bg-terminal-background/60 border-terminal-border text-primary h-9"
/>
<Button
variant="terminal"
onClick={handleMint}
className="text-xs h-9"
>
Mint
</Button>
</div>
</div>
</DialogContent>
</Dialog>
</div>
</div>
</div>
</div>
<DropdownMenuSeparator className="bg-terminal-border m-0" />
<DropdownMenuItem
onClick={disconnectWallet}
className="m-0 rounded-none text-xs py-2.5 text-destructive focus:text-destructive cursor-pointer flex items-center gap-2"
>
<LogOut className="h-3.5 w-3.5" />
Disconnect Wallet
</DropdownMenuItem>
</DropdownMenuContent>
</DropdownMenu>
);
}