Files
Admin_dash/components/widgets/GlobeCard.tsx
Akshay Kolli 3f72118348 Initial commit
2026-02-08 02:32:45 -05:00

71 lines
2.5 KiB
TypeScript

'use client';
import createGlobe from 'cobe';
import { useEffect, useRef } from 'react';
import { Globe } from 'lucide-react';
export function GlobeCard() {
const canvasRef = useRef<HTMLCanvasElement>(null);
useEffect(() => {
let phi = 0;
if (!canvasRef.current) return;
const globe = createGlobe(canvasRef.current, {
devicePixelRatio: 2,
width: 600 * 2,
height: 600 * 2,
phi: 0,
theta: 0,
dark: 1,
diffuse: 1.2,
mapSamples: 16000,
mapBrightness: 6,
baseColor: [0.3, 0.3, 0.3],
markerColor: [0.1, 0.8, 1],
glowColor: [0.1, 0.1, 0.2],
markers: [
// Mock data points
{ location: [40.7128, -74.0060], size: 0.05 }, // NY
{ location: [51.5074, -0.1278], size: 0.05 }, // London
{ location: [35.6762, 139.6503], size: 0.05 }, // Tokyo
{ location: [22.3193, 114.1694], size: 0.05 }, // HK
{ location: [-33.8688, 151.2093], size: 0.05 }, // Sydney
],
onRender: (state) => {
// Called on every animation frame.
// `state` will be an empty object, return updated params.
state.phi = phi;
phi += 0.01;
},
});
return () => {
globe.destroy();
};
}, []);
return (
<div className="col-span-1 md:col-span-2 lg:col-span-2 row-span-2 bg-neutral-900 border border-neutral-800 rounded-xl relative overflow-hidden group hover:border-neutral-700 transition-colors">
<div className="absolute top-6 left-6 z-10 pointer-events-none">
<div className="flex items-center gap-2 text-neutral-400">
<Globe size={18} />
<span className="text-sm font-medium">Active Visitors</span>
</div>
<div className="mt-1">
<span className="text-2xl font-bold text-white tracking-tight">1,240</span>
<span className="text-xs text-neutral-500 ml-2 font-mono">LIVE</span>
</div>
</div>
<div className="absolute inset-0 flex items-center justify-center opacity-80 mt-10">
<canvas
ref={canvasRef}
style={{ width: 600, height: 600, maxWidth: '100%', aspectRatio: 1 }}
/>
</div>
</div>
);
}