One advantage of using React with TypeScript is that you can easily type the props of your (function) components. You don't have to use React's PropTypes because TypeScript already has its own typing system.
In the following, I will show you how to define custom props for a component in connection with already existing props like children
.
Starting Example
PostPreview.tsx
import React from 'react';
export interface Props {
heading: string;
}
const PostPreview = (props: Props) => {
return (
<div>
<h1>{props.heading}</h1>
{props.children}
</div>
);
};
export default PostPreview;
As you can see, our PostPreview
component has a heading
property. The component is supposed to render the heading
and other components (children
) below the heading. In technical terms this is called Containment.
Because our Props
interface only defines the heading, the following error shows up:
TS2339: Property 'children' does not exist on type 'Props'.
Let me show you three different ways to solve this problem.
Solution 1: PropsWithChildren
The easiest way to solve the problem is to use the generic type PropsWithChildren
. It supports a generic type variable, so that we can use our Props
with it:
import React, {PropsWithChildren} from 'react';
export interface Props {
heading: string;
}
const PostPreview = (props: PropsWithChildren<Props>) => {
return (
<div>
<h1>{props.heading}</h1>
{props.children}
</div>
);
};
export default PostPreview;
The solution is simple, but it doesn't describe our component very well. The compiler knows that our component can have children, but it doesn't know whether our component has other tag-specific properties. We also have to remind ourselves to import React. So let's take a look at a more advanced solution.
Solution 2: React.FC
React.FC
specifies a function component and lets us also assign a type variable. It uses PropsWithChildren
behind the scenes, so we don't have to worry about connecting our Props
with it:
import React from 'react';
export interface Props {
heading: string;
}
const PostPreview: React.FC<Props> = (props) => {
return (
<div>
<h1>{props.heading}</h1>
{props.children}
</div>
);
};
export default PostPreview;
Thanks to the use of React.FC
, the TypeScript compiler now knows that our PostPreview
constant is a React component. We no longer have to think about importing React ourselves, as the compiler already prompts us to do so. However, the compiler still does not know how our component looks like in detail. It cannot tell whether it is a <div>
element or a <p>
element or something else. Hence we come to solution number three.
Solution 3: React.HTMLProps
The most specialized version is to extend React.HTMLProps
. The HTMLProps
support a variety of tags (HTMLDivElement
, HTMLFormElement
, HTMLInputElement
, etc.). Make sure that the type variable matches the outmost tag (the first tag, that is mentioned after return
):
import React from 'react';
export interface Props extends React.HTMLProps<HTMLDivElement> {
heading: string;
}
const PostPreview: React.FC<Props> = (props: Props) => {
return (
<div>
<h1>{props.heading}</h1>
{props.children}
</div>
);
};
export default PostPreview;
With this variant our component inherits all properties of a <div>
element and extends them with custom props like heading
.
Our PostPreview
component can now be used as follows:
IndexPage.tsx
import React from 'react';
import PostPreview from './PostPreview';
const IndexPage: React.FC = () => {
return (
<div>
<PostPreview heading="First Post">
<p>#1</p>
</PostPreview>
<PostPreview heading="Second Post">
<p>#2</p>
</PostPreview>
</div>
);
};
export default IndexPage;
Tested with: React v17.0.2
Get connected π
Please follow me on Twitter or subscribe to my YouTube channel if you liked this post. I would love to hear from you what you are building. π Best, Benny
Top comments (6)
If you want you can also just add children as an optional field with the type
React.ReactNode
Good idea! That's a self-made
PropsWithChildren
solution. :)React.FC does not use PropsWithChildren implicitly anymore. We need to add it explicitly to use it inside React.FC like this
React.FC<PropsWithChildren<Prop>>
to get children property in our prop or alternatively we can extend PropsWithChildren with Prop interface.Thanks for this article. I always use React.FC for children and key props.
I am glad to hear that you could find something interesting about this article. :) Are there any other topics related to React & TypeScript that you would like to read more about?
Thank you for sharing this solution with us! π Since the "title" attribute really is a global HTML element attribute, I will also update my tutorial to use "heading" instead of "title". π