2024-09-13 09:36:41 +07:00
|
|
|
<script setup lang="ts">
|
|
|
|
|
import { Icon } from '@iconify/vue';
|
2024-09-25 17:47:01 +07:00
|
|
|
import { computed } from 'vue';
|
2024-09-16 14:47:37 +07:00
|
|
|
|
2024-09-13 09:36:41 +07:00
|
|
|
type Node = {
|
|
|
|
|
[key: string]: any;
|
2024-09-16 13:25:06 +07:00
|
|
|
opened?: boolean;
|
2024-09-13 09:36:41 +07:00
|
|
|
selected?: boolean;
|
|
|
|
|
children?: Node[];
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
type Props = {
|
|
|
|
|
level?: number;
|
|
|
|
|
keyTitle?: string;
|
|
|
|
|
keySubtitle?: string;
|
2024-09-13 09:50:27 +07:00
|
|
|
expandable?: boolean;
|
2024-09-25 17:47:01 +07:00
|
|
|
hideCheckBox?: boolean;
|
|
|
|
|
iconSize?: string;
|
2024-09-13 09:36:41 +07:00
|
|
|
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-25 17:47:01 +07:00
|
|
|
const maxLevel = computed(() =>
|
|
|
|
|
props.decoration?.reduce((max, v) => {
|
|
|
|
|
return v.level && v.level > max ? v.level : max;
|
|
|
|
|
}, 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;
|
|
|
|
|
}
|
2024-09-13 09:36:41 +07:00
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
<template>
|
2024-09-25 17:47:01 +07:00
|
|
|
<div
|
|
|
|
|
class="tree-container"
|
|
|
|
|
:class="{
|
|
|
|
|
'q-pl-lg': level && level > 0,
|
|
|
|
|
'last-children': level && level === maxLevel,
|
|
|
|
|
}"
|
|
|
|
|
>
|
|
|
|
|
<div
|
|
|
|
|
v-for="(node, i) in nodes"
|
|
|
|
|
class="tree-item"
|
|
|
|
|
:class="{
|
|
|
|
|
'q-pt-sm': level !== 0 && level !== maxLevel && i !== 0,
|
|
|
|
|
'q-pt-xs': level === maxLevel && i === 0,
|
|
|
|
|
}"
|
|
|
|
|
:key="i"
|
|
|
|
|
>
|
2024-09-13 09:36:41 +07:00
|
|
|
<slot
|
|
|
|
|
v-if="$slots['item']"
|
|
|
|
|
name="item"
|
2024-09-18 18:02:18 +07:00
|
|
|
:data="{ node, toggleExpand, toggleCheck }"
|
2024-09-13 09:36:41 +07:00
|
|
|
/>
|
2024-09-13 09:43:21 +07:00
|
|
|
|
2024-09-13 09:36:41 +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-25 17:47:01 +07:00
|
|
|
<div
|
|
|
|
|
v-if="level !== maxLevel"
|
|
|
|
|
class="q-mr-md"
|
|
|
|
|
style="color: var(--stone-4)"
|
|
|
|
|
>
|
|
|
|
|
<q-icon
|
|
|
|
|
name="mdi-chevron-down-circle"
|
|
|
|
|
size="sm"
|
|
|
|
|
:style="`transform: rotate(${node.opened ? '180deg' : '0'})`"
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<label
|
|
|
|
|
v-if="!hideCheckBox"
|
|
|
|
|
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"
|
2024-09-25 17:47:01 +07:00
|
|
|
:style="`background: ${dec?.bg}; color: ${dec?.fg}; height: ${iconSize}; width: ${iconSize}`"
|
2024-09-16 10:32:06 +07:00
|
|
|
>
|
2024-09-25 17:47:01 +07:00
|
|
|
<div
|
|
|
|
|
:style="`height: calc(${iconSize} - 40%); width: calc(${iconSize} - 40%)`"
|
|
|
|
|
>
|
|
|
|
|
<Icon
|
|
|
|
|
v-if="dec && dec.icon"
|
|
|
|
|
:icon="dec.icon"
|
|
|
|
|
class="full-width full-height"
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
2024-09-16 10:32:06 +07:00
|
|
|
</div>
|
2024-09-13 09:43:21 +07:00
|
|
|
|
2024-09-13 09:36:41 +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>
|
2024-09-13 09:36:41 +07:00
|
|
|
</template>
|
2024-09-25 17:47:01 +07:00
|
|
|
<q-separator v-if="!level" spaced="md"></q-separator>
|
2024-09-13 09:43:21 +07:00
|
|
|
|
2024-09-16 10:32:06 +07:00
|
|
|
<transition name="slide">
|
2024-09-25 17:47:01 +07:00
|
|
|
<div v-if="node.opened && node.children && node.children.length > 0">
|
2024-09-16 10:32:06 +07:00
|
|
|
<TreeView
|
2024-09-25 17:47:01 +07:00
|
|
|
:iconSize
|
|
|
|
|
:hideCheckBox
|
2024-09-16 10:32:06 +07:00
|
|
|
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>
|
2024-09-13 09:36:41 +07:00
|
|
|
</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;
|
2024-09-13 09:36:41 +07:00
|
|
|
|
|
|
|
|
& .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);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-13 09:36:41 +07:00
|
|
|
& .item__checkbox {
|
2024-09-13 09:50:27 +07:00
|
|
|
padding: 0.1rem 0.5rem;
|
2024-09-13 09:36:41 +07:00
|
|
|
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%;
|
2024-09-13 09:36:41 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
& .item__title {
|
|
|
|
|
font-weight: 600;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
& .item__subtitle {
|
|
|
|
|
font-size: 80%;
|
|
|
|
|
color: hsla(var(--text-mute-2));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-09-16 10:32:06 +07:00
|
|
|
|
2024-09-25 17:47:01 +07:00
|
|
|
.last-children {
|
|
|
|
|
margin-left: 50px;
|
|
|
|
|
}
|
|
|
|
|
|
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;
|
|
|
|
|
}
|
2024-09-13 09:36:41 +07:00
|
|
|
</style>
|