npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

react-craftform

v1.7.4

Published

React form with full customizable and extensible. This library help you create form with your own style.

Downloads

31

Readme

react-craftform

React form library with full customizable and extensible. This library help you create form with your own style.

This library doesn't add any style to your form component. It let style of form to your imagination.

Installation

npm install react-craftform --save

Usage

react-craftform created with HOC in mind. react-craftform provide function called withForm for providing existing form power!

import React from 'react';
import {withForm, validator, hasError} from 'react-craftform';

const MyForm = (props) => {
  return (
     <form>
          <section className="name_section">
              <label>Name</label>
              <input type="text" className="name" onChange={props.form.handleChange("name")} value={props.form.values['name']}/>
          </section>
          <p>Name Value: {props.form.values['name']} </p>
          <section className="code_section">
              <label>Last Name</label>
              <input type="text" className="code" onChange={props.form.handleChange("lastname")} value={props.form.values['lastname']} />
              
          </section>
          <p>LastName Value: {props.form.values['lastname']} </p>
          {hasError(props.form.errors) ? "There is an error" : "No Error, Huley"}
      </form>
  );
}

export default withForm(MyForm,{
  name:["",["required"]],
  lastname:["myinitalCode"]
});

Plunker example

Creating form with withForm

withForm take 2 parameters. First parameter it takes is your form component. Second parameter is description about your controls.

Let's take a look at withForm function.

 export default withForm(MyForm,{
    name:["",["required"]],
    lastname:["myinitilCode"]
 });

MyForm component will receive object called forn via props for managing control (eg. handling onChange event, geting value and errors).

Second parameter is object which property name is your control name. Each control name's value is array which for specifying initial value at index 0 and Validators at index 1.

In above code, we tell withForm that it has 2 controls which name are name and lastname. name control have no initial value and is required and lastname control have initial value which is "initialCode" and is opional.

Handling change on input

withForm doesn't update value for you so that you can customize as you want. Instead, it help you by providing form object via props with have method for update value.

form props has handleChange method for updating value. handleChange accept control name.

   <input type="text" className="name" onChange={props.form.handleChange("name")} value={props.form.values['name']}/>
   <input type="text" className="code" onChange={props.form.handleChange("lastname")} value={props.form.values['lastname']} />

Getting value of control

You can get value of control by call props.form.values[controlname].

As above exmaple, we pass props.form.values['name'] to input attribute value for name control and props.form.values['lastname'] for lastname control.

Validation

You can add validator easily in description object.

   import { validator } from 'react-craftform';
   
   ...
   
   export default withForm(MyForm,{
     name:["",["required",validator.minLength(3)]],
     lastname:["myinitilCode",validator.minLength(5)]
   });

