mirror of
				https://github.com/hay-kot/homebox.git
				synced 2025-10-26 19:06:43 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			213 lines
		
	
	
	
		
			6 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			213 lines
		
	
	
	
		
			6 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
| <template>
 | |
|   <div>
 | |
|     <!--
 | |
|     Confirmation Modal is a singleton used by all components so we render
 | |
|     it here to ensure it's always available. Possibly could move this further
 | |
|     up the tree
 | |
|    -->
 | |
|     <ModalConfirm />
 | |
|     <ItemCreateModal v-model="modals.item" />
 | |
|     <LabelCreateModal v-model="modals.label" />
 | |
|     <LocationCreateModal v-model="modals.location" />
 | |
|     <AppToast />
 | |
|     <div class="drawer drawer-mobile">
 | |
|       <input id="my-drawer-2" v-model="drawerToggle" type="checkbox" class="drawer-toggle" />
 | |
|       <div class="drawer-content justify-center bg-base-300 pt-20 lg:pt-0">
 | |
|         <AppHeaderDecor class="-mt-10 hidden lg:block" />
 | |
|         <!-- Button -->
 | |
|         <div class="navbar z-[99] lg:hidden top-0 fixed bg-primary shadow-md drawer-button">
 | |
|           <label for="my-drawer-2" class="btn btn-square btn-ghost text-base-100 drawer-button lg:hidden">
 | |
|             <MdiMenu class="h-6 w-6" />
 | |
|           </label>
 | |
|           <NuxtLink to="/home">
 | |
|             <h2 class="text-3xl font-bold tracking-tight text-base-100 flex">
 | |
|               HomeB
 | |
|               <AppLogo class="w-8 -mb-3" />
 | |
|               x
 | |
|             </h2>
 | |
|           </NuxtLink>
 | |
|         </div>
 | |
| 
 | |
|         <slot></slot>
 | |
|       </div>
 | |
| 
 | |
|       <!-- Sidebar -->
 | |
|       <div class="drawer-side shadow-lg">
 | |
|         <label for="my-drawer-2" class="drawer-overlay"></label>
 | |
| 
 | |
|         <!-- Top Section -->
 | |
|         <div class="w-60 py-5 md:py-10 bg-base-200 flex flex-grow-1 flex-col">
 | |
|           <div class="space-y-8">
 | |
|             <div class="flex flex-col items-center gap-4">
 | |
|               <p>Welcome, {{ username }}</p>
 | |
|               <NuxtLink class="avatar placeholder" to="/home">
 | |
|                 <div class="bg-base-300 text-neutral-content rounded-full w-24 p-4">
 | |
|                   <AppLogo />
 | |
|                 </div>
 | |
|               </NuxtLink>
 | |
|             </div>
 | |
|             <div class="flex flex-col bg-base-200">
 | |
|               <div class="mx-auto w-40 mb-6">
 | |
|                 <div class="dropdown overflow visible w-40">
 | |
|                   <label tabindex="0" class="btn btn-primary btn-block text-lg text-no-transform">
 | |
|                     <span>
 | |
|                       <MdiPlus class="mr-1 -ml-1" />
 | |
|                     </span>
 | |
|                     Create
 | |
|                   </label>
 | |
|                   <ul tabindex="0" class="dropdown-content menu p-2 shadow bg-base-100 rounded-box w-40">
 | |
|                     <li v-for="btn in dropdown" :key="btn.name">
 | |
|                       <button @click="btn.action">
 | |
|                         {{ btn.name }}
 | |
|                       </button>
 | |
|                     </li>
 | |
|                   </ul>
 | |
|                 </div>
 | |
|               </div>
 | |
|               <ul class="flex flex-col mx-auto gap-2 w-40 menu">
 | |
|                 <li v-for="n in nav" :key="n.id" class="text-xl" @click="unfocus">
 | |
|                   <NuxtLink
 | |
|                     v-if="n.to"
 | |
|                     class="rounded-btn"
 | |
|                     :to="n.to"
 | |
|                     :class="{
 | |
|                       'bg-secondary text-secondary-content': n.active?.value,
 | |
|                     }"
 | |
|                   >
 | |
|                     <component :is="n.icon" class="h-6 w-6 mr-4" />
 | |
|                     {{ n.name }}
 | |
|                   </NuxtLink>
 | |
|                 </li>
 | |
|               </ul>
 | |
|             </div>
 | |
|           </div>
 | |
| 
 | |
|           <!-- Bottom -->
 | |
