<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="ph-code ph-bold ph-lg"></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 { 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 "@/account";
import { Theme, 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";

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",
	},
];

let theme = $ref<Partial<Theme>>({
	base: "light",
	props: lightTheme.props,
});
let description = $ref<string | null>(null);
let themeCode = $ref<string | null>(null);
let changed = $ref(false);

useLeaveGuard($$(changed));

function showPreview() {
	os.pageWindow("/preview");
}

function setBgColor(color: (typeof bgColors)[number]) {
	if (theme.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.props[prop] = base.props[prop];
		}
	}
	theme.base = color.kind;
	theme.props.bg = color.color;

	if (theme.props.fg) {
		const matchedFgColor = fgColors.find((x) =>
			[
				tinycolor(x.forLight).toRgbString(),
				tinycolor(x.forDark).toRgbString(),
			].includes(tinycolor(theme.props.fg).toRgbString()),
		);
		if (matchedFgColor) setFgColor(matchedFgColor);
	}
}

function setAccentColor(color) {
	theme.props.accent = color;
}

function setFgColor(color) {
	theme.props.fg = theme.base === "light" ? color.forLight : color.forDark;
}

function apply() {
	themeCode = JSON5.stringify(theme, null, "\t");
	applyTheme(theme, false);
	changed = true;
}

function applyThemeCode() {
	let parsed;

	try {
		parsed = JSON5.parse(themeCode);
	} catch (err) {
		os.alert({
			type: "error",
			text: i18n.ts._theme.invalid,
		});
		return;
	}

	theme = parsed;
}

async function saveAs() {
	const { canceled, result: name } = await os.inputText({
		title: i18n.ts.name,
		allowEmpty: false,
	});
	if (canceled) return;

	theme.id = uuid();
	theme.name = name;
	theme.author = `@${$i.username}@${toUnicode(host)}`;
	if (description) theme.desc = description;
	await addTheme(theme);
	applyTheme(theme);
	if (defaultStore.state.darkMode) {
		ColdDeviceStorage.set("darkTheme", theme);
	} else {
		ColdDeviceStorage.set("lightTheme", theme);
	}
	changed = false;
	os.alert({
		type: "success",
		text: i18n.t("_theme.installed", { name: theme.name }),
	});
}

watch($$(theme), apply, { deep: true });

const headerActions = $computed(() => [
	{
		asFullButton: true,
		icon: "ph-eye ph-bold ph-lg",
		text: i18n.ts.preview,
		handler: showPreview,
	},
	{
		asFullButton: true,
		icon: "ph-check ph-bold ph-lg",
		text: i18n.ts.saveAs,
		handler: saveAs,
	},
]);

const headerTabs = $computed(() => []);

definePageMetadata({
	title: i18n.ts.themeEditor,
	icon: "ph-palette ph-bold ph-lg",
});
</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>