import React, { useEffect, useMemo, useRef, useState } from &quot;react&quot;; import { motion, AnimatePresence } from &quot;framer-motion&quot;; import { Shuffle, Trophy, Sparkles, Volume2, VolumeX, Flame, Shield, Star, Repeat2, Gift, Wand2, Lock, Share2, Heart, History, BadgeInfo, Megaphone, } from &quot;lucide-react&quot;;

const CLUBS = [ { id: &quot;mci&quot;, name: &quot;Manchester City&quot;, type: &quot;club&quot;, league: &quot;Premier League&quot;, region: &quot;Europe&quot;, strength: 95, logo: &quot;🔵&quot;, colors: &quot;from-sky-500 to-blue-700&quot; }, { id: &quot;liv&quot;, name: &quot;Liverpool&quot;, type: &quot;club&quot;, league: &quot;Premier League&quot;, region: &quot;Europe&quot;, strength: 92, logo: &quot;🔴&quot;, colors: &quot;from-red-500 to-rose-700&quot; }, { id: &quot;ars&quot;, name: &quot;Arsenal&quot;, type: &quot;club&quot;, league: &quot;Premier League&quot;, region: &quot;Europe&quot;, strength: 90, logo: &quot;🔴&quot;, colors: &quot;from-rose-500 to-orange-600&quot; }, { id: &quot;rma&quot;, name: &quot;Real Madrid&quot;, type: &quot;club&quot;, league: &quot;La Liga&quot;, region: &quot;Europe&quot;, strength: 96, logo: &quot;👑&quot;, colors: &quot;from-zinc-100 to-zinc-400&quot; }, { id: &quot;bar&quot;, name: &quot;FC Barcelona&quot;, type: &quot;club&quot;, league: &quot;La Liga&quot;, region: &quot;Europe&quot;, strength: 94, logo: &quot;🔵&quot;, colors: &quot;from-blue-600 to-red-600&quot; }, { id: &quot;atm&quot;, name: &quot;Atl&eacute;tico Madrid&quot;, type: &quot;club&quot;, league: &quot;La Liga&quot;, region: &quot;Europe&quot;, strength: 89, logo: &quot;⚪&quot;, colors: &quot;from-red-700 to-slate-900&quot; }, { id: &quot;juv&quot;, name: &quot;Juventus&quot;, type: &quot;club&quot;, league: &quot;Serie A&quot;, region: &quot;Europe&quot;, strength: 88, logo: &quot;⚫&quot;, colors: &quot;from-slate-900 to-slate-700&quot; }, { id: &quot;mil&quot;, name: &quot;AC Milan&quot;, type: &quot;club&quot;, league: &quot;Serie A&quot;, region: &quot;Europe&quot;, strength: 89, logo: &quot;🔴&quot;, colors: &quot;from-red-600 to-black&quot; }, { id: &quot;int&quot;, name: &quot;Inter&quot;, type: &quot;club&quot;, league: &quot;Serie A&quot;, region: &quot;Europe&quot;, strength: 91, logo: &quot;🔵&quot;, colors: &quot;from-sky-700 to-black&quot; }, { id: &quot;bvb&quot;, name: &quot;Borussia Dortmund&quot;, type: &quot;club&quot;, league: &quot;Bundesliga&quot;, region: &quot;Europe&quot;, strength: 87, logo: &quot;🟡&quot;, colors: &quot;from-yellow-400 to-black&quot; }, { id: &quot;bay&quot;, name: &quot;Bayern Munich&quot;, type: &quot;club&quot;, league: &quot;Bundesliga&quot;, region: &quot;Europe&quot;, strength: 95, logo: &quot;🔴&quot;, colors: &quot;from-red-600 to-rose-800&quot; }, { id: &quot;psg&quot;, name: &quot;PSG&quot;, type: &quot;club&quot;, league: &quot;Ligue 1&quot;, region: &quot;Europe&quot;, strength: 93, logo: &quot;🔵&quot;, colors: &quot;from-blue-800 to-pink-600&quot; }, { id: &quot;mon&quot;, name: &quot;Monaco&quot;, type: &quot;club&quot;, league: &quot;Ligue 1&quot;, region: &quot;Europe&quot;, strength: 84, logo: &quot;🔴&quot;, colors: &quot;from-red-600 to-white&quot; }, { id: &quot;por&quot;, name: &quot;FC Porto&quot;, type: &quot;club&quot;, league: &quot;Primeira Liga&quot;, region: &quot;Europe&quot;, strength: 85, logo: &quot;🐉&quot;, colors: &quot;from-blue-700 to-sky-400&quot; }, { id: &quot;aja&quot;, name: &quot;Ajax&quot;, type: &quot;club&quot;, league: &quot;Eredivisie&quot;, region: &quot;Europe&quot;, strength: 83, logo: &quot;⭐&quot;, colors: &quot;from-red-500 to-white&quot; }, { id: &quot;aln&quot;, name: &quot;Al Nassr&quot;, type: &quot;club&quot;, league: &quot;Saudi Pro League&quot;, region: &quot;Asia&quot;, strength: 84, logo: &quot;🟡&quot;, colors: &quot;from-yellow-400 to-amber-700&quot; }, { id: &quot;hil&quot;, name: &quot;Al Hilal&quot;, type: &quot;club&quot;, league: &quot;Saudi Pro League&quot;, region: &quot;Asia&quot;, strength: 86, logo: &quot;🔷&quot;, colors: &quot;from-sky-500 to-blue-900&quot; }, { id: &quot;Uls&quot;, name: &quot;Ulsan HD&quot;, type: &quot;club&quot;, league: &quot;K League&quot;, region: &quot;Asia&quot;, strength: 78, logo: &quot;🟦&quot;, colors: &quot;from-blue-500 to-slate-800&quot; }, { id: &quot;bfc&quot;, name: &quot;Beijing Guoan&quot;, type: &quot;club&quot;, league: &quot;Chinese Super League&quot;, region: &quot;Asia&quot;, strength: 76, logo: &quot;🟩&quot;, colors: &quot;from-emerald-500 to-green-800&quot; }, { id: &quot;san&quot;, name: &quot;Santos&quot;, type: &quot;club&quot;, league: &quot;Brasileir&atilde;o&quot;, region: &quot;South America&quot;, strength: 79, logo: &quot;⚪&quot;, colors: &quot;from-white to-black&quot; }, { id: &quot;pal&quot;, name: &quot;Palmeiras&quot;, type: &quot;club&quot;, league: &quot;Brasileir&atilde;o&quot;, region: &quot;South America&quot;, strength: 87, logo: &quot;🟢&quot;, colors: &quot;from-emerald-500 to-green-900&quot; }, { id: &quot;rio&quot;, name: &quot;River Plate&quot;, type: &quot;club&quot;, league: &quot;Liga Argentina&quot;, region: &quot;South America&quot;, strength: 88, logo: &quot;🔴&quot;, colors: &quot;from-red-600 to-white&quot; }, { id: &quot;boo&quot;, name: &quot;Boca Juniors&quot;, type: &quot;club&quot;, league: &quot;Liga Argentina&quot;, region: &quot;South America&quot;, strength: 86, logo: &quot;💙&quot;, colors: &quot;from-blue-700 to-yellow-400&quot; }, { id: &quot;afr1&quot;, name: &quot;Al Ahly&quot;, type: &quot;club&quot;, league: &quot;Egypt Premier League&quot;, region: &quot;Africa&quot;, strength: 80, logo: &quot;🟥&quot;, colors: &quot;from-red-700 to-amber-700&quot; }, { id: &quot;afr2&quot;, name: &quot;Wydad AC&quot;, type: &quot;club&quot;, league: &quot;Botola Pro&quot;, region: &quot;Africa&quot;, strength: 77, logo: &quot;⬛&quot;, colors: &quot;from-red-800 to-black&quot; }, { id: &quot;gal&quot;, name: &quot;Galatasaray&quot;, type: &quot;club&quot;, league: &quot;S&uuml;per Lig&quot;, region: &quot;Europe&quot;, strength: 84, logo: &quot;🟠&quot;, colors: &quot;from-orange-500 to-red-700&quot; }, { id: &quot;new&quot;, name: &quot;Newcastle United&quot;, type: &quot;club&quot;, league: &quot;Premier League&quot;, region: &quot;Europe&quot;, strength: 83, logo: &quot;⚫&quot;, colors: &quot;from-zinc-900 to-emerald-700&quot; }, ];

