SearchPage.vue 10.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291
  1. <!-- eslint-disable @intlify/vue-i18n/no-raw-text -->
  2. <template>
  3. <q-page class="bg-page">
  4. <div class="row items-center q-px-md q-pt-md q-pb-sm bg-white shadow-search">
  5. <q-btn flat round dense icon="mdi-chevron-left" color="primary" @click="router.back()" />
  6. <div class="col text-center text-subtitle1 text-weight-bold text-primary">
  7. {{ $t('search_page.title') }}
  8. </div>
  9. <div style="width: 36px" />
  10. </div>
  11. <div class="q-px-md q-pt-md q-pb-sm">
  12. <q-card class="custom-schedule-card bg-surface shadow-card q-pa-sm" :flat="false">
  13. <q-card-section class="row items-center no-wrap q-pa-sm q-gutter-x-sm">
  14. <span class="col text-text fonte-hint">
  15. {{ $t('search_page.custom_schedule_description') }}
  16. </span>
  17. <q-btn
  18. color="secondary"
  19. no-caps
  20. unelevated
  21. padding="8px 16px"
  22. class="text-weight-bold custom-schedule-btn card-border"
  23. @click="router.push({ name: 'SobMedidaPage' })"
  24. >
  25. <template #default>
  26. <div class="column items-center q-gutter-y-xs">
  27. <q-icon name="mdi-scissors-cutting" size="16px" />
  28. <span>{{ $t('search_page.custom_schedule_btn') }}</span>
  29. </div>
  30. </template>
  31. </q-btn>
  32. </q-card-section>
  33. </q-card>
  34. </div>
  35. <div class="row items-center q-px-md q-py-md q-gutter-x-sm">
  36. <q-input
  37. v-model="searchName"
  38. :placeholder="$t('search_page.search_placeholder')"
  39. outlined
  40. rounded
  41. dense
  42. clearable
  43. debounce="400"
  44. class="col bg-white search-input"
  45. input-class="text-text"
  46. @update:model-value="onNameChange"
  47. >
  48. <template #append>
  49. <q-icon name="mdi-magnify" color="grey-5" />
  50. </template>
  51. </q-input>
  52. <q-btn
  53. flat round dense
  54. icon="mdi-tune-variant"
  55. color="grey-6"
  56. size="md"
  57. :class="{ 'filter-active': activeSort || activeDate }"
  58. @click="openFilterDialog"
  59. />
  60. </div>
  61. <div class="row items-center justify-between no-wrap q-px-md q-pb-sm">
  62. <div class="dashboard-section-title gradient-diarista">{{ $t('search_page.choose_provider') }}</div>
  63. <div class="row items-center no-wrap text-text">
  64. <q-btn flat dense round icon="mdi-chevron-left" color="text" size="sm" @click="setPeriodTypePrevious" />
  65. <span class="text-caption text-weight-medium">{{ periodLabel }}</span>
  66. <q-btn flat dense round icon="mdi-chevron-right" color="text" size="sm" @click="setPeriodTypeNext" />
  67. </div>
  68. </div>
  69. <div v-if="loading" class="row items-center justify-center q-py-xl">
  70. <q-spinner-dots color="primary" size="40px" />
  71. </div>
  72. <template v-else>
  73. <div v-if="sortedProviders.length === 0" class="text-center text-grey-6 q-px-md q-py-lg text-body2">
  74. {{ $t('search_page.no_results') }}
  75. </div>
  76. <div v-else class="column q-px-md q-pb-xl">
  77. <q-card
  78. v-for="p in sortedProviders"
  79. :key="p.provider_id"
  80. class="card-border bg-page text-text q-mb-sm"
  81. :flat="false"
  82. >
  83. <q-card-section class="row no-wrap q-pa-sm">
  84. <div class="row no-wrap full-width">
  85. <div class="col-2">
  86. <q-avatar :style="avatarColors[p.provider_id % avatarColors.length]" class="text-weight-bold">
  87. {{ p.provider_name?.slice(0,1).toUpperCase() ?? '—' }}
  88. </q-avatar>
  89. </div>
  90. <div class="col-10 row">
  91. <div class="column col-9 justify-between">
  92. <span class="text-provider-close-name">{{ p.provider_name ?? 'Prestador' }}</span>
  93. <span class="text-provider-close-region">{{ p.district }}</span>
  94. <div class="row items-center justify-between q-pr-lg">
  95. <div class="row items-center">
  96. <q-icon name="mdi-star" color="warning" size="16px" />
  97. <span class="text-provider-close-rating">
  98. {{ p.average_rating != null ? (Number(p.average_rating).toFixed(1) + ' (' + (p.total_reviews ?? 0) + ')') : ('(' + (p.total_reviews ?? 0) + ')') }}
  99. </span>
  100. </div>
  101. <div class="row items-center">
  102. <q-icon name="mdi-broom" color="secondary" size="16px" />
  103. <span class="text-provider-close-jobs">{{ p.total_services ?? 0 }}</span>
  104. </div>
  105. <div class="row items-center">
  106. <q-icon name="mdi-map-marker-outline" color="text" size="16px" />
  107. <span class="text-provider-close-jobs">{{ 0 + ' km' }}</span>
  108. </div>
  109. </div>
  110. </div>
  111. <div class="column col-3 justify-between text-center items-center">
  112. <span class="text-provider-close-price">{{ priceByPeriod(p) }}</span>
  113. <div class="full-width">
  114. <q-btn
  115. unelevated rounded no-caps
  116. color="primary"
  117. size="sm"
  118. padding="3px 12px"
  119. :label="$t('search_page.schedule_btn')"
  120. @click="goToScheduling(p)"
  121. />
  122. </div>
  123. </div>
  124. </div>
  125. </div>
  126. </q-card-section>
  127. </q-card>
  128. </div>
  129. </template>
  130. </q-page>
  131. </template>
  132. <script setup>
  133. import { ref, computed, onMounted } from 'vue';
  134. import { useRouter } from 'vue-router';
  135. import { useI18n } from 'vue-i18n';
  136. import { useQuasar } from 'quasar';
  137. import { buscaPrestadores } from 'src/api/dashboard';
  138. import { formatCurrency } from 'src/helpers/utils';
  139. import SearchFilterDialog from 'src/pages/search/components/SearchFilterDialog.vue';
  140. import SchedulingDialog from 'src/pages/search/components/SchedulingDialog.vue';
  141. const { t } = useI18n();
  142. const router = useRouter();
  143. const $q = useQuasar();
  144. const allProviders = ref([]);
  145. const loading = ref(true);
  146. const searchName = ref('');
  147. const activeDate = ref(null);
  148. const activeSort = ref(null);
  149. const currentPeriodType = ref(8);
  150. const periodTypeMap = { 2: 'daily_price_2h', 4: 'daily_price_4h', 6: 'daily_price_6h', 8: 'daily_price_8h' };
  151. const periodLabel = computed(() => {
  152. const labels = { 8: t('search_page.until_8h'), 6: t('search_page.until_6h'), 4: t('search_page.until_4h'), 2: t('search_page.until_2h') };
  153. return labels[currentPeriodType.value] ?? '';
  154. });
  155. const priceByPeriod = (p) => {
  156. const key = periodTypeMap[currentPeriodType.value];
  157. return p[key] ? formatCurrency(p[key]) : t('search_page.no_price');
  158. };
  159. const setPeriodTypePrevious = () => {
  160. const prev = currentPeriodType.value - 2;
  161. if (periodTypeMap[prev]) currentPeriodType.value = prev;
  162. };
  163. const setPeriodTypeNext = () => {
  164. const next = currentPeriodType.value + 2;
  165. if (periodTypeMap[next]) currentPeriodType.value = next;
  166. };
  167. const sortedProviders = computed(() => {
  168. const list = [...allProviders.value];
  169. const priceKey = periodTypeMap[currentPeriodType.value];
  170. switch (activeSort.value) {
  171. case 'price_asc':
  172. return list.sort((a, b) => Number(a[priceKey] ?? 0) - Number(b[priceKey] ?? 0));
  173. case 'price_desc':
  174. return list.sort((a, b) => Number(b[priceKey] ?? 0) - Number(a[priceKey] ?? 0));
  175. case 'rating_desc':
  176. return list.sort((a, b) => Number(b.average_rating ?? 0) - Number(a.average_rating ?? 0));
  177. case 'rating_asc':
  178. return list.sort((a, b) => Number(a.average_rating ?? 0) - Number(b.average_rating ?? 0));
  179. case 'reviews_desc':
  180. return list.sort((a, b) => Number(b.total_reviews ?? 0) - Number(a.total_reviews ?? 0));
  181. case 'reviews_asc':
  182. return list.sort((a, b) => Number(a.total_reviews ?? 0) - Number(b.total_reviews ?? 0));
  183. case 'services_desc':
  184. return list.sort((a, b) => Number(b.total_services ?? 0) - Number(a.total_services ?? 0));
  185. case 'services_asc':
  186. return list.sort((a, b) => Number(a.total_services ?? 0) - Number(b.total_services ?? 0));
  187. case 'oldest':
  188. return list.sort((a, b) => new Date(a.created_at) - new Date(b.created_at));
  189. case 'newest':
  190. return list.sort((a, b) => new Date(b.created_at) - new Date(a.created_at));
  191. default:
  192. return list.sort((a, b) => Number(b.average_rating ?? 0) - Number(a.average_rating ?? 0));
  193. }
  194. });
  195. const loadProviders = async () => {
  196. loading.value = true;
  197. try {
  198. allProviders.value = await buscaPrestadores({
  199. name: searchName.value,
  200. date: activeDate.value ?? '',
  201. }) ?? [];
  202. } catch {
  203. allProviders.value = [];
  204. } finally {
  205. loading.value = false;
  206. }
  207. };
  208. const onNameChange = () => loadProviders();
  209. const openFilterDialog = () => {
  210. $q.dialog({
  211. component: SearchFilterDialog,
  212. componentProps: {
  213. initialSort: activeSort.value,
  214. initialDate: activeDate.value,
  215. },
  216. }).onOk(({ sort, date }) => {
  217. const dateChanged = date !== activeDate.value;
  218. activeSort.value = sort;
  219. activeDate.value = date;
  220. if (dateChanged) loadProviders();
  221. });
  222. };
  223. const goToScheduling = (provider) => {
  224. $q.dialog({
  225. component: SchedulingDialog,
  226. componentProps: { provider },
  227. });
  228. };
  229. const avatarColors = [
  230. { background: '#ffd5df', color: '#932e57' },
  231. { background: '#d7e8ff', color: '#2158a8' },
  232. { background: '#dfd', color: '#2a7a3b' },
  233. { background: '#ffe5cc', color: '#8a4500' },
  234. ];
  235. onMounted(() => loadProviders());
  236. </script>
  237. <style scoped lang="scss">
  238. .shadow-search {
  239. box-shadow: 0 1px 4px rgba(0, 0, 0, 0.08);
  240. }
  241. .search-input {
  242. :deep(.q-field__control) {
  243. border-radius: 28px;
  244. }
  245. }
  246. .custom-schedule-card {
  247. border-radius: 12px;
  248. }
  249. .custom-schedule-btn {
  250. flex-shrink: 0;
  251. min-width: 72px;
  252. }
  253. .filter-active {
  254. color: var(--q-primary) !important;
  255. }
  256. .fonte-hint {
  257. font-family: Inter;
  258. font-weight: 500;
  259. font-size: 14px;
  260. line-height: 100%;
  261. letter-spacing: -0.04em;
  262. vertical-align: middle;
  263. }
  264. </style>