shoelace/docs/frameworks/react.md

5.8 KiB

React

Shoelace offers a React version of every component to provide an idiomatic experience for React users. You can easily toggle between HTML and React examples throughout the documentation.

Installation

To add Shoelace to your React app, install the package from npm.

npm install @shoelace-style/shoelace

Next, include a theme and set the base path for icons and other assets. In this example, we'll import the light theme and use the CDN as a base path.

// App.jsx
import '@shoelace-style/shoelace/dist/themes/light.css';
import { setBasePath } from '@shoelace-style/shoelace/dist/utilities/base-path';

setBasePath('https://cdn.jsdelivr.net/npm/@shoelace-style/shoelace@%VERSION%/dist/');

?> If you'd rather not use the CDN for assets, you can create a build task that copies node_modules/@shoelace-style/shoelace/dist/assets into your app's public directory. Then you can point the base path to that folder instead.

Now you can start using components!

Usage

Importing Components

Every Shoelace component is available to import as a React component. Note that we're importing the <SlButton> React component instead of the <sl-button> custom element in the example below.

import { SlButton } from '@shoelace-style/shoelace/dist/react';

const MyComponent = () => <SlButton variant="primary">Click me</SlButton>;

export default MyComponent;

You can find a copy + paste import for each component in the "importing" section of its documentation.

Event Handling

Many Shoelace components emit custom events. For example, the input component emits the sl-input event when it receives input. In React, you can listen for the event using onSlInput.

Here's how you can bind the input's value to a state variable.

import { useState } from 'react';
import { SlInput } from '@shoelace-style/shoelace/dist/react';

function MyComponent() {
  const [value, setValue] = useState('');

  return <SlInput value={value} onSlInput={event => setValue(event.target.value)} />;
}

export default MyComponent;

If you're using TypeScript, it's important to note that event.target will be a reference to the underlying custom element. You can use (event.target as any).value as a quick fix, or you can strongly type the event target as shown below.

import { useState } from 'react';
import { SlInput } from '@shoelace-style/shoelace/dist/react';
import type SlInputElement from '@shoelace-style/shoelace/dist/components/input/input';

function MyComponent() {
  const [value, setValue] = useState('');

  return <SlInput value={value} onSlInput={event => setValue((event.target as SlInputElement).value)} />;
}

export default MyComponent;

Testing with Jest

Testing with web components can be challenging if your test environment runs in a Node environment (i.e. it doesn't run in a real browser). Fortunately, Jest has made a number of strides to support web components and provide additional browser APIs. However, it's still not a complete replication of a browser environment.

Here are some tips that will help smooth things over if you're having trouble with Jest + Shoelace.

?> If you're looking for a fast, modern testing alternative, consider Web Test Runner.

Upgrade Jest

Jest underwent a major revamp and received support for web components in version 26.5.0 when it introduced JSDOM 16.2.0. This release also included a number of mocks for built-in browser functions such as MutationObserver, document.createRange, and others.

If you're using Create React App, you can update react-scripts which will also update Jest.

npm install react-scripts@latest

Mock Missing APIs

Some components use window.matchMedia, but this function isn't supported by JSDOM so you'll need to mock it yourself.

In src/setupTests.js, add the following.

Object.defineProperty(window, 'matchMedia', {
  writable: true,
  value: jest.fn().mockImplementation(query => ({
    matches: false,
    media: query,
    onchange: null,
    addListener: jest.fn(), // deprecated
    removeListener: jest.fn(), // deprecated
    addEventListener: jest.fn(),
    removeEventListener: jest.fn(),
    dispatchEvent: jest.fn()
  }))
});

For more details, refer to Jest's manual mocking documentation.

Transform ES Modules

ES Modules are a well-supported browser standard. This is how Shoelace is distributed, but most React apps expect CommonJS. As a result, you'll probably run into the following error.

Error: Unable to import outside of a module

To fix this, add the following to your package.json which tells the transpiler to process Shoelace modules.

{
  "jest": {
    "transformIgnorePatterns": ["node_modules/?!(@shoelace)"]
  }
}

These instructions are for apps created via Create React App. If you're using Jest directly, you can add transformIgnorePatterns directly into jest.config.js.

For more details, refer to Jest's transformIgnorePatterns customization documentation.

?> Are you using Shoelace with React? Help us improve this page!