const NATIONALS = [ { id: &quot;fra&quot;, name: &quot;France&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;Europe&quot;, strength: 95, logo: &quot;🇫🇷&quot;, colors: &quot;from-blue-700 to-red-600&quot; }, { id: &quot;spa&quot;, name: &quot;Spain&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;Europe&quot;, strength: 91, logo: &quot;🇪🇸&quot;, colors: &quot;from-red-600 to-yellow-400&quot; }, { id: &quot;ger&quot;, name: &quot;Germany&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;Europe&quot;, strength: 90, logo: &quot;🇩🇪&quot;, colors: &quot;from-black to-red-600&quot; }, { id: &quot;ita&quot;, name: &quot;Italy&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;Europe&quot;, strength: 89, logo: &quot;🇮🇹&quot;, colors: &quot;from-green-600 to-blue-600&quot; }, { id: &quot;eng&quot;, name: &quot;England&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;Europe&quot;, strength: 90, logo: &quot;🏴&quot;, colors: &quot;from-slate-800 to-slate-500&quot; }, { id: &quot;porN&quot;, name: &quot;Portugal&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;Europe&quot;, strength: 88, logo: &quot;🇵🇹&quot;, colors: &quot;from-emerald-600 to-red-700&quot; }, { id: &quot;jpn&quot;, name: &quot;Japan&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;Asia&quot;, strength: 85, logo: &quot;🇯🇵&quot;, colors: &quot;from-rose-500 to-white&quot; }, { id: &quot;kor&quot;, name: &quot;South Korea&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;Asia&quot;, strength: 84, logo: &quot;🇰🇷&quot;, colors: &quot;from-red-600 to-sky-600&quot; }, { id: &quot;sau&quot;, name: &quot;Saudi Arabia&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;Asia&quot;, strength: 79, logo: &quot;🇸🇦&quot;, colors: &quot;from-green-700 to-emerald-500&quot; }, { id: &quot;aus&quot;, name: &quot;Australia&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;Asia&quot;, strength: 78, logo: &quot;🇦🇺&quot;, colors: &quot;from-sky-700 to-amber-500&quot; }, { id: &quot;nga&quot;, name: &quot;Nigeria&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;Africa&quot;, strength: 82, logo: &quot;🇳🇬&quot;, colors: &quot;from-green-700 to-black&quot; }, { id: &quot;mar&quot;, name: &quot;Morocco&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;Africa&quot;, strength: 84, logo: &quot;🇲🇦&quot;, colors: &quot;from-red-700 to-green-700&quot; }, { id: &quot;gha&quot;, name: &quot;Ghana&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;Africa&quot;, strength: 78, logo: &quot;🇬🇭&quot;, colors: &quot;from-red-600 to-green-600&quot; }, { id: &quot;sen&quot;, name: &quot;Senegal&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;Africa&quot;, strength: 81, logo: &quot;🇸🇳&quot;, colors: &quot;from-green-700 to-yellow-400&quot; }, { id: &quot;bra&quot;, name: &quot;Brazil&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;South America&quot;, strength: 94, logo: &quot;🇧🇷&quot;, colors: &quot;from-emerald-500 to-yellow-400&quot; }, { id: &quot;arg&quot;, name: &quot;Argentina&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;South America&quot;, strength: 93, logo: &quot;🇦🇷&quot;, colors: &quot;from-sky-500 to-white&quot; }, { id: &quot;uru&quot;, name: &quot;Uruguay&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;South America&quot;, strength: 87, logo: &quot;🇺🇾&quot;, colors: &quot;from-sky-600 to-white&quot; }, { id: &quot;col&quot;, name: &quot;Colombia&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;South America&quot;, strength: 84, logo: &quot;🇨🇴&quot;, colors: &quot;from-yellow-400 to-red-600&quot; }, { id: &quot;mex&quot;, name: &quot;Mexico&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;North America&quot;, strength: 82, logo: &quot;🇲🇽&quot;, colors: &quot;from-green-700 to-red-700&quot; }, { id: &quot;usa&quot;, name: &quot;USA&quot;, type: &quot;national&quot;, league: &quot;National Team&quot;, region: &quot;North America&quot;, strength: 83, logo: &quot;🇺🇸&quot;, colors: &quot;from-blue-700 to-red-700&quot; }, ];

const LEAGUES = [ &quot;Premier League&quot;, &quot;La Liga&quot;, &quot;Serie A&quot;, &quot;Bundesliga&quot;, &quot;Ligue 1&quot;, &quot;Primeira Liga&quot;, &quot;Eredivisie&quot;, &quot;Saudi Pro League&quot;, &quot;K League&quot;, &quot;Chinese Super League&quot;, &quot;Brasileir&atilde;o&quot;, &quot;Liga Argentina&quot;, &quot;Egypt Premier League&quot;, &quot;Botola Pro&quot;, &quot;S&uuml;per Lig&quot;, ];

