-
Notifications
You must be signed in to change notification settings - Fork 651
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: An alternative to ReactDOM.findDOMNode #606
Comments
The 3rd solution just came to my mind while I was writing the RFC, I might have overlooked something disqualifying. |
I'm in favor of proposal two:
Number 3 is problematic since it's easier to spot wrong usage. By requiring |
With third solution we can't get a ref direct to element, we have to use one provided by transition import React from 'react'
import { Transition } from 'react-transition-group'
const StrictModeTransition = () => {
const divRef = React.useRef(null)
return (
<Transition appear in>
{(status, childRef) => (
// how we pass `divRef` here? we have to use `childRef` instead
<div ref={childRef}>
{status}
</div>
)}
</Transition>
)
} |
Let's think this problem also from perspective of adding hooks version of library e.g. |
@iamandrewluca we can do this
|
Here is an implementation of merged refs |
The solution in #606 (comment) is quite excessive since it re-creates ref callbacks on every render which means the we need to cleanup and re-attach on every render. Solution number two does not have these problems since the ref callback responsible for forking can be memoized. |
By "solution number two" you're talking about |
Anyway, regardless of which strategy for merging refs is best to use, this is possible and should be done by the user if they need to. |
I think a combo of both is probably the most common and ergonomic? const divRef = React.useRef(null);
return (
<Transition appear in ref={ref}>
{(status, innerRef) => (
<div ref={innerRef}>
{status}
</div>
)}
</Transition>
) Alternatively, i've been dragging my feet on this mostly because i think that Transition likely shouldn't be a component at all and should look more like const status = useTransition({
in,
appear: true,
})
// use an effect instead of callbacks
useEffect(() => {
if (status === 'exited') onExited()
}, [status]) |
Definitely, I also think it shouldn't be a component, IMO same goes for CSSTransition, but maybe a unified shape of the result: const transition = useCssTransition({
in: inProp,
appear: true,
classNames: 'fade'
})
return transition.status !== 'exited'
? (
<div
ref={transition.ref}
className={transition.className}
/>
: null EDIT: Or possibly not having |
About the proposition to pass |
I have another solution, might not be the popular one because it requires to use Here the patch we applied with And here is how it goes in general function Main() {
return (
<Transition in appear>
{transitionStatus => {
return transitionStatus !== 'exited' ? <h1>Some title</h1> : null
}}
</Transition>
)
} and how it goes when your html is deeper in components const Home = forwardRef(function Home({transitionStatus}, forwardedRef) {
return (
<div ref={forwardedRef}>
<Background transitionStatus={transitionStatus} />
<Foreground transitionStatus={transitionStatus} />
</div>
)
})
function Main() {
return (
<Transition in appear>
{transitionStatus => <Home transitionStatus={transitionStatus} />}
</Transition>
)
} So if the user is in the 1st case, no ref to handle :) |
I can understand the appeal of it because it can fix a lot of warnings without users having to do anything, but there will be some cases where forwarding ref is necessary, and for that reason I think it's better if passing the ref was explicit. |
Yep, I know That said I like solution 3 for it's simplicity and proximity to what I'm using now. I really think it is the best DX The only drawback in 3 that 2 solves, is that you are not the creator of the Maybe mixing solution 2 into 3 by allowing the user to give the ref to prevent the lib to create it for you, as an optional prop would be the better of the two worlds. It would only be for advanced usage. |
It's not that, with the 3rd solution we'd need
Why? What's the use case? 🤔
What would the usage look like? |
const divRef = React.useRef(null);
return (
<Transition appear in domRef={divRef}>
{(status, innerRef) => (
// here `innerRef` is not created by `Transition`
// but it is `divRef` that was passed to `Transition`
// so `innerRef === divRef`
// if `domRef` is not passed to `Transition`
// then `innerRef` will be created internaly by `Transition`
<div ref={innerRef}>
{status}
</div>
)}
</Transition>
) But here with all this examples we forget about the case when children is not a function. Or it will be dropped children as react element? |
Again, very advanced usage and maybe very rare but I was thinking about others libs that need a Also it could be added as a feature in a N+1 release, minor version, not breaking the API :)
@iamandrewluca was faster, see above ⬆️ |
So I keep my vote on solution 3 and hope for a solution 2 as a nice addition :) |
To keep in mind that 3rd solution does not work for this case <Transition>
<div>content</div>
</Transition> While 2nd solution will work for this case. |
Why not keeping the actual code/DX when |
For advanced usage that you described I would like to allow this behavior: const childRef = React.useRef(null)
return (
<Transition in appear ref={childRef}>
{(status, ref) => (
// ref === childRef
<div ref={ref} />
)}
</Transition>
) but not this: const childRef = React.useRef(null)
return (
<Transition in appear ref={childRef}>
{/* ref is passed internally */}
<div />
</Transition>
) because someone with the edge case that you described, where they can't combine refs (I don't know how that can happen, but let's say), might attempt to do this: const childRef = React.useRef(null)
const anotherRef = React.useRef(null)
return (
<Transition in appear ref={anotherRef}>
<div ref={childRef} />
</Transition>
) Now refs would be combined internally and things wouldn't work as expected. The 1st snippet is combining solutions 2 and 3 because you're the owner of the ref. What do you think? The reason why I want to avoid passing ref internally is because I want to make a step towards the potential new hooks API. |
Agreed ! |
I love it! It has the key benefits of both solutions IMO. We'll see what others think. |
Just to set a deadline for objections: Wednesday, then I'll start working on this. @iamandrewluca if you're particularly interested in pivoting from your PR to implement this in a separate PR, let me know. 😉 |
Ok. So from what I understood. This is the desired API? function App() {
const divRef = React.useRef(null)
return (
<Transition in appear domRef={divRef}>
{/* ref === divRef */}
{(status, ref) => <div ref={ref}>{status}</div>}
</Transition>
)
}
function App1() {
return (
<Transition in appear>
{/* ref is created internally in Transition */}
{(status, ref) => <div ref={ref}>{status}</div>}
</Transition>
)
}
function App2() {
return (
<Transition in appear>
{/* still use findDOMNode */}
<div />
</Transition>
)
} |
Almost, I think it's cleaner if the outside ref object is being passed through <Transition in appear ref={divRef}> Simply because it's the only ref that |
This should imply a breaking change. https://reactjs.org/docs/forwarding-refs.html#note-for-component-library-maintainers There are may be some users who use |
Oh, that's right 🤦 I've been dealing with function components so much that I forgot about what ref does for class components. Now that the And, now that some time has passed, for the 3rd solution I no longer feel that the semantic benefit (scoping the ref only to children) outweighs the downside of its cluttered syntax (having to use a render prop and skip the often unused I would like to proceed with the 2nd solution, #559, I believe it's the easiest to use. Considering that I have made a decision, I'm closing this RFC and continuing the conversation in that PR, but people are still free to share their opinions about this here. |
@silvenon I investigated this more, and it seems we don't have to add more code, we actually kind of have to remove code, and simplify API. What we are trying to do in most components, is to get a ref from the user, and pass it back to him. ReplaceTransition
|
Yes, now that we’d be passing in the ref ourselves, returning it from callbacks doesn’t really make sense, but in my opinion adding |
But it makes sense to return the node back only if |
Totally agree.
Is not what I meant, but actually it makes sense. Then I'll go with the implementation of |
Hi everyone 👋
The time has come to implement an alternative to
ReactDOM.findDOMNode
, as explained in React docs:The goal is to come up with an API that is optional and falls back to
ReactDOM.findDOMNode
. I think that we should eventually change the entire API into something more sustainable, probably hooks, but that's a separate discussion.So far we have three proposed solutions:
1.
findDOMNode
, a function that returns a node (#457)@eps1lon's solution would look like this:
2.
domRef
, a ref object (#559)@iamandrewluca's solution would look like this:
3. ref object as a 2nd argument in render prop
My solution would look like this:
So let's begin!
Updates
For those who are just catching up with the discussion, one notable proposition is to extend the 3rd solution with the ability to pass your own ref, which would then be forwarded to the render prop:
This way we can avoid having to merge refs, which is the key benefit of the 2nd solution. (If you don't pass a ref to
Transition
, it would create one internally.)The text was updated successfully, but these errors were encountered: