Polish explorer frontend validation and utility pages

This commit is contained in:
defiQUG
2026-03-28 13:26:42 -07:00
parent 59eee21a3f
commit 1e3a3f00ef
7 changed files with 201 additions and 5 deletions

View File

@@ -63,7 +63,7 @@ function DropdownItem({
if (external) {
return (
<li role="none">
<a href={href} className={className} role="menuitem">
<a href={href} className={className} role="menuitem" target="_blank" rel="noopener noreferrer">
{icon}
<span>{children}</span>
</a>

View File

@@ -5,7 +5,7 @@ import { Card } from '@/libs/frontend-ui-primitives'
const poolCards = [
{
title: 'Canonical PMM Matrix',
title: 'Canonical PMM routes',
description: 'Review the public Chain 138 DODO PMM route matrix, live pool freshness, and payload examples.',
href: '/liquidity',
label: 'Open liquidity access',
@@ -25,6 +25,24 @@ const poolCards = [
},
]
const shortcutCards = [
{
title: 'cUSDT / USDT',
description: 'Open the canonical direct stable route coverage and compare the live pool snapshot.',
href: '/liquidity',
},
{
title: 'cUSDC / USDC',
description: 'Check the public stable bridge route and inspect the live reserves block.',
href: '/liquidity',
},
{
title: 'cUSDT / cXAUC',
description: 'Review one of the live gold-backed route families from the liquidity access page.',
href: '/liquidity',
},
]
export default function PoolsPage() {
return (
<div className="container mx-auto px-4 py-8">
@@ -48,6 +66,23 @@ export default function PoolsPage() {
</Card>
))}
</div>
<div className="mt-8">
<Card title="Pool operation shortcuts">
<div className="grid gap-4 md:grid-cols-3">
{shortcutCards.map((card) => (
<Link
key={card.title}
href={card.href}
className="rounded-lg border border-gray-200 p-4 transition hover:border-primary-400 hover:shadow-sm dark:border-gray-700"
>
<div className="text-sm font-semibold text-gray-900 dark:text-white">{card.title}</div>
<p className="mt-2 text-sm text-gray-600 dark:text-gray-400">{card.description}</p>
</Link>
))}
</div>
</Card>
</div>
</div>
)
}

View File

@@ -5,6 +5,15 @@ import { useRouter } from 'next/router'
import { useState } from 'react'
import { Card } from '@/libs/frontend-ui-primitives'
const quickSearches = [
{ label: 'cUSDT', description: 'Canonical bridged USDT liquidity and address results.' },
{ label: 'cUSDC', description: 'Canonical bridged USDC routes and address coverage.' },
{ label: 'cXAUC', description: 'Gold-backed cXAUC pools and token references.' },
{ label: 'cXAUT', description: 'Gold-backed cXAUT references and search coverage.' },
{ label: 'cEURT', description: 'EUR liquidity and cXAUC-connected route coverage.' },
{ label: 'USDT', description: 'Native-side USDT routes and address discovery.' },
]
function normalizeAddress(value: string) {
const trimmed = value.trim()
return /^0x[a-fA-F0-9]{40}$/.test(trimmed) ? trimmed : ''
@@ -75,6 +84,23 @@ export default function TokensPage() {
</div>
</Card>
</div>
<div className="mt-8">
<Card title="Common token searches">
<div className="grid gap-4 md:grid-cols-2 xl:grid-cols-3">
{quickSearches.map((token) => (
<Link
key={token.label}
href={`/search?q=${encodeURIComponent(token.label)}`}
className="rounded-lg border border-gray-200 p-4 transition hover:border-primary-400 hover:shadow-sm dark:border-gray-700"
>
<div className="text-sm font-semibold text-gray-900 dark:text-white">{token.label}</div>
<p className="mt-2 text-sm text-gray-600 dark:text-gray-400">{token.description}</p>
</Link>
))}
</div>
</Card>
</div>
</div>
)
}

View File

@@ -27,15 +27,72 @@ export default function WatchlistPage() {
})
}
const exportWatchlist = () => {
try {
const blob = new Blob([JSON.stringify(entries, null, 2)], { type: 'application/json' })
const url = URL.createObjectURL(blob)
const anchor = document.createElement('a')
anchor.href = url
anchor.download = 'explorer-watchlist.json'
anchor.click()
URL.revokeObjectURL(url)
} catch {}
}
const importWatchlist = (event: React.ChangeEvent<HTMLInputElement>) => {
const file = event.target.files?.[0]
if (!file) return
file.text().then((text) => {
try {
const parsed = JSON.parse(text)
const next = Array.isArray(parsed)
? parsed.filter((entry): entry is string => typeof entry === 'string')
: []
setEntries(next)
window.localStorage.setItem('explorerWatchlist', JSON.stringify(next))
} catch {}
}).catch(() => {})
event.target.value = ''
}
return (
<div className="container mx-auto px-4 py-8">
<h1 className="text-3xl font-bold mb-6">Watchlist</h1>
<Card title="Saved Addresses">
{entries.length === 0 ? (
<div className="mb-4 flex flex-col gap-3 md:flex-row md:items-center md:justify-between">
<p className="text-sm text-gray-600 dark:text-gray-400">
Your watchlist is empty. Add an address from its detail page to keep it here.
{entries.length === 0 ? 'No saved entries yet.' : `${entries.length} saved ${entries.length === 1 ? 'address' : 'addresses'}.`}
</p>
<div className="flex flex-wrap gap-2">
<label className="cursor-pointer rounded-lg bg-gray-100 px-3 py-1.5 text-sm text-gray-700 hover:bg-gray-200 dark:bg-gray-800 dark:text-gray-200 dark:hover:bg-gray-700">
Import JSON
<input type="file" accept="application/json" className="hidden" onChange={importWatchlist} />
</label>
<button
type="button"
onClick={exportWatchlist}
disabled={entries.length === 0}
className="rounded-lg bg-primary-600 px-3 py-1.5 text-sm text-white hover:bg-primary-700 disabled:opacity-50"
>
Export JSON
</button>
</div>
</div>
{entries.length === 0 ? (
<div className="space-y-3 text-sm text-gray-600 dark:text-gray-400">
<p>Your watchlist is empty. Add an address from its detail page to keep it here.</p>
<div className="flex flex-wrap gap-3">
<Link href="/addresses" className="text-primary-600 hover:underline">
Browse addresses
</Link>
<Link href="/search" className="text-primary-600 hover:underline">
Search the explorer
</Link>
</div>
</div>
) : (
<div className="space-y-3">
{entries.map((entry) => (