RelatoriosPage.vue 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. <template>
  2. <div>
  3. <DefaultHeaderPage>
  4. <template #after>
  5. <q-btn
  6. unelevated
  7. class="btn-export"
  8. :label="$t('relatorio.exportar_excel')"
  9. icon="mdi-download"
  10. padding="8px 16px"
  11. :loading="exportLoading"
  12. @click="onExport"
  13. />
  14. </template>
  15. </DefaultHeaderPage>
  16. <div class="relatorio-tabs-row q-mb-md">
  17. <div
  18. v-for="tab in tabs"
  19. :key="tab.name"
  20. class="relatorio-tab-card"
  21. :class="{ active: activeTab === tab.name }"
  22. @click="selectTab(tab.name)"
  23. >
  24. <q-icon :name="tab.icon" size="22px" color="violet-normal" />
  25. <div class="tab-body">
  26. <span class="tab-count">
  27. {{ counters[tab.counterKey] !== undefined ? counters[tab.counterKey] : '—' }}
  28. </span>
  29. <span class="tab-label">{{ tab.label }}</span>
  30. </div>
  31. </div>
  32. </div>
  33. <div v-if="activeTab === 'novos-associados'">
  34. <DefaultTableServerSide
  35. :key="tableKey"
  36. :columns="columnsNovosAssociados"
  37. :api-call="getNovoAssociadosPaginated"
  38. :add-item="false"
  39. :show-search-field="true"
  40. />
  41. </div>
  42. <div v-else-if="activeTab === 'contatos-associados'">
  43. <DefaultTableServerSide
  44. :key="tableKey"
  45. :columns="columnsContatos"
  46. :api-call="getContatosAssociadosPaginated"
  47. :add-item="false"
  48. :show-search-field="true"
  49. />
  50. </div>
  51. <div v-else-if="activeTab === 'exclusoes-mes'">
  52. <DefaultTableServerSide
  53. :key="tableKey"
  54. :columns="columnsExclusoes"
  55. :api-call="getExclusoesMesPaginated"
  56. :add-item="false"
  57. :show-search-field="true"
  58. />
  59. </div>
  60. </div>
  61. </template>
  62. <script setup>
  63. import { ref, onMounted } from "vue";
  64. import { useQuasar } from "quasar";
  65. import { useI18n } from "vue-i18n";
  66. import DefaultHeaderPage from "src/components/layout/DefaultHeaderPage.vue";
  67. import DefaultTableServerSide from "src/components/defaults/DefaultTableServerSide.vue";
  68. import {
  69. getRelatorioCounters,
  70. getNovoAssociadosPaginated,
  71. getContatosAssociadosPaginated,
  72. getExclusoesMesPaginated,
  73. exportRelatorio,
  74. } from "src/api/relatorio.js";
  75. const $q = useQuasar();
  76. const { t } = useI18n();
  77. const activeTab = ref("novos-associados");
  78. const tableKey = ref(0);
  79. const exportLoading = ref(false);
  80. const counters = ref({
  81. novos_associados: undefined,
  82. contatos: undefined,
  83. exclusoes_mes: undefined,
  84. });
  85. const tabs = [
  86. {
  87. name: "novos-associados",
  88. label: t("relatorio.novos_associados"),
  89. icon: "mdi-account-multiple-outline",
  90. counterKey: "novos_associados",
  91. },
  92. {
  93. name: "contatos-associados",
  94. label: t("relatorio.contatos"),
  95. icon: "mdi-phone-outline",
  96. counterKey: "contatos",
  97. },
  98. {
  99. name: "exclusoes-mes",
  100. label: t("relatorio.exclusoes_mes"),
  101. icon: "mdi-account-remove-outline",
  102. counterKey: "exclusoes_mes",
  103. },
  104. ];
  105. const columnsNovosAssociados = [
  106. {
  107. name: "name",
  108. label: t("common.terms.name"),
  109. field: "name",
  110. align: "left",
  111. sortable: true,
  112. },
  113. {
  114. name: "cpf",
  115. label: t("associado.cpf"),
  116. field: "cpf",
  117. align: "left",
  118. sortable: false,
  119. },
  120. {
  121. name: "created_at",
  122. label: t("relatorio.col.cadastro"),
  123. field: "created_at",
  124. align: "left",
  125. sortable: false,
  126. },
  127. ];
  128. const columnsContatos = [
  129. {
  130. name: "name",
  131. label: t("common.terms.name"),
  132. field: "name",
  133. align: "left",
  134. sortable: true,
  135. },
  136. {
  137. name: "phone",
  138. label: t("associado.phone"),
  139. field: "phone",
  140. align: "left",
  141. sortable: false,
  142. },
  143. {
  144. name: "email",
  145. label: t("associado.email"),
  146. field: "email",
  147. align: "left",
  148. sortable: false,
  149. },
  150. ];
  151. const columnsExclusoes = [
  152. {
  153. name: "name",
  154. label: t("common.terms.name"),
  155. field: "name",
  156. align: "left",
  157. sortable: true,
  158. },
  159. {
  160. name: "cpf",
  161. label: t("associado.cpf"),
  162. field: "cpf",
  163. align: "left",
  164. sortable: false,
  165. },
  166. {
  167. name: "excluded_at",
  168. label: t("relatorio.col.exclusao"),
  169. field: "excluded_at",
  170. align: "left",
  171. sortable: false,
  172. },
  173. ];
  174. const selectTab = (name) => {
  175. activeTab.value = name;
  176. tableKey.value++;
  177. };
  178. const onExport = async () => {
  179. exportLoading.value = true;
  180. try {
  181. await exportRelatorio(activeTab.value);
  182. } catch {
  183. $q.notify({ type: "negative", message: t("http.errors.failed") });
  184. } finally {
  185. exportLoading.value = false;
  186. }
  187. };
  188. onMounted(async () => {
  189. try {
  190. const data = await getRelatorioCounters();
  191. counters.value = data;
  192. } catch {
  193. // counters permanecem como undefined, exibindo '—'
  194. }
  195. });
  196. </script>
  197. <style lang="scss" scoped>
  198. @use "src/css/quasar.variables.scss" as vars;
  199. .relatorio-tabs-row {
  200. display: flex;
  201. gap: 12px;
  202. }
  203. .relatorio-tab-card {
  204. flex: 1;
  205. display: flex;
  206. align-items: center;
  207. gap: 12px;
  208. padding: 14px 20px;
  209. border-radius: 8px;
  210. border: 1.5px solid vars.$color-border;
  211. background: vars.$surface;
  212. cursor: pointer;
  213. transition: border-color 0.15s, background 0.15s;
  214. user-select: none;
  215. &:hover:not(.active) {
  216. border-color: vars.$violet-light-active;
  217. background: vars.$violet-light;
  218. }
  219. &.active {
  220. border-color: vars.$violet-normal;
  221. background: vars.$violet-light;
  222. }
  223. }
  224. .tab-body {
  225. display: flex;
  226. align-items: baseline;
  227. gap: 8px;
  228. }
  229. .tab-count {
  230. font-size: 1.375rem;
  231. font-weight: 700;
  232. color: vars.$violet-normal;
  233. line-height: 1;
  234. }
  235. .tab-label {
  236. font-size: 0.875rem;
  237. color: vars.$color-text-2;
  238. font-weight: 400;
  239. }
  240. .btn-export {
  241. background: linear-gradient(90deg, #4d1658 0%, #8b30a5 100%) !important;
  242. color: white !important;
  243. border-radius: 8px !important;
  244. }
  245. .btn-export :deep(.q-icon) {
  246. color: white !important;
  247. }
  248. </style>