ghostguild-org/server/api/onboarding/track.post.js
Jennie Robinson Faber 7292b11c0b feat(member): account/profile polish + tier upgrade flow
- Timezone: curated USelectMenu dropdown (app/config/timezones.js), preserves unknown saved values
- Profile save now uses useToast() for success/error; remove inline save banner
- Nav onboarding dot nudged down 1px for optical alignment with lowercase text
- Onboarding: skip a suggestion with POST /api/onboarding/track {skip}; member.onboarding.skipped map; does not affect graduation
- CirclePicker takes :saved-value so 'Current' badge stays until save completes
- PrivacyToggle is binary (USwitch labeled Private); member schema enum reduced to ['members','private']; zod coerces legacy 'public'
- New /member/payment-setup page: HelcimPay $0 verify + update-contribution, wired from account.vue via requiresPaymentSetup redirect
- Helcim portal: NUXT_PUBLIC_HELCIM_PORTAL_URL env + account.vue 'Manage billing in Helcim' link
- Migration script: scripts/migrate-privacy-public-to-members.js
2026-04-14 20:35:37 +01:00

62 lines
2 KiB
JavaScript

import { requireAuth } from '../../utils/auth.js'
import { validateBody } from '../../utils/validateBody.js'
import { onboardingTrackSchema } from '../../utils/schemas.js'
import Member from '../../models/member.js'
import BoardPost from '../../models/boardPost.js'
import { logActivity } from '../../utils/activityLog.js'
export default defineEventHandler(async (event) => {
const member = await requireAuth(event)
const { goal, skip } = await validateBody(event, onboardingTrackSchema)
// Already graduated — no-op
if (member.onboarding?.completedAt) {
return { success: true }
}
// Skip path: mark suggestion as skipped so the UI advances.
if (skip) {
await Member.findByIdAndUpdate(member._id, {
$set: { [`onboarding.skipped.${skip}`]: true },
})
return { success: true }
}
// Idempotent — already tracked
if (member.onboarding?.[goal]) {
return { success: true }
}
// Set the boolean
await Member.findByIdAndUpdate(member._id, {
$set: { [`onboarding.${goal}`]: true },
})
// Log the individual goal completion
await logActivity(member._id, 'member_onboarding_goal_completed', { goal }, { visibility: 'admin' })
// Must have at least one board post to graduate
const hasPosted = await BoardPost.exists({ author: member._id })
// Graduation check — atomic so concurrent requests can't double-graduate
const graduated = hasPosted
? await Member.findOneAndUpdate(
{
_id: member._id,
'onboarding.completedAt': null,
'onboarding.eventPageVisited': true,
'onboarding.boardPageVisited': true,
'onboarding.wikiClicked': true,
'craftTags.0': { $exists: true },
},
{ $set: { 'onboarding.completedAt': new Date() } },
{ new: true }
)
: null
if (graduated) {
await logActivity(member._id, 'member_onboarding_completed', {}, { visibility: 'admin' })
}
return { success: true, graduated: !!graduated }
})