feat(crypto): add SurfSense 2.0 Crypto Co-Pilot UI components

Frontend - Web Dashboard:
- Add crypto dashboard page with Watchlist, Alerts, Market, Profile tabs
- Add 11 tool-ui components for inline chat display
- Add crypto components (ChainIcon, SafetyBadge, PriceDisplay, etc.)
- Add modals (AddTokenModal, CreateAlertModal)
- Add mock data for development

Frontend - Browser Extension:
- Add shared components (ChainIcon, RiskBadge, PriceDisplay, SuggestionCard)
- Add crypto components (SafetyScoreDisplay, WatchlistPanel, AlertConfigModal)
- Add chat enhancements (WelcomeScreen, ThinkingStepsDisplay)
- Add widget components for inline display
- Enhance TokenInfoCard, ChatHeader, ChatInput, ChatInterface

Documentation:
- Add conversational UX specification
- Add UX analysis report
- Update extension UX design

This implements the Conversational UX paradigm where crypto features
are AI-callable tools that render inline in the chat interface.
This commit is contained in:
API Test Bot 2026-02-04 02:19:57 +07:00
parent ad795eb830
commit e4d020799b
58 changed files with 11315 additions and 661 deletions

View file

@ -0,0 +1,58 @@
"use client";
import { cn } from "@/lib/utils";
import { TrendingUp, TrendingDown, Minus } from "lucide-react";
import { formatPrice, formatPercent } from "@/lib/mock/cryptoMockData";
interface PriceDisplayProps {
price: number;
priceChange?: number;
size?: "sm" | "md" | "lg";
showIcon?: boolean;
className?: string;
}
const sizeClasses = {
sm: { price: "text-sm font-medium", change: "text-xs" },
md: { price: "text-lg font-semibold", change: "text-sm" },
lg: { price: "text-2xl font-bold", change: "text-base" },
};
export function PriceDisplay({
price,
priceChange,
size = "md",
showIcon = true,
className,
}: PriceDisplayProps) {
const isPositive = priceChange !== undefined && priceChange > 0;
const isNegative = priceChange !== undefined && priceChange < 0;
const isNeutral = priceChange === undefined || priceChange === 0;
return (
<div className={cn("flex items-baseline gap-2", className)}>
<span className={sizeClasses[size].price}>{formatPrice(price)}</span>
{priceChange !== undefined && (
<span
className={cn(
"flex items-center gap-0.5",
sizeClasses[size].change,
isPositive && "text-green-500",
isNegative && "text-red-500",
isNeutral && "text-muted-foreground"
)}
>
{showIcon && (
<>
{isPositive && <TrendingUp className="h-3 w-3" />}
{isNegative && <TrendingDown className="h-3 w-3" />}
{isNeutral && <Minus className="h-3 w-3" />}
</>
)}
{formatPercent(priceChange)}
</span>
)}
</div>
);
}