mirror of
https://github.com/sysadminsmedia/homebox.git
synced 2026-01-04 20:14:54 +01:00
* feat: migrate tools page and label generator to shadcn * chore: lint issues * feat: also do profile page * feat: shadcn 404 page * feat: login page shadcn * fix: daisyui ironically breaks the z height for the login page * feat: componentise the language selector and add it to the login page * feat: use nuxtlink * feat: card and table made more shadcn * feat: shadcn statscard * chore: lint * feat: shadcn labelchip and locationcard * feat: shadcn locations page * refactor: remove unused new item page * chore: lint * feat: shadcn item card * fix: wrapping of location and lint * feat: ctrl enter in text area in form submits form * feat: begin shadcn locations page and remove pageqrcode comp in favour of integrating it into labelmaker * chore: lint + remove unused code * fix: remove uneeded margin * feat: shadcn labels page and fix some issues with location * feat: shadcn scanner * chore: lint * feat: begin shadcning item pages * feat: shadcn maintenance page * feat: begin shadcn search page * fix: quick switch blurry text and crashing page when switching + incorrect z height for create menu * feat: finish shadcn search page * chore: lint * feat: shadcn edit item page * fix: quickmenumodal bug * feat: shadcn item details page * feat: remove all non-color related daisyui classes * fix: type error * fix: quick menu modal again :(
126 lines
3.8 KiB
Vue
126 lines
3.8 KiB
Vue
<script setup lang="ts">
|
|
import { BrowserMultiFormatReader, NotFoundException } from "@zxing/library";
|
|
import { useI18n } from "vue-i18n";
|
|
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card";
|
|
import { Select, SelectTrigger, SelectValue, SelectContent, SelectItem } from "@/components/ui/select";
|
|
import MdiAlertCircleOutline from "~icons/mdi/alert-circle-outline";
|
|
|
|
definePageMeta({
|
|
middleware: ["auth"],
|
|
});
|
|
useHead({
|
|
title: "Homebox | Scanner",
|
|
});
|
|
|
|
const { t } = useI18n();
|
|
|
|
const sources = ref<MediaDeviceInfo[]>([]);
|
|
const selectedSource = ref<string | null>(null);
|
|
const loading = ref(false);
|
|
const video = ref<HTMLVideoElement>();
|
|
const codeReader = new BrowserMultiFormatReader();
|
|
const errorMessage = ref<string | null>(null);
|
|
|
|
const handleError = (error: unknown) => {
|
|
console.error("Scanner error:", error);
|
|
errorMessage.value = t("scanner.error");
|
|
};
|
|
|
|
onMounted(async () => {
|
|
if (!(navigator && navigator.mediaDevices && "enumerateDevices" in navigator.mediaDevices)) {
|
|
errorMessage.value = t("scanner.unsupported");
|
|
return;
|
|
}
|
|
|
|
try {
|
|
const devices = await codeReader.listVideoInputDevices();
|
|
sources.value = devices;
|
|
|
|
if (devices.length > 0) {
|
|
for (let i = 0; i < devices.length; i++) {
|
|
if (devices[i].label.toLowerCase().includes("back")) {
|
|
selectedSource.value = devices[i].deviceId;
|
|
}
|
|
}
|
|
if (!selectedSource.value) {
|
|
selectedSource.value = devices[0].deviceId;
|
|
}
|
|
} else {
|
|
errorMessage.value = t("scanner.no_sources");
|
|
}
|
|
} catch (err) {
|
|
handleError(err);
|
|
}
|
|
});
|
|
|
|
// stop the code reader when navigating away
|
|
onBeforeUnmount(() => codeReader.reset());
|
|
|
|
watch(selectedSource, async newSource => {
|
|
codeReader.reset();
|
|
|
|
try {
|
|
await codeReader.decodeFromVideoDevice(newSource, video.value!, (result, err) => {
|
|
if (result && !loading.value) {
|
|
loading.value = true;
|
|
try {
|
|
const url = new URL(result.getText());
|
|
if (!url.pathname.startsWith("/")) {
|
|
throw new Error(t("scanner.invalid_url"));
|
|
}
|
|
const sanitizedPath = url.pathname.replace(/[^a-zA-Z0-9-_/]/g, "");
|
|
navigateTo(sanitizedPath);
|
|
} catch (err) {
|
|
loading.value = false;
|
|
handleError(err);
|
|
}
|
|
}
|
|
if (err && !(err instanceof NotFoundException)) {
|
|
console.error(err);
|
|
handleError(err);
|
|
}
|
|
});
|
|
} catch (err) {
|
|
handleError(err);
|
|
}
|
|
});
|
|
</script>
|
|
|
|
<template>
|
|
<Card class="mx-auto w-full max-w-screen-md">
|
|
<CardHeader>
|
|
<CardTitle>{{ t("scanner.title") }}</CardTitle>
|
|
</CardHeader>
|
|
<CardContent>
|
|
<div
|
|
v-if="errorMessage"
|
|
class="mb-5 flex items-center gap-2 rounded-md border border-destructive bg-destructive/10 p-4 text-destructive"
|
|
role="alert"
|
|
>
|
|
<MdiAlertCircleOutline class="text-destructive" />
|
|
<span class="text-sm font-medium">{{ errorMessage }}</span>
|
|
</div>
|
|
<!-- eslint-disable-next-line tailwindcss/no-custom-classname -->
|
|
<video ref="video" class="aspect-video w-full rounded-lg bg-muted shadow" poster="data:image/gif,AAAA"></video>
|
|
<div class="mt-4">
|
|
<Select v-model="selectedSource">
|
|
<SelectTrigger class="w-full">
|
|
<SelectValue :placeholder="t('scanner.select_video_source')" />
|
|
</SelectTrigger>
|
|
<SelectContent>
|
|
<SelectItem v-for="source in sources" :key="source.deviceId" :value="source.deviceId">
|
|
{{ source.label }}
|
|
</SelectItem>
|
|
</SelectContent>
|
|
</Select>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
</template>
|
|
|
|
<style scoped>
|
|
video {
|
|
object-fit: cover;
|
|
}
|
|
</style>
|