Files
ezbookkeeping/src/components/desktop/IconSelect.vue
T

104 lines
3.6 KiB
Vue

<template>
<v-select
density="comfortable"
item-title="icon"
item-value="id"
persistent-placeholder
:disabled="disabled"
:label="label"
v-model="icon"
@update:menu="onMenuStateChanged"
>
<template #selection>
<v-label class="cursor-pointer">
<ItemIcon :icon-type="iconType" :icon-id="icon" :color="color" />
</v-label>
</template>
<template #no-data>
<div class="icon-select-dropdown px-2" ref="dropdownMenu">
<div class="icon-item" :class="{ 'row-has-selected-item': hasSelectedIcon(row) }"
:style="`grid-template-columns: repeat(${itemPerRow}, minmax(0, 1fr));`"
:key="idx" v-for="(row, idx) in allIconRows">
<div class="text-center" :key="iconInfo.id" v-for="iconInfo in row">
<div class="cursor-pointer" @click="icon = iconInfo.id">
<ItemIcon class="ma-2" icon-type="fixed" :icon-id="iconInfo.icon" :color="color" v-if="!modelValue || modelValue !== iconInfo.id" />
<v-badge class="right-bottom-icon" color="primary"
offset-x="8" offset-y="10"
:location="`bottom ${textDirection === TextDirection.LTR ? 'right' : 'left'}`"
:icon="mdiCheck"
v-if="modelValue && modelValue === iconInfo.id">
<ItemIcon class="ma-2" icon-type="fixed" :icon-id="iconInfo.icon" :color="color" />
</v-badge>
</div>
</div>
</div>
</div>
</template>
</v-select>
</template>
<script setup lang="ts">
import { ref, computed, useTemplateRef, nextTick } from 'vue';
import { useI18n } from '@/locales/helpers.ts';
import { TextDirection } from '@/core/text.ts';
import type { ColorValue } from '@/core/color.ts';
import type { IconInfo, IconInfoWithId } from '@/core/icon.ts';
import { arrayContainsFieldValue } from '@/lib/common.ts';
import { getIconsInRows } from '@/lib/icon.ts';
import { scrollToSelectedItem } from '@/lib/ui/common.ts';
import {
mdiCheck
} from '@mdi/js';
const props = defineProps<{
modelValue: string;
disabled?: boolean;
label?: string;
iconType: string;
color: ColorValue;
columnCount?: number;
allIconInfos: Record<string, IconInfo>;
}>();
const emit = defineEmits<{
(e: 'update:modelValue', value: string): void;
}>();
const { getCurrentLanguageTextDirection } = useI18n();
const dropdownMenu = useTemplateRef<HTMLElement>('dropdownMenu');
const itemPerRow = ref<number>(props.columnCount || 7);
const textDirection = computed<TextDirection>(() => getCurrentLanguageTextDirection());
const allIconRows = computed<IconInfoWithId[][]>(() => getIconsInRows(props.allIconInfos, itemPerRow.value));
const icon = computed<string>({
get: () => props.modelValue,
set: (value: string) => emit('update:modelValue', value)
});
function hasSelectedIcon(row: IconInfoWithId[]): boolean {
return arrayContainsFieldValue(row, 'id', props.modelValue);
}
function onMenuStateChanged(state: boolean): void {
if (state) {
nextTick(() => {
if (dropdownMenu.value && dropdownMenu.value.parentElement) {
scrollToSelectedItem(dropdownMenu.value.parentElement, null, null, '.row-has-selected-item');
}
});
}
}
</script>
<style>
.icon-select-dropdown .icon-item {
display: grid;
}
</style>