You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
Whenever the root elements have different types, React will tear down the old tree and build the new tree from scratch.
In most cases, <LightMode /> and <DarkMode /> are the root or near to the root component and re-building its child components takes high cost.
In addition, forcing the children to re-render causes some undesirable/unexpected effects.
Please see the exapmle below.
In the root (App.tsx), the light/dark modes are toggled by changing the root <LightMode /> and <DarkMode /> components.
In the child component (DataFetcher.tsx), it fetches data from an API in async way with useEffect. In this example, the API is dummy and configured to take 1 second for example.
App.tsx
importReact,{useState}from'react';import{LightMode,DarkMode,Button}from'sancho';importDataFetcherfrom'./DataFetcher';constApp: React.FC=()=>{const[darkMode,setDarkMode]=useState(false);constMode=darkMode ? DarkMode : LightModereturn(<Mode><div><DataFetcher/><ButtononClick={()=>setDarkMode(!darkMode)}>Toggle Dark Mode</Button></div></Mode>);}exportdefaultApp;
When you run this code, you will see that each time you toggle the color mode, the fetching occurs and there are 1s delays for the fetched data to be shown:
This is because, as I wrote above, changing the root component (<LightMode /> and <DarkMode />) forces the child component to re-render and it triggers the fetching in useEffect.
To solve this, I suggest that you provide a single theming component and switching color mode is done by changing its props.
I think ColorMode component defined in https://github.com/bmcmahen/sancho/blob/master/src/Theme/Providers.tsx , which is currently not public is good fit for this purpose.
For example, if the App.tsx is changed like below, the problem is solved.
importReact,{useState,useRef}from'react';import{useTheme,Theme,ThemeColors,ThemeProvider,Button}from'sancho';importDataFetcherfrom'./DataFetcher';constColorMode= ... // ColorMode and its deps are now not public and just copied from the source code./** * Main app */constApp: React.FC=()=>{const[darkMode,setDarkMode]=useState(false);consttheme=useTheme();constcolors=darkMode ? theme.modes.dark : theme.modes.light;constcolorModeRef=useRef();// I think this ref should be optional when `ColorMode` is public API.return(<ColorModecolors={colors}ref={colorModeRef}><div><DataFetcher/><ButtononClick={()=>setDarkMode(!darkMode)}>Toggle Dark Mode</Button></div></ColorMode>);}exportdefaultApp;
This shows that unnecessary re-rendering of the child does not occur.
The text was updated successfully, but these errors were encountered:
I'm actually surprised that changing the color mode actually causes the component to remount, but it makes sense given your explanation. That's definitely not ideal, and really shouldn't be necessary. I'd like to maintain the current API if possible - so let me explore potential ways to do that, and if I can't figure anything else we can fallback to your solution.
I think the current way of switching dark/light modes described in https://sancho-ui.com/#theme has a drawback that it forces the child components to re-render
bacause the top-level component changes to the different one, e.g. from
LightMode
toDarkMode
and vice versa.This React's behavior is described in
https://reactjs.org/docs/reconciliation.html#elements-of-different-types
In most cases,
<LightMode />
and<DarkMode />
are the root or near to the root component and re-building its child components takes high cost.In addition, forcing the children to re-render causes some undesirable/unexpected effects.
Please see the exapmle below.
In the root (
App.tsx
), the light/dark modes are toggled by changing the root<LightMode />
and<DarkMode />
components.In the child component (
DataFetcher.tsx
), it fetches data from an API in async way withuseEffect
. In this example, the API is dummy and configured to take 1 second for example.App.tsx
DataFetcher.tsx
When you run this code, you will see that each time you toggle the color mode, the fetching occurs and there are 1s delays for the fetched data to be shown:
This is because, as I wrote above, changing the root component (
<LightMode />
and<DarkMode />
) forces the child component to re-render and it triggers the fetching inuseEffect
.To solve this, I suggest that you provide a single theming component and switching color mode is done by changing its props.
I think
ColorMode
component defined in https://github.com/bmcmahen/sancho/blob/master/src/Theme/Providers.tsx , which is currently not public is good fit for this purpose.For example, if the
App.tsx
is changed like below, the problem is solved.This shows that unnecessary re-rendering of the child does not occur.
The text was updated successfully, but these errors were encountered: