MOPC-App/src/types/wizard-config.ts

156 lines
5.4 KiB
TypeScript
Raw Normal View History

import { z } from 'zod'
// --- Step Configuration ---
export const WIZARD_STEP_IDS = ['welcome', 'contact', 'project', 'team', 'additional', 'review'] as const
export type WizardStepId = (typeof WIZARD_STEP_IDS)[number]
export const wizardStepSchema = z.object({
id: z.enum(WIZARD_STEP_IDS),
enabled: z.boolean().default(true),
order: z.number().int().min(0),
title: z.string().optional(),
conditionalOn: z
.object({
field: z.string(),
operator: z.enum(['equals', 'notEquals', 'in', 'notIn']),
value: z.union([z.string(), z.array(z.string())]),
})
.optional(),
})
// --- Field Configuration ---
export const wizardFieldValidationSchema = z.object({
min: z.number().optional(),
max: z.number().optional(),
pattern: z.string().optional(),
patternMessage: z.string().optional(),
})
export const wizardFieldConfigSchema = z.object({
required: z.boolean().optional(),
visible: z.boolean().optional(),
label: z.string().optional(),
helpText: z.string().optional(),
placeholder: z.string().optional(),
validation: wizardFieldValidationSchema.optional(),
})
// --- Dropdown Option ---
export const dropdownOptionSchema = z.object({
value: z.string().min(1),
label: z.string().min(1).max(100),
description: z.string().max(300).optional(),
icon: z.string().optional(),
})
// --- Custom Field ---
export const customFieldSchema = z.object({
id: z.string().min(1),
type: z.enum(['text', 'textarea', 'number', 'select', 'multiselect', 'checkbox', 'date']),
label: z.string().min(1).max(100),
placeholder: z.string().optional(),
helpText: z.string().optional(),
required: z.boolean().default(false),
options: z.array(z.string()).optional(),
validation: wizardFieldValidationSchema.optional(),
stepId: z.enum(WIZARD_STEP_IDS),
order: z.number().int().default(0),
})
// --- Welcome Message ---
export const welcomeMessageSchema = z.object({
title: z.string().max(200).optional(),
description: z.string().max(1000).optional(),
imageUrl: z.string().url().optional(),
})
// --- Feature Flags ---
export const wizardFeaturesSchema = z.object({
enableWhatsApp: z.boolean().optional(),
enableMentorship: z.boolean().optional(),
enableTeamMembers: z.boolean().optional(),
requireInstitution: z.boolean().optional(),
})
// --- Main Config Schema ---
export const wizardConfigSchema = z.object({
steps: z.array(wizardStepSchema).default([]),
fields: z.record(z.string(), wizardFieldConfigSchema).default({}),
competitionCategories: z.array(dropdownOptionSchema).optional(),
oceanIssues: z.array(dropdownOptionSchema).optional(),
features: wizardFeaturesSchema.optional(),
welcomeMessage: welcomeMessageSchema.optional(),
customFields: z.array(customFieldSchema).optional(),
})
// --- Exported Types ---
export type WizardStep = z.infer<typeof wizardStepSchema>
export type WizardFieldConfig = z.infer<typeof wizardFieldConfigSchema>
export type WizardFieldValidation = z.infer<typeof wizardFieldValidationSchema>
export type DropdownOption = z.infer<typeof dropdownOptionSchema>
export type CustomField = z.infer<typeof customFieldSchema>
export type WizardFeatures = z.infer<typeof wizardFeaturesSchema>
export type WelcomeMessage = z.infer<typeof welcomeMessageSchema>
export type WizardConfig = z.infer<typeof wizardConfigSchema>
// --- Default Configuration ---
// Must match current hardcoded behavior exactly for backward compatibility
export const DEFAULT_COMPETITION_CATEGORIES: DropdownOption[] = [
{
value: 'BUSINESS_CONCEPT',
label: 'Business Concepts',
description: 'For students and recent graduates with innovative ocean-focused business ideas',
icon: 'GraduationCap',
},
{
value: 'STARTUP',
label: 'Start-ups',
description: 'For established companies working on ocean protection solutions',
icon: 'Rocket',
},
]
export const DEFAULT_OCEAN_ISSUES: DropdownOption[] = [
{ value: 'POLLUTION_REDUCTION', label: 'Reduction of pollution (plastics, chemicals, noise, light,...)' },
{ value: 'CLIMATE_MITIGATION', label: 'Mitigation of climate change and sea-level rise' },
{ value: 'TECHNOLOGY_INNOVATION', label: 'Technology & innovations' },
{ value: 'SUSTAINABLE_SHIPPING', label: 'Sustainable shipping & yachting' },
{ value: 'BLUE_CARBON', label: 'Blue carbon' },
{ value: 'HABITAT_RESTORATION', label: 'Restoration of marine habitats & ecosystems' },
{ value: 'COMMUNITY_CAPACITY', label: 'Capacity building for coastal communities' },
{ value: 'SUSTAINABLE_FISHING', label: 'Sustainable fishing and aquaculture & blue food' },
{ value: 'CONSUMER_AWARENESS', label: 'Consumer awareness and education' },
{ value: 'OCEAN_ACIDIFICATION', label: 'Mitigation of ocean acidification' },
{ value: 'OTHER', label: 'Other' },
]
export const DEFAULT_WIZARD_CONFIG: WizardConfig = {
steps: [
{ id: 'welcome', enabled: true, order: 0, title: 'Category' },
{ id: 'contact', enabled: true, order: 1, title: 'Contact' },
{ id: 'project', enabled: true, order: 2, title: 'Project' },
{ id: 'team', enabled: true, order: 3, title: 'Team' },
{ id: 'additional', enabled: true, order: 4, title: 'Details' },
{ id: 'review', enabled: true, order: 5, title: 'Review' },
],
fields: {},
competitionCategories: DEFAULT_COMPETITION_CATEGORIES,
oceanIssues: DEFAULT_OCEAN_ISSUES,
features: {
enableWhatsApp: false,
enableMentorship: true,
enableTeamMembers: true,
requireInstitution: false,
},
customFields: [],
}