2023-01-12 21:40:33 -07:00
|
|
|
import { ref } from "vue";
|
|
|
|
import tinycolor from "tinycolor2";
|
|
|
|
import { globalEvents } from "@/events";
|
2020-01-29 12:37:25 -07:00
|
|
|
|
|
|
|
export type Theme = {
|
|
|
|
id: string;
|
|
|
|
name: string;
|
|
|
|
author: string;
|
|
|
|
desc?: string;
|
2023-01-12 21:40:33 -07:00
|
|
|
base?: "dark" | "light";
|
2020-10-18 22:17:11 -06:00
|
|
|
props: Record<string, string>;
|
2020-01-29 12:37:25 -07:00
|
|
|
};
|
|
|
|
|
2023-01-12 21:40:33 -07:00
|
|
|
import lightTheme from "@/themes/_light.json5";
|
|
|
|
import darkTheme from "@/themes/_dark.json5";
|
|
|
|
import { deepClone } from "./clone";
|
2023-08-11 10:20:30 -06:00
|
|
|
import { defaultStore } from "@/store";
|
2023-01-12 21:40:33 -07:00
|
|
|
|
|
|
|
export const themeProps = Object.keys(lightTheme.props).filter(
|
|
|
|
(key) => !key.startsWith("X"),
|
2022-05-28 06:59:23 -06:00
|
|
|
);
|
|
|
|
|
2023-01-12 21:40:33 -07:00
|
|
|
export const getBuiltinThemes = () =>
|
|
|
|
Promise.all(
|
|
|
|
[
|
2023-09-12 11:19:28 -06:00
|
|
|
"l-iceshrimp",
|
2023-01-12 21:40:33 -07:00
|
|
|
"l-rosepinedawn",
|
|
|
|
"l-light",
|
2023-03-19 19:44:08 -06:00
|
|
|
"l-nord",
|
|
|
|
"l-gruvbox",
|
2024-06-17 10:47:11 -06:00
|
|
|
"l-catppuccin-latte-blue",
|
|
|
|
"l-catppuccin-latte-flamingo",
|
|
|
|
"l-catppuccin-latte-green",
|
|
|
|
"l-catppuccin-latte-lavender",
|
|
|
|
"l-catppuccin-latte-maroon",
|
|
|
|
"l-catppuccin-latte-mauve",
|
|
|
|
"l-catppuccin-latte-peach",
|
|
|
|
"l-catppuccin-latte-pink",
|
|
|
|
"l-catppuccin-latte-red",
|
|
|
|
"l-catppuccin-latte-rosewater",
|
|
|
|
"l-catppuccin-latte-sapphire",
|
|
|
|
"l-catppuccin-latte-sky",
|
|
|
|
"l-catppuccin-latte-teal",
|
|
|
|
"l-catppuccin-latte-yellow",
|
2023-01-12 21:40:33 -07:00
|
|
|
"l-coffee",
|
|
|
|
"l-apricot",
|
|
|
|
"l-rainy",
|
|
|
|
"l-vivid",
|
|
|
|
"l-cherry",
|
|
|
|
"l-sushi",
|
|
|
|
"l-u0",
|
|
|
|
|
2023-09-12 11:19:28 -06:00
|
|
|
"d-iceshrimp",
|
2023-01-12 21:40:33 -07:00
|
|
|
"d-rosepine",
|
|
|
|
"d-rosepinemoon",
|
|
|
|
"d-dark",
|
2023-03-19 19:44:08 -06:00
|
|
|
"d-nord",
|
|
|
|
"d-gruvbox",
|
2024-06-17 10:47:11 -06:00
|
|
|
"d-catppuccin-frappe-blue",
|
|
|
|
"d-catppuccin-frappe-flamingo",
|
|
|
|
"d-catppuccin-frappe-green",
|
|
|
|
"d-catppuccin-frappe-lavender",
|
|
|
|
"d-catppuccin-frappe-maroon",
|
|
|
|
"d-catppuccin-frappe-mauve",
|
|
|
|
"d-catppuccin-frappe-peach",
|
|
|
|
"d-catppuccin-frappe-pink",
|
|
|
|
"d-catppuccin-frappe-red",
|
|
|
|
"d-catppuccin-frappe-rosewater",
|
|
|
|
"d-catppuccin-frappe-sapphire",
|
|
|
|
"d-catppuccin-frappe-sky",
|
|
|
|
"d-catppuccin-frappe-teal",
|
|
|
|
"d-catppuccin-frappe-yellow",
|
|
|
|
"d-catppuccin-mocha-blue",
|
|
|
|
"d-catppuccin-mocha-flamingo",
|
|
|
|
"d-catppuccin-mocha-green",
|
|
|
|
"d-catppuccin-mocha-lavender",
|
|
|
|
"d-catppuccin-mocha-maroon",
|
|
|
|
"d-catppuccin-mocha-mauve",
|
|
|
|
"d-catppuccin-mocha-peach",
|
|
|
|
"d-catppuccin-mocha-pink",
|
|
|
|
"d-catppuccin-mocha-red",
|
|
|
|
"d-catppuccin-mocha-rosewater",
|
|
|
|
"d-catppuccin-mocha-sapphire",
|
|
|
|
"d-catppuccin-mocha-sky",
|
|
|
|
"d-catppuccin-mocha-teal",
|
|
|
|
"d-catppuccin-mocha-yellow",
|
|
|
|
"d-catppuccin-macchiato-blue",
|
|
|
|
"d-catppuccin-macchiato-flamingo",
|
|
|
|
"d-catppuccin-macchiato-green",
|
|
|
|
"d-catppuccin-macchiato-lavender",
|
|
|
|
"d-catppuccin-macchiato-maroon",
|
|
|
|
"d-catppuccin-macchiato-mauve",
|
|
|
|
"d-catppuccin-macchiato-peach",
|
|
|
|
"d-catppuccin-macchiato-pink",
|
|
|
|
"d-catppuccin-macchiato-red",
|
|
|
|
"d-catppuccin-macchiato-rosewater",
|
|
|
|
"d-catppuccin-macchiato-sapphire",
|
|
|
|
"d-catppuccin-macchiato-sky",
|
|
|
|
"d-catppuccin-macchiato-teal",
|
|
|
|
"d-catppuccin-macchiato-yellow",
|
2023-01-12 21:40:33 -07:00
|
|
|
"d-persimmon",
|
|
|
|
"d-astro",
|
|
|
|
"d-future",
|
|
|
|
"d-botanical",
|
|
|
|
"d-green-lime",
|
|
|
|
"d-green-orange",
|
|
|
|
"d-cherry",
|
|
|
|
"d-ice",
|
2023-10-25 12:30:32 -06:00
|
|
|
"d-u0"
|
2023-01-12 21:40:33 -07:00
|
|
|
].map((name) =>
|
|
|
|
import(`../themes/${name}.json5`).then(
|
|
|
|
({ default: _default }): Theme => _default,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
|
2022-05-28 06:59:23 -06:00
|
|
|
export const getBuiltinThemesRef = () => {
|
|
|
|
const builtinThemes = ref<Theme[]>([]);
|
2023-01-12 21:40:33 -07:00
|
|
|
getBuiltinThemes().then((themes) => (builtinThemes.value = themes));
|
2022-05-28 06:59:23 -06:00
|
|
|
return builtinThemes;
|
2022-06-09 23:36:55 -06:00
|
|
|
};
|
2020-01-29 12:37:25 -07:00
|
|
|
|
|
|
|
let timeout = null;
|
|
|
|
|
|
|
|
export function applyTheme(theme: Theme, persist = true) {
|
2022-01-15 18:14:14 -07:00
|
|
|
if (timeout) window.clearTimeout(timeout);
|
2020-01-29 12:37:25 -07:00
|
|
|
|
2023-01-12 21:40:33 -07:00
|
|
|
document.documentElement.classList.add("_themeChanging_");
|
2020-01-29 12:37:25 -07:00
|
|
|
|
2022-01-15 18:14:14 -07:00
|
|
|
timeout = window.setTimeout(() => {
|
2023-01-12 21:40:33 -07:00
|
|
|
document.documentElement.classList.remove("_themeChanging_");
|
2020-01-29 12:37:25 -07:00
|
|
|
}, 1000);
|
|
|
|
|
2023-01-12 21:40:33 -07:00
|
|
|
const colorSchema = theme.base === "dark" ? "dark" : "light";
|
2022-08-26 09:39:59 -06:00
|
|
|
|
2020-01-29 12:37:25 -07:00
|
|
|
// Deep copy
|
2022-11-16 17:31:07 -07:00
|
|
|
const _theme = deepClone(theme);
|
2020-01-29 12:37:25 -07:00
|
|
|
|
|
|
|
if (_theme.base) {
|
2023-01-12 21:40:33 -07:00
|
|
|
const base = [lightTheme, darkTheme].find((x) => x.id === _theme.base);
|
2022-04-02 22:56:00 -06:00
|
|
|
if (base) _theme.props = Object.assign({}, base.props, _theme.props);
|
2020-01-29 12:37:25 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
const props = compile(_theme);
|
|
|
|
|
|
|
|
for (const tag of document.head.children) {
|
2023-01-12 21:40:33 -07:00
|
|
|
if (tag.tagName === "META" && tag.getAttribute("name") === "theme-color") {
|
|
|
|
tag.setAttribute("content", props["htmlThemeColor"]);
|
2020-01-29 12:37:25 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const [k, v] of Object.entries(props)) {
|
|
|
|
document.documentElement.style.setProperty(`--${k}`, v.toString());
|
|
|
|
}
|
|
|
|
|
2023-01-12 21:40:33 -07:00
|
|
|
document.documentElement.style.setProperty("color-schema", colorSchema);
|
2022-08-26 09:39:59 -06:00
|
|
|
|
2020-01-29 12:37:25 -07:00
|
|
|
if (persist) {
|
2023-01-12 21:40:33 -07:00
|
|
|
localStorage.setItem("theme", JSON.stringify(props));
|
|
|
|
localStorage.setItem("colorSchema", colorSchema);
|
2020-01-29 12:37:25 -07:00
|
|
|
}
|
2021-10-10 09:36:47 -06:00
|
|
|
|
2022-12-14 13:17:06 -07:00
|
|
|
// Site-wide notification that the theme has changed
|
2023-01-12 21:40:33 -07:00
|
|
|
globalEvents.emit("themeChanged");
|
2020-01-29 12:37:25 -07:00
|
|
|
}
|
|
|
|
|
2020-03-29 01:09:44 -06:00
|
|
|
function compile(theme: Theme): Record<string, string> {
|
2023-08-11 10:20:30 -06:00
|
|
|
function getColor(val: string, key?: string): tinycolor.Instance {
|
2020-03-29 01:09:44 -06:00
|
|
|
// ref (prop)
|
2023-01-12 21:40:33 -07:00
|
|
|
if (val[0] === "@") {
|
2023-07-13 00:56:22 -06:00
|
|
|
return getColor(theme.props[val.slice(1)]);
|
2020-03-29 01:09:44 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// ref (const)
|
2023-01-12 21:40:33 -07:00
|
|
|
else if (val[0] === "$") {
|
2020-03-29 01:09:44 -06:00
|
|
|
return getColor(theme.props[val]);
|
2020-01-29 12:37:25 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// func
|
2023-01-12 21:40:33 -07:00
|
|
|
else if (val[0] === ":") {
|
|
|
|
const parts = val.split("<");
|
2023-07-13 00:56:22 -06:00
|
|
|
const func = parts.shift().slice(1);
|
2020-01-29 12:37:25 -07:00
|
|
|
const arg = parseFloat(parts.shift());
|
2023-01-12 21:40:33 -07:00
|
|
|
const color = getColor(parts.join("<"));
|
2020-01-29 12:37:25 -07:00
|
|
|
|
2023-08-11 11:59:52 -06:00
|
|
|
const ignoreAlphaForKeys = ["windowHeader", "acrylicPanel", "pageHeader"];
|
2023-08-11 10:20:30 -06:00
|
|
|
|
2020-01-29 12:37:25 -07:00
|
|
|
switch (func) {
|
2023-01-12 21:40:33 -07:00
|
|
|
case "darken":
|
|
|
|
return color.darken(arg);
|
|
|
|
case "lighten":
|
|
|
|
return color.lighten(arg);
|
|
|
|
case "alpha":
|
2023-10-25 12:30:32 -06:00
|
|
|
if (!defaultStore.state.useBlurEffect && key && ignoreAlphaForKeys.includes(key)) {
|
2023-08-11 10:20:30 -06:00
|
|
|
return color.setAlpha(1.0);
|
2023-10-25 12:30:32 -06:00
|
|
|
}
|
|
|
|
else {
|
2023-08-11 10:20:30 -06:00
|
|
|
return color.setAlpha(arg);
|
|
|
|
}
|
2023-01-12 21:40:33 -07:00
|
|
|
case "hue":
|
|
|
|
return color.spin(arg);
|
|
|
|
case "saturate":
|
|
|
|
return color.saturate(arg);
|
2020-01-29 12:37:25 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-29 01:09:44 -06:00
|
|
|
// other case
|
|
|
|
return tinycolor(val);
|
2020-01-29 12:37:25 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
const props = {};
|
|
|
|
|
|
|
|
for (const [k, v] of Object.entries(theme.props)) {
|
2023-01-12 21:40:33 -07:00
|
|
|
if (k.startsWith("$")) continue; // ignore const
|
2020-03-29 01:09:44 -06:00
|
|
|
|
2023-01-12 21:40:33 -07:00
|
|
|
props[k] = v.startsWith('"')
|
|
|
|
? v.replace(/^"\s*/, "")
|
2023-08-11 10:20:30 -06:00
|
|
|
: genValue(getColor(v, k));
|
2020-01-29 12:37:25 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return props;
|
|
|
|
}
|
|
|
|
|
|
|
|
function genValue(c: tinycolor.Instance): string {
|
|
|
|
return c.toRgbString();
|
|
|
|
}
|
2020-03-23 04:06:46 -06:00
|
|
|
|
|
|
|
export function validateTheme(theme: Record<string, any>): boolean {
|
2023-01-12 21:40:33 -07:00
|
|
|
if (theme.id == null || typeof theme.id !== "string") return false;
|
|
|
|
if (theme.name == null || typeof theme.name !== "string") return false;
|
|
|
|
if (theme.base == null || !["light", "dark"].includes(theme.base))
|
|
|
|
return false;
|
|
|
|
if (theme.props == null || typeof theme.props !== "object") return false;
|
2020-03-23 04:06:46 -06:00
|
|
|
return true;
|
|
|
|
}
|