Text.vue 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. <!--
  2. Champs de saisie de texte, à placer dans un composant `UiForm`
  3. @see https://vuetifyjs.com/en/components/text-fields/
  4. -->
  5. <template>
  6. <v-text-field
  7. :model-value="modelValue"
  8. :label="(label || field) ? $t(label ?? field) : undefined"
  9. :rules="rules"
  10. :disabled="readonly"
  11. :type="(type === 'password' && show) ? 'text' : type"
  12. :error="error || !!fieldViolations"
  13. :error-messages="errorMessage || (fieldViolations ? $t(fieldViolations) : '')"
  14. :append-icon="type === 'password' ? (show ? 'mdi-eye' : 'mdi-eye-off') : ''"
  15. :variant="variant"
  16. @click:append="show = !show"
  17. @update:model-value="onUpdate($event)"
  18. @change="onChange($event)"
  19. />
  20. <!-- v-cleave="mask"-->
  21. </template>
  22. <script setup lang="ts">
  23. import {type Ref, ref} from "@vue/reactivity";
  24. import {useFieldViolation} from "~/composables/form/useFieldViolation";
  25. import type {PropType} from "@vue/runtime-core";
  26. const props = defineProps({
  27. /**
  28. * v-model
  29. */
  30. modelValue: {
  31. type: [String, Number] as PropType<string | number | null>,
  32. required: false,
  33. default: null
  34. },
  35. /**
  36. * Nom de la propriété d'une entité lorsque l'input concerne cette propriété
  37. * - Utilisé par la validation
  38. * - Laisser null si le champ ne s'applique pas à une entité
  39. */
  40. field: {
  41. type: String,
  42. required: false,
  43. default: null
  44. },
  45. /**
  46. * Label du champ
  47. * Si non défini, c'est le nom de propriété qui est utilisé
  48. */
  49. label: {
  50. type: String,
  51. required: false,
  52. default: null
  53. },
  54. /**
  55. * Type d'input HTML
  56. * @see https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/button
  57. */
  58. type: {
  59. type: String,
  60. required: false,
  61. default: null
  62. },
  63. /**
  64. * Définit si le champ est en lecture seule
  65. */
  66. readonly: {
  67. type: Boolean,
  68. required: false,
  69. default: false
  70. },
  71. /**
  72. * Règles de validation
  73. * @see https://vuetify.cn/en/components/forms/#validation-with-submit-clear
  74. */
  75. rules: {
  76. type: Array as PropType<any>,
  77. required: false,
  78. default: () => []
  79. },
  80. /**
  81. * Le champ est-il actuellement en état d'erreur
  82. */
  83. error: {
  84. type: Boolean,
  85. required: false,
  86. default: false
  87. },
  88. /**
  89. * Si le champ est en état d'erreur, quel est le message d'erreur?
  90. */
  91. errorMessage: {
  92. type: String,
  93. required: false,
  94. default: null
  95. },
  96. /**
  97. * Masque de saisie
  98. * @see https://github.com/nosir/cleave.js/
  99. */
  100. mask: {
  101. type: [Array, Boolean],
  102. required: false,
  103. default: false
  104. },
  105. /**
  106. * @see https://vuetifyjs.com/en/api/v-autocomplete/#props-variant
  107. */
  108. variant: {
  109. type: String as PropType<"filled" | "outlined" | "plain" | "underlined" | "solo" | "solo-inverted" | "solo-filled" | undefined>,
  110. required: false,
  111. default: 'filled'
  112. }
  113. })
  114. const { fieldViolations, updateViolationState } = useFieldViolation(props.field)
  115. const show: Ref<boolean> = ref(false)
  116. const emit = defineEmits(['update:model-value', 'change'])
  117. const onUpdate = (event: string) => {
  118. emit('update:model-value', event)
  119. }
  120. const onChange = (event: Event | undefined) => {
  121. updateViolationState(event)
  122. emit('change', event)
  123. }
  124. // const label = computed(() => {
  125. // if (props.label)
  126. // })
  127. </script>
  128. <style scoped lang="scss">
  129. input:read-only{
  130. color: rgb(var(--v-theme-neutral));
  131. }
  132. </style>