I'm kind of new to SolidJS, and now I'm trying to work out how SUID's ThemeProvider
differs from SolidJS's (e.g. this docs provider/context example). A simple example would be how to switch between light and dark modes from a child component. What's the proper way?
I've tried something like this so far, but it hasn't worked, I thought signal reactivity would be enough, but I think I'm misunderstanding a core concept somewhere.
If the example below is successful, then the background should switch between black and white.
import { createSignal } from "solid-js";
import { createTheme } from "@suid/material";
import { green, orange } from "@suid/material/colors";
const lightTheme = createTheme({
palette: {
mode: "light",
background: {
default: "#000",
},
}
});
const darkTheme = createTheme({
palette: {
mode: "dark",
background: {
default: "#fff",
},
}
});
const [theme, setTheme] = createSignal(darkTheme);
function toggleTheme() {
theme() === lightTheme
? setTheme(darkTheme)
: setTheme(lightTheme);
}
function ChildComponent() {
return (
<FormControlLabel
onChange={(e) => {
toggleTheme();
}}
label=""
control={<Switch />}
/>
)
}
function App() {
return (
<>
<CssBaseline />
<ThemeProvider theme={theme()}>
<ChildComponent />
</ThemeProvider>
</>
);
}
The way SolidJS works is pretty different from React. It tracks updates or reactivity based on function calls essentially, and we've to make use of reactive primitives in SolidJS to be able to make sure that the UI is reactive. So, you'll need to create your theme by using a reactive primitive.
The SolidJS's guide to reactivity outlines all the types of reactive primitives and how they work.
Reactivity in SolidJS only works if any reactive primitive is accessed in the a reactive scope, so to say, which is, any other reactive primitive (using a signal inside of a effect for instance) or inside of the JSX code. If this is the case, re-render occurs, and updates are reflected. To quote SolidJS docs -
- All reactivity is tracked from function calls whether directly or hidden beneath getter/proxy and triggered by property access. This means where you access properties on reactive objects is important.
- Components and callbacks from control flows are not tracking scopes and only execute once. This means destructuring or doing logic top-level in your components will not re-execute. You must access these Signals, Stores, and props from within other reactive primitives or the JSX for that part of the code to re-evaluate.
The Signals & Memos return an Accessor<T>
type which, when called, in a reactive scope causes an update. You'd think -
But I'm using the signal inside of the JSX (in my example), and that's a reactive context, so what's wrong?
Yes, you're using it like so, and it's not your fault that it doesn't work, it's SUID
's fault. Remember the rule about destructuring from the docs -
This means destructuring or doing logic top-level in your components will not re-execute.
This is the reason it doesn't work. You see, the SUID
library is just MUI using SolidJS instead of React. The architecture is same as MUI. To inject styles into all components, the MUI library makes use of styled
utility, if no ThemeProvider
is used by user, a default theme is injected, otherwise, if a ThemeProvider
is there, useTheme
is used to get that theme and create components. Same is the case with SUID
, it also makes use of useTheme
under the hood, for creating the components, which is nothing but destructuring like so -
const theme = useTheme();
So,the theme is getting updated, but there are no updates. Let's move on to debugging step by step.
In your example, you're trying to do it way it's done in React. I don't blame you. Let's use your example to compare with React to see what's really happening.
Here's an example of React MUI, and here's an example of SolidJS with SUID. When we click on the button in front of "From Signal" the handler fires in both the examples. However, the behavior isn't same. What's missing?
We see that in MUI, the theme state reflects properly on buttons and the theme mode reflects properly as well, however, in SolidJS the value of theme mode reflects properly from the button using the signal value but the button using theme from useTheme
inside of the context, doesn't seem to be able to get it right. But it actually is getting the correct theme, SolidJS passes the objects as is through the useContext
hook. This is happening because of destructuring, the UI doesn't re-render. Let me prove that to you.
Let's change some piece of code in our SolidJS example, to make it look like this, we'll pass the theme
signal wrapped inside of an array instead of passing it plainly in the provider -- with that we'll need to make use of vanilla html components, because with an array type of theme, the lib components will error from not being able to access the theme.
What? Both the buttons now reflect correct theme mode in SolidJS as well?
Yes, because this time, we're destructuring an array, and it has the reference to the signal, so we just create a wrapper to get that value, and it works!
Enough of all this now, how do I fix the theming then?
Let's move on to that.
As mentioned at the start, we'll need to make use of reactive primitives, and as already seen, signals don't really help, so, we'll make use of the createMemo
reactive primitive and this is the approach we'll take -
import {
createTheme,
ThemeProvider,
Typography,
useTheme,
} from '@suid/material';
import Button from '@suid/material/Button';
import { createPalette } from '@suid/material/styles/createPalette';
import { createSignal, createMemo, createEffect } from 'solid-js';
function Btn(props: any) {
const theme = useTheme();
return (
<div
style={{
display: 'flex',
gap: '1rem',
alignItems: 'center',
margin: '10px',
}}
>
<Typography>From Context: </Typography>
<Button variant="contained" color="success">
{theme.palette.mode}
</Button>
</div>
);
}
export default function Counter(props: any) {
const [themeMode, setThemeMode] = createSignal<'light' | 'dark'>('light');
const palette = createMemo(() => {
return createPalette({ mode: themeMode() });
});
const theme = createTheme({ palette: palette });
function onClick() {
setThemeMode((prev) => (prev === 'light' ? 'dark' : 'light'));
}
return (
<ThemeProvider theme={theme}>
<div
style={{
display: 'flex',
gap: '1rem',
alignItems: 'center',
margin: '10px',
}}
>
<Typography>From Signal: </Typography>
<Button variant="contained" color="success" onClick={onClick}>
{themeMode()}
</Button>
</div>
<Btn />
<div
style={{
display: 'flex',
gap: '1rem',
alignItems: 'center',
justifyContent: 'center',
margin: '15px 50px',
fontWeight: 'bold',
}}
>
SolidJS
</div>
</ThemeProvider>
);
}
Now it works! We finally have our theming!
Wait, why do we have to do it like this?
The createMemo
reactive primitive is a signal and an effect at the same time. It outputs a reactive value, and in all the functions where that reactive value is used, the function is registered on the SolidJS's runtime internal stack as a subscriber.
So, can't I just use the
createTheme
inside of thecreateMemo
?
No, you can't. Reason being, if you use the createMemo
like that, you'll get a theme
object that's memoized and since the useContext
hook passes the data as is, the theme object will be destructured like before by useTheme
and we'll lose reactivity.
The reason why this approach works is, when the themeMode
signal changes, the palette
is created again, and with that, the theme
variable is updated. After all that's done and the nodes in the tree are there, an effect is run to update the tree, and hence we see the theming work.