Skip to content
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

Closed

Conversation

eps1lon
Copy link
Collaborator

@eps1lon eps1lon commented Oct 25, 2022

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.

const RenderNumber = () => 42;
const RenderString = () => 'react';
const RenderArray = () => [19];

// Allowed with proposed change. Previously rejected by type-checker.
<RenderNumber />;
// Allowed with proposed change. Previously rejected by type-checker.
<RenderString />;
// Allowed with proposed change. Previously rejected by type-checker.
<RenderArray />;

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.

Select one of these and delete the others:

If changing an existing definition:

  • Provide a URL to documentation or source code which provides context for the suggested changes:
  • [ ] 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.

Copy link
Contributor

@sebmarkbage sebmarkbage left a 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 />;
}

@eps1lon
Copy link
Collaborator Author

eps1lon commented Oct 25, 2022

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.

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.

@eps1lon eps1lon changed the title React: Allow anything to be rendered by components React: Allow anything to be rendered by function components Oct 26, 2022
@eps1lon eps1lon force-pushed the feat/react/yolo-render branch from 08a2b82 to cf2f891 Compare October 26, 2022 08:30
Comment on lines -95 to -96
// $ExpectType ReactElement<any, any> | null
Subscription({ model: useVirtual(), events: [EVT_FROM], children: <>Abc</> });
Copy link
Collaborator Author

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
Copy link
Collaborator Author

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

@DangerBotOSS
Copy link

Inspecting the JavaScript source for this package found some properties that are not in the .d.ts files.
The check for missing properties isn't always right, so take this list as advice, not a requirement.

af-utils__react-virtual-headless (unpkg)

was missing the following properties:

  1. EVT_RANGE
  2. EVT_RANGE

carbon__icons-react (unpkg)

was missing the following properties:

  1. Icon
  2. Caution
  3. CautionInverted
  4. CircleFill
  5. CircleStroke
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:

  1. Icon
  2. Icon

mjml-react (unpkg)

was missing the following properties:

  1. renderToJSON
  2. renderToJSON2

react (unpkg)

was missing the following properties:

  1. unstable_act

wordpress__components (unpkg)

was missing the following properties:

  1. CustomGradientPicker
  2. DuotonePicker
  3. DuotoneSwatch
  4. GradientPicker
  5. GuidePage
as well as these 4 other properties...

SearchControl, TextHighlight, ToolbarDropdownMenu, ToolbarItem

Generated by 🚫 dangerJS against cf2f891

@eps1lon eps1lon marked this pull request as ready for review October 26, 2022 13:57
@eps1lon eps1lon changed the title React: Allow anything to be rendered by function components RFC: React: Allow anything to be rendered by function components Oct 26, 2022
@typescript-bot
Copy link
Contributor

typescript-bot commented Oct 26, 2022

@eps1lon Thank you for submitting this PR!

This is a live comment which I will keep updated.

14 packages in this PR

Code Reviews

Because 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

  • ❌ No merge conflicts
  • ✅ Continuous integration tests have passed
  • ✅ A DT maintainer needs to approve changes which affect more than one package

Once every item on this list is checked, I'll ask you for permission to merge and publish the changes.

Inactive

This 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"
}

@typescript-bot
Copy link
Contributor

⚠️ There are too many reviewers for this PR change (50). Merging can only be handled by a DT maintainer.

People who would have been pinged huner2 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

@typescript-bot
Copy link
Contributor

@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?

@typescript-bot typescript-bot added Maintainer Approved Self Merge This PR can now be self-merged by the PR author or an owner labels Oct 26, 2022
@eps1lon
Copy link
Collaborator Author

eps1lon commented Oct 26, 2022

FYI please don't merge this yet. Want to have an extended period of time for comments.

@eps1lon
Copy link
Collaborator Author

eps1lon commented Oct 27, 2022

@Retsam
Copy link
Contributor

Retsam commented Nov 4, 2022

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 any.

  1. AFAIK, there's still stuff that React actually does reject, like undefined. Particularly because these sort of mistakes are fairly common:
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.

  1. It's annoying, but if someone wants to "open up" the types, they can. In the handful of cases I've found returning arrays to be very important to a particular component, I've used a trick like:
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 any back into a useful type.

