231 lines
9.1 KiB
Vue
231 lines
9.1 KiB
Vue
<template>
|
|
<Drawer v-model:visible="$appState.designer.active" position="right" class="designer !w-screen md:!w-[48rem]" :modal="false" :dismissable="false" @after-show="onShow" @after-hide="onHide" :dt="drawerTokens">
|
|
<template #header>
|
|
<div class="flex items-center gap-2">
|
|
<Button v-if="$appState.designer.activeView !== 'dashboard'" variant="text" severity="secondary" rounded type="button" icon="pi pi-chevron-left" @click="openDashboard" />
|
|
<span class="font-bold text-xl">{{ viewTitle }}</span>
|
|
</div>
|
|
</template>
|
|
|
|
<DesignDashboard v-if="$appState.designer.activeView === 'dashboard'" />
|
|
<DesignCreateTheme v-else-if="$appState.designer.activeView === 'create_theme'" />
|
|
<DesignEditor v-else-if="$appState.designer.activeView === 'editor'" :deferred="deferredTabs" />
|
|
|
|
<template #footer>
|
|
<DesignEditorFooter v-if="$appState.designer.activeView === 'editor'" />
|
|
</template>
|
|
</Drawer>
|
|
<ConfirmDialog group="designer"></ConfirmDialog>
|
|
</template>
|
|
|
|
<script>
|
|
import EventBus from '@/app/AppEventBus';
|
|
import { $dt, updatePreset } from '@primevue/themes';
|
|
|
|
export default {
|
|
setup() {
|
|
const runtimeConfig = useRuntimeConfig();
|
|
|
|
return {
|
|
designerApiBase: runtimeConfig.public.designerApiBase
|
|
};
|
|
},
|
|
provide() {
|
|
return {
|
|
designerService: {
|
|
refreshACTokens: this.refreshACTokens,
|
|
saveTheme: this.saveTheme,
|
|
downloadTheme: this.downloadTheme,
|
|
applyTheme: this.applyTheme,
|
|
applyFont: this.applyFont,
|
|
replaceColorPalette: this.replaceColorPalette
|
|
}
|
|
};
|
|
},
|
|
data() {
|
|
return {
|
|
deferredTabs: true
|
|
};
|
|
},
|
|
methods: {
|
|
onShow() {
|
|
this.deferredTabs = false;
|
|
},
|
|
onHide() {
|
|
this.deferredTabs = true;
|
|
},
|
|
async downloadTheme(theme) {
|
|
if (!this.$appState.designer.licenseKey) {
|
|
this.$toast.add({ severity: 'error', summary: 'Not Available', detail: 'A license is required to download', life: 3000 });
|
|
} else {
|
|
try {
|
|
const response = await $fetch(this.designerApiBase + '/theme/download/' + theme.t_key, {
|
|
responseType: 'blob',
|
|
headers: {
|
|
Authorization: `Bearer ${this.$appState.designer.ticket}`,
|
|
'X-License-Key': this.$appState.designer.licenseKey
|
|
}
|
|
});
|
|
|
|
if (response.error) {
|
|
this.$toast.add({ severity: 'error', summary: 'An Error Occurred', detail: error.message, life: 3000 });
|
|
} else {
|
|
const blobUrl = window.URL.createObjectURL(response);
|
|
const link = document.createElement('a');
|
|
|
|
link.href = blobUrl;
|
|
link.download = theme.t_name + '.zip';
|
|
document.body.appendChild(link);
|
|
link.click();
|
|
document.body.removeChild(link);
|
|
window.URL.revokeObjectURL(blobUrl);
|
|
}
|
|
} catch (err) {
|
|
this.$toast.add({ severity: 'error', summary: 'An Error Occurred', detail: 'Failed to download file', life: 3000 });
|
|
}
|
|
}
|
|
},
|
|
async saveTheme(theme) {
|
|
const { error } = await $fetch(this.designerApiBase + '/theme/update', {
|
|
method: 'PATCH',
|
|
headers: {
|
|
Authorization: `Bearer ${this.$appState.designer.ticket}`,
|
|
'X-License-Key': this.$appState.designer.licenseKey
|
|
},
|
|
body: {
|
|
key: theme.key,
|
|
preset: theme.preset,
|
|
config: theme.config
|
|
}
|
|
});
|
|
|
|
if (error) {
|
|
this.$toast.add({ severity: 'error', summary: 'An error occured', detail: error.message, life: 3000 });
|
|
}
|
|
},
|
|
applyTheme(theme) {
|
|
if (this.$appState.designer.licenseKey) {
|
|
this.saveTheme(theme);
|
|
}
|
|
|
|
updatePreset(theme.preset);
|
|
EventBus.emit('theme-palette-change');
|
|
},
|
|
camelCaseToDotCase(name) {
|
|
return name.replace(/([a-z])([A-Z])/g, '$1.$2').toLowerCase();
|
|
},
|
|
generateACTokens(parentPath, obj) {
|
|
for (let key in obj) {
|
|
if (key === 'dark' || key === 'components' || key === 'directives') {
|
|
continue;
|
|
}
|
|
|
|
if (key === 'primitive' || key === 'semantic' || key === 'colorScheme' || key === 'light' || key === 'extend') {
|
|
this.generateACTokens(null, obj[key]);
|
|
} else {
|
|
if (typeof obj[key] === 'object') {
|
|
this.generateACTokens(parentPath ? parentPath + '.' + key : key, obj[key]);
|
|
} else {
|
|
const regex = /\.\d+$/;
|
|
|
|
const tokenName = this.camelCaseToDotCase(parentPath ? parentPath + '.' + key : key);
|
|
const tokenValue = $dt(tokenName).value;
|
|
const isColor = tokenName.includes('color') || tokenName.includes('background') || regex.test(tokenName);
|
|
|
|
this.$appState.designer.acTokens.push({ token: tokenName, label: '{' + tokenName + '}', variable: $dt(tokenName).variable, value: tokenValue, isColor: isColor });
|
|
}
|
|
}
|
|
}
|
|
},
|
|
refreshACTokens() {
|
|
this.$appState.designer.acTokens = [];
|
|
this.generateACTokens(null, this.$appState.designer.theme.preset);
|
|
},
|
|
openDashboard() {
|
|
this.$appState.designer.activeView = 'dashboard';
|
|
},
|
|
applyFont(fontFamily) {
|
|
if (fontFamily !== 'Inter var') {
|
|
this.loadFont(fontFamily, 400);
|
|
this.loadFont(fontFamily, 500);
|
|
this.loadFont(fontFamily, 600);
|
|
this.loadFont(fontFamily, 700);
|
|
} else {
|
|
document.body.style.fontFamily = `"Inter var", sans-serif`;
|
|
}
|
|
},
|
|
async loadFont(fontFamily, weight) {
|
|
try {
|
|
const fontFamilyPath = fontFamily.toLowerCase().replace(/\s+/g, '-');
|
|
const fontUrl = `https://fonts.bunny.net/${fontFamilyPath}/files/${fontFamilyPath}-latin-${weight}-normal.woff2`;
|
|
const font = new FontFace(fontFamily, `url(${fontUrl})`, {
|
|
weight: weight.toString(),
|
|
style: 'normal'
|
|
});
|
|
|
|
const loadedFont = await font.load();
|
|
|
|
document.fonts.add(loadedFont);
|
|
document.body.style.fontFamily = `"${fontFamily}", sans-serif`;
|
|
|
|
return loadedFont;
|
|
} catch (error) {
|
|
// silent fail as some fonts may have not all the font weights
|
|
}
|
|
},
|
|
replaceColorPalette() {
|
|
this.$appState.designer.theme.preset.semantic.primary = this.$appState.designer.theme.preset.primitive.emerald;
|
|
this.$appState.designer.theme.preset.semantic.colorScheme.light.surface = { ...{ 0: '#ffffff' }, ...this.$appState.designer.theme.preset.primitive.slate };
|
|
this.$appState.designer.theme.preset.semantic.colorScheme.dark.surface = { ...{ 0: '#ffffff' }, ...this.$appState.designer.theme.preset.primitive.zinc };
|
|
}
|
|
},
|
|
computed: {
|
|
viewTitle() {
|
|
const view = this.$appState.designer.activeView;
|
|
|
|
if (view === 'dashboard') {
|
|
return 'Theme Designer';
|
|
} else if (view === 'create_theme') {
|
|
return 'Create Theme';
|
|
} else if (view === 'editor') {
|
|
return this.$appState.designer.theme.name;
|
|
}
|
|
|
|
return null;
|
|
},
|
|
drawerTokens() {
|
|
return {
|
|
root: {
|
|
shadow: '0 20px 25px -5px rgba(0, 0, 0, 0.1), 0 8px 10px -6px rgba(0, 0, 0, 0.1)'
|
|
},
|
|
header: {
|
|
padding: '1.25rem'
|
|
},
|
|
title: {
|
|
fontSize: '1.5rem',
|
|
fontWeight: '600'
|
|
},
|
|
content: {
|
|
padding: '0 1.25rem 1.25rem 1.25rem'
|
|
},
|
|
footer: {
|
|
padding: '1.25rem'
|
|
},
|
|
colorScheme: {
|
|
light: {
|
|
background: '{surface.0}',
|
|
borderColor: '{surface.200}',
|
|
color: '#09090b'
|
|
},
|
|
dark: {
|
|
background: '{surface.900}',
|
|
borderColor: '{surface.700}',
|
|
color: '#ffffff'
|
|
}
|
|
}
|
|
};
|
|
}
|
|
}
|
|
};
|
|
</script>
|