jws-frontend/src/components/shared/TreeView.vue

170 lines
3.7 KiB
Vue
Raw Normal View History

<script setup lang="ts">
import { Icon } from '@iconify/vue';
2024-09-16 14:47:37 +07:00
type Node = {
[key: string]: any;
2024-09-16 13:25:06 +07:00
opened?: boolean;
selected?: boolean;
children?: Node[];
};
type Props = {
level?: number;
keyTitle?: string;
keySubtitle?: string;
2024-09-13 09:50:27 +07:00
expandable?: boolean;
decoration?: {
level?: number;
bg?: string;
fg?: string;
icon?: string;
}[];
};
const props = defineProps<Props>();
const nodes = defineModel<Node[]>('nodes', { required: true });
2024-09-13 09:43:21 +07:00
2024-09-18 18:02:18 +07:00
const emits = defineEmits<{ (e: 'checked'): void }>();
2024-09-13 09:43:21 +07:00
const dec = props.decoration?.find((v) => v.level === (props.level || 0));
2024-09-18 18:02:18 +07:00
function recursiveDeselect(node: Node) {
if (node.children) {
node.children.forEach((v) => {
v.selected = false;
recursiveDeselect(v);
});
}
}
function toggleCheck(node: Node) {
node.selected = !node.selected;
if (node.selected === false) recursiveDeselect(node);
if (node.selected === true) emits('checked');
}
function toggleExpand(node: Node) {
node.opened = !node.opened;
}
</script>
<template>
<div class="tree-container">
2024-09-16 10:32:06 +07:00
<div v-for="(node, i) in nodes" class="tree-item" :key="i">
<slot
v-if="$slots['item']"
name="item"
2024-09-18 18:02:18 +07:00
:data="{ node, toggleExpand, toggleCheck }"
/>
2024-09-13 09:43:21 +07:00
<template v-else>
2024-09-13 09:50:27 +07:00
<div
2024-09-16 10:32:06 +07:00
class="item__content row items-center no-wrap"
2024-09-18 18:02:18 +07:00
@click="toggleExpand(node)"
2024-09-13 09:50:27 +07:00
>
2024-09-16 10:32:06 +07:00
<label class="flex items-center item__checkbox" @click.stop>
2024-09-18 18:02:18 +07:00
<input
type="checkbox"
v-model="node.selected"
@click="toggleCheck(node)"
/>
2024-09-13 09:50:27 +07:00
</label>
2024-09-13 09:43:21 +07:00
2024-09-16 10:32:06 +07:00
<div
class="item__icon flex items-center justify-center"
:style="`background: ${dec?.bg}; color: ${dec?.fg}`"
>
<Icon v-if="dec && dec.icon" :icon="dec.icon" />
</div>
2024-09-13 09:43:21 +07:00
<div class="column">
<span class="item__title">
{{ node[keyTitle || 'title'] || 'No Title' }}
</span>
<span class="item__subtitle">
{{ node[keySubtitle || 'subtitle'] || 'No Subtitle' }}
</span>
</div>
2024-09-13 09:50:27 +07:00
</div>
</template>
2024-09-16 10:32:06 +07:00
<q-separator v-if="!level"></q-separator>
2024-09-13 09:43:21 +07:00
2024-09-16 10:32:06 +07:00
<transition name="slide">
<div
class="q-pl-lg q-pt-sm"
2024-09-16 13:25:06 +07:00
v-if="node.opened && node.children && node.children.length > 0"
2024-09-16 10:32:06 +07:00
>
<TreeView
class="item__children"
v-if="node.children"
v-model:nodes="node.children"
2024-09-18 18:02:18 +07:00
@checked="
() => {
node.selected = true;
$emit('checked');
}
"
2024-09-16 10:32:06 +07:00
:level="(level || 0) + 1"
:expandable
:decoration
/>
</div>
</transition>
</div>
</div>
</template>
<style lang="css">
.tree-container {
display: flex;
flex-direction: column;
2024-09-13 09:43:21 +07:00
user-select: none;
2024-09-16 10:32:06 +07:00
gap: 8px;
& .tree-item {
2024-09-13 09:43:21 +07:00
& .item__content {
2024-09-13 09:50:27 +07:00
padding: 0.1rem 0.5rem;
2024-09-13 09:43:21 +07:00
&:hover {
background: hsla(0 0% 0% / 0.1);
border-radius: var(--radius-2);
}
}
& .item__checkbox {
2024-09-13 09:50:27 +07:00
padding: 0.1rem 0.5rem;
margin-right: 1rem;
}
& .item__icon {
margin-right: 1rem;
2024-09-16 10:32:06 +07:00
width: 1.5rem;
height: 1.5rem;
border-radius: 50%;
}
& .item__title {
font-weight: 600;
}
& .item__subtitle {
font-size: 80%;
color: hsla(var(--text-mute-2));
}
}
}
2024-09-16 10:32:06 +07:00
.slide-enter-active {
transition: all 0.1s ease-out;
}
.slide-leave-active {
transition: all 0.1s cubic-bezier(1, 0.5, 0.8, 1);
}
.slide-enter-from,
.slide-leave-to {
transform: translateY(-20px);
opacity: 0;
}
</style>