-
Notifications
You must be signed in to change notification settings - Fork 350
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
typed keys 2 #813
base: main
Are you sure you want to change the base?
typed keys 2 #813
Conversation
<div key={author.Author.name}> | ||
<div> <img src={author.imageUrl} /> </div> | ||
</div>; | ||
[| |
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.
I guess this exposes the downside you talked about in the description.
I think this is a pretty big limitation, as you can't write this anymore:
module Foo = {
[@react.component]
let make = (~id) => {
<div key=id />;
};
};
module Bar = {
[@react.component]
let make = () => {
let ks = Array.init(10, string_of_int);
<> {Array.map(ks, id => <Foo id />)} </>;
};
};
Check the error:
379 | <> {Array.map(ks, id => <Foo id />)} </>;
^^^^^^^^^^
Error: This expression has type < id : string > Js.t -> React.elementKeyed
but an expression was expected of type
< id : string > Js.t React.component =
< id : string > Js.t -> React.element
Type React.elementKeyed is not compatible with type React.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.
I believe the keys have to be added directly in the elements that are part of the list, so the keys added to the div
in Foo
in your example are useless.
Check the console warnings from React in this JS snippet, adapted from yours:
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 compiles fine, and it'll guarantee the keys are passed correctly to React:
module Foo = {
[@react.component]
let make = () => {
<div />;
};
};
module Bar = {
[@react.component]
let make = () => {
let ks = Array.init(10, string_of_int);
<div> {Array.map(ks, id => <Foo key=id />)->React.array} </div>;
};
};
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.
Indeed, key needs to be on the call-site: <Foo id />
-> <Foo id key="" />
Sometimes it's useful to have |
[@mel.splice] [@mel.module "react"] | ||
external createElementVariadic: | ||
(component('props), 'props, array(element)) => element = | ||
"createElement"; | ||
|
||
[@mel.module "react/jsx-runtime"] | ||
external jsxKeyed: | ||
(component('props), 'props, ~key: string=?, unit) => 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.
Having element
and elementKeyed
as a completely different type was the main problem that @davesnx and I were running into. There are legit use-cases for supplying a key to an element that is outside of an array and to prevent that at this level would be wrong IMO.
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.
There are legit use-cases for supplying a key to an element that is outside of an array
Can you share some specific examples please? I could only find 1 example on the whole Ahrefs codebase, of a function that was generating some element, and it could be easily fixed.
to prevent that at this level would be wrong IMO.
To me it's more a line of tradeoffs between approaches than right or wrong. Adding a type variable to element
has some implications as well in terms of error messages, type complexity, and general dev experience. It also will break more existing code. Some of the impact can be seen already in #812, where a lot more code has to be modified. I can imagine several 3rd party libs out there expect element
to be a type without variables.
I would love to see the approach on #812 applied on Ahrefs monorepo so that we can have more information about how a migration would look like (both for the monorepo code and the opam libs) and how the errors will show with element('a)
. That way, we can compare both approaches in a more complete way?
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.
Can you share some specific examples please? I could only find 1 example on the whole Ahrefs codebase, of a function that was generating some element, and it could be easily fixed.
The most common usages of key
in React that I have used (outside of arrays) are:
- to completely re-initialize a component which will clear the state, for example an
<input>
should be tied to a stateful value from its parent and reset if that state is changed. - help rendering performance for components with many children that are dynamic, for example if you have a component where it has static children that are rendered or not rendered based on some value and those children are the same element type (i.e. all
<div>
) then React might cause unnecessary re-renders when a components sibling changes.
I would love to see the approach on #812 applied on Ahrefs monorepo so that we can have more information about how a migration would look like (both for the monorepo code and the opam libs) and how the errors will show with element('a). That way, we can compare both approaches in a more complete way?
Yeah! This is a great call and I can try and whip something up for you!
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.
@jchavarri I've been hitting enough blockers to try and get a PoC going for you today that I'm going to take a little break and come back to it later. I did a few things on my fork of the PR as well if you want to check them out.
Alternative take to having typed keys based on the ideas from #812.
Centers the typing around the
React.array
function. The only case where keys should be added is when children are variable size arrays, and not literals. SoReact.array
might be the best spot to place the restriction.The issue is that
React.array
is used in the ppx to convert literals of children like<div> foo bar </div>
. To cover that case, a new functionReact.unsafeArray
is added.The type errors look like:
One downside is that usages of
key
outside arrays now would trigger a type error. This makes sense, because why wouldkey
be added if no needed, but who knows what could break. Also, in all cases the error happens far away from where the component is defined.E.g. for this code:
the error is: