mirror of
https://github.com/pezkuwichain/pwap.git
synced 2026-04-23 02:27:55 +00:00
dc35e19099
- Added DKstate.png to public folder (719KB PNG) - Replaced Kurdish Flag CDN image with local DKstate background - Updated alt text to "DKstate Background" - Maintained gradient overlay styling Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
155 lines
6.2 KiB
TypeScript
155 lines
6.2 KiB
TypeScript
import React, { useState, useEffect } from 'react';
|
|
import { useTranslation } from 'react-i18next';
|
|
import { ChevronRight, Cpu, GitBranch, Shield } from 'lucide-react';
|
|
import { NetworkStats } from './NetworkStats';
|
|
import { usePolkadot } from '../contexts/PolkadotContext';
|
|
import { formatBalance } from '../lib/wallet';
|
|
|
|
const HeroSection: React.FC = () => {
|
|
const { t } = useTranslation();
|
|
const { api, isApiReady } = usePolkadot();
|
|
const [stats, setStats] = useState({
|
|
activeProposals: 0,
|
|
totalVoters: 0,
|
|
tokensStaked: '0',
|
|
trustScore: 0
|
|
});
|
|
|
|
useEffect(() => {
|
|
const fetchStats = async () => {
|
|
if (!api || !isApiReady) return;
|
|
|
|
try {
|
|
// Fetch active referenda
|
|
let activeProposals = 0;
|
|
try {
|
|
const referendaCount = await api.query.referenda.referendumCount();
|
|
activeProposals = referendaCount.toNumber();
|
|
} catch (err) {
|
|
console.warn('Failed to fetch referenda:', err);
|
|
}
|
|
|
|
// Fetch total staked tokens
|
|
let tokensStaked = '0';
|
|
try {
|
|
const currentEra = await api.query.staking.currentEra();
|
|
if (currentEra.isSome) {
|
|
const eraIndex = currentEra.unwrap().toNumber();
|
|
const totalStake = await api.query.staking.erasTotalStake(eraIndex);
|
|
const formatted = formatBalance(totalStake.toString());
|
|
tokensStaked = `${formatted} HEZ`;
|
|
}
|
|
} catch (err) {
|
|
console.warn('Failed to fetch total stake:', err);
|
|
}
|
|
|
|
// Count total voters from conviction voting
|
|
let totalVoters = 0;
|
|
try {
|
|
// Get all voting keys and count unique voters
|
|
const votingKeys = await api.query.convictionVoting.votingFor.keys();
|
|
// Each key represents a unique (account, track) pair
|
|
// Count unique accounts
|
|
const uniqueAccounts = new Set(votingKeys.map(key => key.args[0].toString()));
|
|
totalVoters = uniqueAccounts.size;
|
|
} catch (err) {
|
|
console.warn('Failed to fetch voters:', err);
|
|
}
|
|
|
|
// Update stats
|
|
setStats({
|
|
activeProposals,
|
|
totalVoters,
|
|
tokensStaked,
|
|
trustScore: 0 // TODO: Calculate trust score
|
|
});
|
|
|
|
console.log('✅ Hero stats updated:', {
|
|
activeProposals,
|
|
totalVoters,
|
|
tokensStaked
|
|
});
|
|
} catch (error) {
|
|
console.error('Failed to fetch hero stats:', error);
|
|
}
|
|
};
|
|
|
|
fetchStats();
|
|
}, [api, isApiReady]);
|
|
|
|
return (
|
|
<section className="relative min-h-screen flex items-center justify-start overflow-hidden bg-gray-950">
|
|
{/* Background Image */}
|
|
<div className="absolute inset-0">
|
|
<img
|
|
src="/DKstate.png"
|
|
alt="DKstate Background"
|
|
className="w-full h-full object-cover opacity-30"
|
|
/>
|
|
<div className="absolute inset-0 bg-gradient-to-b from-green-600/20 via-gray-950/70 to-gray-950"></div>
|
|
</div>
|
|
|
|
{/* Content */}
|
|
<div className="relative z-10 w-full text-center">
|
|
<div className="mb-8 inline-flex items-center px-4 py-2 rounded-full bg-green-600/20 backdrop-blur-sm border border-green-500/30">
|
|
<Shield className="w-4 h-4 text-yellow-400 mr-2" />
|
|
<span className="text-yellow-400 text-sm font-medium">Substrate Parachain v1.0</span>
|
|
</div>
|
|
|
|
<h1 className="text-5xl md:text-7xl font-bold mb-6 bg-gradient-to-r from-green-500 via-yellow-400 to-red-500 bg-clip-text text-transparent">
|
|
PezkuwiChain
|
|
</h1>
|
|
|
|
<p className="text-xl md:text-2xl text-gray-300 mb-8 max-w-3xl mx-auto">
|
|
{t('hero.title')}
|
|
</p>
|
|
<p className="text-lg text-gray-400 mb-8 max-w-2xl mx-auto">
|
|
{t('hero.subtitle')}
|
|
</p>
|
|
|
|
{/* Live Network Stats */}
|
|
<div className="mb-12">
|
|
<NetworkStats />
|
|
</div>
|
|
|
|
<div className="grid grid-cols-2 md:grid-cols-4 gap-4 mb-12 max-w-4xl mx-auto">
|
|
<div className="bg-gray-900/50 backdrop-blur-sm rounded-lg border border-green-500/30 p-4">
|
|
<div className="text-2xl font-bold text-green-400">{stats.activeProposals}</div>
|
|
<div className="text-sm text-gray-400">{t('hero.stats.activeProposals')}</div>
|
|
</div>
|
|
<div className="bg-gray-900/50 backdrop-blur-sm rounded-lg border border-yellow-400/30 p-4">
|
|
<div className="text-2xl font-bold text-yellow-400">{stats.totalVoters || '-'}</div>
|
|
<div className="text-sm text-gray-400">{t('hero.stats.totalVoters')}</div>
|
|
</div>
|
|
<div className="bg-gray-900/50 backdrop-blur-sm rounded-lg border border-red-500/30 p-4">
|
|
<div className="text-2xl font-bold text-red-400">{stats.tokensStaked || '-'}</div>
|
|
<div className="text-sm text-gray-400">{t('hero.stats.tokensStaked')}</div>
|
|
</div>
|
|
<div className="bg-gray-900/50 backdrop-blur-sm rounded-lg border border-green-500/30 p-4">
|
|
<div className="text-2xl font-bold text-green-400">{stats.trustScore ? `${stats.trustScore}%` : '-'}</div>
|
|
<div className="text-sm text-gray-400">{t('hero.stats.trustScore')}</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="flex flex-col sm:flex-row gap-4 justify-center">
|
|
<button
|
|
onClick={() => document.getElementById('governance')?.scrollIntoView({ behavior: 'smooth' })}
|
|
className="px-8 py-4 bg-gradient-to-r from-green-600 to-yellow-400 text-white font-semibold rounded-lg hover:from-green-700 hover:to-yellow-500 transition-all transform hover:scale-105 flex items-center justify-center group"
|
|
>
|
|
{t('hero.exploreGovernance')}
|
|
<ChevronRight className="ml-2 w-5 h-5 group-hover:translate-x-1 transition-transform" />
|
|
</button>
|
|
<button
|
|
onClick={() => document.getElementById('identity')?.scrollIntoView({ behavior: 'smooth' })}
|
|
className="px-8 py-4 bg-gray-900/50 backdrop-blur-sm text-white font-semibold rounded-lg border border-red-500/50 hover:bg-red-500/10 transition-all"
|
|
>
|
|
{t('hero.learnMore')}
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
);
|
|
};
|
|
|
|
export default HeroSection;
|