-
Notifications
You must be signed in to change notification settings - Fork 30.4k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
React components should return ReactNode, or children should be ReactElement #18051
Comments
I tried the following fixes to
but I got errors all over the place when trying to lint that I'm not sure how to fix.
|
Nope, this change is incorrect. You cannot return multiple elements in React currently. Even if you do it somewhere, most probably it it is encapsulated in a single one somehow. |
not sure if they changed react behavior, but the redux / react author showcased the ability to return this.props.children for creating thin wrappers (only to add childContext to elements), as seen in https://egghead.io/lessons/javascript-redux-passing-the-store-down-implicitly-via-context react-native does allow you to return an array of JSX elements from render, though. an work around is to wrap the |
Right, you can use
is valid and reasonable React code. |
You can return an array of elements in react 16, but TS throws an error because it thinks you cant. Edit: do this |
From my experience so far, it seems ReactNode should be valid. const Id: React.SFC<{}> = (props) => props.children
const App = ()= > <Id>This should work but instead fails</Id> Is there any chance this will be reflected in |
Without this I am unable to type React children |
I am currently wrapping |
Any updates on it? |
Not sure if this is any more or less correct than wrapping return (children as React.ReactElement); |
I had to workaround another type issue with this change. See DefinitelyTyped/DefinitelyTyped#18051
|
This is causing issues for me as well. TL;DR:I have an HOC that adds props to a given component, and I am trying to use it with a functional component that has More details:Here is a simplified version of the HOC: export interface ExtraProps {
extraProp1: string;
}
export const withExtraProps = <P extends object>(Component: React.ComponentType<P>) => {
return class extends React.Component<Omit<P, keyof ExtraProps>> {
render() {
const extraProps = {
extraProp1: 'test'
};
return (
<Component {...this.props as P} {...extraProps} />
)
}
}
} Here is a simplified version of the functional component: interface ComponentProps extends ExtraProps {
children?: React.ReactNode;
loadingComponent?: ReactElement;
}
export function ExampleComponent(props: ComponentProps) {
const { children, loadingComponent } = props;
const [loading, setLoading] = useState(false);
useEffect(() => {
const fetchData = async () => {
setLoading(true);
await someAsyncCall();
setLoading(false);
}
fetchData();
});
if (loading) {
return loadingComponent || null;
}
return children;
}
// This line errors:
export const FeatureFlag = withExtraProps(ExampleComponent); I get the following error: Argument of type '(props: ComponentProps) => {} | null | undefined' is not assignable to parameter of type 'ComponentType<ComponentProps>'.
Type '(props: ComponentProps) => {} | null | undefined' is not assignable to type 'FunctionComponent<ComponentProps>'.
Type '{} | null | undefined' is not assignable to type 'ReactElement<any, string | ((props: any) => ReactElement<any, string | ... | (new (props: any) => Component<any, any, any>)> | null) | (new (props: any) => Component<any, any, any>)> | null'.
Type 'undefined' is not assignable to type 'ReactElement<any, string | ((props: any) => ReactElement<any, string | ... | (new (props: any) => Component<any, any, any>)> | null) | (new (props: any) => Component<any, any, any>)> | null'. If I add a return type of export function ExampleComponent(props: ComponentProps): ReactElement | null {
...
...
return children; // Type 'ReactNode' is not assignable to type 'ReactElement<any, string | ((props: any) => ReactElement<any, string | ... | (new (props: any) => Component<any, any, any>)> | null) | (new (props: any) => Component<any, any, any>)> | null'.
} As with the others here, if I change it to |
I have the exact same issue as @kyrstenkelly - An HOC function being passed a The |
Also having this problem! Currently doing the suggest solution of wrapping solution with React.Fragment |
Same here interface IIntlMessageProps {
id: string;
valuesGiven?: {[key:string]:string};
}
// this doesn't work - -- TS2769 Type 'null' is not assignable to type '(nodes:ReactNodeARray) => ReactNode' | ... | undefined
export const IntlMessage: React.FC<IIntlMessageProps> = (props) => {
return (
<FormattedMessage id={props.id} values={props.valuesGiven}>
{props.children}
</FormattedMessage>
)
};
//Nope -- TS2769 Type 'Element' is not assignable to type '(nodes:ReactNodeARray) => ReactNode'
export const IntlMessage: React.FC<IIntlMessageProps> = (props) => {
return (
<FormattedMessage id={props.id} values={props.valuesGiven}>
<React.Fragment>{props.children}</React.Fragment>
</FormattedMessage>
)
};
// Nope -- TS2769 Type 'Element' is not assignable to type '(nodes:ReactNodeARray) => ReactNode'
export const IntlMessage: React.FC<IIntlMessageProps> | null = (props) => {
return (
<FormattedMessage id={props.id} values={props.valuesGiven}>
<>{props.children}</>
</FormattedMessage>
)
};
// Nope -- TS2769 Type '{}' is not assignable to type '(nodes:ReactNodeARray) => ReactNode'
export const IntlMessage: React.FC<IIntlMessageProps> = (props) => {
return (
<FormattedMessage id={props.id} values={props.valuesGiven}>
{props.children ? props.children : undefined}
</FormattedMessage>
)
};
I believe I have tried everything that was adviced here... :( Any clue ? |
The solution is you need to wrap it in fragment.
this will fix the |
If possible, I'd like to avoid fragments and make the JSX type system support the same kinds of constructs that JavaScript and Flow React supports, which helps TypeScript work better with 3rd party library patterns. Another non-ideal I find is that fragments deepen the component tree, especially in utility/wrapper components (case-in-point: react-placeholder). Casting to a I tried an approach to change that type, but ran stuck with JSX typechecking using a functional component of the new type. I still have the branch, could have sworn I made a PR asking for help, can't find it back... |
Okay, if type needs to disallow arrays of nodes it does not mean that it should disallow What the rational of prohibiting these additional types? |
Might only be relevant for function components, but here is an advice from Kent C. Dodds: |
The error is too cryptic to make that an obvious solution. Thanks |
In a similar spirit to #18051 (comment) Using a fragment to wrap |
Ran into this issue when trying to implement custom rendering for React Testing Library. The React Fragment around |
It's not a proper fix it's workaround. React can return children array or single child from children prop without fragment and it's totally legit option. Adding fragment sometimes required sometimes it's not necessary. As on @alexandrebrg comment about ESLint warning. |
For those looking for an eslint solution,
|
What if I want to keep the rule because it's useful but false positives doesn't make sense? |
Just a heads up that this has been merged into |
@mattdarveniza it was released in v7.25.0 3 days ago, but there's a bugfix to it that I'm planning on releasing as a patch in the next week. |
@ljharb my mistake! I was looking at github releases and didn't actually check the changelog or NPM 😅. Thankyou! |
It's almost the end of 2022. Over 5 years at this point! When will |
It only needs to be wrapped when it is not ReactElement const children = isValidElement(props.children) ? props.children : <Fragment children={props.children} /> |
Because at this time TypeScript's React types rejects function components that does not return null | JSX.Element. See microsoft/TypeScript#51328 RFC and follow related links to known more. Other links: * DefinitelyTyped/DefinitelyTyped#18051 * DefinitelyTyped/DefinitelyTyped#62876 * https://stackoverflow.com/a/70895599
Only renders anchor tags in a compatible device card if a device url is provided. In the product overview (aka sidebar), we want the whole card to link to #compatibility. In the compatibility section, we want the image and the title to link to a device page. The fragments in CompatibilityWrapper are used to appease typescript. See more here: DefinitelyTyped/DefinitelyTyped#18051
I'm still having issues with this. |
I migrate from create-react-script to vite + update tsconfig, set min. ES2020 and update package react, type/react, typescript. |
@types/react
package and had problems.Definitions by:
inindex.d.ts
) so they can respond.The following components
give the errors
It works if you wrap
children
in anElement
.Returning
children
is quite common when making Provider components that just add something tocontext
.The text was updated successfully, but these errors were encountered: