Building and Testing a Select Component

Debbie O'Brien - Jan 10 '22 - - Dev Community

I have created a component that is actually made up of 2 smaller components, a Select element and a Label. Together these two components make up a component I have named select-size and is the component used in the demo e-commerce project I have created in order for users to select the size of the product.

Building the Select Component

Importing React, Testing Library and Components

The component is built in React and TypeScript and imports React, useState and the two components needed to build this component as well as the styles.

import React, { useState } from 'react'
import { Select } from '@learn-bit-react/base-ui.ui.forms.select'
import { Label } from '@learn-bit-react/base-ui.ui.forms.label'
import styles from './select-size.module.scss'
Enter fullscreen mode Exit fullscreen mode

Prop Types

The props being passed down are the availableSizes which is an array of numbers and the sizeSelected which is a function that passes in the size of the product. As we are using Typescript we first export our types, this ensures our user can only use the types specified such as the array of available sizes can only be a number and not a string.

export type SelectSizeProps = {
  /**
   * sizes as an array of numbers
   */
  availableSizes: number[],
  /**
   * a function that registers the selected size.
   */
  sizeSelected: size => void
} & React.SelectHTMLAttributes<HTMLSelectElement>
Enter fullscreen mode Exit fullscreen mode

Passing down Props

We then pass down the props into our SelectSize component as well as ...rest which gives access to all other props that a html select element can have.

export function SelectSize({
  availableSizes,
  sizeSelected,
  ...rest
}: SelectSizeProps) {}
Enter fullscreen mode Exit fullscreen mode

Adding State

Our component uses the useState hook to set the size of the product. size is the value of the select element and setSize is the function that allows us to set a new value. The default state will be the first number of the availableSizes array.

export function SelectSize({
  availableSizes,
  sizeSelected,
  ...rest
}: SelectSizeProps) {
  const [size, setSize] = useState(availableSizes[0])
}
Enter fullscreen mode Exit fullscreen mode

Using the Select and Label Components

We can now add the return statement to our component and return the Select and Label components that we have imported. The Label component is pretty straight forward and just adds some styles and the htmlFor attribute with the value of size. For the Select component we need to add the id of size, the className for the styles, and the options for the select component which is equal to the value of the availableSizes array.

The Select component takes in a prop of options and will map over the array to give us an <option> for each number in the array. We then need an onChange function to handle the change for every time the user changes the size. And of course we pass in the ...rest of the props that a html select element can take.

const [size, setSize] = useState(availableSizes[0])
return (
  <div className={styles.selectSize}>
    <Label className={styles.label} htmlFor="size">
      Choose a size:
    </Label>
    <Select
      id="size"
      className={styles.select}
      options={availableSizes}
      onChange={handleChange}
      {...rest}
    />
  </div>
)
Enter fullscreen mode Exit fullscreen mode

Creating the handleChange function

We can now create our handleChange function which will set the state of size to be the value of the select element as well as call the sizeSelected function with the value of the select element.

function handleChange(e) {
  setSize(e.target.value)
  sizeSelected(e.target.value)
}
Enter fullscreen mode Exit fullscreen mode

Final Code

The full code for our component will look like this:

import React, { useState } from 'react'
import { Select } from '@learn-bit-react/base-ui.ui.forms.select'
import { Label } from '@learn-bit-react/base-ui.ui.forms.label'
import styles from './select-size.module.scss'

export type SelectSizeProps = {
  /**
   * sizes as an array of numbers
   */
  availableSizes: number[],
  /**
   * a function that registers the selected size.
   */
  sizeSelected: size => void
} & React.SelectHTMLAttributes<HTMLSelectElement>

export function SelectSize({
  availableSizes,
  sizeSelected,
  ...rest
}: SelectSizeProps) {
  const [size, setSize] = useState(availableSizes[0])

  function handleChange(e) {
    setSize(e.target.value)
    sizeSelected(e.target.value)
  }

  return (
    <div className={styles.selectSize}>
      <Label className={styles.label} htmlFor="size">
        Choose a size:
      </Label>
      <Select
        id="size"
        className={styles.select}
        options={availableSizes}
        onChange={handleChange}
        {...rest}
      />
    </div>
  )
}
Enter fullscreen mode Exit fullscreen mode

Creating Compositions

We now need to make compositions for our component to see the component in action. Compositions are a feature of Bit that allows you to see your component in isolation. If not using Bit then you can create mocks to test your component.

My compositions imports React and useState as well as the component we just created. We then create a SelectSizeAndShowSelectedSize component that will render the SelectSize component. We first create a const of sizes equal to an array of numbers, the sizes the want to show. We then use the useState hook to set the state of the selectedSize giving it the default of the first value from our sizes array.

Then in our component we make the prop of sizeSelected equal to a function that passes in the argument of size and sets the state of selectedSize to be the value of size. This give us access to the value of the size selected so as we can use it in another component.

We also add the value of our sizes array to the availableSizes prop of the SelectSize component.

And finally we add a <p> tag with the value of the selectedSize so we can see the size of the product updated in the UI as we change it.

import React, { useState } from 'react'
import { SelectSize } from './select-size'

export function SelectSizeAndShowSelectedSize() {
  const sizes = [36, 37, 38, 39, 40, 45, 46, 47]
  const [selectedSize, setSelectedSize] = useState(sizes[0])

  return (
    <>
      <SelectSize
        sizeSelected={size => {
          setSelectedSize(parseInt(size))
        }}
        availableSizes={sizes}
      />
      <p>You're selected size is: {selectedSize}</p>
    </>
  )
}
Enter fullscreen mode Exit fullscreen mode

We can now see our component works as we would expect it to. As we are building this component using Bit I have a dev server that shows me the component running in isolation. If you are not using Bit then you will need to import it into another component to see it working.


