form-pills v1.1.0
form-pills
 
💊 Encapsulates form fields like a pill!
Features
- All-in-One Definition: Define a field's name, validation schema, default values, and UI in a single, unified declaration.
- Modular Complexity: Effortlessly build complex forms by nesting field definitions with providers and dynamic name generation.
- Smart Type Inference: Harness the power of TypeScript and Standard Schema for automatic type inference and robust data validation.
- Reusable Components: Create self-contained field components that encapsulate logic and minimize code duplication.
- Seamless Integration: Easily integrate with popular libraries like react-hook-form and Zod for efficient form state management and validation.
- Intuitive API: Enjoy a clear, straightforward API that makes form development a breeze.
- Ultra-Lightweight: Benefit from an exceptionally small bundle size, ensuring fast load times and minimal overhead.
Installation
$ npm i form-pills
$ yarn add form-pills
$ pnpm add form-pillsQuick Start
import { defineField, useFieldName, type InferFieldShape } from 'form-pills';
const UsernameField = defineField()({
  name: 'username',
  // You can use any schema validation library that supports Standard Schema!
  // Zod:
  schema: z.string().min(1),
  // Valibot:
  schema: v.pipe(v.string(), v.nonEmpty()),
  getDefaultValues: () => 'John',
  render: (context) => {
    type FieldShape = InferFieldShape<typeof context>
    // ^? { username: string }
    const name = useFieldName();
    // name() === 'username'
    // Here you can write integrated with form and library UI code
  },
})API References
Function
defineField
defineField is a generic function used to create a field component by specifying its name, validation schema, default values,
and UI render function—all in one place. It returns a function that not only renders the field component but also exposes:
- A fieldShapeproperty representing the field's schema.
- A getDefaultValuesfunction to retrieve the default values for the field.
- An extendsmethod to extend or override parts of the field configuration.
Example:
const MyField = defineField<{ label: string }>()({
  name: 'myField',
  schema: mySchema,
  getDefaultValues: () => defaultValue,
  render: (context, props /* { label: string } */) => {
    // Your rendering logic here
  },
});
<MyField label="Hello!" /> // Render as a Component
MyField.fieldShape // { "myField": mySchema }
MyField.getDefaultValue() // Return default value
MyField.extends({ /* ... */ }) // Extend or override parts of the fieldHooks
useFieldName(context)
The useFieldName hook provides a helper function to generate full field names based on a given context. This is particularly useful when working with nested fields, as it dynamically constructs field names by combining a prefix with the specific field identifier.
Example:
const MyField = defineField()({
  name: 'myField',
  // ...
  render: (context, props) => {
    const fieldName = useFieldName(context);
    invariant(fieldName() === 'myField')
    invariant(fieldName('subField') === 'myField.subField')
  },
});Types
InferFieldShape<T>
InferFieldShape<T> is a utility type that infers the shape of a field from either a defined field result or a render context. It produces an object type where the key is the field's name and the value is the type derived from the field's validation schema.
Example:
const MyField = defineField()({
  name: 'myField',
  schema: z.string(),
  render: (context, props) => {
    type FieldShape = InferFieldShape<typeof context>
    // ^? { myField: string }
  },
});
type FieldShape = InferFieldShape<typeof MyField>
// ^? { myField: string }InferFieldSchema<T>
InferFieldSchema<T> is a utility type that infers the schema of a field from either a defined field result or a render context. It returns the type that is derived from the field's validation schema, helping ensure that your form components remain type-safe.
Example:
const MyField = defineField()({
  name: 'myField',
  schema: z.string(),
  render: (context, props) => {
    type FieldSchema = InferFieldSchema<typeof context>
    // ^? string
  },
});
type FieldSchema = InferFieldSchema<typeof MyField>
// ^? stringComponents
FieldNameProvider
The FieldNameProvider component sets a field name prefix for nested fields. By wrapping nested field components, it dynamically composes their names based on the provided prefix, ensuring unique and context-aware naming across your form.
Props:
- name (string): The prefix to apply to all nested field names.
Example:
<FieldNameProvider name="address">
  <CityField />
  <ZipField />
</FieldNameProvider>In this example, CityField and ZipField will have their field names automatically prefixed, resulting in names like address.city and address.zip. This is useful when working with array fields.
Examples
License
MIT
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago