mirror of
https://github.com/sysadminsmedia/homebox.git
synced 2025-12-27 23:46:37 +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 :(
105 lines
2.6 KiB
Vue
105 lines
2.6 KiB
Vue
<template>
|
|
<div v-if="!inline" class="flex w-full flex-col gap-1.5">
|
|
<Label :for="id" class="flex w-full px-1">
|
|
<span>{{ label }}</span>
|
|
<span class="grow"></span>
|
|
<span :class="{ 'text-red-600': isLengthInvalid }">
|
|
{{ lengthIndicator }}
|
|
</span>
|
|
</Label>
|
|
<Textarea
|
|
:id="id"
|
|
v-model="value"
|
|
:placeholder="placeholder"
|
|
class="min-h-[112px] w-full resize-none"
|
|
@keydown="handleKeyDown"
|
|
/>
|
|
</div>
|
|
<div v-else class="sm:grid sm:grid-cols-4 sm:items-start sm:gap-4">
|
|
<Label :for="id" class="flex w-full px-1 py-2">
|
|
<span>{{ label }}</span>
|
|
<span class="grow"></span>
|
|
<span :class="{ 'text-red-600': isLengthInvalid }">
|
|
{{ lengthIndicator }}
|
|
</span>
|
|
</Label>
|
|
<Textarea
|
|
:id="id"
|
|
v-model="value"
|
|
autosize
|
|
:placeholder="placeholder"
|
|
class="col-span-3 mt-2 w-full resize-none"
|
|
@keydown="handleKeyDown"
|
|
/>
|
|
</div>
|
|
</template>
|
|
|
|
<script lang="ts" setup>
|
|
import { computed } from "vue";
|
|
import { Label } from "~/components/ui/label";
|
|
import { Textarea } from "~/components/ui/textarea";
|
|
|
|
const props = defineProps({
|
|
label: {
|
|
type: String,
|
|
required: true,
|
|
},
|
|
modelValue: {
|
|
type: String,
|
|
required: true,
|
|
},
|
|
placeholder: {
|
|
type: String,
|
|
default: "",
|
|
},
|
|
inline: {
|
|
type: Boolean,
|
|
default: false,
|
|
},
|
|
maxLength: {
|
|
type: Number,
|
|
default: -1,
|
|
required: false,
|
|
},
|
|
minLength: {
|
|
type: Number,
|
|
default: -1,
|
|
required: false,
|
|
},
|
|
});
|
|
|
|
const id = useId();
|
|
const value = useVModel(props, "modelValue");
|
|
|
|
const isLengthInvalid = computed(() => {
|
|
if (typeof value.value !== "string") return false;
|
|
const len = value.value.length;
|
|
const max = props.maxLength;
|
|
const min = props.minLength;
|
|
// invalid if max length exists and is exceeded OR min length exists and is not met
|
|
return (max !== -1 && len > max) || (min !== -1 && len < min);
|
|
});
|
|
|
|
const lengthIndicator = computed(() => {
|
|
if (typeof value.value !== "string") return "";
|
|
const max = props.maxLength;
|
|
if (max !== -1) {
|
|
return `${value.value.length}/${max}`;
|
|
}
|
|
return "";
|
|
});
|
|
|
|
const handleKeyDown = (event: KeyboardEvent) => {
|
|
if (event.ctrlKey && event.key === "Enter") {
|
|
// find the closest ancestor form element
|
|
const targetElement = event.target as HTMLElement;
|
|
const form = targetElement.closest("form");
|
|
|
|
if (form) {
|
|
event.preventDefault();
|
|
form.requestSubmit();
|
|
}
|
|
}
|
|
};
|
|
</script>
|