What is React.forwardRef() ?
React.forwardRef() is a higher-order component in React that allows you to forward refs to child components. It’s particularly useful when you want to pass a ref through a component to one of its children. Here’s a brief explanation of its key aspects and use cases:
Purpose: Enables passing refs through components that don’t normally accept refs. Allows parent components to directly interact with child DOM elements or class components. Syntax:
1
2
3
const ForwardedComponent = React.forwardRef((props, ref) => {
return <ChildComponent ref={ref} {...props} />;
});
Key features:
- Takes a render function as an argument.
- The render function receives props and ref as parameters.
- Returns a React component that can be given a ref.
Common use cases:
- Forwarding refs to DOM elements in reusable component libraries.
- Exposing imperative methods of child components to parent components. Integrating with third-party DOM libraries.
Give me an example that what could go wrong if we don't have React.forwardRef() ?
Certainly! Let’s consider a scenario where not having React.forwardRef() could lead to issues. Imagine we have a custom input component that we want to use in a form, and we need to programmatically focus this input from a parent component. Here’s an example of what could go wrong without React.forwardRef():
1
2
3
4
5
6
7
8
9
10
11
const CustomInput = (props) => {
return (
<input
type="text"
className="custom-input"
{...props}
/>
);
};
export default CustomInput;
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
import React, { useRef, useEffect } from 'react';
import CustomInput from './CustomInput';
const Form = () => {
const inputRef = useRef(null);
useEffect(() => {
// Attempt to focus the input on component mount
inputRef.current.focus();
}, []);
return (
<form>
<CustomInput ref={inputRef} placeholder="Enter your name" />
<button type="submit">Submit</button>
</form>
);
};
export default Form;
In this scenario, we’re trying to pass a ref to our CustomInput component to focus it when the Form component mounts. However, this won’t work as expected because:
- The ref is being passed to CustomInput, not to the underlying input element.
- CustomInput is a functional component, which doesn’t accept ref as a prop by default.
As a result, when we try to call inputRef.current.focus(), we’ll get an error because inputRef.current is undefined.
To fix this issue, we need to use React.forwardRef(). Here’s how we could modify CustomInput to make it work:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
import React from 'react';
const CustomInput = React.forwardRef((props, ref) => {
return (
<input
type="text"
className="custom-input"
ref={ref}
{...props}
/>
);
});
export default CustomInput;
Now, the ref passed to CustomInput will be correctly forwarded to the underlying input element, and the focus() method will work as expected in the Form component.
This example illustrates that without React.forwardRef():
- We can’t pass refs through custom components to underlying DOM elements.
- It’s difficult to create reusable component libraries that allow parent components to directly interact with specific child elements.
- We lose the ability to have fine-grained control over child components in certain scenarios, like managing focus programmatically. React.forwardRef() solves these issues by providing a way to pass refs through components, maintaining both the encapsulation of custom components and the flexibility needed for certain use cases.