hippofish/packages/client/src/pages/theme-editor.vue
2024-02-12 02:50:57 +09:00

411 lines
9.5 KiB
Vue

<template>
<MkStickyContainer>
<template #header
><MkPageHeader :actions="headerActions" :tabs="headerTabs"
/></template>
<MkSpacer :content-max="800" :margin-min="16" :margin-max="32">
<div class="cwepdizn _formRoot">
<FormFolder :default-open="true" class="_formBlock">
<template #label>{{ i18n.ts.backgroundColor }}</template>
<div class="cwepdizn-colors">
<div class="row">
<button
v-for="color in bgColors.filter(
(x) => x.kind === 'light',
)"
:key="color.color"
class="color _button"
:class="{
active: theme.props.bg === color.color,
}"
@click="setBgColor(color)"
>
<div
class="preview"
:style="{ background: color.forPreview }"
></div>
</button>
</div>
<div class="row">
<button
v-for="color in bgColors.filter(
(x) => x.kind === 'dark',
)"
:key="color.color"
class="color _button"
:class="{
active: theme.props.bg === color.color,
}"
@click="setBgColor(color)"
>
<div
class="preview"
:style="{ background: color.forPreview }"
></div>
</button>
</div>
</div>
</FormFolder>
<FormFolder :default-open="true" class="_formBlock">
<template #label>{{ i18n.ts.accentColor }}</template>
<div class="cwepdizn-colors">
<div class="row">
<button
v-for="color in accentColors"
:key="color"
class="color rounded _button"
:class="{
active: theme.props.accent === color,
}"
@click="setAccentColor(color)"
>
<div
class="preview"
:style="{ background: color }"
></div>
</button>
</div>
</div>
</FormFolder>
<FormFolder :default-open="true" class="_formBlock">
<template #label>{{ i18n.ts.textColor }}</template>
<div class="cwepdizn-colors">
<div class="row">
<button
v-for="color in fgColors"
:key="color"
class="color char _button"
:class="{
active:
theme.props.fg === color.forLight ||
theme.props.fg === color.forDark,
}"
@click="setFgColor(color)"
>
<div
class="preview"
:style="{
color: color.forPreview
? color.forPreview
: theme.base === 'light'
? '#5f5f5f'
: '#dadada',
}"
>
A
</div>
</button>
</div>
</div>
</FormFolder>
<FormFolder :default-open="false" class="_formBlock">
<template #icon><i :class="icon('ph-code')"></i></template>
<template #label>{{ i18n.ts.editCode }}</template>
<div class="_formRoot">
<FormTextarea
v-model="themeCode"
tall
class="_formBlock"
>
<template #label>{{
i18n.ts._theme.code
}}</template>
</FormTextarea>
<FormButton
primary
class="_formBlock"
@click="applyThemeCode"
>{{ i18n.ts.apply }}</FormButton
>
</div>
</FormFolder>
<FormFolder :default-open="false" class="_formBlock">
<template #label>{{ i18n.ts.addDescription }}</template>
<div class="_formRoot">
<FormTextarea v-model="description">
<template #label>{{
i18n.ts._theme.description
}}</template>
</FormTextarea>
</div>
</FormFolder>
</div>
</MkSpacer>
</MkStickyContainer>
</template>
<script lang="ts" setup>
import { computed, ref, watch } from "vue";
import { toUnicode } from "punycode/";
import tinycolor from "tinycolor2";
import { v4 as uuid } from "uuid";
import JSON5 from "json5";
import FormButton from "@/components/MkButton.vue";
import FormTextarea from "@/components/form/textarea.vue";
import FormFolder from "@/components/form/folder.vue";
import { $i } from "@/reactiveAccount";
import type { Theme } from "@/scripts/theme";
import { applyTheme } from "@/scripts/theme";
import lightTheme from "@/themes/_light.json5";
import darkTheme from "@/themes/_dark.json5";
import { host } from "@/config";
import * as os from "@/os";
import { ColdDeviceStorage, defaultStore } from "@/store";
import { addTheme } from "@/theme-store";
import { i18n } from "@/i18n";
import { useLeaveGuard } from "@/scripts/use-leave-guard";
import { definePageMetadata } from "@/scripts/page-metadata";
import icon from "@/scripts/icon";
const bgColors = [
{ color: "#f5f5f5", kind: "light", forPreview: "#f5f5f5" },
{ color: "#f0eee9", kind: "light", forPreview: "#f3e2b9" },
{ color: "#e9eff0", kind: "light", forPreview: "#bfe3e8" },
{ color: "#f0e9ee", kind: "light", forPreview: "#f1d1e8" },
{ color: "#dce2e0", kind: "light", forPreview: "#a4dccc" },
{ color: "#e2e0dc", kind: "light", forPreview: "#d8c7a5" },
{ color: "#d5dbe0", kind: "light", forPreview: "#b0cae0" },
{ color: "#dad5d5", kind: "light", forPreview: "#d6afaf" },
{ color: "#2b2b2b", kind: "dark", forPreview: "#444444" },
{ color: "#362e29", kind: "dark", forPreview: "#735c4d" },
{ color: "#303629", kind: "dark", forPreview: "#506d2f" },
{ color: "#293436", kind: "dark", forPreview: "#258192" },
{ color: "#2e2936", kind: "dark", forPreview: "#504069" },
{ color: "#252722", kind: "dark", forPreview: "#3c462f" },
{ color: "#212525", kind: "dark", forPreview: "#303e3e" },
{ color: "#191919", kind: "dark", forPreview: "#272727" },
] as const;
const accentColors = [
"#e36749",
"#f29924",
"#98c934",
"#34c9a9",
"#34a1c9",
"#606df7",
"#8d34c9",
"#e84d83",
];
const fgColors = [
{
color: "none",
forLight: "#5f5f5f",
forDark: "#dadada",
forPreview: null,
},
{
color: "red",
forLight: "#7f6666",
forDark: "#e4d1d1",
forPreview: "#ca4343",
},
{
color: "yellow",
forLight: "#736955",
forDark: "#e0d5c0",
forPreview: "#d49923",
},
{
color: "green",
forLight: "#586d5b",
forDark: "#d1e4d4",
forPreview: "#4cbd5c",
},
{
color: "cyan",
forLight: "#5d7475",
forDark: "#d1e3e4",
forPreview: "#2abdc3",
},
{
color: "blue",
forLight: "#676880",
forDark: "#d1d2e4",
forPreview: "#7275d8",
},
{
color: "pink",
forLight: "#84667d",
forDark: "#e4d1e0",
forPreview: "#b12390",
},
];
const theme = ref<Partial<Theme>>({
base: "light",
props: lightTheme.props,
});
const description = ref<string | null>(null);
const themeCode = ref<string | null>(null);
const changed = ref(false);
useLeaveGuard(changed);
function showPreview() {
os.pageWindow("/preview");
}
function setBgColor(color: (typeof bgColors)[number]) {
if (theme.value.base !== color.kind) {
const base = color.kind === "dark" ? darkTheme : lightTheme;
for (const prop of Object.keys(base.props)) {
if (prop === "accent") continue;
if (prop === "fg") continue;
theme.value.props[prop] = base.props[prop];
}
}
theme.value.base = color.kind;
theme.value.props.bg = color.color;
if (theme.value.props.fg) {
const matchedFgColor = fgColors.find((x) =>
[
tinycolor(x.forLight).toRgbString(),
tinycolor(x.forDark).toRgbString(),
].includes(tinycolor(theme.value.props.fg).toRgbString()),
);
if (matchedFgColor) setFgColor(matchedFgColor);
}
}
function setAccentColor(color) {
theme.value.props.accent = color;
}
function setFgColor(color) {
theme.value.props.fg =
theme.value.base === "light" ? color.forLight : color.forDark;
}
function apply() {
themeCode.value = JSON5.stringify(theme.value, null, "\t");
applyTheme(theme.value, false);
changed.value = true;
}
function applyThemeCode() {
let parsed;
try {
parsed = JSON5.parse(themeCode.value);
} catch (err) {
os.alert({
type: "error",
text: i18n.ts._theme.invalid,
});
return;
}
theme.value = parsed;
}
async function saveAs() {
const { canceled, result: name } = await os.inputText({
title: i18n.ts.name,
allowEmpty: false,
});
if (canceled) return;
theme.value.id = uuid();
theme.value.name = name;
theme.value.author = `@${$i.username}@${toUnicode(host)}`;
if (description.value) theme.value.desc = description.value;
await addTheme(theme.value);
applyTheme(theme.value);
if (defaultStore.state.darkMode) {
ColdDeviceStorage.set("darkTheme", theme.value);
} else {
ColdDeviceStorage.set("lightTheme", theme.value);
}
changed.value = false;
os.alert({
type: "success",
text: i18n.t("_theme.installed", { name: theme.value.name }),
});
}
watch(theme, apply, { deep: true });
const headerActions = computed(() => [
{
asFullButton: true,
icon: `${icon("ph-eye")}`,
text: i18n.ts.preview,
handler: showPreview,
},
{
asFullButton: true,
icon: `${icon("ph-check")}`,
text: i18n.ts.saveAs,
handler: saveAs,
},
]);
const headerTabs = computed(() => []);
definePageMetadata({
title: i18n.ts.themeEditor,
icon: `${icon("ph-palette")}`,
});
</script>
<style lang="scss" scoped>
.cwepdizn {
::v-deep(.cwepdizn-colors) {
text-align: center;
> .row {
> .color {
display: inline-block;
position: relative;
width: 64px;
height: 64px;
border-radius: 8px;
> .preview {
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
margin: auto;
width: 42px;
height: 42px;
border-radius: 4px;
box-shadow: 0 2px 4px rgb(0 0 0 / 30%);
transition: transform 0.15s ease;
}
&:hover {
> .preview {
transform: scale(1.1);
}
}
&.active {
box-shadow: 0 0 0 2px var(--divider) inset;
}
&.rounded {
border-radius: 999px;
> .preview {
border-radius: 999px;
}
}
&.char {
line-height: 42px;
}
}
}
}
}
</style>