- 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
62 lines
2 KiB
JavaScript
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 }
|
|
})
|