mirror of
https://github.com/hay-kot/homebox.git
synced 2025-08-04 00:30:27 +00:00
initial tree location elements
This commit is contained in:
parent
ad6115770d
commit
922b0a1623
3 changed files with 142 additions and 0 deletions
105
frontend/components/Location/Tree/Node.vue
Normal file
105
frontend/components/Location/Tree/Node.vue
Normal file
|
@ -0,0 +1,105 @@
|
|||
<script setup lang="ts">
|
||||
import { useTreeState } from "./tree-state";
|
||||
import { ItemSummary, TreeItem } from "~~/lib/api/types/data-contracts";
|
||||
|
||||
type Props = {
|
||||
type?: "location" | "item";
|
||||
treeId: string;
|
||||
item: TreeItem;
|
||||
};
|
||||
const props = withDefaults(defineProps<Props>(), {
|
||||
type: "location",
|
||||
});
|
||||
|
||||
const link = computed(() => {
|
||||
return props.type === "location" ? `/location/${props.item.id}` : `/item/${props.item.id}`;
|
||||
});
|
||||
|
||||
const hasChildren = computed(() => {
|
||||
return props.item.children.length > 0;
|
||||
});
|
||||
|
||||
const state = useTreeState(props.treeId);
|
||||
|
||||
const openRef = computed({
|
||||
get() {
|
||||
return state.value[nodeHash.value] ?? false;
|
||||
},
|
||||
set(value) {
|
||||
state.value[nodeHash.value] = value;
|
||||
},
|
||||
});
|
||||
|
||||
const nodeHash = computed(() => {
|
||||
// converts a UUID to a short hash
|
||||
return props.item.id.replace(/-/g, "").substring(0, 8);
|
||||
});
|
||||
|
||||
const api = useUserApi();
|
||||
|
||||
const hasFetched = ref(false);
|
||||
const items = ref<ItemSummary[]>([]);
|
||||
|
||||
async function fetchItems() {
|
||||
const { data, error } = await api.items.getAll({
|
||||
locations: [props.item.id],
|
||||
});
|
||||
|
||||
if (error) {
|
||||
return;
|
||||
}
|
||||
|
||||
hasFetched.value = true;
|
||||
items.value = data.items;
|
||||
|
||||
console.log("fetched items", items.value);
|
||||
}
|
||||
|
||||
watch(
|
||||
openRef,
|
||||
async value => {
|
||||
if (value && !hasFetched.value && props.type === "location") {
|
||||
await fetchItems();
|
||||
}
|
||||
},
|
||||
{ immediate: true }
|
||||
);
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div>
|
||||
<div
|
||||
class="node flex items-center gap-1 rounded p-1"
|
||||
:class="{
|
||||
'cursor-pointer hover:bg-base-200': hasChildren,
|
||||
}"
|
||||
@click="openRef = !openRef"
|
||||
>
|
||||
<div
|
||||
class="p-1/2 rounded mr-1 flex items-center justify-center"
|
||||
:class="{
|
||||
'hover:bg-base-200': hasChildren,
|
||||
}"
|
||||
>
|
||||
<div v-if="!hasChildren" class="h-6 w-6"></div>
|
||||
<label
|
||||
v-else
|
||||
class="swap swap-rotate"
|
||||
:class="{
|
||||
'swap-active': openRef,
|
||||
}"
|
||||
>
|
||||
<Icon name="mdi-chevron-right" class="h-6 w-6 swap-off" />
|
||||
<Icon name="mdi-chevron-down" class="h-6 w-6 swap-on" />
|
||||
</label>
|
||||
</div>
|
||||
<Icon name="mdi-map-marker" class="h-4 w-4" />
|
||||
<NuxtLink class="hover:link text-lg" :to="link" @click.stop>{{ item.name }} </NuxtLink>
|
||||
</div>
|
||||
<div v-if="openRef" class="ml-4">
|
||||
<LocationTreeNode v-for="child in item.children" :key="child.id" :item="child" :tree-id="treeId" />
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped></style>
|
20
frontend/components/Location/Tree/Root.vue
Normal file
20
frontend/components/Location/Tree/Root.vue
Normal file
|
@ -0,0 +1,20 @@
|
|||
<script setup lang="ts">
|
||||
import { TreeItem } from "~~/lib/api/types/data-contracts";
|
||||
|
||||
type Props = {
|
||||
locs: TreeItem[];
|
||||
treeId: string;
|
||||
};
|
||||
|
||||
defineProps<Props>();
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<BaseCard class="p-4">
|
||||
<div class="p-4 border-2 root">
|
||||
<LocationTreeNode v-for="item in locs" :key="item.id" :item="item" :tree-id="treeId" />
|
||||
</div>
|
||||
</BaseCard>
|
||||
</template>
|
||||
|
||||
<style></style>
|
17
frontend/components/Location/Tree/tree-state.ts
Normal file
17
frontend/components/Location/Tree/tree-state.ts
Normal file
|
@ -0,0 +1,17 @@
|
|||
import type { Ref } from "vue";
|
||||
|
||||
type TreeState = Record<string, boolean>;
|
||||
|
||||
const store: Record<string, Ref<TreeState>> = {};
|
||||
|
||||
export function newTreeKey(): string {
|
||||
return Math.random().toString(36).substring(2);
|
||||
}
|
||||
|
||||
export function useTreeState(key: string): Ref<TreeState> {
|
||||
if (!store[key]) {
|
||||
store[key] = ref({});
|
||||
}
|
||||
|
||||
return store[key];
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue