DefaultInputDatePicker.vue 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. <template>
  2. <DefaultInput
  3. v-model="treatedDate"
  4. v-model:error="error"
  5. v-bind="$attrs"
  6. :input-ref="inputRef"
  7. :label="label"
  8. :mask="inputMask"
  9. clearable
  10. >
  11. <template #append>
  12. <q-icon
  13. :name="time ? 'mdi-calendar-clock' : 'mdi-calendar'"
  14. class="cursor-pointer"
  15. >
  16. <q-popup-proxy cover transition-show="scale" transition-hide="scale">
  17. <template v-if="!time">
  18. <q-date v-model="date" mask="YYYY-MM-DD">
  19. <div class="row items-center justify-end">
  20. <q-btn v-close-popup label="OK" color="primary" flat />
  21. </div>
  22. </q-date>
  23. </template>
  24. <template v-else>
  25. <q-tab-panels
  26. v-model="activePanel"
  27. animated
  28. transition-prev="slide-right"
  29. transition-next="slide-left"
  30. class="bg-white"
  31. >
  32. <q-tab-panel name="date" class="q-pa-none">
  33. <q-date
  34. v-model="date"
  35. mask="YYYY-MM-DD HH:mm"
  36. @update:model-value="handleDateSelection"
  37. />
  38. </q-tab-panel>
  39. <q-tab-panel name="time" class="q-pa-none">
  40. <q-time v-model="date" mask="YYYY-MM-DD HH:mm" format24h />
  41. </q-tab-panel>
  42. </q-tab-panels>
  43. </template>
  44. </q-popup-proxy>
  45. </q-icon>
  46. </template>
  47. </DefaultInput>
  48. </template>
  49. <script setup>
  50. import { watch, ref, computed } from "vue";
  51. import { useI18n } from "vue-i18n";
  52. import masks from "src/helpers/masks";
  53. import DefaultInput from "./DefaultInput.vue";
  54. const { label, time } = defineProps({
  55. label: {
  56. type: String,
  57. default: () => useI18n().t("common.terms.date"),
  58. },
  59. time: {
  60. type: Boolean,
  61. default: false,
  62. },
  63. });
  64. const treatedDate = defineModel({ type: [String, null] });
  65. const untreatedDate = defineModel("untreatedDate", { type: [String, null] });
  66. const error = defineModel("error", {
  67. type: [String, Object, Array, Boolean, null],
  68. });
  69. const inputRef = ref(null);
  70. const activePanel = ref("date");
  71. const date = ref();
  72. const handleDateSelection = () => {
  73. if (time) {
  74. activePanel.value = "time";
  75. }
  76. };
  77. const formatDate = (value) => {
  78. if (!value) return null;
  79. const [datePart, timePart] = value.split(" ");
  80. const formattedDate = datePart.split("-").reverse().join("/");
  81. return time && timePart ? `${formattedDate} ${timePart}` : formattedDate;
  82. };
  83. const unformatDate = (value) => {
  84. if (!value) return null;
  85. const [datePart, timePart] = value.split(" ");
  86. const formattedDate = datePart.split("/").reverse().join("-");
  87. return time && timePart ? `${formattedDate} ${timePart}` : formattedDate;
  88. };
  89. const inputMask = computed(() => {
  90. if (!inputRef.value) return "";
  91. if (time) {
  92. return masks.Brasil.datetime;
  93. }
  94. return masks.Brasil.date;
  95. });
  96. watch(date, (value) => {
  97. if (!value) return;
  98. untreatedDate.value = value;
  99. treatedDate.value = formatDate(value);
  100. });
  101. watch(treatedDate, (value) => {
  102. if (!value) {
  103. date.value = null;
  104. untreatedDate.value = null;
  105. treatedDate.value = null;
  106. activePanel.value = "date";
  107. return;
  108. }
  109. date.value = unformatDate(value);
  110. });
  111. watch(
  112. untreatedDate,
  113. (value) => {
  114. if (!value) {
  115. date.value = null;
  116. untreatedDate.value = null;
  117. treatedDate.value = null;
  118. activePanel.value = "date";
  119. return;
  120. }
  121. date.value = value;
  122. treatedDate.value = formatDate(value);
  123. },
  124. { immediate: true }
  125. );
  126. </script>