Is your feature request related to a problem? Please describe
Overlays, when not passed an explicit container prop, attach themselves to document.body. This doesn't work well if you nest overlays as the child overlay isn't actually inside the parent.
Specifically:
If you've got an overlay with rootClose that contains another overlay, interacting with the inner overlay will close the outer overlay.
If you have an overlay inside a modal you can't focus on the contents of the overlay, since they're attached to the root and outside the modal.
When an overlay is inside of another overlay, default its container to a child of the parent overlay instead of to document.body.
This would be accomplished with React context. Each overlay:
Looks in context for a parent overlay ref and uses that as its default container if present.*
Wraps its children in a provider that passes a ref to itself as the parent overlay ref to be used by its children.
*Except modals, which should always be attached to document.body.
Roughly like this:
const OverlayContext = createContext(null)
const Overlay = () => {
// ... overlay stuff
// grab parent ref from overlay context as the default container
const containerRef = useContext<OverlayContext>()
container ||= containerRef
// ... more overlay stuff
const overlayRef = useWaitForDOMRef(outerRef) // ish
// inject ref of the overlay into context for any children
<OverlayContext.Provider value={overlayRef}>
{child}
</OverlayContext.Provider>
}
Describe alternatives you've considered
Set enforceFocus to false on modals. This avoids the second problem but at the cost of allowing focus on things that are actually outside the modal, which sort of throws the baby out with the bath water.
Explicitly set the container prop on all overlays. This breaks composability - how is the overlay to know whether it's inside a modal or not?
Do the above solution with an OverlayContext in the application code. This requires either either adding a bunch of boilerplate inside and outside of every modal/overlay, or making a wrapper for each of the overlay-based components in react-bootstrap.
Additional context
Perhaps you are wondering "what sort of monster would want to nest overlays like this?" A reasonable question! One specific place this has come up is inline UI for filtering a table. Here's an example of such an element from material UI:
Here we've got a popover-esque with controls inside of it. The controls may themselves have overlays (popovers or multi-pickers) or the popover and table may be inside a modal.
Is your feature request related to a problem? Please describe
Overlays, when not passed an explicit
container
prop, attach themselves to document.body. This doesn't work well if you nest overlays as the child overlay isn't actually inside the parent.Specifically:
focus
on the contents of the overlay, since they're attached to the root and outside the modal.Repro (with bs5 react-bootstrap, also affects bs4): https://codesandbox.io/s/great-montalcini-s2yebc?file=/src/App.js
See:
836
Describe the solution you'd like
When an overlay is inside of another overlay, default its container to a child of the parent overlay instead of to
document.body
.This would be accomplished with React context. Each overlay:
container
if present.**Except modals, which should always be attached to
document.body
.Roughly like this:
Describe alternatives you've considered
enforceFocus
to false on modals. This avoids the second problem but at the cost of allowing focus on things that are actually outside the modal, which sort of throws the baby out with the bath water.container
prop on all overlays. This breaks composability - how is the overlay to know whether it's inside a modal or not?OverlayContext
in the application code. This requires either either adding a bunch of boilerplate inside and outside of every modal/overlay, or making a wrapper for each of the overlay-based components in react-bootstrap.Additional context
Perhaps you are wondering "what sort of monster would want to nest overlays like this?" A reasonable question! One specific place this has come up is inline UI for filtering a table. Here's an example of such an element from material UI:
Here we've got a popover-esque with controls inside of it. The controls may themselves have overlays (popovers or multi-pickers) or the popover and table may be inside a modal.