veza/apps/web/src/components/views/SocialView.tsx

190 lines
7.1 KiB
TypeScript
Raw Normal View History

import React, { useState, useEffect } from 'react';
import { Card } from '../ui/card';
import { Button } from '../ui/button';
import { TrendingUp, Users, Hash, MoreHorizontal, Play } from 'lucide-react';
import { trackService } from '@/services/trackService';
import { useToast } from '@/context/ToastContext';
import { useAudio } from '@/context/AudioContext';
import { Track } from '@/types/api';
import { logger } from '@/utils/logger';
interface SocialViewProps {
onViewProfile: (userId: string | null) => void;
}
export const SocialView: React.FC<SocialViewProps> = ({ onViewProfile }) => {
const { addToast: _addToast } = useToast();
const { playTrack } = useAudio();
const [activeTab, setActiveTab] = useState('feed');
const [feedTracks, setFeedTracks] = useState<Track[]>([]);
const [loading, setLoading] = useState(true);
useEffect(() => {
const loadFeed = async () => {
setLoading(true);
try {
// Using recent tracks as the "Feed"
const res = await trackService.list({
limit: 10,
sort_by: 'created_at',
});
setFeedTracks(res.tracks);
} catch (e) {
logger.error('Error loading feed tracks', {
error: e instanceof Error ? e.message : String(e),
stack: e instanceof Error ? e.stack : undefined,
});
} finally {
setLoading(false);
}
};
loadFeed();
}, []);
return (
<div className="grid grid-cols-1 lg:grid-cols-12 gap-8 animate-fadeIn pb-20">
{/* Sidebar */}
<div className="hidden lg:block lg:col-span-3 space-y-8">
<Card variant="glass" className="p-0 overflow-hidden">
<div className="h-20 bg-gradient-gaming"></div>
<div className="px-4 pb-4">
<div
className="relative -mt-10 mb-3 cursor-pointer"
onClick={() => onViewProfile(null)}
>
<div className="w-20 h-20 rounded-full border-4 border-kodo-graphite overflow-hidden bg-black">
<img
src="https://picsum.photos/id/237/200/200"
className="w-full h-full object-cover"
/>
</div>
</div>
<h3 className="font-bold text-white text-lg">My Profile</h3>
<p className="text-sm text-kodo-content-dim mb-4">View your stats</p>
</div>
</Card>
2026-01-07 18:39:21 +00:00
<Card variant="default" className="p-2">
<nav className="space-y-1">
<Button
variant={activeTab === 'feed' ? 'outline' : 'ghost'}
size="sm"
className="w-full justify-start"
onClick={() => setActiveTab('feed')}
>
<TrendingUp className="w-4 h-4" /> Fresh Tracks
</Button>
<Button variant="ghost" size="sm" className="w-full justify-start">
<Users className="w-4 h-4" /> Communities
</Button>
</nav>
</Card>
</div>
2026-01-07 18:39:21 +00:00
{/* Feed Content */}
<div className="col-span-1 lg:col-span-6 space-y-8">
<div className="mb-4">
<h2 className="text-2xl font-bold text-white mb-1">Community Feed</h2>
<p className="text-kodo-content-dim text-xs">New uploads from the network</p>
</div>
2026-01-07 18:39:21 +00:00
{loading ? (
<div className="text-center py-12">Loading feed...</div>
) : (
feedTracks.map((track) => (
<Card
key={track.id}
variant="default"
className="p-0 overflow-hidden mb-4 border-transparent hover:border-kodo-steel/50"
>
<div className="p-4 flex items-center gap-4">
<div className="w-10 h-10 rounded-full bg-kodo-steel overflow-hidden">
<img
src={track.coverUrl}
className="w-full h-full object-cover"
/>
</div>
<div>
<div className="font-bold text-white text-sm">
{track.artist}
</div>
<div className="text-xs text-kodo-content-dim">
uploaded a new track
</div>
</div>
<Button variant="ghost" size="sm" className="ml-auto">
<MoreHorizontal className="w-4 h-4" />
</Button>
</div>
{/* Track Embed Look */}
<div className="px-4 pb-4">
<div
className="bg-kodo-ink p-4 rounded-xl flex items-center gap-4 border border-kodo-steel group cursor-pointer"
onClick={() => playTrack(track)}
>
<div className="w-16 h-16 rounded overflow-hidden relative">
<img
src={track.coverUrl}
className="w-full h-full object-cover"
/>
<div className="absolute inset-0 bg-black/30 flex items-center justify-center opacity-0 group-hover:opacity-100 transition-opacity">
<Play className="w-6 h-6 text-white fill-current" />
</div>
</div>
<div className="flex-1">
<h4 className="font-bold text-white">{track.title}</h4>
<p className="text-xs text-kodo-content-dim">
{track.genre || 'Electronic'}
</p>
</div>
<div className="text-xs text-kodo-content-dim font-mono pr-2">
{track.duration}
</div>
</div>
</div>
<div className="px-4 py-4 border-t border-kodo-steel/30 flex gap-4 text-xs text-kodo-content-dim">
<Button variant="ghost" size="sm" className="text-xs h-auto py-0">
Like ({track.like_count})
</Button>
<Button variant="ghost" size="sm" className="text-xs h-auto py-0">
Comment
</Button>
<Button variant="ghost" size="sm" className="text-xs h-auto py-0">
Share
</Button>
</div>
</Card>
))
)}
{feedTracks.length === 0 && !loading && (
<div className="text-center py-24 text-kodo-content-dim">
No recent activity.
</div>
)}
</div>
2026-01-07 18:39:21 +00:00
{/* Right Sidebar */}
<div className="hidden lg:block lg:col-span-3 space-y-8">
<Card variant="manga">
<h3 className="font-bold text-sm text-kodo-text-main uppercase tracking-wider mb-4 flex items-center gap-2">
<Hash className="w-4 h-4 text-kodo-magenta" /> Trending Tags
</h3>
<div className="flex flex-wrap gap-2">
{['#Techno', '#Synthwave', '#NewGear', '#Tutorial'].map((t) => (
<span
key={t}
className="text-xs bg-black/20 px-2 py-1 rounded text-kodo-content-dim cursor-pointer hover:text-white hover:bg-black/40"
>
{t}
</span>
))}
</div>
</Card>
</div>
</div>
);
};