This commit is contained in:
gpt-engineer-app[bot]
2025-12-19 16:09:43 +00:00
parent df9dbe5cbb
commit 04cae4fc54
4 changed files with 422 additions and 116 deletions

View File

@@ -5,6 +5,7 @@ import { useCart } from '@/contexts/CartContext';
import { useBranding } from '@/hooks/useBranding'; import { useBranding } from '@/hooks/useBranding';
import { Button } from '@/components/ui/button'; import { Button } from '@/components/ui/button';
import { Sheet, SheetContent, SheetTrigger } from '@/components/ui/sheet'; import { Sheet, SheetContent, SheetTrigger } from '@/components/ui/sheet';
import { Footer } from '@/components/Footer';
import { cn } from '@/lib/utils'; import { cn } from '@/lib/utils';
import { import {
LayoutDashboard, LayoutDashboard,
@@ -101,7 +102,7 @@ export function AppLayout({ children }: AppLayoutProps) {
if (!user) { if (!user) {
// Public layout for non-authenticated pages // Public layout for non-authenticated pages
return ( return (
<div className="min-h-screen bg-background"> <div className="min-h-screen bg-background flex flex-col">
<header className="border-b-2 border-border bg-background sticky top-0 z-50"> <header className="border-b-2 border-border bg-background sticky top-0 z-50">
<div className="container mx-auto px-4 py-4 flex items-center justify-between"> <div className="container mx-auto px-4 py-4 flex items-center justify-between">
<Link to="/" className="text-2xl font-bold flex items-center gap-2"> <Link to="/" className="text-2xl font-bold flex items-center gap-2">
@@ -113,7 +114,7 @@ export function AppLayout({ children }: AppLayoutProps) {
</Link> </Link>
<nav className="flex items-center gap-4"> <nav className="flex items-center gap-4">
<Link to="/products" className="hover:underline font-medium">Produk</Link> <Link to="/products" className="hover:underline font-medium">Produk</Link>
<Link to="/events" className="hover:underline font-medium">Kalender</Link> <Link to="/calendar" className="hover:underline font-medium">Kalender</Link>
<Link to="/auth"> <Link to="/auth">
<Button variant="outline" size="sm" className="border-2"> <Button variant="outline" size="sm" className="border-2">
<User className="w-4 h-4 mr-2" /> <User className="w-4 h-4 mr-2" />
@@ -133,7 +134,8 @@ export function AppLayout({ children }: AppLayoutProps) {
</nav> </nav>
</div> </div>
</header> </header>
<main>{children}</main> <main className="flex-1">{children}</main>
<Footer />
</div> </div>
); );
} }

67
src/components/Footer.tsx Normal file
View File

@@ -0,0 +1,67 @@
import { Link } from 'react-router-dom';
import { useBranding } from '@/hooks/useBranding';
export function Footer() {
const branding = useBranding();
const brandName = branding.brand_name || 'LearnHub';
const currentYear = new Date().getFullYear();
return (
<footer className="border-t-2 border-border bg-muted/50 mt-auto">
<div className="container mx-auto px-4 py-8">
<div className="grid grid-cols-1 md:grid-cols-4 gap-8">
{/* Brand */}
<div className="md:col-span-2">
<h3 className="font-bold text-lg mb-2">{brandName}</h3>
<p className="text-sm text-muted-foreground max-w-md">
{branding.brand_tagline || 'Platform pembelajaran online untuk mengembangkan skill dan karir Anda.'}
</p>
</div>
{/* Quick Links */}
<div>
<h4 className="font-semibold mb-3">Tautan</h4>
<ul className="space-y-2 text-sm">
<li>
<Link to="/products" className="text-muted-foreground hover:text-foreground transition-colors">
Produk
</Link>
</li>
<li>
<Link to="/calendar" className="text-muted-foreground hover:text-foreground transition-colors">
Kalender Event
</Link>
</li>
<li>
<Link to="/consulting" className="text-muted-foreground hover:text-foreground transition-colors">
Konsultasi
</Link>
</li>
</ul>
</div>
{/* Legal */}
<div>
<h4 className="font-semibold mb-3">Legal</h4>
<ul className="space-y-2 text-sm">
<li>
<Link to="/privacy" className="text-muted-foreground hover:text-foreground transition-colors">
Kebijakan Privasi
</Link>
</li>
<li>
<Link to="/terms" className="text-muted-foreground hover:text-foreground transition-colors">
Syarat & Ketentuan
</Link>
</li>
</ul>
</div>
</div>
<div className="border-t border-border mt-8 pt-6 text-center text-sm text-muted-foreground">
© {currentYear} {brandName}. All rights reserved.
</div>
</div>
</footer>
);
}

View File

@@ -137,7 +137,7 @@ export default function Bootcamp() {
}; };
const markAsCompleted = async () => { const markAsCompleted = async () => {
if (!selectedLesson || !user) return; if (!selectedLesson || !user || !product) return;
const { error } = await supabase const { error } = await supabase
.from('lesson_progress') .from('lesson_progress')
@@ -152,7 +152,34 @@ export default function Bootcamp() {
return; return;
} }
setProgress([...progress, { lesson_id: selectedLesson.id, completed_at: new Date().toISOString() }]); const newProgress = [...progress, { lesson_id: selectedLesson.id, completed_at: new Date().toISOString() }];
setProgress(newProgress);
// Calculate completion percentage for notification
const completedCount = newProgress.length;
const completionPercent = Math.round((completedCount / totalLessons) * 100);
// Trigger progress notification at milestones
if (completionPercent === 25 || completionPercent === 50 || completionPercent === 75 || completionPercent === 100) {
try {
await supabase.functions.invoke('send-notification', {
body: {
template_key: 'bootcamp_progress',
recipient_email: user.email,
recipient_name: user.user_metadata?.name || 'Peserta',
variables: {
bootcamp_title: product.title,
progress_percent: completionPercent.toString(),
completed_lessons: completedCount.toString(),
total_lessons: totalLessons.toString(),
},
},
});
} catch (err) {
console.log('Progress notification skipped:', err);
}
}
toast({ title: 'Selesai!', description: 'Pelajaran ditandai selesai' }); toast({ title: 'Selesai!', description: 'Pelajaran ditandai selesai' });
goToNextLesson(); goToNextLesson();
}; };

View File

@@ -10,10 +10,12 @@ import { Badge } from '@/components/ui/badge';
import { Skeleton } from '@/components/ui/skeleton'; import { Skeleton } from '@/components/ui/skeleton';
import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow } from '@/components/ui/table'; import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow } from '@/components/ui/table';
import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogDescription } from '@/components/ui/dialog'; import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogDescription } from '@/components/ui/dialog';
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select';
import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs';
import { toast } from '@/hooks/use-toast'; import { toast } from '@/hooks/use-toast';
import { formatIDR } from '@/lib/format'; import { formatIDR } from '@/lib/format';
import { Video, Calendar, Clock, User, Link as LinkIcon, ExternalLink } from 'lucide-react'; import { Video, Calendar, Clock, User, Link as LinkIcon, ExternalLink, CheckCircle, XCircle, Loader2 } from 'lucide-react';
import { format, parseISO } from 'date-fns'; import { format, parseISO, isToday, isTomorrow, isPast } from 'date-fns';
import { id } from 'date-fns/locale'; import { id } from 'date-fns/locale';
interface ConsultingSlot { interface ConsultingSlot {
@@ -34,6 +36,11 @@ interface ConsultingSlot {
}; };
} }
interface PlatformSettings {
integration_n8n_base_url?: string;
integration_google_calendar_id?: string;
}
const statusLabels: Record<string, { label: string; variant: 'default' | 'secondary' | 'destructive' | 'outline' }> = { const statusLabels: Record<string, { label: string; variant: 'default' | 'secondary' | 'destructive' | 'outline' }> = {
pending_payment: { label: 'Menunggu Pembayaran', variant: 'secondary' }, pending_payment: { label: 'Menunggu Pembayaran', variant: 'secondary' },
confirmed: { label: 'Dikonfirmasi', variant: 'default' }, confirmed: { label: 'Dikonfirmasi', variant: 'default' },
@@ -46,17 +53,23 @@ export default function AdminConsulting() {
const navigate = useNavigate(); const navigate = useNavigate();
const [slots, setSlots] = useState<ConsultingSlot[]>([]); const [slots, setSlots] = useState<ConsultingSlot[]>([]);
const [settings, setSettings] = useState<PlatformSettings>({});
const [loading, setLoading] = useState(true); const [loading, setLoading] = useState(true);
const [selectedSlot, setSelectedSlot] = useState<ConsultingSlot | null>(null); const [selectedSlot, setSelectedSlot] = useState<ConsultingSlot | null>(null);
const [meetLink, setMeetLink] = useState(''); const [meetLink, setMeetLink] = useState('');
const [dialogOpen, setDialogOpen] = useState(false); const [dialogOpen, setDialogOpen] = useState(false);
const [saving, setSaving] = useState(false); const [saving, setSaving] = useState(false);
const [creatingMeet, setCreatingMeet] = useState(false);
const [activeTab, setActiveTab] = useState('upcoming');
useEffect(() => { useEffect(() => {
if (!authLoading) { if (!authLoading) {
if (!user) navigate('/auth'); if (!user) navigate('/auth');
else if (!isAdmin) navigate('/dashboard'); else if (!isAdmin) navigate('/dashboard');
else fetchSlots(); else {
fetchSlots();
fetchSettings();
}
} }
}, [user, isAdmin, authLoading]); }, [user, isAdmin, authLoading]);
@@ -74,6 +87,15 @@ export default function AdminConsulting() {
setLoading(false); setLoading(false);
}; };
const fetchSettings = async () => {
const { data } = await supabase
.from('platform_settings')
.select('integration_n8n_base_url, integration_google_calendar_id')
.single();
if (data) setSettings(data);
};
const openMeetDialog = (slot: ConsultingSlot) => { const openMeetDialog = (slot: ConsultingSlot) => {
setSelectedSlot(slot); setSelectedSlot(slot);
setMeetLink(slot.meet_link || ''); setMeetLink(slot.meet_link || '');
@@ -96,31 +118,98 @@ export default function AdminConsulting() {
setDialogOpen(false); setDialogOpen(false);
fetchSlots(); fetchSlots();
// TODO: Trigger notification with meet link // Send notification to client with meet link
console.log('Would trigger consulting_scheduled notification with meet_link:', meetLink); if (meetLink && selectedSlot.profiles?.email) {
try {
await supabase.functions.invoke('send-notification', {
body: {
template_key: 'consulting_scheduled',
recipient_email: selectedSlot.profiles.email,
recipient_name: selectedSlot.profiles.full_name,
variables: {
consultation_date: format(parseISO(selectedSlot.date), 'd MMMM yyyy', { locale: id }),
consultation_time: `${selectedSlot.start_time.substring(0, 5)} - ${selectedSlot.end_time.substring(0, 5)}`,
meet_link: meetLink,
topic_category: selectedSlot.topic_category,
},
},
});
} catch (err) {
console.log('Notification skipped:', err);
}
}
} }
setSaving(false); setSaving(false);
}; };
const createMeetLink = async () => { const createMeetLink = async () => {
// Placeholder for Google Calendar API integration if (!selectedSlot) return;
const GOOGLE_CLIENT_ID = import.meta.env.VITE_GOOGLE_CLIENT_ID;
if (!GOOGLE_CLIENT_ID) { // Check if n8n webhook is configured
const webhookUrl = settings.integration_n8n_base_url;
if (!webhookUrl) {
toast({ toast({
title: 'Info', title: 'Info',
description: 'VITE_GOOGLE_CLIENT_ID belum dikonfigurasi. Masukkan link Meet secara manual.', description: 'Webhook URL belum dikonfigurasi di Pengaturan Integrasi. Masukkan link Meet secara manual.',
}); });
return; return;
} }
// TODO: Implement actual Google Calendar API call setCreatingMeet(true);
// For now, log what would happen
console.log('Would call Google Calendar API to create Meet link for slot:', selectedSlot); try {
toast({ // Call the webhook to create Google Meet link
title: 'Info', const response = await fetch(`${webhookUrl}/create-meet`, {
description: 'Integrasi Google Calendar akan tersedia setelah konfigurasi OAuth selesai.', method: 'POST',
}); headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({
slot_id: selectedSlot.id,
date: selectedSlot.date,
start_time: selectedSlot.start_time,
end_time: selectedSlot.end_time,
topic: selectedSlot.topic_category,
client_name: selectedSlot.profiles?.full_name || 'Client',
client_email: selectedSlot.profiles?.email,
calendar_id: settings.integration_google_calendar_id,
}),
});
if (!response.ok) {
throw new Error('Webhook request failed');
}
const data = await response.json();
if (data.meet_link) {
setMeetLink(data.meet_link);
toast({ title: 'Berhasil', description: 'Link Google Meet dibuat' });
} else {
throw new Error('No meet_link in response');
}
} catch (error) {
console.error('Error creating meet link:', error);
toast({
title: 'Gagal',
description: 'Gagal membuat link Meet. Pastikan webhook sudah dikonfigurasi dengan benar.',
variant: 'destructive',
});
} finally {
setCreatingMeet(false);
}
};
const updateSlotStatus = async (slotId: string, newStatus: string) => {
const { error } = await supabase
.from('consulting_slots')
.update({ status: newStatus })
.eq('id', slotId);
if (error) {
toast({ title: 'Error', description: error.message, variant: 'destructive' });
} else {
toast({ title: 'Berhasil', description: `Status diubah ke ${statusLabels[newStatus]?.label || newStatus}` });
fetchSlots();
}
}; };
if (authLoading || loading) { if (authLoading || loading) {
@@ -134,8 +223,10 @@ export default function AdminConsulting() {
); );
} }
const confirmedSlots = slots.filter(s => s.status === 'confirmed'); const today = new Date().toISOString().split('T')[0];
const pendingSlots = slots.filter(s => s.status === 'pending_payment'); const upcomingSlots = slots.filter(s => s.date >= today && (s.status === 'confirmed' || s.status === 'pending_payment'));
const pastSlots = slots.filter(s => s.date < today || s.status === 'completed' || s.status === 'cancelled');
const todaySlots = slots.filter(s => isToday(parseISO(s.date)) && s.status === 'confirmed');
return ( return (
<AppLayout> <AppLayout>
@@ -148,113 +239,214 @@ export default function AdminConsulting() {
Kelola jadwal dan link Google Meet untuk sesi konsultasi Kelola jadwal dan link Google Meet untuk sesi konsultasi
</p> </p>
{/* Today's Sessions Alert */}
{todaySlots.length > 0 && (
<Card className="border-2 border-primary bg-primary/5 mb-6">
<CardContent className="py-4">
<h3 className="font-bold flex items-center gap-2">
<Calendar className="w-5 h-5" />
Sesi Hari Ini ({todaySlots.length})
</h3>
<div className="mt-2 space-y-2">
{todaySlots.map(slot => (
<div key={slot.id} className="flex items-center justify-between text-sm">
<span>
{slot.start_time.substring(0, 5)} - {slot.profiles?.full_name || 'N/A'} ({slot.topic_category})
</span>
{slot.meet_link ? (
<a href={slot.meet_link} target="_blank" rel="noopener noreferrer" className="text-primary underline flex items-center gap-1">
<ExternalLink className="w-3 h-3" /> Join
</a>
) : (
<Button size="sm" variant="outline" onClick={() => openMeetDialog(slot)}>
Tambah Link
</Button>
)}
</div>
))}
</div>
</CardContent>
</Card>
)}
{/* Stats */} {/* Stats */}
<div className="grid grid-cols-1 md:grid-cols-3 gap-4 mb-8"> <div className="grid grid-cols-1 md:grid-cols-4 gap-4 mb-8">
<Card className="border-2 border-border"> <Card className="border-2 border-border">
<CardContent className="pt-6"> <CardContent className="pt-6">
<div className="text-2xl font-bold">{confirmedSlots.length}</div> <div className="text-2xl font-bold">{todaySlots.length}</div>
<p className="text-sm text-muted-foreground">Hari Ini</p>
</CardContent>
</Card>
<Card className="border-2 border-border">
<CardContent className="pt-6">
<div className="text-2xl font-bold">{upcomingSlots.filter(s => s.status === 'confirmed').length}</div>
<p className="text-sm text-muted-foreground">Dikonfirmasi</p> <p className="text-sm text-muted-foreground">Dikonfirmasi</p>
</CardContent> </CardContent>
</Card> </Card>
<Card className="border-2 border-border"> <Card className="border-2 border-border">
<CardContent className="pt-6"> <CardContent className="pt-6">
<div className="text-2xl font-bold">{pendingSlots.length}</div> <div className="text-2xl font-bold">{upcomingSlots.filter(s => !s.meet_link && s.status === 'confirmed').length}</div>
<p className="text-sm text-muted-foreground">Menunggu Pembayaran</p> <p className="text-sm text-muted-foreground">Perlu Link Meet</p>
</CardContent> </CardContent>
</Card> </Card>
<Card className="border-2 border-border"> <Card className="border-2 border-border">
<CardContent className="pt-6"> <CardContent className="pt-6">
<div className="text-2xl font-bold"> <div className="text-2xl font-bold">{pastSlots.filter(s => s.status === 'completed').length}</div>
{confirmedSlots.filter(s => !s.meet_link).length} <p className="text-sm text-muted-foreground">Selesai</p>
</div>
<p className="text-sm text-muted-foreground">Perlu Link Meet</p>
</CardContent> </CardContent>
</Card> </Card>
</div> </div>
{/* Slots Table */} {/* Tabs */}
<Card className="border-2 border-border"> <Tabs value={activeTab} onValueChange={setActiveTab}>
<CardHeader> <TabsList className="mb-4">
<CardTitle>Jadwal Konsultasi</CardTitle> <TabsTrigger value="upcoming">Mendatang ({upcomingSlots.length})</TabsTrigger>
<CardDescription>Daftar semua booking konsultasi</CardDescription> <TabsTrigger value="past">Riwayat ({pastSlots.length})</TabsTrigger>
</CardHeader> </TabsList>
<CardContent className="p-0">
<Table> <TabsContent value="upcoming">
<TableHeader> <Card className="border-2 border-border">
<TableRow> <CardContent className="p-0">
<TableHead>Tanggal</TableHead> <Table>
<TableHead>Waktu</TableHead> <TableHeader>
<TableHead>Klien</TableHead> <TableRow>
<TableHead>Kategori</TableHead> <TableHead>Tanggal</TableHead>
<TableHead>Status</TableHead> <TableHead>Waktu</TableHead>
<TableHead>Link Meet</TableHead> <TableHead>Klien</TableHead>
<TableHead className="text-right">Aksi</TableHead> <TableHead>Kategori</TableHead>
</TableRow> <TableHead>Status</TableHead>
</TableHeader> <TableHead>Link Meet</TableHead>
<TableBody> <TableHead className="text-right">Aksi</TableHead>
{slots.map((slot) => ( </TableRow>
<TableRow key={slot.id}> </TableHeader>
<TableCell className="font-medium"> <TableBody>
{format(parseISO(slot.date), 'd MMM yyyy', { locale: id })} {upcomingSlots.map((slot) => (
</TableCell> <TableRow key={slot.id}>
<TableCell> <TableCell className="font-medium">
{slot.start_time.substring(0, 5)} - {slot.end_time.substring(0, 5)} <div>
</TableCell> {format(parseISO(slot.date), 'd MMM yyyy', { locale: id })}
<TableCell> {isToday(parseISO(slot.date)) && <Badge className="ml-2 bg-primary">Hari Ini</Badge>}
<div> {isTomorrow(parseISO(slot.date)) && <Badge className="ml-2 bg-accent">Besok</Badge>}
<p className="font-medium">{slot.profiles?.full_name || '-'}</p> </div>
<p className="text-sm text-muted-foreground">{slot.profiles?.email}</p> </TableCell>
</div> <TableCell>
</TableCell> {slot.start_time.substring(0, 5)} - {slot.end_time.substring(0, 5)}
<TableCell> </TableCell>
<Badge variant="outline">{slot.topic_category}</Badge> <TableCell>
</TableCell> <div>
<TableCell> <p className="font-medium">{slot.profiles?.full_name || '-'}</p>
<Badge variant={statusLabels[slot.status]?.variant || 'secondary'}> <p className="text-sm text-muted-foreground">{slot.profiles?.email}</p>
{statusLabels[slot.status]?.label || slot.status} </div>
</Badge> </TableCell>
</TableCell> <TableCell>
<TableCell> <Badge variant="outline">{slot.topic_category}</Badge>
{slot.meet_link ? ( </TableCell>
<a <TableCell>
href={slot.meet_link} <Badge variant={statusLabels[slot.status]?.variant || 'secondary'}>
target="_blank" {statusLabels[slot.status]?.label || slot.status}
rel="noopener noreferrer" </Badge>
className="text-primary hover:underline flex items-center gap-1" </TableCell>
> <TableCell>
<ExternalLink className="w-4 h-4" /> {slot.meet_link ? (
Buka <a
</a> href={slot.meet_link}
) : ( target="_blank"
<span className="text-muted-foreground">-</span> rel="noopener noreferrer"
)} className="text-primary hover:underline flex items-center gap-1"
</TableCell> >
<TableCell className="text-right"> <ExternalLink className="w-4 h-4" />
{slot.status === 'confirmed' && ( Buka
<Button </a>
variant="outline" ) : (
size="sm" <span className="text-muted-foreground">-</span>
onClick={() => openMeetDialog(slot)} )}
className="border-2" </TableCell>
> <TableCell className="text-right space-x-2">
<LinkIcon className="w-4 h-4 mr-2" /> {slot.status === 'confirmed' && (
{slot.meet_link ? 'Edit Link' : 'Tambah Link'} <>
</Button> <Button
)} variant="outline"
</TableCell> size="sm"
</TableRow> onClick={() => openMeetDialog(slot)}
))} className="border-2"
{slots.length === 0 && ( >
<TableRow> <LinkIcon className="w-4 h-4 mr-1" />
<TableCell colSpan={7} className="text-center py-8 text-muted-foreground"> {slot.meet_link ? 'Edit' : 'Link'}
Belum ada booking konsultasi </Button>
</TableCell> <Button
</TableRow> variant="outline"
)} size="sm"
</TableBody> onClick={() => updateSlotStatus(slot.id, 'completed')}
</Table> className="border-2 text-green-600"
</CardContent> >
</Card> <CheckCircle className="w-4 h-4" />
</Button>
<Button
variant="outline"
size="sm"
onClick={() => updateSlotStatus(slot.id, 'cancelled')}
className="border-2 text-destructive"
>
<XCircle className="w-4 h-4" />
</Button>
</>
)}
</TableCell>
</TableRow>
))}
{upcomingSlots.length === 0 && (
<TableRow>
<TableCell colSpan={7} className="text-center py-8 text-muted-foreground">
Tidak ada jadwal mendatang
</TableCell>
</TableRow>
)}
</TableBody>
</Table>
</CardContent>
</Card>
</TabsContent>
<TabsContent value="past">
<Card className="border-2 border-border">
<CardContent className="p-0">
<Table>
<TableHeader>
<TableRow>
<TableHead>Tanggal</TableHead>
<TableHead>Waktu</TableHead>
<TableHead>Klien</TableHead>
<TableHead>Kategori</TableHead>
<TableHead>Status</TableHead>
</TableRow>
</TableHeader>
<TableBody>
{pastSlots.slice(0, 20).map((slot) => (
<TableRow key={slot.id}>
<TableCell>{format(parseISO(slot.date), 'd MMM yyyy', { locale: id })}</TableCell>
<TableCell>{slot.start_time.substring(0, 5)} - {slot.end_time.substring(0, 5)}</TableCell>
<TableCell>{slot.profiles?.full_name || '-'}</TableCell>
<TableCell><Badge variant="outline">{slot.topic_category}</Badge></TableCell>
<TableCell>
<Badge variant={statusLabels[slot.status]?.variant || 'secondary'}>
{statusLabels[slot.status]?.label || slot.status}
</Badge>
</TableCell>
</TableRow>
))}
{pastSlots.length === 0 && (
<TableRow>
<TableCell colSpan={5} className="text-center py-8 text-muted-foreground">
Belum ada riwayat konsultasi
</TableCell>
</TableRow>
)}
</TableBody>
</Table>
</CardContent>
</Card>
</TabsContent>
</Tabs>
{/* Meet Link Dialog */} {/* Meet Link Dialog */}
<Dialog open={dialogOpen} onOpenChange={setDialogOpen}> <Dialog open={dialogOpen} onOpenChange={setDialogOpen}>
@@ -286,13 +478,31 @@ export default function AdminConsulting() {
</div> </div>
<div className="flex gap-2"> <div className="flex gap-2">
<Button onClick={createMeetLink} variant="outline" className="flex-1 border-2"> <Button
Buat Link Meet onClick={createMeetLink}
variant="outline"
className="flex-1 border-2"
disabled={creatingMeet}
>
{creatingMeet ? (
<>
<Loader2 className="w-4 h-4 mr-2 animate-spin" />
Membuat...
</>
) : (
'Buat Link Meet'
)}
</Button> </Button>
<Button onClick={saveMeetLink} disabled={saving} className="flex-1 shadow-sm"> <Button onClick={saveMeetLink} disabled={saving} className="flex-1 shadow-sm">
{saving ? 'Menyimpan...' : 'Simpan'} {saving ? 'Menyimpan...' : 'Simpan'}
</Button> </Button>
</div> </div>
{!settings.integration_n8n_base_url && (
<p className="text-xs text-muted-foreground text-center">
Tip: Konfigurasi webhook di Pengaturan Integrasi untuk pembuatan otomatis
</p>
)}
</div> </div>
</DialogContent> </DialogContent>
</Dialog> </Dialog>