dev #13

Merged
Michilis merged 2 commits from dev into main 2026-02-19 02:23:19 +00:00
11 changed files with 48 additions and 16 deletions
Showing only changes of commit bbfaa1172a - Show all commits

View File

@@ -75,7 +75,7 @@ export const sqliteEvents = sqliteTable('events', {
price: real('price').notNull().default(0), price: real('price').notNull().default(0),
currency: text('currency').notNull().default('PYG'), currency: text('currency').notNull().default('PYG'),
capacity: integer('capacity').notNull().default(50), capacity: integer('capacity').notNull().default(50),
status: text('status', { enum: ['draft', 'published', 'cancelled', 'completed', 'archived'] }).notNull().default('draft'), status: text('status', { enum: ['draft', 'published', 'unlisted', 'cancelled', 'completed', 'archived'] }).notNull().default('draft'),
bannerUrl: text('banner_url'), bannerUrl: text('banner_url'),
externalBookingEnabled: integer('external_booking_enabled', { mode: 'boolean' }).notNull().default(false), externalBookingEnabled: integer('external_booking_enabled', { mode: 'boolean' }).notNull().default(false),
externalBookingUrl: text('external_booking_url'), externalBookingUrl: text('external_booking_url'),

View File

@@ -75,7 +75,7 @@ const baseEventSchema = z.object({
price: z.union([z.number(), z.string()]).transform(parsePrice).pipe(z.number().min(0)).default(0), price: z.union([z.number(), z.string()]).transform(parsePrice).pipe(z.number().min(0)).default(0),
currency: z.string().default('PYG'), currency: z.string().default('PYG'),
capacity: z.union([z.number(), z.string()]).transform((val) => typeof val === 'string' ? parseInt(val, 10) || 50 : val).pipe(z.number().min(1)).default(50), capacity: z.union([z.number(), z.string()]).transform((val) => typeof val === 'string' ? parseInt(val, 10) || 50 : val).pipe(z.number().min(1)).default(50),
status: z.enum(['draft', 'published', 'cancelled', 'completed', 'archived']).default('draft'), status: z.enum(['draft', 'published', 'unlisted', 'cancelled', 'completed', 'archived']).default('draft'),
// Accept relative paths (/uploads/...) or full URLs // Accept relative paths (/uploads/...) or full URLs
bannerUrl: z.string().optional().nullable().or(z.literal('')), bannerUrl: z.string().optional().nullable().or(z.literal('')),
// External booking support - accept boolean or number (0/1 from DB) // External booking support - accept boolean or number (0/1 from DB)

View File

@@ -69,7 +69,7 @@ ticketsRouter.post('/', zValidator('json', createTicketSchema), async (c) => {
return c.json({ error: 'Event not found' }, 404); return c.json({ error: 'Event not found' }, 404);
} }
if (event.status !== 'published') { if (!['published', 'unlisted'].includes(event.status)) {
return c.json({ error: 'Event is not available for booking' }, 400); return c.json({ error: 'Event is not available for booking' }, 400);
} }

View File

@@ -145,7 +145,7 @@ export default function BookingPage() {
paymentOptionsApi.getForEvent(params.eventId as string), paymentOptionsApi.getForEvent(params.eventId as string),
]) ])
.then(([eventRes, paymentRes]) => { .then(([eventRes, paymentRes]) => {
if (!eventRes.event || eventRes.event.status !== 'published') { if (!eventRes.event || !['published', 'unlisted'].includes(eventRes.event.status)) {
toast.error('Event is not available for booking'); toast.error('Event is not available for booking');
router.push('/events'); router.push('/events');
return; return;

View File

@@ -60,7 +60,7 @@ export default function EventDetailClient({ eventId, initialEvent }: EventDetail
const isCancelled = event.status === 'cancelled'; const isCancelled = event.status === 'cancelled';
// Only calculate isPastEvent after mount to avoid hydration mismatch // Only calculate isPastEvent after mount to avoid hydration mismatch
const isPastEvent = mounted ? new Date(event.startDatetime) < new Date() : false; const isPastEvent = mounted ? new Date(event.startDatetime) < new Date() : false;
const canBook = !isSoldOut && !isCancelled && !isPastEvent && event.status === 'published'; const canBook = !isSoldOut && !isCancelled && !isPastEvent && (event.status === 'published' || event.status === 'unlisted');
// Booking card content - reused for mobile and desktop positions // Booking card content - reused for mobile and desktop positions
const BookingCardContent = () => ( const BookingCardContent = () => (

View File

@@ -20,7 +20,7 @@ interface Event {
price: number; price: number;
currency: string; currency: string;
capacity: number; capacity: number;
status: 'draft' | 'published' | 'cancelled' | 'completed' | 'archived'; status: 'draft' | 'published' | 'unlisted' | 'cancelled' | 'completed' | 'archived';
bannerUrl?: string; bannerUrl?: string;
availableSeats?: number; availableSeats?: number;
bookedCount?: number; bookedCount?: number;

View File

@@ -568,7 +568,7 @@ export default function AdminEmailsPage() {
className="w-full px-4 py-3 rounded-btn border border-secondary-light-gray" className="w-full px-4 py-3 rounded-btn border border-secondary-light-gray"
> >
<option value="">Choose an event</option> <option value="">Choose an event</option>
{events.filter(e => e.status === 'published').map((event) => ( {events.filter(e => e.status === 'published' || e.status === 'unlisted').map((event) => (
<option key={event.id} value={event.id}> <option key={event.id} value={event.id}>
{event.title} - {new Date(event.startDatetime).toLocaleDateString(locale === 'es' ? 'es-ES' : 'en-US', { timeZone: 'America/Asuncion' })} {event.title} - {new Date(event.startDatetime).toLocaleDateString(locale === 'es' ? 'es-ES' : 'en-US', { timeZone: 'America/Asuncion' })}
</option> </option>

View File

@@ -10,7 +10,7 @@ import Button from '@/components/ui/Button';
import Input from '@/components/ui/Input'; import Input from '@/components/ui/Input';
import MediaPicker from '@/components/MediaPicker'; import MediaPicker from '@/components/MediaPicker';
import { MoreMenu, DropdownItem, AdminMobileStyles } from '@/components/admin/MobileComponents'; import { MoreMenu, DropdownItem, AdminMobileStyles } from '@/components/admin/MobileComponents';
import { PlusIcon, PencilIcon, TrashIcon, EyeIcon, PhotoIcon, DocumentDuplicateIcon, ArchiveBoxIcon, StarIcon, XMarkIcon } from '@heroicons/react/24/outline'; import { PlusIcon, PencilIcon, TrashIcon, EyeIcon, PhotoIcon, DocumentDuplicateIcon, ArchiveBoxIcon, StarIcon, XMarkIcon, LinkIcon } from '@heroicons/react/24/outline';
import { StarIcon as StarIconSolid } from '@heroicons/react/24/solid'; import { StarIcon as StarIconSolid } from '@heroicons/react/24/solid';
import toast from 'react-hot-toast'; import toast from 'react-hot-toast';
import clsx from 'clsx'; import clsx from 'clsx';
@@ -40,7 +40,7 @@ export default function AdminEventsPage() {
price: number; price: number;
currency: string; currency: string;
capacity: number; capacity: number;
status: 'draft' | 'published' | 'cancelled' | 'completed' | 'archived'; status: 'draft' | 'published' | 'unlisted' | 'cancelled' | 'completed' | 'archived';
bannerUrl: string; bannerUrl: string;
externalBookingEnabled: boolean; externalBookingEnabled: boolean;
externalBookingUrl: string; externalBookingUrl: string;
@@ -225,8 +225,8 @@ export default function AdminEventsPage() {
const getStatusBadge = (status: string) => { const getStatusBadge = (status: string) => {
const styles: Record<string, string> = { const styles: Record<string, string> = {
draft: 'badge-gray', published: 'badge-success', cancelled: 'badge-danger', draft: 'badge-gray', published: 'badge-success', unlisted: 'badge-warning',
completed: 'badge-info', archived: 'badge-gray', cancelled: 'badge-danger', completed: 'badge-info', archived: 'badge-gray',
}; };
return <span className={`badge ${styles[status] || 'badge-gray'}`}>{status}</span>; return <span className={`badge ${styles[status] || 'badge-gray'}`}>{status}</span>;
}; };
@@ -359,6 +359,7 @@ export default function AdminEventsPage() {
className="w-full px-4 py-3 rounded-btn border border-secondary-light-gray"> className="w-full px-4 py-3 rounded-btn border border-secondary-light-gray">
<option value="draft">Draft</option> <option value="draft">Draft</option>
<option value="published">Published</option> <option value="published">Published</option>
<option value="unlisted">Unlisted</option>
<option value="cancelled">Cancelled</option> <option value="cancelled">Cancelled</option>
<option value="completed">Completed</option> <option value="completed">Completed</option>
<option value="archived">Archived</option> <option value="archived">Archived</option>
@@ -515,6 +516,21 @@ export default function AdminEventsPage() {
<PencilIcon className="w-4 h-4" /> <PencilIcon className="w-4 h-4" />
</button> </button>
<MoreMenu> <MoreMenu>
{(event.status === 'draft' || event.status === 'published') && (
<DropdownItem onClick={() => handleStatusChange(event, 'unlisted')}>
<LinkIcon className="w-4 h-4 mr-2" /> Make Unlisted
</DropdownItem>
)}
{event.status === 'unlisted' && (
<DropdownItem onClick={() => handleStatusChange(event, 'published')}>
Make Public
</DropdownItem>
)}
{(event.status === 'published' || event.status === 'unlisted') && (
<DropdownItem onClick={() => handleStatusChange(event, 'draft')}>
Unpublish
</DropdownItem>
)}
<DropdownItem onClick={() => handleDuplicate(event)}> <DropdownItem onClick={() => handleDuplicate(event)}>
<DocumentDuplicateIcon className="w-4 h-4 mr-2" /> Duplicate <DocumentDuplicateIcon className="w-4 h-4 mr-2" /> Duplicate
</DropdownItem> </DropdownItem>
@@ -588,6 +604,21 @@ export default function AdminEventsPage() {
Publish Publish
</DropdownItem> </DropdownItem>
)} )}
{(event.status === 'draft' || event.status === 'published') && (
<DropdownItem onClick={() => handleStatusChange(event, 'unlisted')}>
<LinkIcon className="w-4 h-4 mr-2" /> Make Unlisted
</DropdownItem>
)}
{event.status === 'unlisted' && (
<DropdownItem onClick={() => handleStatusChange(event, 'published')}>
Make Public
</DropdownItem>
)}
{(event.status === 'published' || event.status === 'unlisted') && (
<DropdownItem onClick={() => handleStatusChange(event, 'draft')}>
Unpublish
</DropdownItem>
)}
{event.status === 'published' && ( {event.status === 'published' && (
<DropdownItem onClick={() => handleSetFeatured(featuredEventId === event.id ? null : event.id)}> <DropdownItem onClick={() => handleSetFeatured(featuredEventId === event.id ? null : event.id)}>
<StarIcon className="w-4 h-4 mr-2" /> <StarIcon className="w-4 h-4 mr-2" />

View File

@@ -671,10 +671,11 @@ export default function AdminScannerPage() {
// Load events // Load events
useEffect(() => { useEffect(() => {
eventsApi.getAll({ status: 'published' }) eventsApi.getAll()
.then((res) => { .then((res) => {
setEvents(res.events); const bookable = res.events.filter((e) => e.status === 'published' || e.status === 'unlisted');
const upcoming = res.events.filter((e) => new Date(e.startDatetime) >= new Date()); setEvents(bookable);
const upcoming = bookable.filter((e) => new Date(e.startDatetime) >= new Date());
if (upcoming.length === 1) { if (upcoming.length === 1) {
setSelectedEventId(upcoming[0].id); setSelectedEventId(upcoming[0].id);
} }

View File

@@ -168,7 +168,7 @@ export default function AdminTicketsPage() {
onChange={(e) => setCreateForm({ ...createForm, eventId: e.target.value })} onChange={(e) => setCreateForm({ ...createForm, eventId: e.target.value })}
className="w-full px-4 py-3 rounded-btn border border-secondary-light-gray min-h-[44px]" required> className="w-full px-4 py-3 rounded-btn border border-secondary-light-gray min-h-[44px]" required>
<option value="">Select an event</option> <option value="">Select an event</option>
{events.filter(e => e.status === 'published').map((event) => ( {events.filter(e => e.status === 'published' || e.status === 'unlisted').map((event) => (
<option key={event.id} value={event.id}>{event.title} ({event.availableSeats} spots left)</option> <option key={event.id} value={event.id}>{event.title} ({event.availableSeats} spots left)</option>
))} ))}
</select> </select>

View File

@@ -516,7 +516,7 @@ export interface Event {
price: number; price: number;
currency: string; currency: string;
capacity: number; capacity: number;
status: 'draft' | 'published' | 'cancelled' | 'completed' | 'archived'; status: 'draft' | 'published' | 'unlisted' | 'cancelled' | 'completed' | 'archived';
bannerUrl?: string; bannerUrl?: string;
externalBookingEnabled?: boolean; externalBookingEnabled?: boolean;
externalBookingUrl?: string; externalBookingUrl?: string;