2021-06-18 12:57:08 -04:00
|
|
|
import { isTouchscreenDevice } from "../lib/isTouchscreenDevice";
|
2021-06-18 10:35:35 -04:00
|
|
|
import { createGlobalStyle } from "styled-components";
|
2021-06-19 17:37:12 -04:00
|
|
|
import { connectState } from "../redux/connector";
|
2021-06-18 12:57:08 -04:00
|
|
|
import { Children } from "../types/Preact";
|
2021-07-03 10:23:29 -04:00
|
|
|
import { useEffect } from "preact/hooks";
|
2021-06-18 15:21:54 -04:00
|
|
|
import { createContext } from "preact";
|
2021-06-18 12:57:08 -04:00
|
|
|
import { Helmet } from "react-helmet";
|
2021-06-18 10:35:35 -04:00
|
|
|
|
2021-06-18 12:57:08 -04:00
|
|
|
export type Variables =
|
|
|
|
| "accent"
|
|
|
|
| "background"
|
|
|
|
| "foreground"
|
|
|
|
| "block"
|
|
|
|
| "message-box"
|
|
|
|
| "mention"
|
|
|
|
| "success"
|
|
|
|
| "warning"
|
|
|
|
| "error"
|
|
|
|
| "hover"
|
|
|
|
| "scrollbar-thumb"
|
|
|
|
| "scrollbar-track"
|
|
|
|
| "primary-background"
|
|
|
|
| "primary-header"
|
|
|
|
| "secondary-background"
|
|
|
|
| "secondary-foreground"
|
|
|
|
| "secondary-header"
|
|
|
|
| "tertiary-background"
|
|
|
|
| "tertiary-foreground"
|
|
|
|
| "status-online"
|
|
|
|
| "status-away"
|
|
|
|
| "status-busy"
|
|
|
|
| "status-streaming"
|
|
|
|
| "status-invisible";
|
|
|
|
|
|
|
|
export type Theme = {
|
|
|
|
[variable in Variables]: string;
|
|
|
|
} & {
|
|
|
|
light?: boolean;
|
|
|
|
css?: string;
|
|
|
|
};
|
|
|
|
|
|
|
|
export interface ThemeOptions {
|
|
|
|
preset?: string;
|
|
|
|
custom?: Partial<Theme>;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generated from https://gitlab.insrt.uk/revolt/community/themes
|
|
|
|
export const PRESETS: { [key: string]: Theme } = {
|
|
|
|
light: {
|
|
|
|
light: true,
|
|
|
|
accent: "#FD6671",
|
|
|
|
background: "#F6F6F6",
|
|
|
|
foreground: "#101010",
|
|
|
|
block: "#414141",
|
|
|
|
"message-box": "#F1F1F1",
|
|
|
|
mention: "rgba(251, 255, 0, 0.40)",
|
|
|
|
success: "#65E572",
|
|
|
|
warning: "#FAA352",
|
|
|
|
error: "#F06464",
|
|
|
|
hover: "rgba(0, 0, 0, 0.2)",
|
|
|
|
"scrollbar-thumb": "#CA525A",
|
|
|
|
"scrollbar-track": "transparent",
|
|
|
|
"primary-background": "#FFFFFF",
|
|
|
|
"primary-header": "#F1F1F1",
|
|
|
|
"secondary-background": "#F1F1F1",
|
|
|
|
"secondary-foreground": "#888888",
|
|
|
|
"secondary-header": "#F1F1F1",
|
|
|
|
"tertiary-background": "#4D4D4D",
|
|
|
|
"tertiary-foreground": "#646464",
|
|
|
|
"status-online": "#3ABF7E",
|
|
|
|
"status-away": "#F39F00",
|
|
|
|
"status-busy": "#F84848",
|
|
|
|
"status-streaming": "#977EFF",
|
|
|
|
"status-invisible": "#A5A5A5",
|
|
|
|
},
|
|
|
|
dark: {
|
|
|
|
light: false,
|
|
|
|
accent: "#FD6671",
|
|
|
|
background: "#191919",
|
|
|
|
foreground: "#F6F6F6",
|
|
|
|
block: "#2D2D2D",
|
|
|
|
"message-box": "#363636",
|
|
|
|
mention: "rgba(251, 255, 0, 0.06)",
|
|
|
|
success: "#65E572",
|
|
|
|
warning: "#FAA352",
|
|
|
|
error: "#F06464",
|
|
|
|
hover: "rgba(0, 0, 0, 0.1)",
|
|
|
|
"scrollbar-thumb": "#CA525A",
|
|
|
|
"scrollbar-track": "transparent",
|
|
|
|
"primary-background": "#242424",
|
|
|
|
"primary-header": "#363636",
|
|
|
|
"secondary-background": "#1E1E1E",
|
|
|
|
"secondary-foreground": "#C8C8C8",
|
|
|
|
"secondary-header": "#2D2D2D",
|
|
|
|
"tertiary-background": "#4D4D4D",
|
|
|
|
"tertiary-foreground": "#848484",
|
|
|
|
"status-online": "#3ABF7E",
|
|
|
|
"status-away": "#F39F00",
|
|
|
|
"status-busy": "#F84848",
|
|
|
|
"status-streaming": "#977EFF",
|
|
|
|
"status-invisible": "#A5A5A5",
|
|
|
|
},
|
2021-06-18 10:57:08 -04:00
|
|
|
};
|
2021-06-18 10:35:35 -04:00
|
|
|
|
2021-07-01 14:46:50 -04:00
|
|
|
const keys = Object.keys(PRESETS.dark);
|
2021-06-18 12:57:08 -04:00
|
|
|
const GlobalTheme = createGlobalStyle<{ theme: Theme }>`
|
2021-06-18 10:35:35 -04:00
|
|
|
:root {
|
2021-06-18 12:57:08 -04:00
|
|
|
${(props) =>
|
|
|
|
(Object.keys(props.theme) as Variables[]).map((key) => {
|
2021-07-01 14:46:50 -04:00
|
|
|
if (!keys.includes(key)) return;
|
2021-06-18 12:57:08 -04:00
|
|
|
return `--${key}: ${props.theme[key]};`;
|
|
|
|
})}
|
2021-06-18 10:35:35 -04:00
|
|
|
}
|
|
|
|
`;
|
2021-06-18 12:57:08 -04:00
|
|
|
|
2021-06-18 15:21:54 -04:00
|
|
|
export const ThemeContext = createContext<Theme>({} as any);
|
|
|
|
|
2021-06-18 12:57:08 -04:00
|
|
|
interface Props {
|
|
|
|
children: Children;
|
2021-06-19 17:37:12 -04:00
|
|
|
options?: ThemeOptions;
|
2021-06-18 12:57:08 -04:00
|
|
|
}
|
|
|
|
|
2021-06-19 17:37:12 -04:00
|
|
|
function Theme(props: Props) {
|
|
|
|
const theme: Theme = {
|
|
|
|
...PRESETS["dark"],
|
|
|
|
...(PRESETS as any)[props.options?.preset as any],
|
|
|
|
...props.options?.custom
|
|
|
|
};
|
2021-06-18 12:57:08 -04:00
|
|
|
|
2021-07-03 10:23:29 -04:00
|
|
|
useEffect(() => {
|
|
|
|
const resize = () => document.documentElement.style.setProperty('--app-height', `${window.innerHeight}px`);
|
|
|
|
resize();
|
|
|
|
|
|
|
|
window.addEventListener('resize', resize);
|
|
|
|
return () => window.removeEventListener('resize', resize);
|
|
|
|
}, []);
|
|
|
|
|
2021-06-18 12:57:08 -04:00
|
|
|
return (
|
2021-06-18 15:21:54 -04:00
|
|
|
<ThemeContext.Provider value={theme}>
|
2021-06-18 12:57:08 -04:00
|
|
|
<Helmet>
|
|
|
|
<meta
|
|
|
|
name="theme-color"
|
|
|
|
content={
|
|
|
|
isTouchscreenDevice
|
|
|
|
? theme["primary-header"]
|
2021-07-03 09:27:57 -04:00
|
|
|
: theme["background"]
|
2021-06-18 12:57:08 -04:00
|
|
|
}
|
|
|
|
/>
|
|
|
|
</Helmet>
|
|
|
|
<GlobalTheme theme={theme} />
|
2021-06-19 17:37:12 -04:00
|
|
|
{theme.css && (
|
|
|
|
<style dangerouslySetInnerHTML={{ __html: theme.css }} />
|
|
|
|
)}
|
2021-06-18 12:57:08 -04:00
|
|
|
{props.children}
|
2021-06-18 15:21:54 -04:00
|
|
|
</ThemeContext.Provider>
|
2021-06-18 12:57:08 -04:00
|
|
|
);
|
|
|
|
}
|
2021-06-19 17:37:12 -04:00
|
|
|
|
2021-06-19 17:39:30 -04:00
|
|
|
export default connectState<{ children: Children }>(Theme, state => {
|
2021-06-19 17:37:12 -04:00
|
|
|
return {
|
|
|
|
options: state.settings.theme
|
|
|
|
};
|
|
|
|
});
|