Last active
February 11, 2025 00:31
-
-
Save OliverJAsh/e9a588e7e907101affe1a7696a25b1fd to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<SkipRenderOnClient | |
shouldRenderOnClient={() => window.innerWidth <= 500}> | |
<MyComponent /> | |
</SkipRenderOnClient> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import * as React from 'react'; | |
const useIsFirstRender = (): boolean => { | |
const isFirst = React.useRef(true); | |
if (isFirst.current) { | |
isFirst.current = false; | |
return true; | |
} else { | |
return false; | |
} | |
}; | |
/** | |
* When using React's server-side rendering, we often need to render components | |
* on the server even if they are conditional on the client e.g. hidden based on | |
* window width. | |
* | |
* In order for hydration to succeed, the first client render must | |
* match the DOM (which is generated from the HTML returned by the server), | |
* otherwise we will get hydration mismatch errors. This means the component | |
* must be rendered again during the first client render. | |
* | |
* However, hydration is expensive, so we really don't want to pay that penalty | |
* only for the element to be hidden or removed immediately afterwards. | |
* | |
* For example, imagine we have two components: one for mobile and one for | |
* desktop. Usually we would render both components on the server and on the | |
* client (to avoid hydration mismatch errors) and toggle their visibility using | |
* CSS. This means we would be hydrating both components even though only one of | |
* them is currently shown to the user. | |
* | |
* `SkipRenderOnClient` conditionally skips hydrating children by removing them | |
* from the DOM _before the first client render_. Removing them before ensures | |
* hydration is successful and there are no hydration mismatch errors. | |
* | |
* Following on from the example above, this is how we would apply | |
* `SkipRenderOnClient`: | |
* | |
* ```tsx | |
* <SkipRenderOnClient shouldRenderOnClient={() => window.innerWidth <= 500}> | |
* <MyMobileComponent className={styles.showOnMobile} /> | |
* </SkipRenderOnClient> | |
* <SkipRenderOnClient shouldRenderOnClient={() => window.innerWidth > 500}> | |
* <MyDesktopComponent className={styles.showOnDesktop} /> | |
* </SkipRenderOnClient> | |
* ``` | |
*/ | |
export const SkipRenderOnClient: React.FC<{ | |
children: React.ReactNode; | |
shouldRenderOnClient: () => boolean; | |
}> = ({ children, shouldRenderOnClient }) => { | |
const id = React.useId(); | |
const isClient = typeof window !== 'undefined'; | |
const isFirstRender = useIsFirstRender(); | |
if (isClient && isFirstRender && shouldRenderOnClient() === false) { | |
const el = document.getElementById(id); | |
if (el !== null) { | |
el.innerHTML = ''; | |
} | |
} | |
const shouldRender = isClient ? shouldRenderOnClient() : true; | |
return <div id={id}>{shouldRender && children}</div>; | |
}; |
How about using el.remove() instead of el.innerHTML?
How about using el.remove() instead of el.innerHTML?
el.remove()
would remove the wrapper div
element from the document as well. This would cause a hydration error since the wrapper div
is not part of what's being conditionally rendered. innerHTML
on the other hand only affects the children.
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
This is Awesome! thanks. 🔥
I'm thinking maybe adding a
display: contents
to the<div/>
element would be a good idea.