You can use built-in validator provided by this library or create your own validator (I promise it's easy).

Plunker example

Error object

Error object will be generated according to specified validators. You can get error object in props.form.errors[controlname].

Error object has property name indicating what validator that control violate and it's value is detail about the error you can leverage (eg. showing more meaningful error message)

As above example, there'll be initial error for name and lastname controls.

 props.form.errors['name']; // { required: true, minLength: {length: 0, minRequired:3 }}
 props.form.errors['lastname']; // { maxLength:{length: 12, maxRequired:3 }}

You can use error object for showing in error message and/or prevent submit if there's an error. We will explain in Error message section and preventing user submit form section

Built-in validator

| Function | String | example of error object | ------------------- | ---------------------- | ------- | | validator.required | "required" | { required:true } | | validator.minLength(minRequired) | - | { minLength:{ length:3, minRequired:4 } length is current length of control | | validator.maxLength(maxRequired) | - | { maxLength:{ length:6, maxRequired:4 } length is current length of control |

Custom validator

Validator is just a function. You can provide your own easily.

 ...
 const haveToBeCat = (val) => {
   if(val === "cat"){ 
       return false;
   }else{
       return {
           havetoBeCat:true
       }
   }
 };

 export default withForm(MyForm,{
   name:["",[haveToBeCat]],
   lastname:["myinitilCode",validator.maxLength(3)]
 });

Plunker example

In the above example, we create function called haveToBeCat. It will be called and provided current value of control to first parameter by library so you can use it to check current value against your specified rule.

When create your own custom validator, just remember 2 things

  1. if value is valid, just return false.
  2. if value is invalid, return object which propery name is meaningful name. You will be use that name for showing error message.

haveToBeCat works great but you can refactor haveToBeCat to be more reuseable.

   const haveToBeSomething = (word) => {
       return (val) => {
           if(val === word){
               return false;
           }else{
               return {
                    havetoBeSomething:{
                      expectedValue:word,
                      value:val
                     }
               }
           }
       }

   };

   const haveToBeCat = haveToBeSomething("cat");
   const haveToBeDog = haveToBeSomething("dog");
  
   export default withForm(MyForm,{
     name:["",[haveToBeCat]],
     lastname:["myinitilCode",haveToBeDog]
   });

Plunker example

May functional programing be with you.

Showing error message

You can provide error message by calling errorMessage function. It accept one parameter and return another function which accept error object and return component we provide. It use currying function style.

If you're confused, don't worry just let code exlain itself.

  ...
  import {withForm, validator, hasError, errorMessage} from 'react-craftform';

  const MyForm = (props) => {
    return (
     <form>
          <section className="name_section">
              <label>Name</label>
              <input type="text" className="name" onChange={props.form.handleChange("name")} value={props.form.values['name']}/>
          </section>
          <p>Name Value: {props.form.values['name']} </p>
          {
            errorMessage({
                minLength:()=>{
                    return (<p>Please fill control with 5 length</p>);
                },
                required:()=>{
                    return (<p>Please fill input</p>);
                },        
            })(props.form.errors['name'])
          }
          <section className="code_section">
              <label>Last Name</label>
              <input type="text" className="code" onChange={props.form.handleChange("lastname")} value={props.form.values['lastname']} />
              
          </section>
          <p>LastName Value: {props.form.values['lastname']} </p>
          {
            errorMessage({
                minLength:()=>{
                    return (<p>Please fill control with 5 length</p>);
                },
                required:()=>{
                    return (<p>Please fill input</p>);
                },        
            })(props.form.errors['lastname'])
          }
      </form>
  );
}

export default withForm(MyForm,{
  name:["",["required"]],
  lastname:["",[validator.minLength(5)]]
});

Plunker example

Get rid of duplicated code

It works great but there's duplicated code for name and lastname controls. You can get rid of it by creating function and return invoking errorMessage function.

const errorSpec =  errorMessage({
    minLength:()=>{
        return (<p>Please fill control with 5 length</p>);
    },
    required:()=>{
        return (<p>Please fill input</p>);
    }
});

const MyForm = (props) => {
  return (
     <form>
          <section className="name_section">
              <label>Name</label>
              <input type="text" className="name" onChange={props.form.handleChange("name")} value={props.form.values['name']}/>
          </section>
          <p>Name Value: {props.form.values['name']} </p>
          {
            errorSpec(props.form.errors['name'])
          }
          <section className="code_section">
              <label>Last Name</label>
              <input type="text" className="code" onChange={props.form.handleChange("lastname")} value={props.form.values['lastname']} />
              
          </section>
          <p>LastName Value: {props.form.values['lastname']} </p>
          {
            errorSpec(props.form.errors['lastname'])
          }
      </form>
  );
}

Plunker example

Just assign variable to errorMessage with how to render error message.

Showing error with information

You can provide error information easily for showing error message

const errorSpec = (name) => {
    return errorMessage({
        minLength:()=>{
            return (<p>Please fill control {name} with 5 length</p>);
        },
        required:()=>{
            return (<p>Please fill control {name}</p>);
        }
    });
};

...
const MyForm = (props) => {
         return (
          ...
          {
            errorSpec('name')(props.form.errors['name'])
          }
          ..
          )
}

Plunker example

Showing error message with custom validator

It's the same as built-in validator. Let me show you

   const haveToBeSomething = (word) => {
       return (val) => {
           if(val === word){
               return false;
           }else{
               return {
                   havetoBeSomething:{
                     expectedValue:word,
                     value:val
                   }
               }
           }
       }

   };

   const haveToBeCat = haveToBeSomething("cat");
   const haveToBeDog  haveToBeSomething("dog");
  
   const errorSpec = () => {
       return errorMessage({
           ...
           havetoBeSomething:() => {
               return (<p>Please fill cat!</p>)
           }
       });
   };

Plunker example

It's easy but there's one problem, error message for havetoBeSomething assume to be "cat". How about dog, pig, elephant or whatever?

Luckily we can handle this issue in easy way.

Using error object for more meaningful error message

     const errorSpec = () => {
        return errorMessage({
            ...
            havetoBeSomething:(def) => {
                return (<p>Please fill {def.expectedValue}! Now you provide {def.value}</p>)
            }
        });
    };

Plunker example

errorMessage will be provided corresponding error object in parameter of function. So you can use it for providing more meaningful error message.

validator.minLength and validator.maxLength are also provide error object. you can look at here

How to handle form submit

withForm doesn't handle submit form. We want you to have full control.

Provided props to withForm component will go to your form component as well. So you can add callback for onSubmit event via props.

const MyForm = (props) => {
  const onSubmit = (evt) => {
    evt.preventDefault();
    props.onSubmit(props.form.values);
  };
  return (
     <form onSubmit={onSubmit}>
          <section className="name_section" >
              <label>Name</label>
              <input type="text" className="name" onChange={props.form.handleChange("name")} value={props.form.values['name']}/>
          </section>
          <p>Name Value: {props.form.values['name']} </p>
          {
            errorSpec()(props.form.errors['name'])
          }
          <section className="code_section">
              <label>Last Name</label>
              <input type="text" className="code" onChange={props.form.handleChange("lastname")} value={props.form.values['lastname']} />
              
          </section>
          <p>LastName Value: {props.form.values['lastname']} </p>
          {
            errorSpec()(props.form.errors['lastname'])
          }
          <button type="submit">Submit</button>
      </form>
  );
}

Plunker example

There's no difference for handling form in normal way, except you can get value from props.form.values which it's object whose properies is control name and it's value is corresponding control value.

Preventing form submit if there's an error

You can check there's an error easily with hasError function.

 ...
 const MyForm = (props) => {
  const onSubmit = (evt) => {
    evt.preventDefault();
    if(hasError(props.form.errors)){
        window.alert("There's an error");
    }else{
        props.onSubmit(props.form.values);
    }
    
  };
  return (
     <form onSubmit={onSubmit}>
     ...
  );
}

Plunker example