112 lines
No EOL
2.1 KiB
Vue
112 lines
No EOL
2.1 KiB
Vue
<template>
|
|
<div class="block">
|
|
<div class="block__inner">
|
|
<label v-if="label" class="block__label">{{ label }}</label>
|
|
<textarea
|
|
:placeholder="placeholder"
|
|
:value="modelValue"
|
|
@input="onInput"
|
|
class="block__textarea"
|
|
/>
|
|
</div>
|
|
<p v-if="!isValid" class="block__error">{{ errorMessage }}</p>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
type Rule = (value: string) => boolean | string;
|
|
|
|
const emit = defineEmits<{
|
|
(e: 'update:modelValue', value: string | number): void;
|
|
}>();
|
|
const props = defineProps<{
|
|
placeholder: string;
|
|
modelValue?: string;
|
|
rules?: Rule[];
|
|
label?: string;
|
|
}>();
|
|
|
|
const isValid = ref(true);
|
|
const errorMessage = ref('');
|
|
|
|
const onInput = (e: Event) => {
|
|
const target = e.target as HTMLTextAreaElement;
|
|
const value = target.value;
|
|
|
|
isValid.value = true;
|
|
errorMessage.value = '';
|
|
|
|
props.rules?.forEach(rule => {
|
|
const result = rule(value);
|
|
if (result !== true) {
|
|
isValid.value = false;
|
|
errorMessage.value = String(result);
|
|
}
|
|
});
|
|
|
|
emit('update:modelValue', value);
|
|
};
|
|
</script>
|
|
|
|
<style lang="scss" scoped>
|
|
.block {
|
|
width: 100%;
|
|
display: flex;
|
|
flex-direction: column;
|
|
align-items: flex-start;
|
|
gap: 10px;
|
|
position: relative;
|
|
|
|
&__inner {
|
|
width: 100%;
|
|
position: relative;
|
|
display: flex;
|
|
flex-direction: column;
|
|
gap: 10px;
|
|
}
|
|
|
|
&__label {
|
|
color: #374151;
|
|
font-size: 14px;
|
|
font-weight: 500;
|
|
letter-spacing: -0.5px;
|
|
}
|
|
|
|
&__textarea {
|
|
width: 100%;
|
|
height: 150px;
|
|
resize: none;
|
|
padding: 6px 12px;
|
|
border: 1px solid #e0e0e0;
|
|
border-radius: $default_border_radius;
|
|
background-color: $white;
|
|
|
|
color: #1f1f1f;
|
|
font-size: 12px;
|
|
font-weight: 400;
|
|
line-height: 20px;
|
|
|
|
&::placeholder {
|
|
color: #2B2B2B;
|
|
}
|
|
}
|
|
|
|
&__error {
|
|
color: $error;
|
|
font-size: 12px;
|
|
font-weight: 500;
|
|
animation: fadeInUp 0.3s ease;
|
|
|
|
@keyframes fadeInUp {
|
|
0% {
|
|
opacity: 0;
|
|
transform: translateY(-50%);
|
|
}
|
|
100% {
|
|
opacity: 1;
|
|
transform: translateY(0);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
</style> |