Files
blackcanyontickets/src/pages/dashboard.astro
dzinesco 425dfc9348 fix: Remove client-side auth redirects causing dashboard flashing
- Removed checkAuth() function and redirects from dashboard.astro
- Removed checkAuth() function and redirects from events/new.astro
- Updated to use Astro.cookies for better SSR compatibility
- Client-side code now focuses on data loading, not authentication
- Server-side unified auth system handles all protection

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-07-12 20:40:11 -06:00

832 lines
38 KiB
Plaintext

---
import Layout from '../layouts/Layout.astro';
import Navigation from '../components/Navigation.astro';
import { verifyAuth } from '../lib/auth';
// Enable server-side rendering for auth checks
export const prerender = false;
// Server-side auth check using cookies for better SSR compatibility
const auth = await verifyAuth(Astro.cookies);
if (!auth) {
return Astro.redirect('/login');
}
---
<Layout title="Dashboard - Black Canyon Tickets">
<style>
@keyframes fadeInUp {
0% {
opacity: 0;
transform: translateY(20px);
}
100% {
opacity: 1;
transform: translateY(0);
}
}
@keyframes slideIn {
0% {
opacity: 0;
transform: translateX(-20px);
}
100% {
opacity: 1;
transform: translateX(0);
}
}
.animate-fadeInUp {
animation: fadeInUp 0.6s ease-out forwards;
}
.animate-slideIn {
animation: slideIn 0.5s ease-out forwards;
}
.hover-float:hover {
transform: translateY(-2px);
}
/* Dark mode glass effects */
[data-theme="dark"] .glass-effect {
background: rgba(15, 23, 42, 0.8);
backdrop-filter: blur(10px);
border: 1px solid rgba(255, 255, 255, 0.1);
}
[data-theme="dark"] .glass-card {
background: rgba(30, 41, 59, 0.7);
backdrop-filter: blur(10px);
border: 1px solid rgba(255, 255, 255, 0.1);
}
[data-theme="dark"] .glass-card-content {
background: rgba(15, 23, 42, 0.9);
backdrop-filter: blur(5px);
padding: 1rem;
border-radius: 0.75rem;
}
/* Light mode solid effects - No glassmorphism */
[data-theme="light"] .glass-effect,
[data-theme="light"] .glass-card {
background: white !important;
backdrop-filter: none !important;
border: none !important;
box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1), 0 2px 4px -1px rgba(0, 0, 0, 0.06) !important;
border-radius: 1rem !important;
}
[data-theme="light"] .glass-card-content {
background: transparent !important;
backdrop-filter: none !important;
padding: 0;
}
[data-theme="light"] .glass-card:hover {
box-shadow: 0 10px 15px -3px rgba(0, 0, 0, 0.1), 0 4px 6px -2px rgba(0, 0, 0, 0.05) !important;
transform: translateY(-1px);
}
/* Remove all blur and transparency in light mode */
[data-theme="light"] * {
backdrop-filter: none !important;
}
[data-theme="light"] .event-card {
background: white !important;
border: 1px solid rgba(0, 0, 0, 0.1) !important;
}
.bg-grid-pattern {
background-image:
linear-gradient(var(--grid-pattern) 1px, transparent 1px),
linear-gradient(90deg, var(--grid-pattern) 1px, transparent 1px);
background-size: 20px 20px;
}
</style>
<div class="min-h-screen" data-theme-background>
<!-- Animated background elements -->
<div class="fixed inset-0 overflow-hidden pointer-events-none">
<div class="absolute -top-40 -right-40 w-80 h-80 rounded-full blur-3xl animate-pulse" style="background: var(--bg-orb-1);"></div>
<div class="absolute -bottom-40 -left-40 w-80 h-80 rounded-full blur-3xl animate-pulse" style="background: var(--bg-orb-2);"></div>
<div class="absolute top-1/2 left-1/2 transform -translate-x-1/2 -translate-y-1/2 w-96 h-96 rounded-full blur-3xl animate-pulse" style="background: var(--bg-orb-3);"></div>
</div>
<!-- Grid pattern is now integrated into bg-static-pattern -->
<Navigation title="Dashboard" />
<main class="relative max-w-7xl mx-auto py-8 sm:px-6 lg:px-8">
<div class="px-4 sm:px-0">
<!-- Dashboard Header -->
<div class="mb-12 animate-slideIn">
<div class="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-6">
<div>
<h1 class="text-4xl md:text-5xl font-light tracking-wide" style="color: var(--glass-text-primary);">Dashboard</h1>
<p class="mt-2 text-lg font-light" style="color: var(--glass-text-secondary);">Manage your events and track performance</p>
</div>
<div class="flex flex-wrap gap-4">
<button
id="toggle-view-btn"
type="button"
class="glass-card px-6 py-3 rounded-xl text-sm font-medium transition-all duration-200 shadow-lg flex items-center gap-2 hover:shadow-xl hover:scale-105"
style="color: var(--glass-text-primary);"
>
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z" />
</svg>
<span id="view-text">Calendar View</span>
</button>
<a
href="/calendar"
class="glass-card px-6 py-3 rounded-xl text-sm font-medium transition-all duration-200 shadow-lg flex items-center gap-2 hover:shadow-xl hover:scale-105 hover:opacity-80"
style="color: var(--glass-text-primary);"
>
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10.325 4.317c.426-1.756 2.924-1.756 3.35 0a1.724 1.724 0 002.573 1.066c1.543-.94 3.31.826 2.37 2.37a1.724 1.724 0 001.065 2.572c1.756.426 1.756 2.924 0 3.35a1.724 1.724 0 00-1.066 2.573c.94 1.543-.826 3.31-2.37 2.37a1.724 1.724 0 00-2.572 1.065c-.426 1.756-2.924 1.756-3.35 0a1.724 1.724 0 00-2.573-1.066c-1.543.94-3.31-.826-2.37-2.37a1.724 1.724 0 00-1.065-2.572c-1.756-.426-1.756-2.924 0-3.35a1.724 1.724 0 001.066-2.573c-.94-1.543.826-3.31 2.37-2.37.996.608 2.296.07 2.572-1.065z" />
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 12a3 3 0 11-6 0 3 3 0 016 0z" />
</svg>
Fee Settings
</a>
<a
href="/onboarding/stripe"
class="btn-premium-gold px-6 py-3 rounded-xl text-sm font-medium transition-all duration-200 shadow-lg flex items-center gap-2 hover:shadow-xl hover:scale-105"
>
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M3 10h18M7 15h1m4 0h1m-7 4h12a3 3 0 003-3V8a3 3 0 00-3-3H6a3 3 0 00-3 3v8a3 3 0 003 3z" />
</svg>
Stripe Setup
</a>
<a
href="/events/new"
class="px-6 py-3 rounded-xl text-sm font-medium transition-all duration-200 shadow-lg flex items-center gap-2 hover:shadow-xl hover:scale-105"
style="background: var(--glass-text-accent); color: white;"
data-light-shadow
>
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 6v6m0 0v6m0-6h6m-6 0H6" />
</svg>
Create Event
</a>
</div>
</div>
</div>
<!-- Quick Stats Cards -->
<div id="stats-cards" class="grid grid-cols-1 md:grid-cols-3 gap-8 mb-12">
<!-- Stats will be populated here -->
</div>
<!-- Calendar View -->
<div id="calendar-view" class="hidden mb-8">
<div id="calendar-container"></div>
</div>
<!-- List View -->
<div id="list-view">
<div class="glass-card rounded-2xl shadow-lg overflow-hidden">
<div class="glass-card-content">
<div class="px-8 py-6 border-b" style="border-color: var(--glass-border);">
<h3 class="text-xl font-light tracking-wide" style="color: var(--glass-text-primary);">Your Events</h3>
</div>
<div class="p-8">
<div id="events-container" class="grid gap-8 md:grid-cols-2 lg:grid-cols-3">
<!-- Events will be loaded here -->
</div>
</div>
</div>
</div>
</div>
<div id="loading" class="text-center py-16">
<div class="glass-card rounded-2xl shadow-lg p-12 max-w-md mx-auto">
<div class="glass-card-content">
<div class="animate-spin rounded-full h-12 w-12 border-2 border-t-transparent mx-auto mb-6" style="border-color: var(--glass-text-accent); border-top-color: transparent;"></div>
<p class="font-light text-lg" style="color: var(--glass-text-secondary);">Loading your events...</p>
</div>
</div>
</div>
<div id="no-events" class="text-center py-16 hidden">
<div class="glass-card rounded-2xl shadow-lg p-16 max-w-lg mx-auto">
<div class="glass-card-content">
<div class="w-20 h-20 rounded-2xl flex items-center justify-center mx-auto mb-6" style="background: var(--glass-text-accent); opacity: 0.2;">
<svg class="h-10 w-10" style="color: var(--glass-text-accent);" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z" />
</svg>
</div>
<h3 class="text-xl font-light mb-3 tracking-wide" style="color: var(--glass-text-primary);">No events yet</h3>
<p class="mb-8 text-lg font-light leading-relaxed" style="color: var(--glass-text-secondary);">Get started by creating your first event and start selling tickets.</p>
<a
href="/events/new"
class="inline-flex items-center gap-2 px-6 py-3 text-white font-medium rounded-xl transition-all duration-200 shadow-lg hover:shadow-xl hover:scale-105"
style="background: var(--glass-text-accent); box-shadow: 0 4px 15px rgba(139, 92, 246, 0.3);"
>
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 6v6m0 0v6m0-6h6m-6 0H6" />
</svg>
Create Your First Event
</a>
</div>
</div>
</div>
</div>
</main>
</div>
</Layout>
<script>
// Theme persistence and initialization
function initializeTheme() {
// Get saved theme or use system preference, fallback to 'dark'
const savedTheme = localStorage.getItem('theme') ||
(window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches ? 'dark' : 'light') ||
'dark';
document.documentElement.setAttribute('data-theme', savedTheme);
document.documentElement.classList.remove('light', 'dark');
document.documentElement.classList.add(savedTheme);
// Store the theme for consistency
localStorage.setItem('theme', savedTheme);
window.__INITIAL_THEME__ = savedTheme;
updateBackground();
}
// Theme-aware background handling
function updateBackground() {
const theme = document.documentElement.getAttribute('data-theme') || 'dark';
const bgElement = document.querySelector('[data-theme-background]');
if (bgElement) {
if (theme === 'light') {
// Light mode gets solid clean background
bgElement.style.background = '#f8fafc';
} else {
// Dark mode gets rich dark background with gradient - use CSS variable
bgElement.style.background = 'var(--bg-gradient)';
}
}
}
// Listen for theme changes from other components
window.addEventListener('themeChanged', (e) => {
const newTheme = e.detail?.theme || document.documentElement.getAttribute('data-theme');
if (newTheme) {
localStorage.setItem('theme', newTheme);
updateBackground();
}
});
// Initialize immediately
initializeTheme();
// Also initialize on DOM ready as fallback
document.addEventListener('DOMContentLoaded', initializeTheme);
</script>
<script>
import { supabase } from '../lib/supabase';
const eventsContainer = document.getElementById('events-container');
const loading = document.getElementById('loading');
const noEvents = document.getElementById('no-events');
const toggleViewBtn = document.getElementById('toggle-view-btn');
const viewText = document.getElementById('view-text');
const calendarView = document.getElementById('calendar-view');
const listView = document.getElementById('list-view');
const calendarContainer = document.getElementById('calendar-container');
const statsCards = document.getElementById('stats-cards');
let currentView = 'list';
let allEvents = [];
// Handle Stripe onboarding completion from URL parameters
function handleOnboardingSuccess() {
const urlParams = new URLSearchParams(window.location.search);
if (urlParams.get('stripe_onboarding') === 'completed') {
// Show success notification
showSuccessNotification('Payment processing setup completed successfully! You can now start accepting payments for your events.');
// Clean up URL
const newUrl = window.location.pathname;
window.history.replaceState({}, document.title, newUrl);
}
}
// Show success notification
function showSuccessNotification(message) {
// Create notification element
const notification = document.createElement('div');
notification.innerHTML = `
<div class="fixed top-4 right-4 px-6 py-4 rounded-lg shadow-lg z-50 max-w-md" style="background: var(--success-color); color: white;">
<div class="flex items-start space-x-3">
<svg class="w-6 h-6 mt-0.5 flex-shrink-0" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zm3.707-9.293a1 1 0 00-1.414-1.414L9 10.586 7.707 9.293a1 1 0 00-1.414 1.414l2 2a1 1 0 001.414 0l4-4z" clip-rule="evenodd" />
</svg>
<div>
<h4 class="font-semibold mb-1">Success!</h4>
<p class="text-sm">${message}</p>
</div>
<button onclick="this.closest('div').parentElement.remove()" class="ml-4 text-green-200 hover:text-white">
<svg class="w-5 h-5" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M4.293 4.293a1 1 0 011.414 0L10 8.586l4.293-4.293a1 1 0 111.414 1.414L11.414 10l4.293 4.293a1 1 0 01-1.414 1.414L10 11.414l-4.293 4.293a1 1 0 01-1.414-1.414L8.586 10 4.293 5.707a1 1 0 010-1.414z" clip-rule="evenodd" />
</svg>
</button>
</div>
</div>
`;
document.body.appendChild(notification.firstElementChild);
// Auto-remove after 8 seconds
setTimeout(() => {
const element = notification.firstElementChild;
if (element && element.parentNode) {
element.remove();
}
}, 8000);
}
// Note: Authentication is now handled server-side by unified auth system
// Load events
async function loadEvents() {
try {
// Get current user (auth already verified server-side)
const { data: { user } } = await supabase.auth.getUser();
if (!user) {
// This shouldn't happen due to server-side auth, but handle gracefully
console.error('No user found despite server-side auth');
loading.innerHTML = `
<div class="rounded-xl p-6 max-w-md mx-auto" style="background: var(--error-bg); border: 1px solid var(--error-border);">
<p class="font-medium" style="color: var(--error-color);">Session error</p>
<p class="text-sm mt-2" style="color: var(--error-color); opacity: 0.8;">Please refresh the page</p>
</div>
`;
return;
}
const { data: userProfile, error: userError } = await supabase
.from('users')
.select('organization_id, role')
.eq('id', user.id)
.single();
if (userError) {
// Error loading user profile
loading.innerHTML = `
<div class="rounded-xl p-6 max-w-md mx-auto" style="background: var(--error-bg); border: 1px solid var(--error-border);">
<p class="font-medium" style="color: var(--error-color);">Error loading user profile</p>
<p class="text-sm mt-2" style="color: var(--error-color); opacity: 0.8;">${userError.message || userError}</p>
</div>
`;
return;
}
// User profile loaded successfully
// Check if user is admin or has organization_id
const isAdmin = userProfile?.role === 'admin';
if (!isAdmin && !userProfile?.organization_id) {
// User has no organization_id and is not admin, showing no events
loading.classList.add('hidden');
noEvents.classList.remove('hidden');
return;
}
// Load events based on user permissions
let query = supabase.from('events').select('*');
if (!isAdmin) {
// Regular users see only their organization's events
query = query.eq('organization_id', userProfile.organization_id);
}
// Admins see all events (no filter)
const { data: events, error } = await query.order('created_at', { ascending: false });
if (error) {
// Error loading events from database
throw error;
}
// Successfully loaded events
allEvents = events || [];
loading.classList.add('hidden');
if (allEvents.length === 0) {
noEvents.classList.remove('hidden');
return;
}
renderStatsCards();
renderListView();
if (currentView === 'calendar') {
renderCalendarView();
}
} catch (error) {
// Error loading events
loading.innerHTML = `
<div class="rounded-xl p-6 max-w-md mx-auto" style="background: var(--error-bg); border: 1px solid var(--error-border);">
<p class="font-medium" style="color: var(--error-color);">Error loading events</p>
<p class="text-sm mt-2" style="color: var(--error-color); opacity: 0.8;">${error.message || error}</p>
</div>
`;
}
}
// Render stats cards
function renderStatsCards() {
const totalEvents = allEvents.length;
const upcomingEvents = allEvents.filter(event => new Date(event.start_time) > new Date()).length;
const pastEvents = totalEvents - upcomingEvents;
statsCards.innerHTML = `
<div class="glass-card rounded-2xl shadow-lg p-8 premium-hover cursor-pointer group">
<div class="glass-card-content">
<div class="flex items-center justify-between">
<div>
<p class="text-sm font-semibold uppercase tracking-wider" style="color: var(--glass-text-secondary);">Total Events</p>
<p class="text-3xl font-light mt-2 transition-colors animate-countUp" style="color: var(--glass-text-primary);">${totalEvents}</p>
<div class="flex items-center mt-1">
<span class="text-xs flex items-center" style="color: var(--glass-text-tertiary);">
<svg class="w-3 h-3 mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z" />
</svg>
All time
</span>
</div>
</div>
<div class="w-16 h-16 rounded-2xl flex items-center justify-center shadow-lg group-hover:shadow-xl group-hover:scale-110 transition-all duration-300" style="background: var(--glass-text-accent); opacity: 0.15;">
<svg class="w-8 h-8" style="color: white;" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z" />
</svg>
</div>
</div>
</div>
</div>
<div class="glass-card rounded-2xl shadow-lg p-8 premium-hover cursor-pointer group" style="--card-accent: var(--success-color);">
<div class="glass-card-content">
<div class="flex items-center justify-between">
<div>
<p class="text-sm font-semibold uppercase tracking-wider" style="color: var(--success-color);">Upcoming Events</p>
<p class="text-3xl font-light mt-2 transition-colors animate-countUp" style="color: var(--success-color);">${upcomingEvents}</p>
<div class="flex items-center mt-1">
<span class="text-xs flex items-center" style="color: var(--success-color); opacity: 0.8;">
<svg class="w-3 h-3 mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M13 7h8m0 0v8m0-8l-8 8-4-4-6 6" />
</svg>
Ready to launch
</span>
</div>
</div>
<div class="w-16 h-16 rounded-2xl flex items-center justify-center shadow-lg group-hover:shadow-xl group-hover:scale-110 transition-all duration-300" style="background: var(--success-color); opacity: 0.15;">
<svg class="w-8 h-8" style="color: white;" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z" />
</svg>
</div>
</div>
</div>
</div>
<div class="glass-card rounded-2xl shadow-lg p-8 premium-hover cursor-pointer group" style="--card-accent: var(--premium-gold);">
<div class="glass-card-content">
<div class="flex items-center justify-between">
<div>
<p class="text-sm font-semibold uppercase tracking-wider" style="color: var(--premium-gold);">Past Events</p>
<p class="text-3xl font-light mt-2 transition-colors animate-countUp" style="color: var(--premium-gold);">${pastEvents}</p>
<div class="flex items-center mt-1">
<span class="text-xs flex items-center" style="color: var(--premium-gold); opacity: 0.8;">
<svg class="w-3 h-3 mr-1" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z" />
</svg>
Completed
</span>
</div>
</div>
<div class="w-16 h-16 rounded-2xl flex items-center justify-center shadow-lg group-hover:shadow-xl group-hover:scale-110 transition-all duration-300" style="background: var(--premium-gold); opacity: 0.15;">
<svg class="w-8 h-8" style="color: white;" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z" />
</svg>
</div>
</div>
</div>
</div>
`;
}
// Render list view
function renderListView() {
eventsContainer.innerHTML = allEvents.map((event, index) => {
const eventDate = new Date(event.start_time);
const isUpcoming = eventDate > new Date();
const formattedDate = eventDate.toLocaleDateString('en-US', {
weekday: 'short',
year: 'numeric',
month: 'short',
day: 'numeric'
});
const formattedTime = eventDate.toLocaleTimeString('en-US', {
hour: 'numeric',
minute: '2-digit'
});
const animationDelay = index * 0.1;
return `
<div class="glass-card event-card rounded-2xl shadow-lg overflow-hidden group hover:shadow-lg hover:translate-y-px transition-all duration-200">
<div class="glass-card-content">
<div class="p-8">
<div class="flex items-start justify-between mb-6">
<div class="flex-1">
<h3 class="text-xl font-light mb-4 tracking-wide" style="color: var(--glass-text-primary);">${event.title}</h3>
<div class="flex items-center text-sm mb-3" style="color: var(--glass-text-secondary);">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M17.657 16.657L13.414 20.9a1.998 1.998 0 01-2.827 0l-4.244-4.243a8 8 0 1111.314 0z" />
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 11a3 3 0 11-6 0 3 3 0 016 0z" />
</svg>
${event.venue}
</div>
<div class="flex items-center text-sm mb-6" style="color: var(--glass-text-secondary);">
<svg class="w-4 h-4 mr-2" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z" />
</svg>
${formattedDate} at ${formattedTime}
</div>
</div>
<div class="flex items-start ml-4">
<span class="px-3 py-1 text-xs font-semibold rounded-full transition-all duration-200 ease-out" style="${isUpcoming ? 'background: var(--success-bg); color: var(--success-color); border: 1px solid var(--success-border);' : 'background: var(--glass-bg); color: var(--glass-text-tertiary); border: 1px solid var(--glass-border);'}">
${isUpcoming ? 'Upcoming' : 'Past'}
</span>
</div>
</div>
${event.description ? `<p class="text-sm mb-6 leading-relaxed group-hover:opacity-80 transition-colors" style="color: var(--glass-text-tertiary);">${event.description}</p>` : ''}
<div class="flex items-center justify-between pt-6 border-t" style="border-color: var(--glass-border);">
<div class="flex space-x-3">
<a
href="/events/${event.id}/manage"
class="inline-flex items-center gap-2 px-4 py-2 rounded-xl text-sm font-medium transition-all duration-200 shadow-lg hover:shadow-lg hover:-translate-y-0.5 hover:scale-105"
style="background: var(--glass-text-accent); color: white;"
>
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10.325 4.317c.426-1.756 2.924-1.756 3.35 0a1.724 1.724 0 002.573 1.066c1.543-.94 3.31.826 2.37 2.37a1.724 1.724 0 001.065 2.572c1.756.426 1.756 2.924 0 3.35a1.724 1.724 0 00-1.066 2.573c.94 1.543-.826 3.31-2.37 2.37a1.724 1.724 0 00-2.572 1.065c-.426 1.756-2.924 1.756-3.35 0a1.724 1.724 0 00-2.573-1.066c-1.543.94-3.31-.826-2.37-2.37a1.724 1.724 0 00-1.065-2.572c-1.756-.426-1.756-2.924 0-3.35a1.724 1.724 0 001.066-2.573c-.94-1.543.826-3.31 2.37-2.37.996.608 2.296.07 2.572-1.065z" />
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 12a3 3 0 11-6 0 3 3 0 016 0z" />
</svg>
Manage
</a>
<a
href="/e/${event.slug}"
class="glass-card inline-flex items-center gap-2 px-4 py-2 rounded-xl text-sm font-medium transition-all duration-200 shadow-lg hover:shadow-lg hover:-translate-y-0.5 hover:scale-105"
style="color: var(--glass-text-primary);"
target="_blank"
>
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10 6H6a2 2 0 00-2 2v10a2 2 0 002 2h10a2 2 0 002-2v-4M14 4h6m0 0v6m0-6L10 14" />
</svg>
Preview
</a>
</div>
<div class="text-xs font-medium" style="color: var(--glass-text-tertiary);">
Created ${new Date(event.created_at).toLocaleDateString()}
</div>
</div>
</div>
</div>
</div>
`;
}).join('');
}
// Render calendar view
function renderCalendarView() {
// Initialize calendar with current date or previously selected date
calendarContainer.innerHTML = createSimpleCalendar(allEvents, currentCalendarYear, currentCalendarMonth);
}
// Enhanced calendar HTML generator with mobile responsiveness
function createSimpleCalendar(events, year, month) {
const monthNames = [
'January', 'February', 'March', 'April', 'May', 'June',
'July', 'August', 'September', 'October', 'November', 'December'
];
const daysInMonth = new Date(year, month + 1, 0).getDate();
const firstDay = new Date(year, month, 1).getDay();
let html = `
<div class="glass-card rounded-2xl shadow-lg overflow-hidden">
<div class="glass-card-content">
<!-- Calendar Header -->
<div class="px-8 py-6 border-b" style="border-color: var(--glass-border);">
<div class="flex items-center justify-between">
<h3 class="text-2xl font-light tracking-wide" style="color: var(--glass-text-primary);">${monthNames[month]} ${year}</h3>
<div class="flex gap-3">
<button onclick="navigateMonth(-1)" class="glass-card p-3 rounded-xl transition-all duration-200 shadow-lg hover:shadow-xl hover:scale-105" style="color: var(--glass-text-secondary);">
<svg class="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 19l-7-7 7-7" />
</svg>
</button>
<button onclick="navigateMonth(1)" class="glass-card p-3 rounded-xl transition-all duration-200 shadow-lg hover:shadow-xl hover:scale-105" style="color: var(--glass-text-secondary);">
<svg class="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 5l7 7-7 7" />
</svg>
</button>
</div>
</div>
</div>
<!-- Calendar Body -->
<div class="p-4 sm:p-6">
<!-- Day Headers -->
<div class="grid grid-cols-7 gap-1 mb-4">
<div class="text-center text-xs sm:text-sm font-semibold py-2" style="color: var(--glass-text-secondary);">Sun</div>
<div class="text-center text-xs sm:text-sm font-semibold py-2" style="color: var(--glass-text-secondary);">Mon</div>
<div class="text-center text-xs sm:text-sm font-semibold py-2" style="color: var(--glass-text-secondary);">Tue</div>
<div class="text-center text-xs sm:text-sm font-semibold py-2" style="color: var(--glass-text-secondary);">Wed</div>
<div class="text-center text-xs sm:text-sm font-semibold py-2" style="color: var(--glass-text-secondary);">Thu</div>
<div class="text-center text-xs sm:text-sm font-semibold py-2" style="color: var(--glass-text-secondary);">Fri</div>
<div class="text-center text-xs sm:text-sm font-semibold py-2" style="color: var(--glass-text-secondary);">Sat</div>
</div>
<!-- Calendar Grid -->
<div class="grid grid-cols-7 gap-1 sm:gap-2">
`;
// Empty cells for days before month starts
for (let i = 0; i < firstDay; i++) {
html += '<div class="aspect-square"></div>';
}
// Days of the month
for (let day = 1; day <= daysInMonth; day++) {
const dayEvents = events.filter(event => {
const eventDate = new Date(event.start_time);
return eventDate.getDate() === day &&
eventDate.getMonth() === month &&
eventDate.getFullYear() === year;
});
const isToday = new Date().toDateString() === new Date(year, month, day).toDateString();
const hasEvents = dayEvents.length > 0;
html += `
<div class="aspect-square rounded-lg p-1 sm:p-2 transition-colors cursor-pointer ${
isToday ? 'ring-2' : ''
}" style="${
isToday ? 'background: rgba(96, 165, 250, 0.2); border: 1px solid rgb(96, 165, 250); ring-color: rgba(96, 165, 250, 0.3); color: rgb(96, 165, 250);' :
hasEvents ? 'border: 1px solid var(--glass-border); background: var(--glass-bg);' : 'border: 1px solid var(--glass-border);'
}" onclick="showDayEvents(${year}, ${month}, ${day})">
<div class="h-full flex flex-col">
<div class="text-xs sm:text-sm font-semibold mb-1" style="color: ${
isToday ? 'rgb(96, 165, 250)' : 'var(--glass-text-primary)'
};">${day}</div>
<!-- Mobile: Show event dots -->
<div class="flex-1 sm:hidden">
${dayEvents.length > 0 ? `
<div class="flex flex-wrap gap-0.5">
${dayEvents.slice(0, 3).map(() => `
<div class="w-1.5 h-1.5 rounded-full" style="background: var(--glass-text-accent);"></div>
`).join('')}
${dayEvents.length > 3 ? '<div class="text-xs" style="color: var(--glass-text-tertiary);">+</div>' : ''}
</div>
` : ''}
</div>
<!-- Desktop: Show event titles -->
<div class="hidden sm:block flex-1 space-y-1 overflow-hidden">
${dayEvents.slice(0, 2).map(event => `
<div class="text-xs rounded px-1.5 py-0.5 truncate font-medium"
style="background: var(--glass-bg-button); color: var(--glass-text-accent);"
title="${event.title} at ${event.venue}">
${event.title}
</div>
`).join('')}
${dayEvents.length > 2 ? `<div class="text-xs font-medium" style="color: var(--glass-text-tertiary);">+${dayEvents.length - 2} more</div>` : ''}
</div>
</div>
</div>
`;
}
html += `
</div>
</div>
</div>
</div>
</div>
<!-- Mobile Event List (shows when day is clicked) -->
<div id="mobile-day-events" class="hidden sm:hidden mt-4 glass-card rounded-xl shadow-lg p-4">
<div class="glass-card-content">
<div class="flex items-center justify-between mb-3">
<h4 id="selected-day-title" class="font-semibold" style="color: var(--glass-text-primary);"></h4>
<button onclick="hideMobileDayEvents()" class="glass-card p-1 rounded" style="color: var(--glass-text-secondary);">
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12" />
</svg>
</button>
</div>
<div id="selected-day-events" class="space-y-2">
<!-- Events will be populated here -->
</div>
</div>
</div>
`;
return html;
}
// Calendar navigation
let currentCalendarYear = new Date().getFullYear();
let currentCalendarMonth = new Date().getMonth();
window.navigateMonth = function(direction) {
currentCalendarMonth += direction;
if (currentCalendarMonth < 0) {
currentCalendarMonth = 11;
currentCalendarYear--;
} else if (currentCalendarMonth > 11) {
currentCalendarMonth = 0;
currentCalendarYear++;
}
calendarContainer.innerHTML = createSimpleCalendar(allEvents, currentCalendarYear, currentCalendarMonth);
};
// Show day events on mobile
window.showDayEvents = function(year, month, day) {
const dayEvents = allEvents.filter(event => {
const eventDate = new Date(event.start_time);
return eventDate.getDate() === day &&
eventDate.getMonth() === month &&
eventDate.getFullYear() === year;
});
if (dayEvents.length === 0) return;
const monthNames = [
'January', 'February', 'March', 'April', 'May', 'June',
'July', 'August', 'September', 'October', 'November', 'December'
];
const mobileDayEvents = document.getElementById('mobile-day-events');
const selectedDayTitle = document.getElementById('selected-day-title');
const selectedDayEventsContainer = document.getElementById('selected-day-events');
selectedDayTitle.textContent = `${monthNames[month]} ${day}, ${year}`;
selectedDayEventsContainer.innerHTML = dayEvents.map(event => `
<div class="glass-card flex items-start gap-3 p-3 rounded-lg">
<div class="glass-card-content">
<div class="flex items-start gap-3">
<div class="w-2 h-2 rounded-full mt-2 flex-shrink-0" style="background: var(--glass-text-accent);"></div>
<div class="flex-1 min-w-0">
<h5 class="font-medium text-sm" style="color: var(--glass-text-primary);">${event.title}</h5>
<p class="text-xs mt-1" style="color: var(--glass-text-secondary);">${event.venue}</p>
<p class="text-xs mt-1" style="color: var(--glass-text-tertiary);">${new Date(event.start_time).toLocaleTimeString([], {hour: '2-digit', minute:'2-digit'})}</p>
</div>
</div>
</div>
</div>
`).join('');
mobileDayEvents.classList.remove('hidden');
};
window.hideMobileDayEvents = function() {
document.getElementById('mobile-day-events').classList.add('hidden');
};
// Toggle between views
function toggleView() {
if (currentView === 'list') {
currentView = 'calendar';
viewText.textContent = 'List View';
calendarView.classList.remove('hidden');
listView.classList.add('hidden');
renderCalendarView();
} else {
currentView = 'list';
viewText.textContent = 'Calendar View';
calendarView.classList.add('hidden');
listView.classList.remove('hidden');
}
}
// Event listeners
toggleViewBtn.addEventListener('click', toggleView);
// Initialize
// Handle onboarding success on page load
handleOnboardingSuccess();
// Load events directly (auth already verified server-side)
loadEvents();
</script>