Files
homebox/frontend/components/Form/TextArea.vue
Tonya cbaf483788 migrate pages to shadcn (#628)
* 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 :(
2025-04-20 08:58:03 +01:00

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>