@dynamic-field-kit/core
v1.1.0
Published
Core types and field registry for dynamic-field-kit
Downloads
616
Maintainers
Readme
@dynamic-field-kit/core
Core types and shared registries for dynamic-field-kit.
@dynamic-field-kit/core is intentionally framework-agnostic. It does not import React, Vue, or Angular types in its public API. Applications define field schemas in core, then register framework-specific renderers through an adapter package such as @dynamic-field-kit/react, @dynamic-field-kit/vue, or @dynamic-field-kit/angular.
Demo app: https://github.com/vannt-dev/dynamic-field-kit-demo
What this package provides
FieldDescriptionfor schema-driven field definitionsFieldRendererPropsas the shared renderer contractFieldTypeMapfor module augmentation and custom field typingfieldRegistryas the shared runtime registry instance
Install
npm install @dynamic-field-kit/coreNote: This package is framework-agnostic. To render forms in an application, install a framework adapter as well (for example, @dynamic-field-kit/react, @dynamic-field-kit/vue, or @dynamic-field-kit/angular).
Example: npm install @dynamic-field-kit/core @dynamic-field-kit/react
Or: npm install @dynamic-field-kit/core @dynamic-field-kit/vue
Or: npm install @dynamic-field-kit/core @dynamic-field-kit/angular
Core is intended to be a single source of truth for registry and types. Ensure your app uses a single version of @dynamic-field-kit/core to avoid multiple registry instances.
Install a UI adapter alongside it when rendering forms:
npm install @dynamic-field-kit/core @dynamic-field-kit/react
# or
npm install @dynamic-field-kit/core @dynamic-field-kit/vue
# or
npm install @dynamic-field-kit/core @dynamic-field-kit/angularCore idea
The library does not hard-code field types like "text" | "number".
Instead, apps extend FieldTypeMap:
export interface FieldTypeMap {}That keeps the package open for custom field types without modifying the library.
Define field types in your app
Create a declaration file such as src/types/dynamic-field-kit.d.ts:
import '@dynamic-field-kit/core';
declare module '@dynamic-field-kit/core' {
interface FieldTypeMap {
text: string;
number: number;
checkbox: boolean;
select: string;
}
}Make sure that file is included by your app's tsconfig.json.
Shared types
export interface FieldRendererProps<T = any> {
value?: T;
onValueChange?: (value: T) => void;
label?: string;
placeholder?: string;
required?: boolean;
options?: Record<string, any>[];
className?: string;
description?: any;
}export interface FieldDescription<T extends FieldTypeKey = FieldTypeKey> {
name: string;
type: T;
label?: string;
placeholder?: string;
required?: boolean;
appearCondition?: (data: Record<string, any>) => boolean;
options?: Record<string, any>[];
className?: string;
description?: any;
}Register renderers through an adapter
core owns the shared registry instance, but renderer registration should happen through the framework adapter so the adapter can expose the correct renderer type for that framework.
Typical adapter imports:
import { fieldRegistry as reactRegistry } from '@dynamic-field-kit/react';
import { fieldRegistry as vueRegistry } from '@dynamic-field-kit/vue';
import { fieldRegistry as angularRegistry } from '@dynamic-field-kit/angular';Then register a renderer using the adapter that matches your UI framework.
React:
import { fieldRegistry } from '@dynamic-field-kit/react';
fieldRegistry.register('text', ({ value, onValueChange, label }) => (
<label>
<span>{label}</span>
<input
value={value ?? ''}
onChange={(e) => onValueChange?.(e.target.value)}
/>
</label>
));Vue:
import { defineComponent, h } from 'vue';
import { fieldRegistry } from '@dynamic-field-kit/vue';
fieldRegistry.register(
'text',
defineComponent({
setup() {
return () => h('input');
},
})
);Angular:
import { fieldRegistry } from '@dynamic-field-kit/angular';
import { TextFieldComponent } from './text-field.component';
fieldRegistry.register('text', TextFieldComponent as any);Example schema
import type { FieldDescription } from '@dynamic-field-kit/core';
const fields: FieldDescription[] = [
{ name: 'username', type: 'text', label: 'Username' },
{
name: 'age',
type: 'number',
label: 'Age',
appearCondition: (data) => Boolean(data.username),
},
];Notes
coreis runtime-shared across adapters.coredoes not ship UI components.- The registry stores framework-specific renderers, but the framework-specific typing belongs in each adapter package.
Related packages
@dynamic-field-kit/react@dynamic-field-kit/vue@dynamic-field-kit/angular
License
MIT
