Text.vue 3.3 KB

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