Skip to content

Instantly share code, notes, and snippets.

@OliverJAsh
Last active February 11, 2025 00:31
Show Gist options
  • Save OliverJAsh/e9a588e7e907101affe1a7696a25b1fd to your computer and use it in GitHub Desktop.
Save OliverJAsh/e9a588e7e907101affe1a7696a25b1fd to your computer and use it in GitHub Desktop.
<SkipRenderOnClient
shouldRenderOnClient={() => window.innerWidth <= 500}>
<MyComponent />
</SkipRenderOnClient>
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>;
};
@mhsattarian
Copy link

This is Awesome! thanks. 🔥

I'm thinking maybe adding a display: contents to the <div/> element would be a good idea.

@hi-paul
Copy link

hi-paul commented Jan 16, 2024

How about using el.remove() instead of el.innerHTML?

@denk0403
Copy link

denk0403 commented Nov 30, 2024

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