mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-12-18 09:20:15 +01:00
* style: fix prompts icon shrinking in command popover * fix: scroll into view behavior for mentions * fix: always apply default font size if not found * refactor: Update useMessageScrolling threshold and debounceRate
78 lines
2.3 KiB
TypeScript
78 lines
2.3 KiB
TypeScript
//ThemeContext.js
|
|
// source: https://plainenglish.io/blog/light-and-dark-mode-in-react-web-application-with-tailwind-css-89674496b942
|
|
import { useSetRecoilState } from 'recoil';
|
|
import React, { createContext, useState, useEffect } from 'react';
|
|
import { getInitialTheme, applyFontSize } from '~/utils';
|
|
import store from '~/store';
|
|
|
|
type ProviderValue = {
|
|
theme: string;
|
|
setTheme: React.Dispatch<React.SetStateAction<string>>;
|
|
};
|
|
|
|
const defaultContextValue: ProviderValue = {
|
|
theme: getInitialTheme(),
|
|
setTheme: () => {
|
|
return;
|
|
},
|
|
};
|
|
|
|
export const isDark = (theme: string): boolean => {
|
|
if (theme === 'system') {
|
|
return window.matchMedia('(prefers-color-scheme: dark)').matches;
|
|
}
|
|
return theme === 'dark';
|
|
};
|
|
|
|
export const ThemeContext = createContext<ProviderValue>(defaultContextValue);
|
|
|
|
export const ThemeProvider = ({ initialTheme, children }) => {
|
|
const [theme, setTheme] = useState(getInitialTheme);
|
|
const setFontSize = useSetRecoilState(store.fontSize);
|
|
|
|
const rawSetTheme = (rawTheme: string) => {
|
|
const root = window.document.documentElement;
|
|
const darkMode = isDark(rawTheme);
|
|
|
|
root.classList.remove(darkMode ? 'light' : 'dark');
|
|
root.classList.add(darkMode ? 'dark' : 'light');
|
|
|
|
localStorage.setItem('color-theme', rawTheme);
|
|
};
|
|
|
|
useEffect(() => {
|
|
const mediaQuery = window.matchMedia('(prefers-color-scheme: dark)');
|
|
const changeThemeOnSystemChange = () => {
|
|
rawSetTheme(mediaQuery.matches ? 'dark' : 'light');
|
|
};
|
|
|
|
mediaQuery.addEventListener('change', changeThemeOnSystemChange);
|
|
|
|
return () => {
|
|
mediaQuery.removeEventListener('change', changeThemeOnSystemChange);
|
|
};
|
|
}, []);
|
|
|
|
useEffect(() => {
|
|
const fontSize = localStorage.getItem('fontSize');
|
|
if (fontSize == null) {
|
|
setFontSize('text-base');
|
|
applyFontSize('text-base');
|
|
localStorage.setItem('fontSize', 'text-base');
|
|
return;
|
|
}
|
|
applyFontSize(JSON.parse(fontSize));
|
|
// Reason: This effect should only run once, and `setFontSize` is a stable function
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
}, []);
|
|
|
|
if (initialTheme) {
|
|
rawSetTheme(initialTheme);
|
|
}
|
|
|
|
useEffect(() => {
|
|
rawSetTheme(theme);
|
|
}, [theme]);
|
|
|
|
return <ThemeContext.Provider value={{ theme, setTheme }}>{children}</ThemeContext.Provider>;
|
|
};
|