|           <button class="mt-auto mx-2 hover:bg-base-300 p-3 rounded-btn" @click="logout">Sign Out</button>
 | |
|         </div>
 | |
|       </div>
 | |
|     </div>
 | |
|   </div>
 | |
| </template>
 | |
| 
 | |
| <script lang="ts" setup>
 | |
|   import { useLabelStore } from "~~/stores/labels";
 | |
|   import { useLocationStore } from "~~/stores/locations";
 | |
|   import MdiMenu from "~icons/mdi/menu";
 | |
|   import MdiPlus from "~icons/mdi/plus";
 | |
| 
 | |
|   import MdiHome from "~icons/mdi/home";
 | |
|   import MdiFileTree from "~icons/mdi/file-tree";
 | |
|   import MdiMagnify from "~icons/mdi/magnify";
 | |
|   import MdiAccount from "~icons/mdi/account";
 | |
|   import MdiCog from "~icons/mdi/cog";
 | |
| 
 | |
|   const username = computed(() => authCtx.user?.name || "User");
 | |
| 
 | |
|   // Preload currency format
 | |
|   useFormatCurrency();
 | |
| 
 | |
|   const modals = reactive({
 | |
|     item: false,
 | |
|     location: false,
 | |
|     label: false,
 | |
|     import: false,
 | |
|   });
 | |
| 
 | |
|   const dropdown = [
 | |
|     {
 | |
|       name: "Item / Asset",
 | |
|       action: () => {
 | |
|         modals.item = true;
 | |
|       },
 | |
|     },
 | |
|     {
 | |
|       name: "Location",
 | |
|       action: () => {
 | |
|         modals.location = true;
 | |
|       },
 | |
|     },
 | |
|     {
 | |
|       name: "Label",
 | |
|       action: () => {
 | |
|         modals.label = true;
 | |
|       },
 | |
|     },
 | |
|   ];
 | |
| 
 | |
|   const route = useRoute();
 | |
| 
 | |
|   const drawerToggle = ref();
 | |
| 
 | |
|   function unfocus() {
 | |
|     // unfocus current element
 | |
|     drawerToggle.value = false;
 | |
|   }
 | |
| 
 | |
|   const nav = [
 | |
|     {
 | |
|       icon: MdiHome,
 | |
|       active: computed(() => route.path === "/home"),
 | |
|       id: 0,
 | |
|       name: "Home",
 | |
|       to: "/home",
 | |
|     },
 | |
|     {
 | |
|       icon: MdiFileTree,
 | |
|       id: 4,
 | |
|       active: computed(() => route.path === "/locations"),
 | |
|       name: "Locations",
 | |
|       to: "/locations",
 | |
|     },
 | |
|     {
 | |
|       icon: MdiMagnify,
 | |
|       id: 3,
 | |
|       active: computed(() => route.path === "/items"),
 | |
|       name: "Search",
 | |
|       to: "/items",
 | |
|     },
 | |
|     {
 | |
|       icon: MdiAccount,
 | |
|       id: 1,
 | |
|       active: computed(() => route.path === "/profile"),
 | |
|       name: "Profile",
 | |
|       to: "/profile",
 | |
|     },
 | |
|     {
 | |
|       icon: MdiCog,
 | |
|       id: 6,
 | |
|       active: computed(() => route.path === "/tools"),
 | |
|       name: "Tools",
 | |
|       to: "/tools",
 | |
|     },
 | |
|   ];
 | |
| 
 | |
|   const labelStore = useLabelStore();
 | |
| 
 | |
|   const locationStore = useLocationStore();
 | |
| 
 | |
|   onServerEvent(ServerEvent.LabelMutation, () => {
 | |
|     labelStore.refresh();
 | |
|   });
 | |
| 
 | |
|   onServerEvent(ServerEvent.LocationMutation, () => {
 | |
|     locationStore.refreshChildren();
 | |
|     locationStore.refreshParents();
 | |
|     locationStore.refreshTree();
 | |
|   });
 | |
| 
 | |
|   onServerEvent(ServerEvent.ItemMutation, () => {
 | |
|     // item mutations can affect locations counts
 | |
|     // so we need to refresh those as well
 | |
|     locationStore.refreshChildren();
 | |
|     locationStore.refreshParents();
 | |
|   });
 | |
| 
 | |
|   const authCtx = useAuthContext();
 | |
|   const api = useUserApi();
 | |
| 
 | |
|   async function logout() {
 | |
|     await authCtx.logout(api);
 | |
|     navigateTo("/");
 | |
|   }
 | |
| </script>
 |