const NATIONAL_REGIONS = [&quot;Europe&quot;, &quot;Asia&quot;, &quot;Africa&quot;, &quot;North America&quot;, &quot;South America&quot;];

const EFFECTS = [ { id: &quot;wheel&quot;, label: &quot;V&ograve;ng quay may mắn&quot;, icon: &quot;🎡&quot; }, { id: &quot;cards&quot;, label: &quot;Bốc l&aacute; b&agrave;i&quot;, icon: &quot;🃏&quot; }, { id: &quot;gift&quot;, label: &quot;Bốc hộp qu&agrave;&quot;, icon: &quot;🎁&quot; }, { id: &quot;flip&quot;, label: &quot;Lật thẻ&quot;, icon: &quot;🪙&quot; }, ];

const SPECIAL_CARDS = [ { id: &quot;opponent&quot;, label: &quot;Chọn đội cho đối phương&quot;, icon: &lt;Megaphone className=&quot;h-4 w-4&quot; /&gt; }, { id: &quot;swap&quot;, label: &quot;Đổi đội&quot;, icon: &lt;Repeat2 className=&quot;h-4 w-4&quot; /&gt; }, { id: &quot;strongWeak&quot;, label: &quot;Chọn đội mạnh/yếu&quot;, icon: &lt;Flame className=&quot;h-4 w-4&quot; /&gt; }, { id: &quot;reroll&quot;, label: &quot;Random lại&quot;, icon: &lt;Shuffle className=&quot;h-4 w-4&quot; /&gt; }, { id: &quot;extra&quot;, label: &quot;Bốc th&ecirc;m 1 lượt&quot;, icon: &lt;Gift className=&quot;h-4 w-4&quot; /&gt; }, { id: &quot;banLeague&quot;, label: &quot;Cấm chọn một giải đấu&quot;, icon: &lt;Lock className=&quot;h-4 w-4&quot; /&gt; }, ];

function clamp(min, max, value) { return Math.max(min, Math.min(max, value)); }

function pickRandom(arr) { return arr[Math.floor(Math.random() * arr.length)]; }

function shuffle(arr) { return [...arr].sort(() =&gt; Math.random() - 0.5); }

function getDataset(mode) { if (mode === &quot;national&quot;) return NATIONALS; if (mode === &quot;favorites&quot;) return [...CLUBS, ...NATIONALS]; return CLUBS; }

function normalize(str) { return String(str || &quot;&quot;).toLowerCase(); }

