Skip to content

uragecz/json-to-react-form

Repository files navigation

json-to-react-form

Made with create-react-library

NPM JavaScript Style Guide

React form component, that allows to easily generate form inputs from json file using react-hook-form library. Including validations, and high customization.

Install

yarn add json-to-react-form

or

npm install --save json-to-react-form

Basic usage

Example of simple json file:

// registrationForm.json
;[
  {
    component: 'TextInput',
    name: 'email',
    order: 0,
    label: 'E-mail:',
    labelPosition: 'top',
    type: 'email',
    validation: {
      required: 'This field is required'
    }
  },
  {
    component: 'TextInput',
    name: 'password',
    order: 1,
    label: 'Password:',
    type: 'password',
    validation: {
      required: 'This field is required',
      pattern: {
        value: '(?=.*\\d)(?=.*[a-z])(?=.*[A-Z])(?=.*[a-zA-Z]).{8,}$',
        message: 'Password needs to contain at leat 8 letters and one number'
      }
    }
  },
  {
    component: 'Button',
    name: 'submit',
    order: 2,
    title: 'Register'
  }
]

Generating form using our default components:

import React, { Component } from 'react'

import Form from 'json-to-react-form'
import registrationInputs from './registrationForm.json'
// import styles
import 'json-to-react-form/dist/index.css'

const Registration = () => {
  const handleRegister = (values: any) => {
    // do something here
  }
  return (
    <Form
      layout={[2, 8]}
      inputs={registrationInputs}
      onSubmit={handleRegister}
    />
  )
}

Generating form providing own input component:

import React, { Component } from 'react'

import Form, { InputFormProps } from 'json-to-react-form'
import registrationInputs from './registrationForm'
// import styles
import 'json-to-react-form/dist/index.css'

interface Props {
  // This property is injected by our library
  formProps: InputFormProps
}

const TextInput = ({ formProps }: Props) => {
  const { inputProps, form } = formProps
  const { name } = inputProps
  const error = form.formState.errors[name]
  return (
    <div>
      <input {...inputProps} />
      {!!error && <span>{error}</span>}
    </div>
  )
}

const Registration = () => {
  const handleRegister = (values: any) => {}

  return (
    <Form
      layout={[2, 8]}
      inputs={registrationInputs}
      onSubmit={handleRegister}
      components={TextInput}
    />
  )
}

Props

Common props you may need / want to specify include:

// Source of inputs that are rendered, more details see bellow
inputs: Item[]

// Instance of react-hook-form, can be usefull if you want to submit or do some other magic from parent.
form?: UseFormReturn<FieldValues>

// Validation type for form, default is 'onBlur'.
validationMode?: 'onBlur' | 'onChange' | 'onSubmit' | 'onTouched' | 'all'

// It's neccesary if you want to have repeatable form, with possibility of adding and removing forms. This value you get from useFieldArray().
fields?: Record<'id', string>[]

// This is needed only if fields prop is provided
formName?: string

// Layout for input and its label, first value is size of label, second is input, default is [3,7].
layout?: [number, number]

// You can provide your own layout component, that cares of rendering label together with input.
layoutComponent?: FunctionComponent<any>

onSubmit: (values: any) => void

// Events methods, adds listener for every button / input in form
onButtonClick?: (
  e: React.MouseEvent<HTMLButtonElement, MouseEvent>,
  name: string
) => void
onInputChange?: (checked: boolean | number | string, name: string) => void

// You can add your classname for our elements to add your own style
classes?: {
  // for input element
  input?: string
  // for input element in error state
  inputError?: string
  // eye icon button on password type button
  pswdButton?: string
  // wrapper around error icon and error message
  errorContainer?: string
  // for error icon
  errorIcon?: string
  // for error message
  errorMessage?: string
  // for button elements
  button?: string
  // for title
  title?: string
  // for wrapper around children
  childrenWrapper?: string
}

// We provide our own components by default, but if you want you can provide your own with own, it's super easy !
component?: {
  TextInput?: FunctionComponent<InputFormProps>
  SelectInput?: FunctionComponent<InputFormProps>
  Checkbox?: FunctionComponent<InputFormProps>
  Button?: FunctionComponent<ButtonFormProps>
  TextArea?: FunctionComponent<InputFormProps>
  Title?: FunctionComponent<{ children: React.ReactNode }>
}

What are InputFormProps and ButtonFormProps

It contains basic props that needs to be passed into input / button. If you are using some advance library like react-select and so, you should provide these props correctly. This prop may be different for different kind of input.

Items/Inputs type for json

Common properties for all inputs

{
  // If it's not specified, it renders just div, usually there is also added children prop, that renders another inputs
  component?: 'TextInput' | 'TextArea' | 'Checkbox' | 'RadioButton' | 'SelectInput' | 'Button',
  // Usually is used with "conditionalChildrenRender" or when "component" is not defined
  children?: Item[],
  // Every input that has specified "component" needs to have "name"
  name?: string,
  order?: number,
  label?: string,
  labelPosition?: 'top' | 'left' | 'bottom' | 'right',
  // align-items for inputs inside "children"
  align?: 'flex-start' | 'flex-end' | 'space-between' | 'center',
  type?: string,
  defaultValue?: any,
  validation?: Validation,
  // Allows to render children conditionaly - if this input is
  componentProps?: {},
  // These props are passed into component inside FormProps
  customProps?: {},
}

Specific properties

SelectInput:

{
  options: { label: string, value: any }[] | { label: string, options: { label: string, value: any }[] }[]
}

RadioButton:

{
  options: { label: string, value: any }[]
}

Button:

{
  title?: string,
  type?: string | 'submit',
}

Checkbox:

{
  // Allows to render children if checkbox is checked
  conditionalChildrenRender?: true,
  // can contain html tags
  text?: string
}

Validation type

{
  // string represents error message
  required?: string,
  pattern?: {
	// regex
    value: string
    message: string
  },
  min?: {
    value: number
    message: string
  },
  max?: {
    value: number
    message: string
  },
  minLength?: {
    value: number
    message: string
  },
  maxLength?: {
    value: number
    message: string
  },
  validation?: {
    sameAs?: {
      // name of input
      value: string
      message: string
    }
  }
}

License

MIT © uRAGEcz