mirror of
https://github.com/sysadminsmedia/homebox.git
synced 2025-12-21 21:33:02 +01:00
* feat: Add item templates feature (#435) Add ability to create and manage item templates for quick item creation. Templates store default values and custom fields that can be applied when creating new items. Backend changes: - New ItemTemplate and TemplateField Ent schemas - Template CRUD API endpoints - Create item from template endpoint Frontend changes: - Templates management page with create/edit/delete - Template selector in item creation modal - 'Use as Template' action on item detail page - Templates link in navigation menu * refactor: Improve template item creation with a single query - Add `CreateFromTemplate` method to ItemsRepository that creates items with all template data (including custom fields) in a single atomic transaction, replacing the previous two-phase create-then-update pattern - Fix `GetOne` to require group ID parameter so templates can only be accessed by users in the owning group (security fix) - Simplify `HandleItemTemplatesCreateItem` handler using the new transactional method * Refactor item template types and formatting Updated type annotations in CreateModal.vue to use specific ItemTemplate types instead of 'any'. Improved code formatting for template fields and manufacturer display. Also refactored warranty field logic in item details page for better readability. This resolves the linter issues as well that the bot in github keeps nagging at. * Add 'id' property to template fields Introduces an 'id' property to each field object in CreateModal.vue and item details page to support unique identification of fields. This change prepares the codebase for future enhancements that may require field-level identification. * Removed redundant SQL migrations. Removed redundant SQL migrations per @tankerkiller125's findings. * Updates to PR #1099. Regarding pull #1099. Fixed an issue causing some conflict with GUIDs and old rows in the migration files. * Add new fields and location edge to ItemTemplate Addresses recommendations from @tonyaellie. * Relocated add template button * Added more default fields to the template * Added translation of all strings (think so?) * Make oval buttons round * Added duplicate button to the template (this required a rewrite of the migration files, I made sure only 1 exists per DB type) * Added a Save as template button to a item detail view (this creates a template with all the current data of that item) * Changed all occurrences of space to gap and flex where applicable. * Made template selection persistent after item created. * Collapsible template info on creation view. * Updates to translation and fix for labels/locations I also added a test in here because I keep missing small function tests. That should prevent that from happening again. * Linted * Bring up to date with main, fix some lint/type check issues * In theory fix playwright tests * Fix defaults being unable to be nullable/empty (and thus limiting flexibility) * Last few fixes I think * Forgot to fix the golang tests --------- Co-authored-by: Matthew Kilgore <matthew@kilgore.dev>
414 lines
13 KiB
Vue
414 lines
13 KiB
Vue
<template>
|
|
<div id="app">
|
|
<!--
|
|
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 />
|
|
<OutdatedModal v-if="status" :status="status" />
|
|
<ItemCreateModal />
|
|
<LabelCreateModal />
|
|
<LocationCreateModal />
|
|
<ItemBarcodeModal />
|
|
<AppQuickMenuModal :actions="quickMenuActions" />
|
|
<AppScannerModal />
|
|
<SidebarProvider :default-open="sidebarState">
|
|
<Sidebar collapsible="icon">
|
|
<SidebarHeader class="items-center">
|
|
<SidebarGroupLabel class="text-base group-data-[collapsible=icon]:hidden">{{
|
|
$t("global.welcome", { username: username })
|
|
}}</SidebarGroupLabel>
|
|
<NuxtLink class="group-data-[collapsible=icon]:hidden" to="/home">
|
|
<div class="flex size-24 items-center justify-center rounded-full bg-background-accent p-4">
|
|
<AppLogo />
|
|
</div>
|
|
</NuxtLink>
|
|
<DropdownMenu>
|
|
<DropdownMenuTrigger as-child>
|
|
<SidebarMenuButton
|
|
class="flex justify-center bg-primary text-primary-foreground shadow hover:bg-primary/90 group-data-[collapsible=icon]:justify-start"
|
|
:tooltip="$t('global.create')"
|
|
hotkey="Shortcut: Ctrl+`"
|
|
>
|
|
<MdiPlus />
|
|
<span>
|
|
{{ $t("global.create") }}
|
|
</span>
|
|
</SidebarMenuButton>
|
|
</DropdownMenuTrigger>
|
|
<DropdownMenuContent class="z-40 min-w-[var(--reka-dropdown-menu-trigger-width)]">
|
|
<DropdownMenuItem
|
|
v-for="btn in dropdown"
|
|
:key="btn.id"
|
|
class="group cursor-pointer text-lg"
|
|
@click="openDialog(btn.dialogId as NoParamDialogIDs)"
|
|
>
|
|
{{ btn.name.value }}
|
|
<Shortcut
|
|
v-if="btn.shortcut"
|
|
class="ml-auto hidden group-hover:inline"
|
|
:keys="btn.shortcut.replace('Shift', '⇧').split('+')"
|
|
/>
|
|
</DropdownMenuItem>
|
|
</DropdownMenuContent>
|
|
</DropdownMenu>
|
|
</SidebarHeader>
|
|
|
|
<SidebarContent>
|
|
<SidebarGroup>
|
|
<SidebarMenu>
|
|
<SidebarMenuItem v-for="n in nav" :key="n.id">
|
|
<SidebarMenuLink
|
|
:href="n.to"
|
|
:class="{
|
|
'bg-accent text-accent-foreground': n.active?.value,
|
|
'text-nowrap': typeof locale === 'string' && locale.startsWith('zh-'),
|
|
}"
|
|
:tooltip="n.name.value"
|
|
>
|
|
<component :is="n.icon" />
|
|
<span>{{ n.name.value }}</span>
|
|
</SidebarMenuLink>
|
|
</SidebarMenuItem>
|
|
|
|
<!-- makes scanner accessible easily if using legacy header -->
|
|
<SidebarMenuItem v-if="preferences.displayLegacyHeader">
|
|
<SidebarMenuButton
|
|
:class="{
|
|
'text-nowrap': typeof locale === 'string' && locale.startsWith('zh-'),
|
|
}"
|
|
:tooltip="$t('menu.scanner')"
|
|
@click.prevent="openDialog(DialogID.Scanner)"
|
|
>
|
|
<MdiQrcodeScan />
|
|
<span>{{ $t("menu.scanner") }}</span>
|
|
</SidebarMenuButton>
|
|
</SidebarMenuItem>
|
|
</SidebarMenu>
|
|
</SidebarGroup>
|
|
</SidebarContent>
|
|
|
|
<SidebarFooter>
|
|
<SidebarMenuButton
|
|
class="flex justify-center group-data-[collapsible=icon]:justify-start group-data-[collapsible=icon]:bg-destructive group-data-[collapsible=icon]:text-destructive-foreground group-data-[collapsible=icon]:shadow-sm group-data-[collapsible=icon]:hover:bg-destructive/90"
|
|
:tooltip="$t('global.sign_out')"
|
|
data-testid="logout-button"
|
|
@click="logout"
|
|
>
|
|
<MdiLogout />
|
|
<span>
|
|
{{ $t("global.sign_out") }}
|
|
</span>
|
|
</SidebarMenuButton>
|
|
</SidebarFooter>
|
|
|
|
<SidebarRail />
|
|
</Sidebar>
|
|
<SidebarInset class="min-h-dvh max-w-full overflow-hidden bg-background-accent">
|
|
<div class="relative flex h-full flex-col justify-center">
|
|
<div v-if="preferences.displayLegacyHeader">
|
|
<AppHeaderDecor class="-mt-10 hidden lg:block" />
|
|
<SidebarTrigger class="absolute left-2 top-2 hidden lg:flex" variant="default" />
|
|
</div>
|
|
<!-- IMPORTANT: if you change the height of this div, alter the top value in the item edit page-->
|
|
<div
|
|
class="sticky top-0 z-20 flex h-[var(--header-height-mobile)] translate-y-[-0.5px] flex-col bg-secondary p-2 shadow-md sm:h-[var(--header-height)] sm:flex-row"
|
|
:class="{
|
|
'lg:hidden': preferences.displayLegacyHeader,
|
|
}"
|
|
>
|
|
<div class="flex h-1/2 items-center gap-2 sm:h-auto">
|
|
<SidebarTrigger variant="default" />
|
|
<NuxtLink to="/home">
|
|
<AppHeaderText class="h-6" />
|
|
</NuxtLink>
|
|
</div>
|
|
<div class="sm:grow" />
|
|
<div class="flex h-1/2 grow items-center justify-end gap-2 sm:h-auto">
|
|
<Input
|
|
v-model:model-value="search"
|
|
class="h-9 grow sm:max-w-sm"
|
|
:placeholder="$t('global.search')"
|
|
type="search"
|
|
@keyup.enter="triggerSearch"
|
|
/>
|
|
<div>
|
|
<Button size="icon" @click="triggerSearch">
|
|
<MdiMagnify />
|
|
</Button>
|
|
</div>
|
|
<div>
|
|
<Button size="icon" @click="openScanner">
|
|
<MdiQrcodeScan />
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<slot />
|
|
<div class="grow" />
|
|
|
|
<footer v-if="status" class="bottom-0 w-full pb-4 text-center">
|
|
<p class="text-center text-sm">
|
|
<span
|
|
v-html="
|
|
DOMPurify.sanitize(
|
|
$t('global.footer.version_link', { version: status.build.version, build: status.build.commit })
|
|
)
|
|
"
|
|
/>
|
|
~
|
|
<span v-html="DOMPurify.sanitize($t('global.footer.api_link'))" />
|
|
</p>
|
|
</footer>
|
|
</div>
|
|
</SidebarInset>
|
|
</SidebarProvider>
|
|
</div>
|
|
</template>
|
|
|
|
<script lang="ts" setup>
|
|
import { useI18n } from "vue-i18n";
|
|
import DOMPurify from "dompurify";
|
|
import { useLabelStore } from "~~/stores/labels";
|
|
import { useLocationStore } from "~~/stores/locations";
|
|
|
|
import MdiHome from "~icons/mdi/home";
|
|
import MdiFileTree from "~icons/mdi/file-tree";
|
|
import MdiMagnify from "~icons/mdi/magnify";
|
|
import MdiQrcodeScan from "~icons/mdi/qrcode-scan";
|
|
import MdiAccount from "~icons/mdi/account";
|
|
import MdiCog from "~icons/mdi/cog";
|
|
import MdiWrench from "~icons/mdi/wrench";
|
|
import MdiPlus from "~icons/mdi/plus";
|
|
import MdiLogout from "~icons/mdi/logout";
|
|
import MdiFileDocumentMultiple from "~icons/mdi/file-document-multiple";
|
|
|
|
import {
|
|
Sidebar,
|
|
SidebarContent,
|
|
SidebarFooter,
|
|
SidebarGroup,
|
|
SidebarGroupLabel,
|
|
SidebarHeader,
|
|
SidebarInset,
|
|
SidebarMenu,
|
|
SidebarMenuButton,
|
|
SidebarMenuItem,
|
|
SidebarMenuLink,
|
|
SidebarProvider,
|
|
SidebarRail,
|
|
SidebarTrigger,
|
|
} from "@/components/ui/sidebar";
|
|
import {
|
|
DropdownMenu,
|
|
DropdownMenuContent,
|
|
DropdownMenuItem,
|
|
DropdownMenuTrigger,
|
|
} from "@/components/ui/dropdown-menu";
|
|
import { Shortcut } from "~/components/ui/shortcut";
|
|
import { useDialog } from "~/components/ui/dialog-provider";
|
|
import { Input } from "~/components/ui/input";
|
|
import { Button } from "~/components/ui/button";
|
|
import { toast } from "@/components/ui/sonner";
|
|
import { DialogID, type NoParamDialogIDs, type OptionalDialogIDs } from "~/components/ui/dialog-provider/utils";
|
|
import ModalConfirm from "~/components/ModalConfirm.vue";
|
|
import OutdatedModal from "~/components/App/OutdatedModal.vue";
|
|
import ItemCreateModal from "~/components/Item/CreateModal.vue";
|
|
|
|
import LabelCreateModal from "~/components/Label/CreateModal.vue";
|
|
import LocationCreateModal from "~/components/Location/CreateModal.vue";
|
|
import ItemBarcodeModal from "~/components/Item/BarcodeModal.vue";
|
|
import AppQuickMenuModal from "~/components/App/QuickMenuModal.vue";
|
|
import AppScannerModal from "~/components/App/ScannerModal.vue";
|
|
import AppLogo from "~/components/App/Logo.vue";
|
|
import AppHeaderDecor from "~/components/App/HeaderDecor.vue";
|
|
import AppHeaderText from "~/components/App/HeaderText.vue";
|
|
|
|
const { t, locale } = useI18n();
|
|
const username = computed(() => authCtx.user?.name || "User");
|
|
|
|
const { openDialog } = useDialog();
|
|
|
|
const preferences = useViewPreferences();
|
|
|
|
// get sidebar state from cookies
|
|
const sidebarState = useCookie("sidebar:state", {
|
|
readonly: true,
|
|
decode: value => value !== "false",
|
|
});
|
|
|
|
const pubApi = usePublicApi();
|
|
const { data: status } = useAsyncData(async () => {
|
|
const { data } = await pubApi.status();
|
|
|
|
return data;
|
|
});
|
|
|
|
const search = ref("");
|
|
|
|
const triggerSearch = () => {
|
|
if (search.value) {
|
|
navigateTo(`/items?q=${encodeURIComponent(search.value)}`);
|
|
search.value = "";
|
|
// remove focus from input
|
|
if (document.activeElement && "blur" in document.activeElement) {
|
|
(document.activeElement as HTMLElement).blur();
|
|
}
|
|
}
|
|
};
|
|
|
|
const openScanner = () => {
|
|
// request permission
|
|
if (navigator.mediaDevices) {
|
|
navigator.mediaDevices
|
|
.getUserMedia({ video: true })
|
|
.then(() => {
|
|
openDialog(DialogID.Scanner);
|
|
})
|
|
.catch(err => {
|
|
console.error(err);
|
|
toast.error(t("scanner.permission_denied"));
|
|
});
|
|
} else {
|
|
toast.error(t("scanner.unsupported"));
|
|
}
|
|
};
|
|
|
|
// Preload currency format
|
|
useFormatCurrency();
|
|
|
|
type DropdownItem = {
|
|
id: number;
|
|
name: ComputedRef<string>;
|
|
shortcut: string;
|
|
dialogId: NoParamDialogIDs | OptionalDialogIDs;
|
|
};
|
|
|
|
const dropdown: DropdownItem[] = [
|
|
{
|
|
id: 0,
|
|
name: computed(() => t("menu.create_item")),
|
|
shortcut: "Shift+1",
|
|
dialogId: DialogID.CreateItem,
|
|
},
|
|
{
|
|
id: 1,
|
|
name: computed(() => t("menu.create_location")),
|
|
shortcut: "Shift+3",
|
|
dialogId: DialogID.CreateLocation,
|
|
},
|
|
{
|
|
id: 2,
|
|
name: computed(() => t("menu.create_label")),
|
|
shortcut: "Shift+2",
|
|
dialogId: DialogID.CreateLabel,
|
|
},
|
|
];
|
|
|
|
const route = useRoute();
|
|
|
|
const nav = [
|
|
{
|
|
icon: MdiHome,
|
|
active: computed(() => route.path === "/home"),
|
|
id: 0,
|
|
name: computed(() => t("menu.home")),
|
|
to: "/home",
|
|
},
|
|
{
|
|
icon: MdiFileTree,
|
|
id: 1,
|
|
active: computed(() => route.path === "/locations"),
|
|
name: computed(() => t("menu.locations")),
|
|
to: "/locations",
|
|
},
|
|
{
|
|
icon: MdiMagnify,
|
|
id: 2,
|
|
active: computed(() => route.path === "/items"),
|
|
name: computed(() => t("menu.search")),
|
|
to: "/items",
|
|
},
|
|
{
|
|
icon: MdiFileDocumentMultiple,
|
|
id: 3,
|
|
active: computed(() => route.path === "/templates"),
|
|
name: computed(() => t("menu.templates")),
|
|
to: "/templates",
|
|
},
|
|
{
|
|
icon: MdiWrench,
|
|
id: 4,
|
|
active: computed(() => route.path === "/maintenance"),
|
|
name: computed(() => t("menu.maintenance")),
|
|
to: "/maintenance",
|
|
},
|
|
{
|
|
icon: MdiAccount,
|
|
id: 5,
|
|
active: computed(() => route.path === "/profile"),
|
|
name: computed(() => t("menu.profile")),
|
|
to: "/profile",
|
|
},
|
|
{
|
|
icon: MdiCog,
|
|
id: 6,
|
|
active: computed(() => route.path === "/tools"),
|
|
name: computed(() => t("menu.tools")),
|
|
to: "/tools",
|
|
},
|
|
];
|
|
|
|
const quickMenuActions = reactive([
|
|
...dropdown.map(v => ({
|
|
text: computed(() => v.name.value),
|
|
dialogId: v.dialogId as NoParamDialogIDs,
|
|
shortcut: v.shortcut.split("+")[1] as string,
|
|
type: "create" as const,
|
|
})),
|
|
...nav.map(v => ({
|
|
text: computed(() => v.name.value),
|
|
href: v.to,
|
|
type: "navigate" as const,
|
|
})),
|
|
]);
|
|
|
|
const labelStore = useLabelStore();
|
|
labelStore.ensureAllLabelsFetched();
|
|
|
|
const locationStore = useLocationStore();
|
|
locationStore.ensureLocationsFetched();
|
|
|
|
onMounted(() => {
|
|
locationStore.refreshParents();
|
|
locationStore.refreshTree();
|
|
});
|
|
|
|
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>
|