2
0
forked from jmug/cactoide

feat(tmp): invite link feature

This commit is contained in:
Levente Orban
2025-10-15 10:00:26 +02:00
parent f6b51232a7
commit c9c78d0ea6
18 changed files with 1199 additions and 164 deletions

View File

@@ -1,7 +1,8 @@
import { database } from '$lib/database/db';
import { events } from '$lib/database/schema';
import { events, inviteTokens } from '$lib/database/schema';
import { fail, redirect } from '@sveltejs/kit';
import type { Actions } from './$types';
import { generateInviteToken, calculateTokenExpiration } from '$lib/inviteTokenHelpers.js';
// Generate a random URL-friendly ID
function generateEventId(): string {
@@ -25,7 +26,7 @@ export const actions: Actions = {
const locationUrl = formData.get('location_url') as string;
const type = formData.get('type') as 'limited' | 'unlimited';
const attendeeLimit = formData.get('attendee_limit') as string;
const visibility = formData.get('visibility') as 'public' | 'private';
const visibility = formData.get('visibility') as 'public' | 'private' | 'invite-only';
const userId = cookies.get('cactoideUserId');
// Validation
@@ -98,6 +99,7 @@ export const actions: Actions = {
const eventId = generateEventId();
// Create the event
await database
.insert(events)
.values({
@@ -118,6 +120,24 @@ export const actions: Actions = {
throw error;
});
// Generate invite token for invite-only events
if (visibility === 'invite-only') {
const token = generateInviteToken();
const expiresAt = calculateTokenExpiration(date, time);
await database
.insert(inviteTokens)
.values({
eventId: eventId,
token: token,
expiresAt: new Date(expiresAt)
})
.catch((error) => {
console.error('Error creating invite token', error);
throw error;
});
}
throw redirect(303, `/event/${eventId}`);
}
};

View File

@@ -15,7 +15,7 @@
location_url: '',
type: 'unlimited',
attendee_limit: undefined,
visibility: 'public'
visibility: 'public' as 'public' | 'private' | 'invite-only'
};
let errors: Record<string, string> = {};
@@ -317,13 +317,13 @@
{t('create.visibilityLabel')}
<span class="text-red-400">{t('common.required')}</span>
</legend>
<div class="grid grid-cols-2 gap-3">
<div class="grid grid-cols-3 gap-3">
<button
type="button"
class="rounded-sm border-2 px-4 py-3 font-medium transition-all duration-200 {eventData.visibility ===
'public'
? ' border-violet-500 bg-violet-400/20 font-semibold hover:bg-violet-400/70'
: 'border-dark-300 text-dark-700'}"
: 'border-dark-300 text-dark-700 bg-gray-600/20 hover:bg-gray-600/70'}"
on:click={() => (eventData.visibility = 'public')}
>
{t('create.publicOption')}
@@ -338,11 +338,23 @@
>
{t('create.privateOption')}
</button>
<button
type="button"
class="rounded-sm border-2 px-4 py-3 font-medium transition-all duration-200 {eventData.visibility ===
'invite-only'
? ' border-violet-500 bg-violet-400/20 font-semibold hover:bg-violet-400/70'
: 'border-dark-300 text-dark-700 bg-gray-600/20 hover:bg-gray-600/70'}"
on:click={() => (eventData.visibility = 'invite-only')}
>
{t('create.inviteOnlyOption')}
</button>
</div>
<p class="mt-2 text-xs text-slate-400 italic">
{eventData.visibility === 'public'
? t('create.publicDescription')
: t('create.privateDescription')}
: eventData.visibility === 'private'
? t('create.privateDescription')
: 'Event is public but requires a special invite link to attend'}
</p>
</fieldset>
</div>