2024-11-12 09:18:07 +00:00
|
|
|
<template>
|
2025-01-02 22:30:06 +00:00
|
|
|
<div class="group">
|
|
|
|
<div class="flex justify-between justify-items-center">
|
|
|
|
<label :for="inputId" class="text-sm text-zinc-700 dark:text-white block capitalize text-ellipsis overflow-hidden w-full whitespace-nowrap" :title="label">{{ label }}</label>
|
|
|
|
<button type="button" @click="transfer"><i class="pi pi-sort-alt !text-xs text-zinc-400 hidden group-hover:block animate-fadein"></i></button>
|
|
|
|
</div>
|
2024-11-13 16:09:13 +00:00
|
|
|
<div :id="id" class="relative">
|
2024-11-14 07:25:06 +00:00
|
|
|
<AutoComplete
|
|
|
|
:modelValue="modelValue"
|
2024-11-13 16:17:03 +00:00
|
|
|
@input="onInput"
|
2024-11-14 07:25:06 +00:00
|
|
|
:inputId="inputId"
|
|
|
|
:suggestions="items"
|
|
|
|
@complete="search"
|
|
|
|
unstyled
|
2024-11-14 08:16:00 +00:00
|
|
|
optionLabel="label"
|
2024-11-14 07:25:06 +00:00
|
|
|
:showEmptyMessage="false"
|
|
|
|
:pt="{
|
|
|
|
pcInputText: {
|
2024-12-13 20:19:49 +00:00
|
|
|
root: ['border border-surface-300 dark:border-surface-600 rounded-lg py-2 px-2 w-full text-sm', { 'pr-8': type === 'color' }]
|
2024-11-14 07:25:06 +00:00
|
|
|
},
|
|
|
|
overlay: 'border border-surface-200 dark:border-surface-700 bg-surface-0 dark:bg-surface-950 shadow-2 rounded-md',
|
|
|
|
listContainer: 'max-h-40 overflow-auto',
|
2024-11-14 08:18:07 +00:00
|
|
|
list: 'm-0 py-2 px-2 list-none',
|
2024-11-14 08:58:02 +00:00
|
|
|
loader: 'hidden',
|
2024-11-14 08:18:07 +00:00
|
|
|
option: 'cursor-pointer py-1 text-sm text-surface-700 dark:text-white/80 data-[p-focus=true]:bg-surface-100 data-[p-focus=true]:dark:bg-surface-800 rounded-md'
|
2024-11-14 07:25:06 +00:00
|
|
|
}"
|
|
|
|
@option-select="onOptionSelect"
|
2024-11-14 08:16:00 +00:00
|
|
|
>
|
|
|
|
<template #option="slotProps">
|
2024-11-14 08:18:07 +00:00
|
|
|
<div v-tooltip.left="slotProps.option.value" class="flex items-center justify-between gap-4 px-2">
|
2024-11-14 08:16:00 +00:00
|
|
|
<span>{{ slotProps.option.token }}</span>
|
|
|
|
<div v-if="slotProps.option.isColor" class="border border-surface-200 dark:border-surface-700 w-4 h-4 rounded-full" :style="{ backgroundColor: slotProps.option.variable }"></div>
|
|
|
|
<div v-else class="text-xs max-w-16 text-ellipsis whitespace-nowrap overflow-hidden">
|
|
|
|
{{ slotProps.option.value }}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
</AutoComplete>
|
2024-11-13 16:09:13 +00:00
|
|
|
<div v-if="type === 'color'" class="absolute right-[4px] top-1/2 -mt-3 w-6 h-6 rounded-md border border-surface-300 dark:border-surface-600" :style="{ backgroundColor: previewColor }"></div>
|
|
|
|
</div>
|
2024-11-12 09:18:07 +00:00
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script>
|
2024-11-13 16:09:13 +00:00
|
|
|
import { UniqueComponentId } from '@primevue/core/utils';
|
2024-11-12 09:18:07 +00:00
|
|
|
import { $dt } from '@primevue/themes';
|
2025-01-02 22:30:06 +00:00
|
|
|
import set from 'lodash.set';
|
|
|
|
import unset from 'lodash.unset';
|
2024-11-12 09:18:07 +00:00
|
|
|
|
|
|
|
export default {
|
|
|
|
emits: ['update:modelValue'],
|
|
|
|
props: {
|
|
|
|
label: {
|
|
|
|
type: String,
|
|
|
|
default: undefined
|
|
|
|
},
|
|
|
|
type: {
|
|
|
|
type: String,
|
|
|
|
default: undefined
|
|
|
|
},
|
|
|
|
modelValue: {
|
|
|
|
type: null,
|
|
|
|
default: undefined
|
2025-01-02 22:30:06 +00:00
|
|
|
},
|
|
|
|
componentKey: {
|
|
|
|
type: null,
|
|
|
|
default: null
|
|
|
|
},
|
|
|
|
path: {
|
|
|
|
type: String,
|
|
|
|
default: undefined
|
2024-11-12 09:18:07 +00:00
|
|
|
}
|
|
|
|
},
|
2024-11-12 13:44:52 +00:00
|
|
|
data() {
|
|
|
|
return {
|
2024-11-13 16:09:13 +00:00
|
|
|
id: null,
|
|
|
|
items: null
|
2024-11-12 13:44:52 +00:00
|
|
|
};
|
|
|
|
},
|
2024-11-14 09:57:46 +00:00
|
|
|
created() {
|
2024-11-13 16:09:13 +00:00
|
|
|
this.id = 'dt_field_' + UniqueComponentId();
|
|
|
|
},
|
2024-11-12 09:18:07 +00:00
|
|
|
methods: {
|
2024-11-14 07:25:06 +00:00
|
|
|
onOptionSelect(event) {
|
2024-11-14 09:57:46 +00:00
|
|
|
this.$emit('update:modelValue', event.value.label);
|
2024-11-14 07:25:06 +00:00
|
|
|
event.originalEvent.stopPropagation();
|
|
|
|
},
|
2024-11-12 09:18:07 +00:00
|
|
|
onInput(event) {
|
2024-11-14 07:25:06 +00:00
|
|
|
this.$emit('update:modelValue', event.target.value);
|
|
|
|
},
|
|
|
|
search(event) {
|
|
|
|
const query = event.query;
|
2024-11-12 13:44:52 +00:00
|
|
|
|
2024-11-14 07:25:06 +00:00
|
|
|
if (query.startsWith('{')) {
|
2024-11-22 10:02:06 +00:00
|
|
|
this.items = this.$appState.designer.acTokens.filter((t) => t.label.startsWith(query));
|
2024-11-12 13:44:52 +00:00
|
|
|
} else {
|
2024-11-14 08:58:02 +00:00
|
|
|
this.items = [];
|
2024-11-12 13:44:52 +00:00
|
|
|
}
|
2025-01-02 22:30:06 +00:00
|
|
|
},
|
|
|
|
getPathFromColorScheme(colorScheme) {
|
|
|
|
const lightPrefix = 'light.';
|
|
|
|
const darkPrefix = 'dark.';
|
|
|
|
|
|
|
|
if (colorScheme.startsWith(lightPrefix)) {
|
|
|
|
return colorScheme.slice(lightPrefix.length);
|
|
|
|
} else if (colorScheme.startsWith(darkPrefix)) {
|
|
|
|
return colorScheme.slice(darkPrefix.length);
|
|
|
|
}
|
|
|
|
|
|
|
|
return colorScheme;
|
|
|
|
},
|
|
|
|
transfer(event) {
|
|
|
|
let tokens = this.$appState.designer.theme.preset.components[this.componentKey] || this.$appState.designer.theme.preset.directives[this.componentKey];
|
|
|
|
const colorSchemePrefix = 'colorScheme.';
|
|
|
|
|
|
|
|
if (this.path.startsWith(colorSchemePrefix)) {
|
|
|
|
let tokenPath = this.getPathFromColorScheme(this.path.slice(colorSchemePrefix.length));
|
|
|
|
|
|
|
|
set(tokens, tokenPath, this.modelValue);
|
|
|
|
unset(tokens, 'colorScheme.light.' + tokenPath);
|
|
|
|
unset(tokens, 'colorScheme.dark.' + tokenPath);
|
|
|
|
} else {
|
|
|
|
set(tokens, 'colorScheme.light.' + this.path, this.modelValue);
|
|
|
|
set(tokens, 'colorScheme.dark.' + this.path, this.modelValue);
|
|
|
|
unset(tokens, this.path);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.removeEmptyProps(tokens);
|
|
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
},
|
|
|
|
removeEmptyProps(obj) {
|
|
|
|
if (typeof obj !== 'object' || obj === null) {
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const key in obj) {
|
|
|
|
if (obj.hasOwnProperty(key)) {
|
|
|
|
const value = obj[key];
|
|
|
|
|
|
|
|
if (typeof value === 'object' && value !== null) {
|
|
|
|
this.removeEmptyProps(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof value === 'object' && value !== null && Object.keys(value).length === 0) {
|
|
|
|
delete obj[key];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return obj;
|
2024-11-12 09:18:07 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
computed: {
|
|
|
|
previewColor() {
|
2024-11-14 09:57:46 +00:00
|
|
|
const tokenValue = typeof this.modelValue === 'object' ? this.modelValue.label : this.modelValue;
|
|
|
|
|
|
|
|
return tokenValue && tokenValue.trim().length && tokenValue.startsWith('{') && tokenValue.endsWith('}') ? $dt(tokenValue).variable : tokenValue;
|
2024-11-13 16:09:13 +00:00
|
|
|
},
|
|
|
|
inputId() {
|
|
|
|
return this.id + '_input';
|
2024-11-12 09:18:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
</script>
|