Passing a component as a context value
I want to implement a dynamic sidebar which can show different content based on where you are in the component hierarchy. To achieve this I was trying to use the context API to supply the sidebar content. My goal was that descendants could either provide their own sidebar content or use one defined higher up in the hierarchy.
Trying something like this I keep getting hydration errors (see attached image). Is this not possible? Any alternative approaches to achieve my goal that you can think about?

14 Replies
The idea seems solid in my head π Had to throw that pun in
What's
SidebarNavProvider
look like?I think it's because
value
is evaluated in a different context than children usually are
The element is probably being created in the provider but then mounted somewhere else and it's not happyHmmm. Any ideas for a workaround?
One option would be to provide it as an accessor, like
sidebarNav={() => ...}
Let me try that
oh for a situation like this you may want to look into using a portal actually
that'd let you properly portal your element from the page into the sidebar, then you'd use the context to provide the sidebar ref to the portal on the page
That sounds like a plan... I'm still a bit new to solid so I'm gonna have to look up the various pieces. I don't necessarily want to burden you with providing an example snippet.
Mainly around the ref part π
The idea is you create a signal in the context, use the setter of the signal as the ref of the element you want to be the portal container, and then use the getter as the
mount
prop of the portal so that it knows where to put the elementI think that part is similar to my current plan of displaying the actual sidebar content. The thing I was trying to solve with the context is to get the nearest fallback sidebar from the hierarchy if the current nested page does not provide its own.
But I guess I could work around that by doing something like:
- Setup a store for sidebars which are indexed by some id
- Provide the id in the context
- Use the id for looking up the correct sidebar to show
- Then children can add sidebars with a unique id
Since the id would be a string it would probably not run into the hydration issues. But my original idea was cool. Sad it didn't work. I guess it would work if I wasn't using SSR.
Oh wait. I think I may have misunderstood a small part here about the signal βinβ the context. Let me try that out
I think the Accessor approach would work with SSR, tbh the Portal approach may not even server render at all lol
So what I ended up doing was providing the sidebar to the context as an acessor, but only setting it onMount so that I don't have hydrations errors where it is being used.
Context provider stuffs:
Home layout:
Actual Usage:
Thanks for the help @Brendonovich
Passing Solid's
JSX.Element
around routinely causes issues:
https://discord.com/channels/722131463138705510/1238643925988937820
() => JSX.Element
is less an accessor but a means of delaying rendering/execution, i.e. not creating the instance until it is at its final destination (with the supporting environment).