@eps1lon
Copy link
Collaborator Author

eps1lon commented Nov 4, 2022

AFAIK, there's still stuff that React actually does reject, like undefined

undefined is allowed as of React 18.

This is far from ideal, but it's a lot easier to "open up" a sometimes-too-restrictive typ

Note that you would have to do this every time with async components. In that world we defeated type-checking for every case anyway.

@Retsam
Copy link
Contributor

Retsam commented Nov 4, 2022

undefined is allowed as of React 18.

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:

When we first added this error in 2017, type systems and linting in the ecosystem were not as popular or robust as they are today, and there were a lot of runtime checks like prop-types being done to help developers find unsafe issues. Since then, the community has evolved and these tools have grown in popularity, so it’s a good time to re-visit this feature to replace it with something better.

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".

Note that you would have to do this every time with async components. In that world we defeated type-checking for every case anyway.

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 React.FC style is still pretty widespread (despite some efforts to the contrary), so it's not a huge departure from the current status-quo.

I think the jsxReturn-style utility is better and would probably use that myself, and I do not think a world in which async components must use something like that is a world in which we "defeated type-checking".

@sebmarkbage
Copy link
Contributor

sebmarkbage commented Nov 4, 2022

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.

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 function Wrapper({ children }) { return children; }.

The blog post refers to catching mistakes like fall-throughs. This is something TypeScript already does by separating "void" from "undefined".

function Component({x}: {x: boolean}): React.ReactNode {
  if (x) {
    return <div />;
  }
  // typescript error, missing return statement
}
function Component({x}: {x: boolean}): React.ReactNode {
  if (x) {
    return <div />;
  }
  return undefined; // not an error
}

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.

@Retsam
Copy link
Contributor

Retsam commented Nov 4, 2022

The TS rule that catches that error --noImplicitReturns. I do think this helps justify adding undefined as a valid return type for React components. (Even though it's an optional TS config that not every project has enabled or wants to enable)

... but returning JSX as any actually breaks that rule:

This errors, with --noImplicitReturns:

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!
}

TS Playground

@Retsam
Copy link
Contributor

Retsam commented Nov 4, 2022

But to be clear, the whole undefined thing is a digression: that came up because I wrongly used it as an example for something that shouldn't be returned from a React component. I was wrong, and I'm not really trying to argue against that change.


I am trying to argue against making all JSX any. There's still stuff that shouldn't be returned from React components, right? We just improved types by fixing ReactNode to not have a random {} type in it which was letting a lot of wrong stuff through. This would do more than reverse that improvement.

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 @types/react).

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 .d.ts file :

declare namespace Belt {
  type UseMutableArrays = 1
}

Perhaps something similar can be done so that people who want to treat JSX like any can opt-into that behavior in one place?

@typescript-bot
Copy link
Contributor

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.)

@typescript-bot typescript-bot added the Unmerged The author did not merge the PR when it was ready. label Nov 5, 2022
@typescript-bot typescript-bot added Has Merge Conflict This PR can't be merged because it has a merge conflict. The author needs to update it. and removed Unmerged The author did not merge the PR when it was ready. Self Merge This PR can now be self-merged by the PR author or an owner labels Nov 15, 2022
@typescript-bot
Copy link
Contributor

@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!

@typescript-bot
Copy link
Contributor

@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.

@typescript-bot typescript-bot added the Abandoned This PR had no activity for a long time, and is considered abandoned label Nov 27, 2022
@typescript-bot
Copy link
Contributor

@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!

dgdavid added a commit to agama-project/agama that referenced this pull request Dec 29, 2022
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
@eps1lon eps1lon deleted the feat/react/yolo-render branch April 15, 2023 14:33
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Abandoned This PR had no activity for a long time, and is considered abandoned Critical package Edits multiple packages Has Merge Conflict This PR can't be merged because it has a merge conflict. The author needs to update it. Maintainer Approved Too Many Owners
Projects
None yet
6 participants