Applied the editorial look to legal notice and privacy policy pages. Created consistent grid-based layouts for easier reading and a premium feel.
121 lines
5.2 KiB
TypeScript
121 lines
5.2 KiB
TypeScript
"use client";
|
|
|
|
import React, { useEffect, useState } from "react";
|
|
import Image from "next/image";
|
|
import { motion } from "framer-motion";
|
|
import { Code2, Disc3, Gamepad2, Zap, BookOpen, Quote } from "lucide-react";
|
|
|
|
interface StatusData {
|
|
status: { text: string; color: string; };
|
|
music: { isPlaying: boolean; track: string; artist: string; album: string; albumArt: string; url: string; } | null;
|
|
gaming: { isPlaying: boolean; name: string; image: string | null; state?: string | number; details?: string | number; } | null;
|
|
coding: { isActive: boolean; project?: string; file?: string; language?: string; stats?: { time: string; topLang: string; topProject: string; }; } | null;
|
|
customActivities?: Record<string, any>;
|
|
}
|
|
|
|
export default function ActivityFeed({
|
|
onActivityChange,
|
|
idleQuote
|
|
}: {
|
|
onActivityChange?: (active: boolean) => void;
|
|
idleQuote?: string;
|
|
}) {
|
|
const [data, setData] = useState<StatusData | null>(null);
|
|
const [hasActivity, setHasActivity] = useState(false);
|
|
|
|
useEffect(() => {
|
|
const fetchData = async () => {
|
|
try {
|
|
const res = await fetch("/api/n8n/status");
|
|
if (!res.ok) throw new Error();
|
|
const json = await res.json();
|
|
const activityData = Array.isArray(json) ? json[0] : json;
|
|
|
|
setData(activityData);
|
|
const isActive = Boolean(
|
|
activityData.coding?.isActive ||
|
|
activityData.gaming?.isPlaying ||
|
|
activityData.music?.isPlaying ||
|
|
Object.values(activityData.customActivities || {}).some((act: any) => act?.enabled)
|
|
);
|
|
setHasActivity(isActive);
|
|
onActivityChange?.(isActive);
|
|
} catch (error) {
|
|
setHasActivity(false);
|
|
onActivityChange?.(false);
|
|
}
|
|
};
|
|
|
|
fetchData();
|
|
const interval = setInterval(fetchData, 30000);
|
|
return () => clearInterval(interval);
|
|
}, [onActivityChange]);
|
|
|
|
if (!hasActivity) {
|
|
return (
|
|
<motion.div
|
|
initial={{ opacity: 0 }}
|
|
animate={{ opacity: 1 }}
|
|
className="h-full flex flex-col justify-center space-y-6"
|
|
>
|
|
<div className="w-10 h-10 rounded-full bg-liquid-mint/10 flex items-center justify-center">
|
|
<Quote size={18} className="text-liquid-mint" />
|
|
</div>
|
|
<p className="text-xl md:text-2xl font-light leading-tight text-stone-300 italic">
|
|
“{idleQuote || "Gerade am Planen des nächsten großen Projekts."}”
|
|
</p>
|
|
<div className="flex items-center gap-2 text-[10px] font-black uppercase tracking-widest text-stone-500">
|
|
<span className="w-1.5 h-1.5 rounded-full bg-stone-700" /> Currently Idle
|
|
</div>
|
|
</motion.div>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<div className="space-y-4">
|
|
{data?.coding?.isActive && (
|
|
<motion.div initial={{ opacity: 0, y: 10 }} animate={{ opacity: 1, y: 0 }} className="bg-green-500/10 border border-green-500/20 rounded-2xl p-5">
|
|
<div className="flex items-center gap-3 mb-2">
|
|
<Zap size={14} className="text-green-400 animate-pulse" />
|
|
<span className="text-[10px] font-black uppercase tracking-widest text-green-400">Coding Now</span>
|
|
</div>
|
|
<p className="font-bold text-white text-lg truncate">{data.coding.project}</p>
|
|
<p className="text-xs text-white/50 truncate">{data.coding.file}</p>
|
|
</motion.div>
|
|
)}
|
|
|
|
{data?.gaming?.isPlaying && (
|
|
<motion.div initial={{ opacity: 0, y: 10 }} animate={{ opacity: 1, y: 0 }} className="bg-indigo-500/10 border border-indigo-500/20 rounded-2xl p-5">
|
|
<div className="flex items-center gap-3 mb-3">
|
|
<Gamepad2 size={14} className="text-indigo-400" />
|
|
<span className="text-[10px] font-black uppercase tracking-widest text-indigo-400">Gaming</span>
|
|
</div>
|
|
<div className="flex gap-4">
|
|
{data.gaming.image && <div className="w-12 h-12 rounded-xl overflow-hidden shrink-0 shadow-lg"><img src={data.gaming.image} className="w-full h-full object-cover" /></div>}
|
|
<div className="min-w-0 flex flex-col justify-center">
|
|
<p className="font-bold text-white text-base truncate">{data.gaming.name}</p>
|
|
<p className="text-xs text-white/50 truncate">In Game</p>
|
|
</div>
|
|
</div>
|
|
</motion.div>
|
|
)}
|
|
|
|
{data?.music?.isPlaying && (
|
|
<motion.div initial={{ opacity: 0, y: 10 }} animate={{ opacity: 1, y: 0 }} className="bg-white/5 border border-white/10 rounded-2xl p-5">
|
|
<div className="flex items-center gap-2 mb-3">
|
|
<Disc3 size={14} className="text-green-400 animate-spin-slow" />
|
|
<span className="text-[10px] font-black uppercase tracking-widest text-white/40">Listening</span>
|
|
</div>
|
|
<div className="flex gap-4">
|
|
<div className="w-12 h-12 rounded-xl overflow-hidden shrink-0 shadow-2xl"><img src={data.music.albumArt} className="w-full h-full object-cover" /></div>
|
|
<div className="min-w-0 flex flex-col justify-center">
|
|
<p className="font-bold text-white text-base truncate">{data.music.track}</p>
|
|
<p className="text-xs text-white/50 truncate">{data.music.artist}</p>
|
|
</div>
|
|
</div>
|
|
</motion.div>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|