function useLocalStorage(key, initialValue) { const [value, setValue] = useState(() =&gt; { if (typeof window === &quot;undefined&quot;) return initialValue; try { const saved = window.localStorage.getItem(key); return saved ? JSON.parse(saved) : initialValue; } catch { return initialValue; } });

useEffect(() =&gt; { try { window.localStorage.setItem(key, JSON.stringify(value)); } catch { // ignore } }, [key, value]);

return [value, setValue]; }

function makeConfetti(count = 48) { return Array.from({ length: count }).map((_, i) =&gt; ({ id: ${Date.now()}-${i}, left: Math.random() * 100, delay: Math.random() * 0.35, size: 8 + Math.random() * 12, rotate: Math.random() * 360, duration: 1.8 + Math.random() * 1.8, })); }

function strengthLabel(n) { if (n &gt;= 92) return &quot;Elite&quot;; if (n &gt;= 87) return &quot;Mạnh&quot;; if (n &gt;= 80) return &quot;Tốt&quot;; if (n &gt;= 72) return &quot;Kh&aacute;&quot;; return &quot;Vui&quot;; }

function playBeep(type = &quot;win&quot;) { try { const AudioCtx = window.AudioContext || window.webkitAudioContext; if (!AudioCtx) return; const ctx = new AudioCtx(); const now = ctx.currentTime; const tones = type === &quot;win&quot; ? [392, 523.25, 659.25] : [220, 277.18]; tones.forEach((freq, idx) =&gt; { const osc = ctx.createOscillator(); const gain = ctx.createGain(); osc.type = &quot;sine&quot;; osc.frequency.value = freq; gain.gain.value = 0.0001; osc.connect(gain); gain.connect(ctx.destination); osc.start(now + idx * 0.08); gain.gain.exponentialRampToValueAtTime(0.2, now + idx * 0.08 + 0.01); gain.gain.exponentialRampToValueAtTime(0.0001, now + idx * 0.08 + 0.22); osc.stop(now + idx * 0.08 + 0.24); }); setTimeout(() =&gt; ctx.close().catch(() =&gt; {}), 1000); } catch { // ignore audio errors } }

function Card({ title, icon, children, className = &quot;&quot; }) { return ( &lt;div className={rounded-3xl border border-white/10 bg-white/5 backdrop-blur-xl shadow-2xl shadow-black/20 ${className}}&gt; &lt;div className=&quot;flex items-center gap-2 border-b border-white/10 px-4 py-3 text-white/90&quot;&gt; &lt;div className=&quot;rounded-2xl bg-white/10 p-2&quot;&gt;{icon}&lt;/div&gt; &lt;h3 className=&quot;font-semibold&quot;&gt;{title}&lt;/h3&gt; &lt;/div&gt; &lt;div className=&quot;p-4&quot;&gt;{children}&lt;/div&gt; &lt;/div&gt; ); }

function TeamChip({ team, onToggleFavorite, favorite }) { return ( &lt;div className={rounded-2xl border border-white/10 bg-gradient-to-br ${team.colors} p-[1px]}&gt; &lt;div className=&quot;rounded-2xl bg-slate-950/90 p-3 text-white&quot;&gt; &lt;div className=&quot;flex items-start justify-between gap-3&quot;&gt; &lt;div&gt; &lt;div className=&quot;text-2xl&quot;&gt;{team.logo}&lt;/div&gt; &lt;div className=&quot;mt-2 font-semibold leading-tight&quot;&gt;{team.name}&lt;/div&gt; &lt;div className=&quot;mt-1 text-xs text-white/65&quot;&gt;{team.type === &quot;club&quot; ? team.league : team.region} &middot; {strengthLabel(team.strength)}&lt;/div&gt; &lt;/div&gt; &lt;button onClick={() =&gt; onToggleFavorite(team.id)} className={rounded-full p-2 transition ${favorite ? &quot;bg-pink-500/20 text-pink-300&quot; : &quot;bg-white/8 text-white/60 hover:bg-white/12&quot;}} aria-label=&quot;favorite&quot; &gt; &lt;Heart className={h-4 w-4 ${favorite ? &quot;fill-pink-400&quot; : &quot;&quot;}} /&gt; &lt;/button&gt; &lt;/div&gt; &lt;/div&gt; &lt;/div&gt; ); }

export default function PesRandomizerWebsite() { const [mode, setMode] = useState(&quot;2p&quot;); const [effect, setEffect] = useState(&quot;wheel&quot;); const [soundOn, setSoundOn] = useState(true); const [partyMode, setPartyMode] = useState(true); const [uniqueTeams, setUniqueTeams] = useState(true); const [selectedLeague, setSelectedLeague] = useState(&quot;All&quot;); const [selectedRegion, setSelectedRegion] = useState(&quot;All&quot;); const [bannedLeague, setBannedLeague] = useState(&quot;None&quot;); const [favoriteIds, setFavoriteIds] = useLocalStorage(&quot;pes-favorites&quot;, [&quot;mci&quot;, &quot;rma&quot;, &quot;bar&quot;, &quot;fra&quot;, &quot;arg&quot;, &quot;bra&quot;]); const [history, setHistory] = useLocalStorage(&quot;pes-history&quot;, []); const [result, setResult] = useState(null); const [confetti, setConfetti] = useState([]); const [animating, setAnimating] = useState(false); const [specialCard, setSpecialCard] = useState(null); const [forcedMode, setForcedMode] = useState(null); const [partyMessage, setPartyMessage] = useState(&quot;Sẵn s&agrave;ng random chưa? Nhấn n&uacute;t lớn b&ecirc;n dưới!&quot;); const resultRef = useRef(null);

const allTeams = useMemo(() =&gt; { if (mode === &quot;national&quot;) return NATIONALS; if (mode === &quot;favorites&quot;) { const fav = [...CLUBS, ...NATIONALS].filter((t) =&gt; favoriteIds.includes(t.id)); return fav.length ? fav : [...CLUBS, ...NATIONALS]; } return CLUBS; }, [mode, favoriteIds]);

const filteredTeams = useMemo(() =&gt; { let list = [...allTeams]; if (mode === &quot;club&quot; || mode === &quot;2p&quot; || mode === &quot;party&quot;) { if (selectedLeague !== &quot;All&quot;) list = list.filter((t) =&gt; t.type === &quot;club&quot; &amp;&amp; t.league === selectedLeague); if (bannedLeague !== &quot;None&quot;) list = list.filter((t) =&gt; t.league !== bannedLeague); } if (mode === &quot;national&quot;) { if (selectedRegion !== &quot;All&quot;) list = list.filter((t) =&gt; t.region === selectedRegion); } if (forcedMode === &quot;strong&quot;) list = [...list].sort((a, b) =&gt; b.strength - a.strength); if (forcedMode === &quot;weak&quot;) list = [...list].sort((a, b) =&gt; a.strength - b.strength); return list; }, [allTeams, mode, selectedLeague, selectedRegion, bannedLeague, forcedMode]);

const favoriteTeams = useMemo(() =&gt; [...CLUBS, ...NATIONALS].filter((t) =&gt; favoriteIds.includes(t.id)), [favoriteIds]);

const stats = useMemo(() =&gt; { const leagues = new Set(filteredTeams.filter((t) =&gt; t.type === &quot;club&quot;).map((t) =&gt; t.league)); const regions = new Set(filteredTeams.filter((t) =&gt; t.type === &quot;national&quot;).map((t) =&gt; t.region)); return { teams: filteredTeams.length, leagues: leagues.size, regions: regions.size }; }, [filteredTeams]);

useEffect(() =&gt; { if (resultRef.current &amp;&amp; result) { resultRef.current.scrollIntoView({ behavior: &quot;smooth&quot;, block: &quot;center&quot; }); } }, [result]);

function saveHistory(entry) { setHistory((prev) =&gt; [entry, ...prev].slice(0, 8)); }

function toggleFavorite(id) { setFavoriteIds((prev) =&gt; prev.includes(id) ? prev.filter((x) =&gt; x !== id) : [id, ...prev].slice(0, 18) ); }

function announce(text) { setPartyMessage(text); }

function chooseWithConstraints(source, { different = true } = {}) { if (!source.length) return null; const pool = shuffle(source); return pool[0] || null; }

function pickTwoTeams(source) { if (!source.length) return null; const pool = [...source]; if (pool.length === 1) { return uniqueTeams ? null : { a: pool[0], b: pool[0] }; }

const a = pickRandom(pool);
let remaining = pool.filter((t) =&gt; t.id !== a.id);
if (!uniqueTeams &amp;&amp; Math.random() &gt; 0.7) remaining = [...pool];
const b = remaining.length ? pickRandom(remaining) : null;
return b ? { a, b } : null;

}

function runRandom({ fromSpecial = false } = {}) { const source = filteredTeams; if (!source.length) { setResult({ error: &quot;Kh&ocirc;ng c&oacute; đội n&agrave;o ph&ugrave; hợp với bộ lọc hiện tại.&quot; }); return; }

setAnimating(true);
setResult(null);
if (soundOn) playBeep(&quot;spin&quot;);

const delay = effect === &quot;wheel&quot; ? 1050 : effect === &quot;gift&quot; ? 900 : effect === &quot;cards&quot; ? 780 : 680;
setTimeout(() =&gt; {
  let nextResult;
  if (mode === &quot;2p&quot; || mode === &quot;party&quot;) {
    const picked = pickTwoTeams(source);
    if (!picked) {
      nextResult = { error: uniqueTeams ? &quot;Danh s&aacute;ch hiện tại kh&ocirc;ng đủ đội để chọn 2 đội kh&aacute;c nhau.&quot; : &quot;Kh&ocirc;ng thể random.&quot; };
    } else {
      nextResult = { type: &quot;pair&quot;, a: picked.a, b: picked.b };
    }
  } else if (mode === &quot;national&quot;) {
    nextResult = { type: &quot;single&quot;, a: chooseWithConstraints(source) };
  } else if (mode === &quot;favorites&quot;) {
    const picked = pickTwoTeams(source);
    nextResult = picked ? { type: &quot;pair&quot;, a: picked.a, b: picked.b } : { error: &quot;Danh s&aacute;ch y&ecirc;u th&iacute;ch qu&aacute; &iacute;t để random.&quot; };
  } else {
    const picked = pickTwoTeams(source);
    nextResult = picked ? { type: &quot;pair&quot;, a: picked.a, b: picked.b } : { error: &quot;Kh&ocirc;ng đủ dữ liệu.&quot; };
  }

  if (nextResult?.error) {
    setResult(nextResult);
    announce(nextResult.error);
    setAnimating(false);
    return;
  }

  if (partyMode &amp;&amp; mode === &quot;party&quot; &amp;&amp; !fromSpecial) {
    const partyRoll = Math.random();
    if (partyRoll &lt; 0.2) {
      setSpecialCard({ id: &quot;opponent&quot;, label: &quot;Thẻ đặc biệt: Chọn đội cho đối phương&quot; });
      announce(&quot;Party mode: đối phương bị chọn đội bất ngờ!&quot;);
    } else if (partyRoll &lt; 0.35) {
      setSpecialCard({ id: &quot;strongWeak&quot;, label: &quot;Thẻ đặc biệt: Chọn đội mạnh/yếu&quot; });
    } else if (partyRoll &lt; 0.48) {
      setSpecialCard({ id: &quot;extra&quot;, label: &quot;Thẻ đặc biệt: Bốc th&ecirc;m 1 lượt&quot; });
    } else if (partyRoll &lt; 0.58) {
      setSpecialCard({ id: &quot;swap&quot;, label: &quot;Thẻ đặc biệt: Đổi đội&quot; });
    } else {
      setSpecialCard(null);
    }
  }

  if (soundOn) playBeep(&quot;win&quot;);
  const entry = { ...nextResult, mode, effect, createdAt: new Date().toISOString() };
  setResult(nextResult);
  saveHistory(entry);
  setConfetti(makeConfetti(effect === &quot;gift&quot; ? 72 : 54));
  announce(
    nextResult.type === &quot;pair&quot;
      ? `${nextResult.a.name} vs ${nextResult.b.name}`
      : `Bạn nhận được ${nextResult.a.name}`
  );
  setAnimating(false);
  setTimeout(() =&gt; setConfetti([]), 2200);
}, delay);

}

function reroll() { runRandom({ fromSpecial: false }); }

function swapTeams() { if (!result || result.type !== &quot;pair&quot;) return; setResult({ ...result, a: result.b, b: result.a }); const swapped = { ...result, a: result.b, b: result.a, createdAt: new Date().toISOString(), mode, effect }; saveHistory(swapped); setConfetti(makeConfetti(34)); if (soundOn) playBeep(&quot;win&quot;); }

function chooseStrongWeak(kind) { const source = filteredTeams; if (!source.length) return; const ordered = [...source].sort((a, b) =&gt; (kind === &quot;strong&quot; ? b.strength - a.strength : a.strength - b.strength)); const top = ordered.slice(0, Math.max(2, Math.ceil(ordered.length * 0.3))); const picked = pickTwoTeams(top); if (!picked) return; setForcedMode(kind); setResult({ type: &quot;pair&quot;, a: picked.a, b: picked.b }); saveHistory({ type: &quot;pair&quot;, a: picked.a, b: picked.b, mode, effect, createdAt: new Date().toISOString(), special: kind }); setConfetti(makeConfetti(60)); announce(kind === &quot;strong&quot; ? &quot;Đ&atilde; chọn cặp đội mạnh!&quot; : &quot;Đ&atilde; chọn cặp đội yếu hơn cho party mode!&quot;); }

function chooseForOpponent() { const source = filteredTeams; if (!source.length) return; const other = pickRandom(source); setResult({ type: &quot;single&quot;, a: other, note: &quot;Đội d&agrave;nh cho đối phương&quot; }); saveHistory({ type: &quot;single&quot;, a: other, mode, effect, createdAt: new Date().toISOString(), special: &quot;opponent&quot; }); setConfetti(makeConfetti(44)); }

function extraDraw() { runRandom({ fromSpecial: true }); setTimeout(() =&gt; runRandom({ fromSpecial: true }), 250); }

function banLeagueRandom() { setBannedLeague(pickRandom(LEAGUES)); setPartyMessage(&quot;Một giải đấu đ&atilde; bị cấm ngẫu nhi&ecirc;n.&quot;); }

function shareResult() { if (!result) return; const text = result.type === &quot;pair&quot; ? PES/eFootball Random: ${result.a.name} vs ${result.b.name} : PES/eFootball Random: ${result.a.name}; if (navigator.share) { navigator.share({ title: &quot;PES Random&quot;, text }).catch(() =&gt; {}); } else if (navigator.clipboard) { navigator.clipboard.writeText(text).catch(() =&gt; {}); ann
import React, { useEffect, useMemo, useRef, useState } from "react"; import { motion, AnimatePresence } from "framer-motion"; import { Shuffle, Trophy, Sparkles, Volume2, VolumeX, Flame, Shield, Star, Repeat2, Gift, Wand2, Lock, Share2, Heart, History, BadgeInfo, Megaphone, } from "lucide-react";
const CLUBS = [ { id: "mci", name: "Manchester City", type: "club", league: "Premier League", region: "Europe", strength: 95, logo: "🔵", colors: "from-sky-500 to-blue-700" }, { id: "liv", name: "Liverpool", type: "club", league: "Premier League", region: "Europe", strength: 92, logo: "🔴", colors: "from-red-500 to-rose-700" }, { id: "ars", name: "Arsenal", type: "club", league: "Premier League", region: "Europe", strength: 90, logo: "🔴", colors: "from-rose-500 to-orange-600" }, { id: "rma", name: "Real Madrid", type: "club", league: "La Liga", region: "Europe", strength: 96, logo: "👑", colors: "from-zinc-100 to-zinc-400" }, { id: "bar", name: "FC Barcelona", type: "club", league: "La Liga", region: "Europe", strength: 94, logo: "🔵", colors: "from-blue-600 to-red-600" }, { id: "atm", name: "Atlético Madrid", type: "club", league: "La Liga", region: "Europe", strength: 89, logo: "⚪", colors: "from-red-700 to-slate-900" }, { id: "juv", name: "Juventus", type: "club", league: "Serie A", region: "Europe", strength: 88, logo: "⚫", colors: "from-slate-900 to-slate-700" }, { id: "mil", name: "AC Milan", type: "club", league: "Serie A", region: "Europe", strength: 89, logo: "🔴", colors: "from-red-600 to-black" }, { id: "int", name: "Inter", type: "club", league: "Serie A", region: "Europe", strength: 91, logo: "🔵", colors: "from-sky-700 to-black" }, { id: "bvb", name: "Borussia Dortmund", type: "club", league: "Bundesliga", region: "Europe", strength: 87, logo: "🟡", colors: "from-yellow-400 to-black" }, { id: "bay", name: "Bayern Munich", type: "club", league: "Bundesliga", region: "Europe", strength: 95, logo: "🔴", colors: "from-red-600 to-rose-800" }, { id: "psg", name: "PSG", type: "club", league: "Ligue 1", region: "Europe", strength: 93, logo: "🔵", colors: "from-blue-800 to-pink-600" }, { id: "mon", name: "Monaco", type: "club", league: "Ligue 1", region: "Europe", strength: 84, logo: "🔴", colors: "from-red-600 to-white" }, { id: "por", name: "FC Porto", type: "club", league: "Primeira Liga", region: "Europe", strength: 85, logo: "🐉", colors: "from-blue-700 to-sky-400" }, { id: "aja", name: "Ajax", type: "club", league: "Eredivisie", region: "Europe", strength: 83, logo: "⭐", colors: "from-red-500 to-white" }, { id: "aln", name: "Al Nassr", type: "club", league: "Saudi Pro League", region: "Asia", strength: 84, logo: "🟡", colors: "from-yellow-400 to-amber-700" }, { id: "hil", name: "Al Hilal", type: "club", league: "Saudi Pro League", region: "Asia", strength: 86, logo: "🔷", colors: "from-sky-500 to-blue-900" }, { id: "Uls", name: "Ulsan HD", type: "club", league: "K League", region: "Asia", strength: 78, logo: "🟦", colors: "from-blue-500 to-slate-800" }, { id: "bfc", name: "Beijing Guoan", type: "club", league: "Chinese Super League", region: "Asia", strength: 76, logo: "🟩", colors: "from-emerald-500 to-green-800" }, { id: "san", name: "Santos", type: "club", league: "Brasileirão", region: "South America", strength: 79, logo: "⚪", colors: "from-white to-black" }, { id: "pal", name: "Palmeiras", type: "club", league: "Brasileirão", region: "South America", strength: 87, logo: "🟢", colors: "from-emerald-500 to-green-900" }, { id: "rio", name: "River Plate", type: "club", league: "Liga Argentina", region: "South America", strength: 88, logo: "🔴", colors: "from-red-600 to-white" }, { id: "boo", name: "Boca Juniors", type: "club", league: "Liga Argentina", region: "South America", strength: 86, logo: "💙", colors: "from-blue-700 to-yellow-400" }, { id: "afr1", name: "Al Ahly", type: "club", league: "Egypt Premier League", region: "Africa", strength: 80, logo: "🟥", colors: "from-red-700 to-amber-700" }, { id: "afr2", name: "Wydad AC", type: "club", league: "Botola Pro", region: "Africa", strength: 77, logo: "⬛", colors: "from-red-800 to-black" }, { id: "gal", name: "Galatasaray", type: "club", league: "Süper Lig", region: "Europe", strength: 84, logo: "🟠", colors: "from-orange-500 to-red-700" }, { id: "new", name: "Newcastle United", type: "club", league: "Premier League", region: "Europe", strength: 83, logo: "⚫", colors: "from-zinc-900 to-emerald-700" }, ];
const NATIONALS = [ { id: "fra", name: "France", type: "national", league: "National Team", region: "Europe", strength: 95, logo: "🇫🇷", colors: "from-blue-700 to-red-600" }, { id: "spa", name: "Spain", type: "national", league: "National Team", region: "Europe", strength: 91, logo: "🇪🇸", colors: "from-red-600 to-yellow-400" }, { id: "ger", name: "Germany", type: "national", league: "National Team", region: "Europe", strength: 90, logo: "🇩🇪", colors: "from-black to-red-600" }, { id: "ita", name: "Italy", type: "national", league: "National Team", region: "Europe", strength: 89, logo: "🇮🇹", colors: "from-green-600 to-blue-600" }, { id: "eng", name: "England", type: "national", league: "National Team", region: "Europe", strength: 90, logo: "🏴", colors: "from-slate-800 to-slate-500" }, { id: "porN", name: "Portugal", type: "national", league: "National Team", region: "Europe", strength: 88, logo: "🇵🇹", colors: "from-emerald-600 to-red-700" }, { id: "jpn", name: "Japan", type: "national", league: "National Team", region: "Asia", strength: 85, logo: "🇯🇵", colors: "from-rose-500 to-white" }, { id: "kor", name: "South Korea", type: "national", league: "National Team", region: "Asia", strength: 84, logo: "🇰🇷", colors: "from-red-600 to-sky-600" }, { id: "sau", name: "Saudi Arabia", type: "national", league: "National Team", region: "Asia", strength: 79, logo: "🇸🇦", colors: "from-green-700 to-emerald-500" }, { id: "aus", name: "Australia", type: "national", league: "National Team", region: "Asia", strength: 78, logo: "🇦🇺", colors: "from-sky-700 to-amber-500" }, { id: "nga", name: "Nigeria", type: "national", league: "National Team", region: "Africa", strength: 82, logo: "🇳🇬", colors: "from-green-700 to-black" }, { id: "mar", name: "Morocco", type: "national", league: "National Team", region: "Africa", strength: 84, logo: "🇲🇦", colors: "from-red-700 to-green-700" }, { id: "gha", name: "Ghana", type: "national", league: "National Team", region: "Africa", strength: 78, logo: "🇬🇭", colors: "from-red-600 to-green-600" }, { id: "sen", name: "Senegal", type: "national", league: "National Team", region: "Africa", strength: 81, logo: "🇸🇳", colors: "from-green-700 to-yellow-400" }, { id: "bra", name: "Brazil", type: "national", league: "National Team", region: "South America", strength: 94, logo: "🇧🇷", colors: "from-emerald-500 to-yellow-400" }, { id: "arg", name: "Argentina", type: "national", league: "National Team", region: "South America", strength: 93, logo: "🇦🇷", colors: "from-sky-500 to-white" }, { id: "uru", name: "Uruguay", type: "national", league: "National Team", region: "South America", strength: 87, logo: "🇺🇾", colors: "from-sky-600 to-white" }, { id: "col", name: "Colombia", type: "national", league: "National Team", region: "South America", strength: 84, logo: "🇨🇴", colors: "from-yellow-400 to-red-600" }, { id: "mex", name: "Mexico", type: "national", league: "National Team", region: "North America", strength: 82, logo: "🇲🇽", colors: "from-green-700 to-red-700" }, { id: "usa", name: "USA", type: "national", league: "National Team", region: "North America", strength: 83, logo: "🇺🇸", colors: "from-blue-700 to-red-700" }, ];
const LEAGUES = [ "Premier League", "La Liga", "Serie A", "Bundesliga", "Ligue 1", "Primeira Liga", "Eredivisie", "Saudi Pro League", "K League", "Chinese Super League", "Brasileirão", "Liga Argentina", "Egypt Premier League", "Botola Pro", "Süper Lig", ];
const NATIONAL_REGIONS = ["Europe", "Asia", "Africa", "North America", "South America"];
const EFFECTS = [ { id: "wheel", label: "Vòng quay may mắn", icon: "🎡" }, { id: "cards", label: "Bốc lá bài", icon: "🃏" }, { id: "gift", label: "Bốc hộp quà", icon: "🎁" }, { id: "flip", label: "Lật thẻ", icon: "🪙" }, ];
const SPECIAL_CARDS = [ { id: "opponent", label: "Chọn đội cho đối phương", icon: <Megaphone className="h-4 w-4" /> }, { id: "swap", label: "Đổi đội", icon: <Repeat2 className="h-4 w-4" /> }, { id: "strongWeak", label: "Chọn đội mạnh/yếu", icon: <Flame className="h-4 w-4" /> }, { id: "reroll", label: "Random lại", icon: <Shuffle className="h-4 w-4" /> }, { id: "extra", label: "Bốc thêm 1 lượt", icon: <Gift className="h-4 w-4" /> }, { id: "banLeague", label: "Cấm chọn một giải đấu", icon: <Lock className="h-4 w-4" /> }, ];
function clamp(min, max, value) { return Math.max(min, Math.min(max, value)); }
function pickRandom(arr) { return arr[Math.floor(Math.random() * arr.length)]; }
function shuffle(arr) { return [...arr].sort(() => Math.random() - 0.5); }
function getDataset(mode) { if (mode === "national") return NATIONALS; if (mode === "favorites") return [...CLUBS, ...NATIONALS]; return CLUBS; }
function normalize(str) { return String(str || "").toLowerCase(); }
function useLocalStorage(key, initialValue) { const [value, setValue] = useState(() => { if (typeof window === "undefined") return initialValue; try { const saved = window.localStorage.getItem(key); return saved ? JSON.parse(saved) : initialValue; } catch { return initialValue; } });
useEffect(() => { try { window.localStorage.setItem(key, JSON.stringify(value)); } catch { // ignore } }, [key, value]);
return [value, setValue]; }
function makeConfetti(count = 48) { return Array.from({ length: count }).map((_, i) => ({ id: ${Date.now()}-${i}, left: Math.random() * 100, delay: Math.random() * 0.35, size: 8 + Math.random() * 12, rotate: Math.random() * 360, duration: 1.8 + Math.random() * 1.8, })); }
function strengthLabel(n) { if (n >= 92) return "Elite"; if (n >= 87) return "Mạnh"; if (n >= 80) return "Tốt"; if (n >= 72) return "Khá"; return "Vui"; }
function playBeep(type = "win") { try { const AudioCtx = window.AudioContext || window.webkitAudioContext; if (!AudioCtx) return; const ctx = new AudioCtx(); const now = ctx.currentTime; const tones = type === "win" ? [392, 523.25, 659.25] : [220, 277.18]; tones.forEach((freq, idx) => { const osc = ctx.createOscillator(); const gain = ctx.createGain(); osc.type = "sine"; osc.frequency.value = freq; gain.gain.value = 0.0001; osc.connect(gain); gain.connect(ctx.destination); osc.start(now + idx * 0.08); gain.gain.exponentialRampToValueAtTime(0.2, now + idx * 0.08 + 0.01); gain.gain.exponentialRampToValueAtTime(0.0001, now + idx * 0.08 + 0.22); osc.stop(now + idx * 0.08 + 0.24); }); setTimeout(() => ctx.close().catch(() => {}), 1000); } catch { // ignore audio errors } }
function Card({ title, icon, children, className = "" }) { return ( <div className={rounded-3xl border border-white/10 bg-white/5 backdrop-blur-xl shadow-2xl shadow-black/20 ${className}}> <div className="flex items-center gap-2 border-b border-white/10 px-4 py-3 text-white/90"> <div className="rounded-2xl bg-white/10 p-2">{icon}</div> <h3 className="font-semibold">{title}</h3> </div> <div className="p-4">{children}</div> </div> ); }
function TeamChip({ team, onToggleFavorite, favorite }) { return ( <div className={rounded-2xl border border-white/10 bg-gradient-to-br ${team.colors} p-[1px]}> <div className="rounded-2xl bg-slate-950/90 p-3 text-white"> <div className="flex items-start justify-between gap-3"> <div> <div className="text-2xl">{team.logo}</div> <div className="mt-2 font-semibold leading-tight">{team.name}</div> <div className="mt-1 text-xs text-white/65">{team.type === "club" ? team.league : team.region} · {strengthLabel(team.strength)}</div> </div> <button onClick={() => onToggleFavorite(team.id)} className={rounded-full p-2 transition ${favorite ? "bg-pink-500/20 text-pink-300" : "bg-white/8 text-white/60 hover:bg-white/12"}} aria-label="favorite" > <Heart className={h-4 w-4 ${favorite ? "fill-pink-400" : ""}} /> </button> </div> </div> </div> ); }
export default function PesRandomizerWebsite() { const [mode, setMode] = useState("2p"); const [effect, setEffect] = useState("wheel"); const [soundOn, setSoundOn] = useState(true); const [partyMode, setPartyMode] = useState(true); const [uniqueTeams, setUniqueTeams] = useState(true); const [selectedLeague, setSelectedLeague] = useState("All"); const [selectedRegion, setSelectedRegion] = useState("All"); const [bannedLeague, setBannedLeague] = useState("None"); const [favoriteIds, setFavoriteIds] = useLocalStorage("pes-favorites", ["mci", "rma", "bar", "fra", "arg", "bra"]); const [history, setHistory] = useLocalStorage("pes-history", []); const [result, setResult] = useState(null); const [confetti, setConfetti] = useState([]); const [animating, setAnimating] = useState(false); const [specialCard, setSpecialCard] = useState(null); const [forcedMode, setForcedMode] = useState(null); const [partyMessage, setPartyMessage] = useState("Sẵn sàng random chưa? Nhấn nút lớn bên dưới!"); const resultRef = useRef(null);
const allTeams = useMemo(() => { if (mode === "national") return NATIONALS; if (mode === "favorites") { const fav = [...CLUBS, ...NATIONALS].filter((t) => favoriteIds.includes(t.id)); return fav.length ? fav : [...CLUBS, ...NATIONALS]; } return CLUBS; }, [mode, favoriteIds]);
const filteredTeams = useMemo(() => { let list = [...allTeams]; if (mode === "club" || mode === "2p" || mode === "party") { if (selectedLeague !== "All") list = list.filter((t) => t.type === "club" && t.league === selectedLeague); if (bannedLeague !== "None") list = list.filter((t) => t.league !== bannedLeague); } if (mode === "national") { if (selectedRegion !== "All") list = list.filter((t) => t.region === selectedRegion); } if (forcedMode === "strong") list = [...list].sort((a, b) => b.strength - a.strength); if (forcedMode === "weak") list = [...list].sort((a, b) => a.strength - b.strength); return list; }, [allTeams, mode, selectedLeague, selectedRegion, bannedLeague, forcedMode]);
const favoriteTeams = useMemo(() => [...CLUBS, ...NATIONALS].filter((t) => favoriteIds.includes(t.id)), [favoriteIds]);
const stats = useMemo(() => { const leagues = new Set(filteredTeams.filter((t) => t.type === "club").map((t) => t.league)); const regions = new Set(filteredTeams.filter((t) => t.type === "national").map((t) => t.region)); return { teams: filteredTeams.length, leagues: leagues.size, regions: regions.size }; }, [filteredTeams]);
useEffect(() => { if (resultRef.current && result) { resultRef.current.scrollIntoView({ behavior: "smooth", block: "center" }); } }, [result]);
function saveHistory(entry) { setHistory((prev) => [entry, ...prev].slice(0, 8)); }
function toggleFavorite(id) { setFavoriteIds((prev) => prev.includes(id) ? prev.filter((x) => x !== id) : [id, ...prev].slice(0, 18) ); }
function announce(text) { setPartyMessage(text); }
function chooseWithConstraints(source, { different = true } = {}) { if (!source.length) return null; const pool = shuffle(source); return pool[0] || null; }
function pickTwoTeams(source) { if (!source.length) return null; const pool = [...source]; if (pool.length === 1) { return uniqueTeams ? null : { a: pool[0], b: pool[0] }; }
const a = pickRandom(pool);
let remaining = pool.filter((t) => t.id !== a.id);
if (!uniqueTeams && Math.random() > 0.7) remaining = [...pool];
const b = remaining.length ? pickRandom(remaining) : null;
return b ? { a, b } : null;
}
function runRandom({ fromSpecial = false } = {}) { const source = filteredTeams; if (!source.length) { setResult({ error: "Không có đội nào phù hợp với bộ lọc hiện tại." }); return; }
setAnimating(true);
setResult(null);
if (soundOn) playBeep("spin");
const delay = effect === "wheel" ? 1050 : effect === "gift" ? 900 : effect === "cards" ? 780 : 680;
setTimeout(() => {
let nextResult;
if (mode === "2p" || mode === "party") {
const picked = pickTwoTeams(source);
if (!picked) {
nextResult = { error: uniqueTeams ? "Danh sách hiện tại không đủ đội để chọn 2 đội khác nhau." : "Không thể random." };
} else {
nextResult = { type: "pair", a: picked.a, b: picked.b };
}
} else if (mode === "national") {
nextResult = { type: "single", a: chooseWithConstraints(source) };
} else if (mode === "favorites") {
const picked = pickTwoTeams(source);
nextResult = picked ? { type: "pair", a: picked.a, b: picked.b } : { error: "Danh sách yêu thích quá ít để random." };
} else {
const picked = pickTwoTeams(source);
nextResult = picked ? { type: "pair", a: picked.a, b: picked.b } : { error: "Không đủ dữ liệu." };
}
if (nextResult?.error) {
setResult(nextResult);
announce(nextResult.error);
setAnimating(false);
return;
}
if (partyMode && mode === "party" && !fromSpecial) {
const partyRoll = Math.random();
if (partyRoll < 0.2) {
setSpecialCard({ id: "opponent", label: "Thẻ đặc biệt: Chọn đội cho đối phương" });
announce("Party mode: đối phương bị chọn đội bất ngờ!");
} else if (partyRoll < 0.35) {
setSpecialCard({ id: "strongWeak", label: "Thẻ đặc biệt: Chọn đội mạnh/yếu" });
} else if (partyRoll < 0.48) {
setSpecialCard({ id: "extra", label: "Thẻ đặc biệt: Bốc thêm 1 lượt" });
} else if (partyRoll < 0.58) {
setSpecialCard({ id: "swap", label: "Thẻ đặc biệt: Đổi đội" });
} else {
setSpecialCard(null);
}
}
if (soundOn) playBeep("win");
const entry = { ...nextResult, mode, effect, createdAt: new Date().toISOString() };
setResult(nextResult);
saveHistory(entry);
setConfetti(makeConfetti(effect === "gift" ? 72 : 54));
announce(
nextResult.type === "pair"
? `${nextResult.a.name} vs ${nextResult.b.name}`
: `Bạn nhận được ${nextResult.a.name}`
);
setAnimating(false);
setTimeout(() => setConfetti([]), 2200);
}, delay);
}
function reroll() { runRandom({ fromSpecial: false }); }
function swapTeams() { if (!result || result.type !== "pair") return; setResult({ ...result, a: result.b, b: result.a }); const swapped = { ...result, a: result.b, b: result.a, createdAt: new Date().toISOString(), mode, effect }; saveHistory(swapped); setConfetti(makeConfetti(34)); if (soundOn) playBeep("win"); }
function chooseStrongWeak(kind) { const source = filteredTeams; if (!source.length) return; const ordered = [...source].sort((a, b) => (kind === "strong" ? b.strength - a.strength : a.strength - b.strength)); const top = ordered.slice(0, Math.max(2, Math.ceil(ordered.length * 0.3))); const picked = pickTwoTeams(top); if (!picked) return; setForcedMode(kind); setResult({ type: "pair", a: picked.a, b: picked.b }); saveHistory({ type: "pair", a: picked.a, b: picked.b, mode, effect, createdAt: new Date().toISOString(), special: kind }); setConfetti(makeConfetti(60)); announce(kind === "strong" ? "Đã chọn cặp đội mạnh!" : "Đã chọn cặp đội yếu hơn cho party mode!"); }
function chooseForOpponent() { const source = filteredTeams; if (!source.length) return; const other = pickRandom(source); setResult({ type: "single", a: other, note: "Đội dành cho đối phương" }); saveHistory({ type: "single", a: other, mode, effect, createdAt: new Date().toISOString(), special: "opponent" }); setConfetti(makeConfetti(44)); }
function extraDraw() { runRandom({ fromSpecial: true }); setTimeout(() => runRandom({ fromSpecial: true }), 250); }
function banLeagueRandom() { setBannedLeague(pickRandom(LEAGUES)); setPartyMessage("Một giải đấu đã bị cấm ngẫu nhiên."); }
function shareResult() { if (!result) return; const text = result.type === "pair" ? PES/eFootball Random: ${result.a.name} vs ${result.b.name} : PES/eFootball Random: ${result.a.name}; if (navigator.share) { navigator.share({ title: "PES Random", text }).catch(() => {}); } else if (navigator.clipboard) { navigator.clipboard.writeText(text).catch(() => {}); ann