Writing Tests

We can therefore move on to write the tests for this component and use the composition created in order to test it.

Importing what we need

We are using Testing Library to test our component so we need to import render, screen, userEvent from @testing-library/react as well as React from 'react'. We also need to import our composition component as our tests are based on the composition we created earlier.

import React from 'react'
import { render, screen } from '@testing-library/react'
import userEvent from '@testing-library/user-event'
import { SelectSizeAndShowSelectedSize } from './select-size.composition'
Enter fullscreen mode Exit fullscreen mode

Describing our Test

Our test should check that the value changes when the user chooses a new size so we can start with that as a description.

import React from 'react'
import { render, screen } from '@testing-library/react'
import userEvent from '@testing-library/user-event'
import { SelectSizeAndShowSelectedSize } from './select-size.composition'

it('checks value changes when user chooses a new size', () => {})
Enter fullscreen mode Exit fullscreen mode

Rendering our Composition Component

We then render the component we want to test which is the component we created in our composition file which uses our select size component and also adds in a

tag with the value of the selectedSize so we can see the size of the product selected as we change it.

import React from 'react'
import { render, screen } from '@testing-library/react'
import userEvent from '@testing-library/user-event'
import { SelectSizeAndShowSelectedSize } from './select-size.composition'

it('checks value changes when user chooses a new size', () => {
  render(<SelectSizeAndShowSelectedSize />)
})
Enter fullscreen mode Exit fullscreen mode

Checking what Role Exists

In order to see what role is available can use the screen.getByRole function and pass in any string. This will tell us that the role we are looking for doesn't exist but will show us what roles do exist on our component.

import React from 'react'
import { render, screen } from '@testing-library/react'
import userEvent from '@testing-library/user-event';
import { SelectSizeAndShowSelectedSize } from './select-size.composition'

it('checks value changes when user chooses a new size', () => {
  render(<SelectSizeAndShowSelectedSize />)
  const selectSizeAndShowSelectedSize = screen.getByRole('blah')
Enter fullscreen mode Exit fullscreen mode

Getting by Correct Role

As we are running our tests in watch mode we can see that the role blah does not exist but it tells us that combobox does exist meaning we can use this for our role. We can also pass in the name with the value of the label. This also makes sure we have the correct label. Adding in a regex with i at the end means we won't have to worry about case sensitivity.

import React from 'react'
import { render, screen } from '@testing-library/react'
import userEvent from '@testing-library/user-event'
import { SelectSizeAndShowSelectedSize } from './select-size.composition'

it('checks value changes when user chooses a new size', () => {
  render(<SelectSizeAndShowSelectedSize />)
  const selectSizeAndShowSelectedSize = screen.getByRole('combobox', {
    name: /choose a size/i
  })
})
Enter fullscreen mode Exit fullscreen mode

Expecting our Component to Have Correct Value

We now use expect to make sure our component has the correct value which will be the default value we set it to. We can see what value this is by first adding in any value such as 0 and seeing our test fail. The failing test will tell us what value it is expecting which should be the first value in our array that we created in the composition file, 36.

import React from 'react'
import { render, screen } from '@testing-library/react'
import userEvent from '@testing-library/user-event'
import { SelectSizeAndShowSelectedSize } from './select-size.composition'

it('checks value changes when user chooses a new size', () => {
  render(<SelectSizeAndShowSelectedSize />)
  const selectSizeAndShowSelectedSize = screen.getByRole('combobox', {
    name: /choose a size/i
  })
  expect(selectSizeAndShowSelectedSize).toHaveValue('36')
})
Enter fullscreen mode Exit fullscreen mode

Firing an Event and Expecting the Value to Change

As we want to make sure the value is updated when the user chooses a new size we can use the userEvent method with the change function passing in what we want to change and what the target is. In our case it is the const of selectSizeAndShowSelectedSize and the target is the value and we can add in what value we want to change it to. We then use the expect method to make sure the value has been updated correctly to the new value of the userEvent.

import React from 'react'
import { render, screen } from '@testing-library/react'
import userEvent from '@testing-library/user-event'
import { SelectSizeAndShowSelectedSize } from './select-size.composition'

it('checks value changes when user chooses a new size', () => {
  render(<SelectSizeAndShowSelectedSize />)
  const selectSizeAndShowSelectedSize = screen.getByRole('combobox', {
    name: /choose a size/i
  })
  expect(selectSizeAndShowSelectedSize).toHaveValue('36')
  userEvent.selectOptions(selectSizeAndShowSelectedSize, '45')
  expect(selectSizeAndShowSelectedSize).toHaveValue('45')
})
Enter fullscreen mode Exit fullscreen mode

Conclusion

And that's it. We now have a select component that works as we would expect and can now be used in the component where it should be used knowing that it will work correctly. Compositions are a great way of seeing the different states of our components and we can then use the composition file to understand what we need to do to make our component work when using it in our next component/app.

select element changing size on click

We should also document our component so that it contains clear instructions and examples which makes it even easier for our consumer to understand what the component does and how to use it. And of course tests make sure our component not only works as expected but also that if we do make any changes to it our tests ensure that it can not be exported if our tests are broken meaning if we do have any breaking changes we can fix our tests and release a new major version of our component.

Using the Component

The select size component can be found here and is fully open source meaning you can install it in your own project using bit, npm or yarn so feel free to take it for a test drive.

bit install @learn-bit-react/ecommerce.ui.product.select-size

npm i @learn-bit-react/ecommerce.ui.product.select-size

yarn add @learn-bit-react/ecommerce.ui.product.select-size
Enter fullscreen mode Exit fullscreen mode

Useful Links

. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .