* feat: add `vee-validate` * chore: update * chore: update `AccountForm` example - add `FormDescription` component - include `src` in tsconfig * refactor: use radix-vue `Slot` component * chore: refresh lockfile * chore: update `ProfileForm.vue` and `AccountForm` fix vee-validate initialValues on components with `componentField` slotProp * chore: update `AppearanceForm.vue` update pnpm and some deps -_- * refactor: update - add new-york style - off eslint import/first rule - use `useId` from radix-vue * fix: class-name -> class * refactor: simplify validation for `Command` component * fix: v-bind="field" -> v-bind="componentField" * fix: useAttrs to prevent class duplication * docs: add `form.md` - change TabPreview.vue to showcase way of using vee-validate * docs: add form example for `checkbox` `input` and `datepicker` * docs: add `combobox`, `datepicker`, `radio-group`, `select`, `switch` and `textarea` form and some other exmaples * chore: typo, update `zod`, `vite`, and `@vitejs/plugin-vue`
200 lines
6.6 KiB
Vue
200 lines
6.6 KiB
Vue
<script setup lang="ts">
|
|
import { useForm } from 'vee-validate'
|
|
import { toTypedSchema } from '@vee-validate/zod'
|
|
import * as z from 'zod'
|
|
|
|
import { FormControl, FormDescription, FormField, FormItem, FormLabel, FormMessage } from '@/lib/registry/default/ui/form'
|
|
import { Separator } from '@/lib/registry/new-york/ui/separator'
|
|
import { RadioGroup, RadioGroupItem } from '@/lib/registry/new-york/ui/radio-group'
|
|
import { Switch } from '@/lib/registry/new-york/ui/switch'
|
|
import { Checkbox } from '@/lib/registry/new-york/ui/checkbox'
|
|
import { Button } from '@/lib/registry/new-york/ui/button'
|
|
|
|
const notificationsFormSchema = toTypedSchema(z.object({
|
|
type: z.enum(['all', 'mentions', 'none'], {
|
|
required_error: 'You need to select a notification type.',
|
|
}),
|
|
mobile: z.boolean().default(false).optional(),
|
|
communication_emails: z.boolean().default(false).optional(),
|
|
social_emails: z.boolean().default(false).optional(),
|
|
marketing_emails: z.boolean().default(false).optional(),
|
|
security_emails: z.boolean(),
|
|
}))
|
|
|
|
const { handleSubmit } = useForm({
|
|
validationSchema: notificationsFormSchema,
|
|
initialValues: {
|
|
communication_emails: false,
|
|
marketing_emails: false,
|
|
social_emails: true,
|
|
security_emails: true,
|
|
},
|
|
})
|
|
|
|
const onSubmit = handleSubmit((values, { resetForm }) => {
|
|
console.log('Form submitted!', values)
|
|
})
|
|
</script>
|
|
|
|
<template>
|
|
<div>
|
|
<h3 class="text-lg font-medium">
|
|
Notifications
|
|
</h3>
|
|
<p class="text-sm text-muted-foreground">
|
|
Configure how you receive notifications.
|
|
</p>
|
|
</div>
|
|
<Separator />
|
|
<form class="space-y-8" @submit="onSubmit">
|
|
<FormField v-slot="{ componentField }" type="radio" name="type">
|
|
<FormItem class="space-y-3">
|
|
<FormLabel>Notify me about...</FormLabel>
|
|
<FormControl>
|
|
<RadioGroup
|
|
class="flex flex-col space-y-1"
|
|
v-bind="componentField"
|
|
>
|
|
<FormItem class="flex items-center space-x-3 space-y-0">
|
|
<FormControl>
|
|
<RadioGroupItem value="all" />
|
|
</FormControl>
|
|
<FormLabel class="font-normal">
|
|
All new messages
|
|
</FormLabel>
|
|
</FormItem>
|
|
<FormItem class="flex items-center space-x-3 space-y-0">
|
|
<FormControl>
|
|
<RadioGroupItem value="mentions" />
|
|
</FormControl>
|
|
<FormLabel class="font-normal">
|
|
Direct messages and mentions
|
|
</FormLabel>
|
|
</FormItem>
|
|
<FormItem class="flex items-center space-x-3 space-y-0">
|
|
<FormControl>
|
|
<RadioGroupItem value="none" />
|
|
</FormControl>
|
|
<FormLabel class="font-normal">
|
|
Nothing
|
|
</FormLabel>
|
|
</FormItem>
|
|
</RadioGroup>
|
|
</FormControl>
|
|
<FormMessage />
|
|
</FormItem>
|
|
</FormField>
|
|
|
|
<div>
|
|
<h3 class="mb-4 text-lg font-medium">
|
|
Email Notifications
|
|
</h3>
|
|
<div class="space-y-4">
|
|
<FormField v-slot="{ handleChange, value }" type="checkbox" name="communication_emails">
|
|
<FormItem class="flex flex-row items-center justify-between rounded-lg border p-4">
|
|
<div class="space-y-0.5">
|
|
<FormLabel class="text-base">
|
|
Communication emails
|
|
</FormLabel>
|
|
<FormDescription>
|
|
Receive emails about your account activity.
|
|
</FormDescription>
|
|
</div>
|
|
<FormControl>
|
|
<Switch
|
|
:checked="value"
|
|
@update:checked="handleChange"
|
|
/>
|
|
</FormControl>
|
|
</FormItem>
|
|
</FormField>
|
|
|
|
<FormField v-slot="{ handleChange, value }" type="checkbox" name="marketing_emails">
|
|
<FormItem class="flex flex-row items-center justify-between rounded-lg border p-4">
|
|
<div class="space-y-0.5">
|
|
<FormLabel class="text-base">
|
|
Marketing emails
|
|
</FormLabel>
|
|
<FormDescription>
|
|
Receive emails about new products, features, and more.
|
|
</FormDescription>
|
|
</div>
|
|
<FormControl>
|
|
<Switch
|
|
:checked="value"
|
|
@update:checked="handleChange"
|
|
/>
|
|
</FormControl>
|
|
</FormItem>
|
|
</FormField>
|
|
|
|
<FormField v-slot="{ handleChange, value }" type="checkbox" name="social_emails">
|
|
<FormItem class="flex flex-row items-center justify-between rounded-lg border p-4">
|
|
<div class="space-y-0.5">
|
|
<FormLabel class="text-base">
|
|
Social emails
|
|
</FormLabel>
|
|
<FormDescription>
|
|
Receive emails for friend requests, follows, and more.
|
|
</FormDescription>
|
|
</div>
|
|
<FormControl>
|
|
<Switch
|
|
:checked="value"
|
|
@update:checked="handleChange"
|
|
/>
|
|
</FormControl>
|
|
</FormItem>
|
|
</FormField>
|
|
|
|
<FormField v-slot="{ handleChange, value }" type="checkbox" name="security_emails">
|
|
<FormItem class="flex flex-row items-center justify-between rounded-lg border p-4">
|
|
<div class="space-y-0.5">
|
|
<FormLabel class="text-base">
|
|
Security emails
|
|
</FormLabel>
|
|
<FormDescription>
|
|
Receive emails about your account activity and security.
|
|
</FormDescription>
|
|
</div>
|
|
<FormControl>
|
|
<Switch
|
|
:checked="value"
|
|
@update:checked="handleChange"
|
|
/>
|
|
</FormControl>
|
|
</FormItem>
|
|
</FormField>
|
|
</div>
|
|
</div>
|
|
|
|
<FormField v-slot="{ handleChange, value }" type="checkbox" name="mobile">
|
|
<FormItem class="flex flex-row items-start space-x-3 space-y-0">
|
|
<FormControl>
|
|
<Checkbox
|
|
:checked="value"
|
|
@update:checked="handleChange"
|
|
/>
|
|
</FormControl>
|
|
<div class="space-y-1 leading-none">
|
|
<FormLabel>
|
|
Use different settings for my mobile devices
|
|
</FormLabel>
|
|
<FormDescription>
|
|
You can manage your mobile notifications in the
|
|
<a href="/examples/forms">
|
|
mobile settings
|
|
</a> page.
|
|
</FormDescription>
|
|
</div>
|
|
</FormItem>
|
|
</FormField>
|
|
|
|
<div class="flex justify-start">
|
|
<Button type="submit">
|
|
Update notifications
|
|
</Button>
|
|
</div>
|
|
</form>
|
|
</template>
|