DEV Community

Cover image for How to Create React Custom Input & Password Field (show/hide) in 5 minutes
Syakir
Syakir

Posted on • Originally published at devaradise.com

How to Create React Custom Input & Password Field (show/hide) in 5 minutes

This tutorial is originally published at https://devaradise.com/custom-input-password-field-paradise-ui

In the previous post, I shared about how I built Paradise UI, a React component library with Monorepo architecture. I split the library into multiple independent component packages so everyone can pick and install only certain components based on their needs.

In this post, I'm going to share about the TextField component, the first component I made when I started Paradise UI. You might want to save or bookmark this for future reference.

In Paradise UI, TextField component is a native <input> component that is enhanced so it can be used for some use cases quickly, without having to manage the states or add css style.

Without further ado, let's see how to implement it.

Installation & Basic Usage

Just pick one of the commands below based on your favorite package manager.

yarn add @paradise-ui/text-field
# or
npm i @paradise-ui/text-field
# or
pnpm i @paradise-ui/text-field
Enter fullscreen mode Exit fullscreen mode

@paradise-ui/text-field export 1 component and 2 functions:

  • TextField component itself
  • defaultTextFieldElementClass a function that will generate default classes for all elements inside TextField component
  • tailwindTextFieldElementClass a function that will generate tailwind classes for all elements inside TextField component

Default usage

import { TextField } from '@paradise-ui/text-field'
import '@package-ui/text-field/style'

const ParentComponent = () => {
  return <TextField label="Lorem ipsum"/>
}
Enter fullscreen mode Exit fullscreen mode

You need to import the style because all Paradise UI components are unstyled by default.

With Tailwind

import { TextField, tailwindTextFieldElementClass } from '@paradise-ui/text-field'

const ParentComponent = () => {
  return <TextField elementClass={tailwindTextFieldElementClass} label="Lorem ipsum" />
}
Enter fullscreen mode Exit fullscreen mode

If you are using Tailwindcss, you don't need to import the style, but you have to add Paradise UI path into your Tailwindcss config.

/** @type {import('tailwindcss').Config} */
module.exports = {
  content: [
    ...,
    './node_modules/@paradise-ui/**/*.{js,ts}'
  ],
 ...
}
Enter fullscreen mode Exit fullscreen mode

Although the documentation for this component is not ready, you can customize and change all props on this page.

Use cases

Now let's see some use cases where you can implement TextField component

Size and Variants

TextField component come with 3 sizes (sm, md, lg) and 3 variants (outlined, filled, line). The default size is md and the default variant is outlined.

Interactive demo only available in the original post

Sizes and variants

  <div class="flex flex-col gap-2 max-w-[370px] mx-auto">
    <h3>Size</h3>
    <TextField size="sm" placeholder="Small text field"/>
    <TextField placeholder="Medium text field"/>
    <TextField size="lg" placeholder="Large text field"/>
    <h3>Variants</h3>
    <TextField variant="line" placeholder="Line text field"/>
    <TextField placeholder="Outlined text field"/>
    <TextField variant="filled" placeholder="Filled text field"/>
  </div>
Enter fullscreen mode Exit fullscreen mode

Input with Helper & Error message

Paradise UI TextField component also allows you to implement form validation logic and show error messages.

Interactive demo only available in the original post

Input with Helper & Error message

  export const HelperAndErrorMessage = () => {
    const [errorMessage, setErrorMessage] = useState('');
    return (
      <form
        className='flex gap-2 items-center'
        onSubmit={(e) => { 
          e.preventDefault();
          setErrorMessage('Username is already taken') 
        }}
        >
        <TextField
          label='Username'
          prefix={<AtSign size={16} strokeWidth={1} />}
          errorMessage={errorMessage}
          helperText='Check your username availability'
          />
        <Button className='shrink-0' type='submit'>Submit</Button>
      </form>
    )
  }
Enter fullscreen mode Exit fullscreen mode

Input with Custom label

You can also pass a custom element or React component to label props to implement TextField component with a custom label.

Interactive demo only available in the original post

Input with Custom label

  <div class="flex flex-col gap-2 max-w-[370px] mx-auto">
    <TextField
      label={<>
        <div className='font-bold'>Custom label</div>
        <small className='text-[var(--pui-text-secondary)]'>This is a custom label</small>
      </>}
      placeholder='Input placeholder'
    />
  </div>
Enter fullscreen mode Exit fullscreen mode

Input with Custom icon, prefix & suffix

Like label, prefix, and suffix props are also able to accept custom elements, making them customizable.

Interactive demo only available in the original post

Input with Custom icon, prefix & suffix

  <div class="flex flex-col py-2 gap-4 max-w-[370px] mx-auto">
    <TextField
      label='Product Price'
      prefix={<DollarSign size={16} strokeWidth={1.5}/>}
      placeholder='0.00'
    />
    <TextField
      variant='filled'
      label='Subdomain'
      prefix='https://'
      suffix='.devaradise.com'
      placeholder='subdomain'
    />
    <form
      onSubmit={(e) => {
        e.preventDefault();
        alert('trigger search function')
      }}
      >
      <TextField
        variant='line'
        label='Search'
        prefix={<UserSearch size={16} strokeWidth={1.5}/>}
        suffix={
          <Button
            variant='text'
            type='submit'
            size='sm'
            >
            <Search size={16}/>
          </Button>
        }
        placeholder='Type and enter to search ...'
      />
    </form>
  </div>
Enter fullscreen mode Exit fullscreen mode

Show/Hide Password Field

Showing and hiding password input can also be implemented by changing the type props upon clicking the custom suffix.

Interactive demo only available in the original post

Show/Hide Password Field

  export const PasswordShowAndHide = () => {
    const [value, setValue] = useState('');
    const [visible, setVisible] = useState(false);
    return (
      <TextField
        label='Password'
        placeholder='Placeholder'
        type={visible ? 'text' : 'password'}
        value={value} onChange={value => setValue(value)} prefix={<Lock size={16} strokeWidth={1.5} />}
        suffix={
          <a className='cursor-pointer text-inherit flex items-center' onClick={() => setVisible(!visible)}>
            {visible ? <EyeOff size={16} strokeWidth={1.5} /> : <Eye size={16} strokeWidth={1.5} />}
          </a>
        }
      />
    )
  }
Enter fullscreen mode Exit fullscreen mode

Advanced Customization

Since all Paradise UI components are unstyled by default, you can entirely implement your custom style. You can refer to style.scss file before you write your custom css.

You can also change the element class by making a custom element class generator. Since I haven't completed the documentation yet, you can refer to Alert customization documentation to create a custom element class.

Feedback

This component, along with another component in Paradise UI still in development. Feedback is greatly appreciated for the improvement of this project.

Please kindly give me feedback through a comment section below 😄.

Thank you, have a good day.

Top comments (0)