-
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
RFC: React: Allow anything to be rendered by function components #62876
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is the unfortunate but pragmatic solution. It'll still catch errors on props but if you pass the element to the wrong thing or if you pass a component that returns something other than ReactNode.
It'll also still work with things that explicitly type the return value as an ReactElement which is still correct if that's all you're returning.
function Component(): React.ReactElement {
return <div />;
}
To be clear, the opposite is also true: Returning a JSX element from something that shouldn't would now also be allowed: function compute(): number {
// that would make no sense
return <div />;
} So codebases relying more on type inference in return values, might see a drop in type coverage. |
08a2b82
to
cf2f891
Compare
// $ExpectType ReactElement<any, any> | null | ||
Subscription({ model: useVirtual(), events: [EVT_FROM], children: <>Abc</> }); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Subscription
is a component so this wouldn't be allowed at runtime anyway. Refactored to use JSX and removed the $ExpectType
since that's testing how TypeScript behaves not the library.
@@ -23,29 +23,29 @@ import { | |||
Plan, | |||
} from "@carbon/icons-react"; | |||
|
|||
<UserAccess />; // $ExpectType Element |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Remove the $ExpectType
since they're testing TypeScript not this library
Inspecting the JavaScript source for this package found some properties that are not in the .d.ts files. af-utils__react-virtual-headless (unpkg)was missing the following properties:
carbon__icons-react (unpkg)was missing the following properties:
as well as these 13 other properties...Critical, CriticalSeverity, LowSeverity, SquareFill, Icon, Caution, CautionInverted, CircleFill, CircleStroke, Critical, CriticalSeverity, LowSeverity, SquareFill carbon__pictograms-react (unpkg)was missing the following properties:
mjml-react (unpkg)was missing the following properties:
react (unpkg)was missing the following properties:
wordpress__components (unpkg)was missing the following properties:
as well as these 4 other properties...SearchControl, TextHighlight, ToolbarDropdownMenu, ToolbarItem |
@eps1lon Thank you for submitting this PR! This is a live comment which I will keep updated. 14 packages in this PR
Code ReviewsBecause this is a widely-used package, a DT maintainer will need to review it before it can be merged. You can test the changes of this PR in the Playground. Status
Once every item on this list is checked, I'll ask you for permission to merge and publish the changes. InactiveThis PR has been inactive for 31 days — it is considered abandoned, and therefore closed! Diagnostic Information: What the bot saw about this PR{
"type": "info",
"now": "-",
"pr_number": 62876,
"author": "eps1lon",
"headCommitOid": "cf2f891505fada9a1a589fc2765dbb8be5725083",
"mergeBaseOid": "3719cc719a31f13c913615b747270e02113e5b7b",
"lastPushDate": "2022-10-26T08:30:19.000Z",
"lastActivityDate": "2022-11-04T16:17:22.000Z",
"hasMergeConflict": true,
"isFirstContribution": false,
"tooManyFiles": false,
"hugeChange": false,
"popularityLevel": "Critical",
"pkgInfo": [
{
"name": "af-utils__react-virtual-headless",
"kind": "edit",
"files": [
{
"path": "types/af-utils__react-virtual-headless/af-utils__react-virtual-headless-tests.tsx",
"kind": "test"
}
],
"owners": [
"huner2"
],
"addedOwners": [],
"deletedOwners": [],
"popularityLevel": "Well-liked by everyone"
},
{
"name": "carbon__icons-react",
"kind": "edit",
"files": [
{
"path": "types/carbon__icons-react/carbon__icons-react-tests.tsx",
"kind": "test"
}
],
"owners": [
"metonym"
],
"addedOwners": [],
"deletedOwners": [],
"popularityLevel": "Well-liked by everyone"
},
{
"name": "carbon__pictograms-react",
"kind": "edit",
"files": [
{
"path": "types/carbon__pictograms-react/carbon__pictograms-react-tests.tsx",
"kind": "test"
}
],
"owners": [
"metonym"
],
"addedOwners": [],
"deletedOwners": [],
"popularityLevel": "Well-liked by everyone"
},
{
"name": "mjml-react",
"kind": "edit",
"files": [
{
"path": "types/mjml-react/mjml-react-tests.tsx",
"kind": "test"
}
],
"owners": [
"henrinormak",
"IanEdington"
],
"addedOwners": [],
"deletedOwners": [],
"popularityLevel": "Well-liked by everyone"
},
{
"name": "react-aria-menubutton",
"kind": "edit",
"files": [
{
"path": "types/react-aria-menubutton/index.d.ts",
"kind": "definition"
}
],
"owners": [
"forabi",
"crohlfs",
"karmats"
],
"addedOwners": [],
"deletedOwners": [],
"popularityLevel": "Well-liked by everyone"
},
{
"name": "react-form",
"kind": "edit",
"files": [
{
"path": "types/react-form/v1/index.d.ts",
"kind": "definition"
},
{
"path": "types/react-form/v2/index.d.ts",
"kind": "definition"
}
],
"owners": [
"cameron-mcateer",
"TiuSh",
"Toliak"
],
"addedOwners": [],
"deletedOwners": [],
"popularityLevel": "Well-liked by everyone"
},
{
"name": "react-key-handler",
"kind": "edit",
"files": [
{
"path": "types/react-key-handler/react-key-handler-tests.tsx",
"kind": "test"
}
],
"owners": [
"pmqueiroz"
],
"addedOwners": [],
"deletedOwners": [],
"popularityLevel": "Well-liked by everyone"
},
{
"name": "react-syntax-highlighter",
"kind": "edit",
"files": [
{
"path": "types/react-syntax-highlighter/react-syntax-highlighter-tests.tsx",
"kind": "test"
}
],
"owners": [
"NoHomey",
"guoyunhe",
"anirban09",
"michaelyuen",
"DoK6n"
],
"addedOwners": [],
"deletedOwners": [],
"popularityLevel": "Critical"
},
{
"name": "react",
"kind": "edit",
"files": [
{
"path": "types/react/index.d.ts",
"kind": "definition"
},
{
"path": "types/react/test/index.ts",
"kind": "test"
},
{
"path": "types/react/test/tsx.tsx",
"kind": "test"
}
],
"owners": [
"johnnyreilly",
"bbenezech",
"pzavolinsky",
"ericanderson",
"DovydasNavickas",
"theruther4d",
"guilhermehubner",
"ferdaber",
"jrakotoharisoa",
"pascaloliv",
"hotell",
"franklixuefei",
"Jessidhia",
"saranshkataria",
"lukyth",
"eps1lon",
"zieka",
"dancerphil",
"dimitropoulos",
"disjukr",
"vhfmag",
"hellatan",
"priyanshurav"
],
"addedOwners": [],
"deletedOwners": [],
"popularityLevel": "Critical"
},
{
"name": "react_ujs",
"kind": "edit",
"files": [
{
"path": "types/react_ujs/react_ujs-tests.tsx",
"kind": "test"
}
],
"owners": [
"TimothyGu"
],
"addedOwners": [],
"deletedOwners": [],
"popularityLevel": "Well-liked by everyone"
},
{
"name": "rr-notifications",
"kind": "edit",
"files": [
{
"path": "types/rr-notifications/rr-notifications-tests.ts",
"kind": "test"
}
],
"owners": [
"RobbieGM"
],
"addedOwners": [],
"deletedOwners": [],
"popularityLevel": "Well-liked by everyone"
},
{
"name": "simple-react-lightbox",
"kind": "edit",
"files": [
{
"path": "types/simple-react-lightbox/simple-react-lightbox-tests.tsx",
"kind": "test"
}
],
"owners": [
"guilhermefront"
],
"addedOwners": [],
"deletedOwners": [],
"popularityLevel": "Well-liked by everyone"
},
{
"name": "wordpress__components",
"kind": "edit",
"files": [
{
"path": "types/wordpress__components/icon/index.d.ts",
"kind": "definition"
}
],
"owners": [
"dsifford",
"sirreal",
"p-jackson",
"sarayourfriend",
"michaelhthomas",
"manzoorwanijk",
"bastolen"
],
"addedOwners": [],
"deletedOwners": [],
"popularityLevel": "Well-liked by everyone"
},
{
"name": "yaireo__tagify",
"kind": "edit",
"files": [
{
"path": "types/yaireo__tagify/test/yaireo__tagify-react-tests.tsx",
"kind": "test"
}
],
"owners": [
"Brakebein",
"blutorange"
],
"addedOwners": [],
"deletedOwners": [],
"popularityLevel": "Well-liked by everyone"
}
],
"reviews": [
{
"type": "approved",
"reviewer": "johnnyreilly",
"date": "2022-10-26T15:15:27.000Z",
"isMaintainer": true
},
{
"type": "stale",
"reviewer": "sebmarkbage",
"date": "2022-10-25T16:46:14.000Z",
"abbrOid": "08a2b82"
}
],
"mainBotCommentID": 1292095805,
"ciResult": "pass"
} |
People who would have been pingedhuner2 metonym henrinormak IanEdington forabi crohlfs karmats cameron-mcateer TiuSh Toliak pmqueiroz NoHomey guoyunhe anirban09 michaelyuen DoK6n johnnyreilly bbenezech pzavolinsky ericanderson DovydasNavickas theruther4d guilhermehubner ferdaber jrakotoharisoa pascaloliv hotell franklixuefei Jessidhia saranshkataria lukyth zieka dancerphil dimitropoulos disjukr vhfmag hellatan priyanshurav TimothyGu RobbieGM guilhermefront dsifford sirreal p-jackson sarayourfriend michaelhthomas manzoorwanijk bastolen Brakebein blutorange |
@sebmarkbage Thank you for reviewing this PR! The author has pushed new commits since your last review. Could you take another look and submit a fresh review? |
FYI please don't merge this yet. Want to have an extended period of time for comments. |
Alternate proposal: RFC: Consult new JSX.ElementType for valid JSX element types |
Personally, I don't think I'm a big fan. It's unfortunate that the TS typings don't match the real allowed return values, but given the choice, I think I'd rather the typings erred on the side of too-restrictive than just making the type
const MyComponent = () => { // meant to use ( ) not { }
<div>Hello</div>
} If I'm understanding this change, this code would type-check and that seems particularly unfortunate.
function jsxReturn(x: ReactNode[]) { return x as JSX.Element; }
const myComponent = () => {
return jsxReturn([
<div>Hello</div>,
<div>World</div>
]);
} This is far from ideal, but it's a lot easier to "open up" a sometimes-too-restrictive type than it is to wrangle |
Note that you would have to do this every time with async components. In that world we defeated type-checking for every case anyway. |
Sorry, did a quick scan of the React 18 features page, but missed this. I do note that part of the justification for this change is:
It seems unfortunate that an issue justified - in part - on the basis that "the type system can handle it" is leading to the typings being significantly loosened to not handle the issue. I get that they're more focused on eslint (though eslint was pretty widespread in 2017, IME), but it still seems not ideal for something to go from a type error to "maybe a lint error if you add the right rule".
Async components are a brand-new and relatively niche part of the React ecosystem, and it's unclear how widespread they will be. Making the typing experience worse for the entire React ecosystem in order to improve the experience of a small part of it seems... unfortunate. If people want the experience of this PR now, they can have it with four characters: const MyAsyncComponent = (props: MyAsyncComponentProps): any => {
} Or if you want something that looks more respectable: // In the style of React.FC
type AC<T> = (props: T) => any
const MyAsyncComponent: AC<MyAsyncComponentProps> = (props) => {
} The I think the |
The main justification for the change was mainly to simplify types in TypeScript/Flow so that there's not a difference between other child positions. There should only be one type. E.g. the equivalent of an identity function should be totally valid The blog post refers to catching mistakes like fall-throughs. This is something TypeScript already does by separating "void" from "undefined".
So undefined is allowed but void is not. The runtime can't catch this difference and so it's better to let TypeScript handle it than React. This can be caught anytime the component is annotated either way. |
The TS rule that catches that error ... but returning JSX as any actually breaks that rule: This errors, with function Component({x}: {x: boolean}) {
// typescript error: "Not all code paths return a value"
if (x) {
return <div />;
}
} But this doesn't, and I think this reflects the new behavior under this proposed change: const jsxIsAny: any = <div /> // simulate changing the type of JSX to any
function Component({x}: {x: boolean}) {
if (x) {
return jsxIsAny;
}
// no error!
} |
But to be clear, the whole I am trying to argue against making all JSX I understand why some people would want this behavior, but I don't, and it's a lot easier for the people that want loose types to have them. If this merges, it doesn't seem like I'll have any way to get back to the type behavior I expect (other than to pin to an older version of Maybe this can be done behind some sort of toggle? For example, I know the ts-belt repo has some of their behavior behind a type toggle that can be defined in a declare namespace Belt {
type UseMutableArrays = 1
} Perhaps something similar can be done so that people who want to treat JSX like |
Re-ping @eps1lon / @huner2, @metonym, @henrinormak, @IanEdington, @forabi, @crohlfs, @karmats, @cameron-mcateer, @TiuSh, @Toliak, @pmqueiroz, @NoHomey, @guoyunhe, @anirban09, @michaelyuen, @DoK6n, @bbenezech, @pzavolinsky, @ericanderson, @DovydasNavickas, @theruther4d, @guilhermehubner, @ferdaber, @jrakotoharisoa, @pascaloliv, @Hotell, @franklixuefei, @Jessidhia, @saranshkataria, @lukyth, @zieka, @dancerphil, @dimitropoulos, @disjukr, @vhfmag, @hellatan, @priyanshurav, @TimothyGu, @RobbieGM, @guilhermefront, @dsifford, @sirreal, @p-jackson, @sarayourfriend, @michaelhthomas, @manzoorwanijk, @bastolen, @Brakebein, @blutorange: This PR has been ready to merge for over a week, and I haven't seen any requests to merge it. I will close it on Nov 25th (in three weeks) if this doesn't happen. (If there's no reason to avoid merging it, please do so. Otherwise, if it shouldn't be merged or if it needs more time, please close it or turn it into a draft.) |
@eps1lon Unfortunately, this pull request currently has a merge conflict 😥. Please update your PR branch to be up-to-date with respect to master. Have a nice day! |
@eps1lon I haven't seen any activity on this PR in more than three weeks, and it still has problems that prevent it from being merged. The PR will be closed on Dec 4th (in a week) if the issues aren't addressed. |
@eps1lon To keep things tidy, we have to close PRs that aren't mergeable and don't have activity in the last month. No worries, though — please open a new PR if you'd like to continue with this change. Thank you! |
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
FYI please don't merge this yet. Want to have an extended period of time for comments.
Closes #18051
Closes #18912
Allows returning
any
from function components.In React 18 (and even more so in canary releases), there are now way more things that can be returned from render than can't be. Considering microsoft/TypeScript#14729 and microsoft/TypeScript#21699 have been open for years it seems more reasonable to disable type-checking for render return values completely.
The original thinking of TypeScript has been to be pragmatic in the things we can type-check and just give up on the edges that are commonly used at runtime but can't be (performantly) expressed in types. Return values from render seem to fit such a case.
Alternatives:
Not doing the
ReactNode = ReactNode | Promise<ReactNode>
yet since that '@next' only. Just want to test the waters with this change.Please fill in this template.
@ts-expect-error
where we relied on return types being inferred)npm test <package to test>
.Select one of these and delete the others:
If changing an existing definition:
[ ]If this PR brings the type definitions up to date with a new version of the JS library, update the version number in the header.