An easy to use library to quickly create react forms π, its the new level from react-nice-inputs.
npm i -D react-nicer-inputs
Now import the inputs as you would regularly do by:
import { Input } from 'react-nicer-inputs';
Now you are good to go. Create the first input as:
<Input
type="text"
name="example"
className="col-12 example-class"
inputChange={({ value, name, e }) => {
console.log(value, name, e);
}}
attrs={{
placeholder: 'enter a value',
}}
/>
π Nice! You must now have a simple text input like so:
And when you type in the console must display the value you type in, the name of the input and the Syntetic Event. Awesome! π
When you want to cooperate and create or update or patch the components you may fork this repo.
npm i -D
This will install all dev dependencies to start working.
npm i -g yalc
Yalc is a perfect tool to develop libraries, it helps with bad dependencies resolutions between node environments therefore this library needs it in order to work locally by building yalc local repo and linking it into the example folder app.
Inside the root directory create a react app called example
:
npx create-react-app example
This way you can see the actual render in browser of any input you are updating and creating.
To add react-nicer-inputs to yalc local repo just run on root:
yalc add react-nicer-inputs
Now to link this into the example app run:
$ cd example
$ yalc link react-nicer-inputs
To fire the library dev compilation run at the root directory:
npm run dev
To fire jest at watch mode run at the root directory in separate bash window:
npm run test-w
To start the example react app, in yet another bash window run:
$ cd example
$ npm start
Nice! Let's get hackin' π¨π½βπ»
So each component renders different inputs for you to use and each have common props which will be discussed next but for particular props check each component definition in the next section.
-
type="" {string}
This defines the type of the input being rendered, is passed to the native HTML5 type attribute; now some components do not require this or prefer the usage of a dedicated component for an specific input. Refer to each component description below what case it is for the component you are currently using. -
name="fname" {string}
This sets the name of this input as well it is assigned to the HTML5 name and id attributes, additionally it is returned as well on theinputChange
prop. -
className="col-12 col-md-5" {string}
This is as you might know the class this component will have, normally the prop is assigned to the input or to the wrapper of the component such asInputGroup
,Password
,Autocomplete
,Datepicker
for example. Some components do have internal wrap blocks and those can have classNames to but the prop name might differ in each case, refer then to each component specific prop dictionary below.Observation: each input do have the
input
utility class by default so you can paint with CSS the actual input as you see fit. -
inputChange={} {Function}
This is the way the input return the value back. This is a semantic name for onChange (due to the implementation of TS it needed to be named as inputChange). You may destructure the params to:
{
e, name, value;
}
// e: This is the syntetic event from the input
// name: This is the name of the input, helpful if you are storing a model of the form in the state and you have a single function that handles input change
// value: The value from the input
-
attrs={{placeholder: 'Enter your name'}} {Object}
This is the list of HTML5 attributes the input can handle, you can send here as much as you need. -
value={fnameVal} {string}
This is the initial value you need this input to start with, it is ignored as soon the input changes. -
...props
Any other props from react you see you might need to pass along to the input are spreaded internally. The handle or callback is for you to write.
Now we are about to see what input components are in store for you from 'react-nicer-inputs'
.
As we saw in the example above this input is very helpful when you are creating classic text inputs, number, date or textarea, for the latter the input will render a proper <textarea>
tag element. While you can create radios, checkboxes and password with this component you should favor the <InputGroup>
component for radios and checkboxes and <Password>
for well... pwds!
Let's see what props this component may handle in particular:
-
cols {string | number}
When you define this Input to be textarea it will handle the columns the textarea will have -
rows {string | number}
Same as cols. -
mask {Regex}
This prop will help the component to hide certain characters visible on the input based on a regex you provide in order to capture whatever matches on the value string and replace it with themaskChar
value. For example:mask=".{3}$"
this Regex will select the last three characters in the input and replace it with discs each character. So let's say the user types in:reactjs
therefore the value VISIBLE will bereacβββ
while the returned value oninputChange
will be still stringreactjs
.
Observation: Do not missuse this functionality for a password input.
maskChar {string} defaults to 'β'
This prop altogether with mask allows to change the matching characters with whatever character you might need: asterisks, dashes, periods... etc.
This component creates a <select>
dropdown for you. Since this is straight forward there is no more to explain for it, let's dive into the props this component use in particular aside from the common ones described above:
-
options={[ {label: 'Option 1 1οΈβ£', value: '1'} ]} {Array}
This is an array of objects that define the select options, additional you may send 'attrs' a third property on each option Object. That property is spreaded on the<option>
tag. -
defaultLabel="Pick your favorite option" {string}
This is the default option label on the Select.
These two components help you create more semantic accurate inputs rather than just a simple list of options when you send the options prop to the <Select >
. Nest these boys inside the Select component and you are ready to go.
If you need to create checkboxes and radiobutton groups this component will aid you in no time, the component renders a box with rows of radio βΊ or checkbox β inputs aside of a label for each element. So the props this input need are:
options={[ {label: 'Option 1 1οΈβ£', value: '1'} ]} {Array}
Idem as<select>
This is an array of objects that define the select options, additional you may send 'attrs' a third property on each option Object. That property is spreaded on the<option>
tag.
Observation: This component value prop and the value returned on the inputChange function may differ wether it is a radio or checkbox
For checkboxes since conceptually it is a multi option selectable it will return on the inputChange function a string of the selected options separated by a commas, the same as the value prop, it will take a comma separated string of the values to be preselected by the component.
For radio buttons as its a single option selectable out of a group of option this will return on inputChange function a normal string, no commas. The value prop can be a single string of one of the options.
The checkbox and radio button components allow for you to create single checkbox and single radio, providing a more semantic approach for your inputs. The value grouping falls to your hands but you could wrap these <Checkbox>
or <Radio>
components inside a <InputGroup>
component, be aware of the type
prop for InputGroup. Do not mix <InputGroup>
type="radio" with <Checkboxes>
.
labelClassName="col-8" {string}
Optional. Is the class appended to the label element
This component allows a large list of options to be browsable by typing in the textbox, reducing the ammount of options available as the user type by matching the string to any possible option.
This component needs the following props:
options={[ {label: 'Option 1 1οΈβ£', value: '1'} ]} {Array}
Idem as<select> and <InputGroup>
This is an array of objects that define the options available to pick, from there the options are reduced as the user types in.
The password component is unique and helpful, following the way Edge browser allows the behavior natively to toggle show/hide the visibility of the characters on the input, this component includes a native button that allows such toggling the password to be visible or set back to be hidden.
Some props are the same as the input component but this contains the following:
showIcon="π" {ReactNode} defaults to 'hide'
This is the icon to be displayed when the password is hiddenhideIcon="β" {ReactNode} defaults to 'show'
This is the invers as showIcon, it displays when the password is visiblenoToggle {boolean} defaults to false
This defines if you need the toggle button or not.
These are simple components you can use to work along with any input. The label component can take className and also requires only:
htmlFor="fname" {string}
This is of course the id of the input this label belongs to.
The feedback component allows to display a text and its usage is to be part of the FormGroup as a way to display messages about this input, can take className but nothing else, it will render a <span>
element with whatever text you send to it.
This component allows the user to select a date or a date range, depending on the dateRange
prop the behavior of the component do change into two scenarios:
- If no
dateRange
prop is sent then it will display a single text input that when the user sets focus on displays a calendar element below, as soon as the user do select one date it will trigger theinputChange
function and return the formatted date as value. - If
dateRange
is sent then two inputs are displayed and the calendar accepts two dates to be selected, on each selection the component triggers theinputChange
function but this time the value is an object like so:
{
startDate: '05-30-2021',
endDate: '06-02-2021'
}
This component takes a handful of props that mutate its behavior, let's see:
format="DD-MM-YYYY" {string}
Allows formatting for the dates that this component will return.maxDate="05-29-2021" {string}
Is the maximum date allowable to select by this datepickerminDate="07-30-2021" {string}
Is the minimum date allowable to select by this datepickerbottomPanel={} {Function}
Is the panel below the calendar, it prompts the user to clear selection and confirm to close calendar, must return JSX, this function can take an object as parameter, this object contains{clearSelections, confirmSelections}
each functions can close the calendar and confirmSelections will trigger inputChange and return the value of the selected date/s. This way you return a block of JSX for further customization of the bottom panel.minNights={3} {number}
Is the minimum nights allowable to select by this calendarmaxNights={10} {number}
Is the maximum nights allowable to select by this calendarmonthsToDisplay={3} {number}
Is the ammount of months to be displayeddisabledDates={['06-01-2021']} {Array}
Is the array of dates that this calendar will mark as unallowable to be selectedmonthHeader={} {Function}
This allows you to create a custom header that could allow users to change months and years quickly, the function must return JSX, this function can take an object as params which contains{ setYear, setMonth, month }
.- setYear is a function that takes a string of the year you want to set
- setMonth is a function that takes a string of the month you want to set
- month is a momentjs object that you can use to get the current month information.
prevButton={<JSX />} {ReactNode}
Allows to customize the navigation button for previous calendar datesnextButton={<JSX />} {ReactNode}
Allows to customize the navigation button for next calendar datesdisableNavigationOnDateBoundary {boolean}
This prop disables the calendar navigation along with maxDate / minDate when the user has reached either the maxDate meaning that the user cant go any more into the future or cant go into the past if has reached the minDate.calendarComponentClassName="col-12" {string}
Is the class that the calendar wrapper(which contains the month/months, navigation arrows... etc) below the input can havecalendarClassName="col-12 col-md-6" {string}
Is the class that each of the months can have, allows for further customization on each of the months wrappers
This is a calendar element that you can use to make your own implementations of date picker without the <DatePicker>
component in favor of your own ways.
All the props explained in the <DatePicker>
component are almost the same.
className="custom-calendar" {string}
This class is appended to the wrapper of this calendar component.calendarClassName="col-12 col-md-4" {string}
Is the class that each of the months can have, allows for further customization on each of the months wrappers
This component is kinda special, sometimes somewhere there is the need to select a date in a very unique way. Selects. Yep! So, we got you covered, here this component renders three different selects for each of the segments of the date, month, day and year. The value of each select depends of the props as well. Let's take a peek:
format="DD-MM-YY" {string}
Allows formatting for the dates that this component will return.maxDate="05-29-2021" {string}
Is the maximum date allowable to select by this datepickerminDate="07-30-2021" {string}
Is the minimum date allowable to select by this datepickerddClassName="col-3" {string}
This className is appended to the wrapper of the Day selectormmClassName="col-3" {string}
The same as day selectoryyClassName="col-6" {string}
The same as day selectorddLabel="Pick day:" {string}
This is the text of the<label>
element above the day<select>
mmLabel="Pick month:" {string}
The same as day LabelyyLabel="Pick year:" {string}
The same as day LabelddDefaultLabel="Day..." {string}
This prop allows to customize the default text on the<Select>
componentmmDefaultLabel="Month..." {string}
The same as Day Default LabelyyDefaultLabel="Year..." {string}
The same as Day Default LabeldisplayOrder="DD-MM-YYYY" {string}
This allows to customize the order of rendering of each selectormmmm {boolean}
If you need the month element to have words instead of numbers
Observation: The options react to maxDate and minDate and its options are rerendered to disallow user selecting dates out of boundaries.
This component will help you to create a google captcha validation before any form is submitted. It works with v2 and v3 APIs. Let us see the props:
For v2 the purpose is to have a check element that provides a challenge under certain circumstances which are in google side, that said the props you need to create one is:
publicKey="AAAABBBCCCC" {string}
This defines the key to be used for this captcha element, check ππ½ google recaptcha docs.theme="dark" {string} defaults to 'light'
Is the color scheme this captcha element will have, the values could be light | dark as per google recaptcha docscaptchaSize="compact" {string} defaults to 'normal'
Is the size of this captcha element, the values could be compact | normal | invisible as per docsgetResponse={} {Function}
Is the function that returns the token google returned after the challenge is completed
For v3 the challenge is not visible as per the docs, and it must trigger on form submit, so the way we play the game in this one is different, we wrap the <Submit>
component around captcha component, we use the v3 prop.
v3 {boolean}
Helps to define what API version is going to be used
This component provide an in-house approach to submit your forms, this way you can submit forms regularly or with google recaptcha protection. The unique prop is just one, let's take a look at that now:
formSubmit={} {Function}
This function is triggered when you click the submit button, using v3 captcha it will return the captchaToken.
Check the example:
<Grecaptcha publicKey="AAAABBBBCCCCCDDDDEEEE111222333" v3>
<Submit
className="blue-big"
formSubmit={({ captchaToken }) => {
// Your logic to submit the form
}}
>
Send Form
</Submit>
</Grecaptcha>
The submit component is now part of the Grecaptcha protection, this way, when the button is clicked it will fetch to google for the token, then it will run the formSubmit Function returning the captchaToken in v3 captcha API.
Observation: a simple submit does not require to be wrapped around with <Grecaptcha >
component, the formSubmit function does not need to destructure the captchaToken either.
This is a component that is composed of three elemental components.
<Label>
<Input> ||
<Select> ||
<InputGroup> ||
<Password> ||
<Autocomplete> ||
<Datepicker> ||
<DropDownDates>
<Feedback>
So, it utilizes all the props explained for each component above.
This component allows you to create forms, all Input component within the
tags are added to a formModel object, the model updates when user interacts with the inputs, this allows a very flexible access to everything regarding the form, values, validations, used inputs or pristine inputs.model="login" {string}
Optional - Name for the form model, defaults to "defaultModel"formSubmit={() => {}} {Function}
This function will run when button component is added to the formuseModel={() => {}} {Function}
Optional - A pseudoHook that allows the consuming of the form model and its state
A simple login form can be created as:
<Form
model="login"
className="col-12"
formSubmit={handleFormSubmit}
useModel={useModel}
>
<Input
name="username"
type="text"
className="col-6"
data-testid="username"
inputChange={handleInputChange}
validate={['required']}
/>
<Password
name="password"
className="col-6"
data-testid="password"
inputChange={handleInputChange}
validate={['required']}
/>
<Submit className="col-6" data-testid="submit">
Send Form
</Submit>
</Form>
Where handleFormSubmit is a function that can receive back on sync if the form is valid or invalid, destructure the params {isInvalid, isValid, formModel}; isInvalid and isValid those booleans will tell if the form is ready to be posted and the action is on your side, meanwhile formModel contains all the form model properties for you to use in case you want to work with the form model after the user clicks submit.
useModel is invoked everytime the form model updates, telling you what input identifier has changed and from there you could access to each input model values:
{
type: string,
valid: boolean,
invalid: boolean,
pristine: boolean,
touched: boolean,
dirty: boolean,
validate: Array,
value: string | number | null,
summary: Array
}
- type is this input type.
- valid is a boolean that represent what the result of the validation Array is.
- invalid is a boolean and is a more verbose approach and works as helper, inverse of valid.
- pristine is a boolean, indicates if this input has not been interacted with yet.
- touched is the inverse of pristine.
- validate array contains the current validations this input must run each change.
- value, the value the input holds
- summary is an array of results for each element in the validate array.
This is an example of what could be the form model in the forementioned login form:
{
"fields":{
"username":{
"type":"text",
"valid":true,
"invalid":false,
"pristine":false,
"touched":true,
"dirty":false,
"validate":[
"required"
],
"value":"reactdev",
"summary":{
"required":true
}
},
"password":{
"type":"password",
"valid":null,
"invalid":null,
"pristine":true,
"touched":false,
"dirty":false,
"validate":[
"required",
],
"value":null
},
},
"isValid":true,
"isInvalid":false
}
From this example you can determine that username has been filled whereas password has not been touched yet. This object is at your disposal and you can do as you see fit on your side of things.
- Form (Beta)
- Custom options for Autocomplete
- Input type range
Have a spin: check this codesandbox