forked from mirror/misskey
275 lines
8.3 KiB
Vue
275 lines
8.3 KiB
Vue
<template>
|
|
<FormBase class="cwepdizn">
|
|
<div class="_formItem colorPicker">
|
|
<div class="_formLabel">{{ $ts.backgroundColor }}</div>
|
|
<div class="_formPanel colors">
|
|
<div class="row">
|
|
<button v-for="color in bgColors.filter(x => x.kind === 'light')" :key="color.color" @click="bgColor = color" class="color _button" :class="{ active: bgColor?.color === color.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" @click="bgColor = color" class="color _button" :class="{ active: bgColor?.color === color.color }">
|
|
<div class="preview" :style="{ background: color.forPreview }"></div>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="_formItem colorPicker">
|
|
<div class="_formLabel">{{ $ts.accentColor }}</div>
|
|
<div class="_formPanel colors">
|
|
<div class="row">
|
|
<button v-for="color in accentColors" :key="color" @click="accentColor = color" class="color rounded _button" :class="{ active: accentColor === color }">
|
|
<div class="preview" :style="{ background: color }"></div>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="_formItem colorPicker">
|
|
<div class="_formLabel">{{ $ts.textColor }}</div>
|
|
<div class="_formPanel colors">
|
|
<div class="row">
|
|
<button v-for="color in fgColors" :key="color" @click="fgColor = color" class="color char _button" :class="{ active: fgColor === color }">
|
|
<div class="preview" :style="{ color: color.forPreview ? color.forPreview : bgColor?.kind === 'light' ? '#5f5f5f' : '#dadada' }">A</div>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="_formItem preview">
|
|
<div class="_formLabel">{{ $ts.preview }}</div>
|
|
<div class="_formPanel preview">
|
|
<MkSample class="preview"/>
|
|
</div>
|
|
</div>
|
|
<FormButton @click="saveAs" primary>{{ $ts.saveAs }}</FormButton>
|
|
</FormBase>
|
|
</template>
|
|
|
|
<script lang="ts">
|
|
import { defineComponent } from 'vue';
|
|
import { faPalette, faChevronDown, faKeyboard } from '@fortawesome/free-solid-svg-icons';
|
|
import { toUnicode } from 'punycode';
|
|
import * as tinycolor from 'tinycolor2';
|
|
import { v4 as uuid} from 'uuid';
|
|
|
|
import FormBase from '@/components/form/base.vue';
|
|
import FormButton from '@/components/form/button.vue';
|
|
import MkSample from '@/components/sample.vue';
|
|
|
|
import { Theme, applyTheme, validateTheme } from '@/scripts/theme';
|
|
import { host } from '@/config';
|
|
import * as os from '@/os';
|
|
import { ColdDeviceStorage } from '@/store';
|
|
import { addTheme } from '@/theme-store';
|
|
|
|
export default defineComponent({
|
|
components: {
|
|
FormBase,
|
|
FormButton,
|
|
MkSample,
|
|
},
|
|
|
|
data() {
|
|
return {
|
|
INFO: {
|
|
title: this.$ts.themeEditor,
|
|
icon: faPalette,
|
|
},
|
|
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' },
|
|
],
|
|
bgColor: null,
|
|
accentColors: ['#e36749', '#f29924', '#98c934', '#34c9a9', '#34a1c9', '#606df7', '#8d34c9', '#e84d83'],
|
|
accentColor: null,
|
|
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' },
|
|
],
|
|
fgColor: null,
|
|
changed: false,
|
|
faPalette,
|
|
}
|
|
},
|
|
|
|
created() {
|
|
const currentBgColor = getComputedStyle(document.documentElement).getPropertyValue('--bg');
|
|
const matchedBgColor = this.bgColors.find(x => tinycolor(x.color).toRgbString() === tinycolor(currentBgColor).toRgbString());
|
|
if (matchedBgColor) this.bgColor = matchedBgColor;
|
|
const currentAccentColor = getComputedStyle(document.documentElement).getPropertyValue('--accent');
|
|
const matchedAccentColor = this.accentColors.find(x => tinycolor(x).toRgbString() === tinycolor(currentAccentColor).toRgbString());
|
|
if (matchedAccentColor) this.accentColor = matchedAccentColor;
|
|
const currentFgColor = getComputedStyle(document.documentElement).getPropertyValue('--fg');
|
|
const matchedFgColor = this.fgColors.find(x => [tinycolor(x.forLight).toRgbString(), tinycolor(x.forDark).toRgbString()].includes(tinycolor(currentFgColor).toRgbString()));
|
|
if (matchedFgColor) this.fgColor = matchedFgColor;
|
|
|
|
this.$watch('bgColor', this.apply);
|
|
this.$watch('accentColor', this.apply);
|
|
this.$watch('fgColor', this.apply);
|
|
|
|
window.addEventListener('beforeunload', this.beforeunload);
|
|
},
|
|
|
|
beforeUnmount() {
|
|
window.removeEventListener('beforeunload', this.beforeunload);
|
|
},
|
|
|
|
async beforeRouteLeave(to, from) {
|
|
if (this.changed && !(await this.leaveConfirm())) {
|
|
return false;
|
|
}
|
|
},
|
|
|
|
methods: {
|
|
beforeunload(e: BeforeUnloadEvent) {
|
|
if (this.changed) {
|
|
e.preventDefault();
|
|
e.returnValue = '';
|
|
}
|
|
},
|
|
|
|
async leaveConfirm(): Promise<boolean> {
|
|
const { canceled } = await os.dialog({
|
|
type: 'warning',
|
|
text: this.$ts.leaveConfirm,
|
|
showCancelButton: true
|
|
});
|
|
return !canceled;
|
|
},
|
|
|
|
convert(): Theme {
|
|
return {
|
|
name: this.$ts.myTheme,
|
|
base: this.bgColor.kind,
|
|
props: {
|
|
bg: this.bgColor.color,
|
|
fg: this.bgColor.kind === 'light' ? this.fgColor.forLight : this.fgColor.forDark,
|
|
accent: this.accentColor,
|
|
}
|
|
};
|
|
},
|
|
|
|
apply() {
|
|
if (this.bgColor == null) this.bgColor = this.bgColors[0];
|
|
if (this.accentColor == null) this.accentColor = this.accentColors[0];
|
|
if (this.fgColor == null) this.fgColor = this.fgColors[0];
|
|
|
|
const theme = this.convert();
|
|
applyTheme(theme, false);
|
|
this.changed = true;
|
|
},
|
|
|
|
async saveAs() {
|
|
const { canceled, result: name } = await os.dialog({
|
|
title: this.$ts.name,
|
|
input: {
|
|
allowEmpty: false
|
|
}
|
|
});
|
|
if (canceled) return;
|
|
|
|
const theme = this.convert();
|
|
theme.id = uuid();
|
|
theme.name = name;
|
|
theme.author = `@${this.$i.username}@${toUnicode(host)}`;
|
|
addTheme(theme);
|
|
applyTheme(theme);
|
|
if (this.$store.state.darkMode) {
|
|
ColdDeviceStorage.set('darkTheme', theme.id);
|
|
} else {
|
|
ColdDeviceStorage.set('lightTheme', theme.id);
|
|
}
|
|
this.changed = false;
|
|
os.dialog({
|
|
type: 'success',
|
|
text: this.$t('_theme.installed', { name: theme.name })
|
|
});
|
|
}
|
|
}
|
|
});
|
|
</script>
|
|
|
|
<style lang="scss" scoped>
|
|
.cwepdizn {
|
|
max-width: 800px;
|
|
margin: 0 auto;
|
|
|
|
> .colorPicker {
|
|
> .colors {
|
|
padding: 32px;
|
|
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;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
> .preview > .preview > .preview {
|
|
box-shadow: none;
|
|
background: transparent;
|
|
}
|
|
}
|
|
</style>
|