This skill should be used when the user asks to "learn formily basics", "setup formily", "formily core concepts", "formily getting started", "formily fundamentals", "how to use formily", "formily...
This skill provides essential guidance for using Formily as a React form solution. Focus on understanding core concepts, setting up Formily projects, and implementing basic forms with TypeScript.
Formily is a performant, flexible, and extensible form solution for React. It provides:
Install core Formily packages:
npm install @formily/core @formily/react @formily/validate
For TypeScript support, install types:
npm install -D @types/react @types/react-dom
Create a simple contact form to understand Formily fundamentals:
import { createForm } from '@formily/core'
const form = createForm({
initialValues: {
name: '',
email: '',
message: ''
}
})
import { FormProvider, createField } from '@formily/react'
// In your component
const MyForm = () => {
return (
<FormProvider form={form}>
{/* Form fields go here */}
</FormProvider>
)
}
import { Field } from '@formily/react'
const NameField = () => (
<Field name="name">
{(field, state) => (
<div>
<label>Name:</label>
<input
value={state.value || ''}
onChange={e => field.onInput(e.target.value)}
/>
{state.errors && <span>{state.errors}</span>}
</div>
)}
</Field>
)
Formily manages form state through the form instance:
// Get all form values
const values = form.values
// Get specific field value
const name = form.values.name
// Get nested values
const address = form.values.contact.address
// Set single field
form.setValues({
name: 'John Doe'
})
// Set multiple fields
form.setValues({
name: 'John Doe',
email: 'john@example.com'
})
// Set nested values
form.setValues({
'user.profile.name': 'John Doe'
})
// Access field state
const field = form.query('name')
const fieldValue = field.value
const fieldErrors = field.errors
const fieldTouched = field.touched
// Modify field state
form.setFieldState('name', state => {
state.value = 'New value'
state.errors = ['Error message']
})
Formily provides built-in validation capabilities:
import { createForm } from '@formily/core'
const form = createForm({
schema: {
type: 'object',
properties: {
name: {
type: 'string',
title: 'Name',
required: true,
'x-validator': [
{ required: true, message: 'Name is required' },
{ min: 2, message: 'Name must be at least 2 characters' }
]
},
email: {
type: 'string',
title: 'Email',
format: 'email',
'x-validator': [
{ format: 'email', message: 'Invalid email format' }
]
}
}
}
})
import { createForm } from '@formily/core'
const form = createForm({
validateFirst: true,
effects() {
onFieldChange('confirmPassword', ['value'], (field) => {
const password = form.values.password
const confirmPassword = field.value
if (password !== confirmPassword) {
field.errors = ['Passwords do not match']
} else {
field.errors = []
}
})
}
})
Handle form submission with proper validation:
const handleSubmit = async () => {
try {
// Validate entire form
await form.validate()
// Get form values
const values = form.values
console.log('Form submitted:', values)
// Reset form if needed
form.reset()
} catch (errors) {
console.error('Validation errors:', errors)
}
}
Formily provides full TypeScript support:
interface ContactForm {
name: string
email: string
message: string
}
// Create typed form
const form = createForm<ContactForm>({
initialValues: {
name: '',
email: '',
message: ''
}
})
// Access typed values
const name: string = form.values.name
import { Field } from '@formily/react'
interface StringFieldProps {
name: string
label: string
placeholder?: string
}
const StringField: React.FC<StringFieldProps> = ({ name, label, placeholder }) => (
<Field name={name}>
{(field, state) => (
<div>
<label>{label}:</label>
<input
value={state.value || ''}
onChange={e => field.onInput(e.target.value)}
placeholder={placeholder}
/>
{state.errors && <span className="error">{state.errors[0]}</span>}
</div>
)}
</Field>
)
const form = createForm({
effects() {
onFieldChange('hasAccount', ['value'], (field) => {
const hasAccount = field.value
form.setFieldState('password', state => {
state.visible = hasAccount
if (!hasAccount) {
state.value = ''
}
})
})
}
})
// Add item to array
form.pushValues('emails', '')
// Remove item from array
form.removeValues('emails.0')
// Insert item at specific position
form.insertValues('emails.1', 'new@email.com')
import { observer } from '@formily/reactive-react'
const OptimizedField = observer(({ name }: { name: string }) => (
<Field name={name}>
{(field, state) => (
<input
value={state.value || ''}
onChange={e => field.onInput(e.target.value)}
/>
)}
</Field>
))
// Batch multiple updates for better performance
form.batch(() => {
form.setValues({ field1: 'value1' })
form.setValues({ field2: 'value2' })
form.setValues({ field3: 'value3' })
})
const form = createForm({
effects() {
onError((error) => {
console.error('Form error:', error)
// Handle global form errors
})
}
})
const form = createForm({
effects() {
onFieldError('email', (field, errors) => {
if (errors.includes('Invalid format')) {
// Show custom error UI
}
})
}
})
references/types-cheatsheet.md - Common TypeScript types and interfacesreferences/validation-rules.md - Built-in validation rules referenceexamples/simple-contact-form.tsx - Complete working contact form exampleexamples/field-components.tsx - Reusable field component patternsform.submit() for form submission with validationform.reset() for clearing form stateform.setFieldState() for granular field controlx-validator or effects