- Pages: Updated remaining spacing issues (p-6 → p-8, space-y-6 → space-y-8, gap-6 → gap-8) - DashboardPage, DeveloperPage, GearPage, DesignSystemDemoPage, MarketplaceHome - Views: Updated spacing (space-y-6 → space-y-8, gap-6 → gap-8, p-6 → p-8) - FileManagerView, NotificationsView, SocialView, FileDetailsView, UploadView, MarketplaceView, GearView, LiveView, ProfileView - Action 11.5.1.5: Apply direction to all pages - Batch 3 complete (5 pages + 9 views = 14 files)
189 lines
7.1 KiB
TypeScript
189 lines
7.1 KiB
TypeScript
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>
|
|
|
|
<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>
|
|
|
|
{/* 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>
|
|
|
|
{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>
|
|
|
|
{/* 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>
|
|
);
|
|
};
|