AddEditContractDialog.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622
  1. <template>
  2. <q-dialog ref="dialogRef" @hide="onDialogHide">
  3. <q-card
  4. class="q-dialog-plugin overflow-hidden"
  5. style="width: 100%; max-width: 1350px"
  6. >
  7. <DefaultDialogHeader
  8. :title="props.contract ? 'Editar Contrato' : 'Novo Contrato'"
  9. @close="onDialogCancel"
  10. />
  11. <template v-if="props.contract">
  12. <q-tabs
  13. v-model="activeTab"
  14. dense
  15. align="left"
  16. class="q-px-md text-grey-7"
  17. active-color="primary"
  18. indicator-color="primary"
  19. >
  20. <q-tab name="dados" label="Dados do Contrato" />
  21. <q-tab name="midias" label="Mídias do Contrato" />
  22. </q-tabs>
  23. <q-separator />
  24. </template>
  25. <q-card-section class="q-pt-sm" style="height: 65vh; overflow-y: auto">
  26. <div v-show="activeTab === 'dados'">
  27. <div class="text-subtitle1 q-mb-md">Dados do Aluno</div>
  28. <div class="row q-col-gutter-sm">
  29. <div class="col-12">
  30. <DefaultInput
  31. :model-value="props.student.name"
  32. label="Aluno"
  33. disable
  34. />
  35. </div>
  36. <div class="col-6">
  37. <DefaultInput
  38. :model-value="props.student.document_number"
  39. label="CPF"
  40. disable
  41. />
  42. </div>
  43. <div class="col-6">
  44. <DefaultInput
  45. :model-value="formattedBirthDate"
  46. label="Data de Nascimento"
  47. disable
  48. />
  49. </div>
  50. </div>
  51. <div class="text-subtitle1 q-mt-lg q-mb-md">Dados do Contrato</div>
  52. <div class="row q-col-gutter-sm">
  53. <div class="col-4">
  54. <DefaultInput v-model="form.protocol" label="Protocolo" />
  55. </div>
  56. <div class="col-4">
  57. <DefaultInputDatePicker
  58. v-model="form.signature_date"
  59. label="Data Assinatura"
  60. />
  61. </div>
  62. <div class="col-4">
  63. <DefaultInputDatePicker
  64. v-model="form.end_date"
  65. label="Data Encerramento"
  66. />
  67. </div>
  68. <div class="col-5">
  69. <DefaultSelect
  70. v-model="form.package_id"
  71. label="Pacote de Aulas"
  72. :options="packages"
  73. option-value="id"
  74. option-label="name"
  75. emit-value
  76. map-options
  77. />
  78. </div>
  79. <div class="col-7">
  80. <DefaultInput
  81. v-model="form.class_quantity"
  82. label="Qtd. Aulas"
  83. type="number"
  84. disable
  85. />
  86. </div>
  87. <div class="col-4">
  88. <DefaultSelect
  89. v-model="form.weekday"
  90. label="Dia da Semana"
  91. :options="weekdays"
  92. option-value="value"
  93. option-label="label"
  94. emit-value
  95. map-options
  96. />
  97. </div>
  98. <div class="col-4">
  99. <DefaultInput
  100. v-model="form.start_time"
  101. label="Hora de Início"
  102. mask="##:##"
  103. >
  104. <template #append>
  105. <q-icon name="mdi-clock-outline" />
  106. </template>
  107. </DefaultInput>
  108. </div>
  109. <div class="col-4">
  110. <DefaultInput
  111. v-model="form.end_time"
  112. label="Hora de Término"
  113. mask="##:##"
  114. >
  115. <template #append>
  116. <q-icon name="mdi-clock-outline" />
  117. </template>
  118. </DefaultInput>
  119. </div>
  120. <div class="col-4">
  121. <DefaultSelect
  122. v-model="form.second_weekday"
  123. label="2° Dia da Semana"
  124. :options="weekdays"
  125. option-value="value"
  126. option-label="label"
  127. emit-value
  128. map-options
  129. />
  130. </div>
  131. <div class="col-4">
  132. <DefaultInput
  133. v-model="form.second_start_time"
  134. label="Hora de Início"
  135. mask="##:##"
  136. >
  137. <template #append>
  138. <q-icon name="mdi-clock-outline" />
  139. </template>
  140. </DefaultInput>
  141. </div>
  142. <div class="col-4">
  143. <DefaultInput
  144. v-model="form.second_end_time"
  145. label="Hora de Término"
  146. mask="##:##"
  147. >
  148. <template #append>
  149. <q-icon name="mdi-clock-outline" />
  150. </template>
  151. </DefaultInput>
  152. </div>
  153. </div>
  154. <div class="text-subtitle1 q-mt-lg q-mb-md">Dados Financeiros</div>
  155. <div class="row q-col-gutter-sm">
  156. <div class="col-4">
  157. <DefaultInput
  158. v-model="form.due_day"
  159. label="Dia de Vencimento"
  160. type="number"
  161. />
  162. </div>
  163. </div>
  164. <div class="text-subtitle2 q-mt-md q-mb-sm text-grey-8">Matrícula</div>
  165. <div class="row q-col-gutter-sm">
  166. <div class="col-3">
  167. <DefaultCurrencyInput
  168. v-model="form.enrollment_fee"
  169. label="Valor da Matrícula"
  170. disable
  171. />
  172. </div>
  173. <div class="col-3">
  174. <DefaultSelect
  175. v-model="form.installments"
  176. label="Qtde Parcelas"
  177. :options="enrollmentInstallmentOptions"
  178. option-value="value"
  179. option-label="label"
  180. emit-value
  181. map-options
  182. />
  183. </div>
  184. <div class="col-3">
  185. <DefaultCurrencyInput
  186. :model-value="enrollmentInstallmentValue"
  187. label="Valor da Parcela"
  188. disable
  189. />
  190. </div>
  191. <div class="col-3">
  192. <DefaultInputDatePicker
  193. v-model="form.enrollment_due_date"
  194. label="Data Vencimento"
  195. />
  196. </div>
  197. </div>
  198. <div class="text-subtitle2 q-mt-md q-mb-sm text-grey-8">Pacote</div>
  199. <div class="row q-col-gutter-sm">
  200. <div class="col-3">
  201. <DefaultCurrencyInput
  202. v-model="form.package_value"
  203. label="Valor do Pacote"
  204. disable
  205. />
  206. </div>
  207. <div class="col-3">
  208. <DefaultSelect
  209. v-model="form.package_installments"
  210. label="Qtde Parcelas"
  211. :options="packageInstallmentOptions"
  212. option-value="value"
  213. option-label="label"
  214. emit-value
  215. map-options
  216. />
  217. </div>
  218. <div class="col-3">
  219. <DefaultCurrencyInput
  220. :model-value="packageInstallmentValue"
  221. label="Valor da Parcela"
  222. disable
  223. />
  224. </div>
  225. <div class="col-3">
  226. <DefaultInputDatePicker
  227. v-model="form.package_due_date"
  228. label="Data 1ª Parcela"
  229. />
  230. </div>
  231. </div>
  232. <div class="row q-col-gutter-sm q-mt-xs">
  233. <div class="col-6">
  234. <DefaultInput
  235. v-model="form.early_payment_discount"
  236. label="Desconto até o vencimento (%)"
  237. type="number"
  238. />
  239. </div>
  240. <div class="col-6">
  241. <DefaultSelect
  242. v-model="form.payment_method"
  243. label="Forma de Pagamento"
  244. :options="paymentMethods"
  245. option-value="value"
  246. option-label="label"
  247. emit-value
  248. map-options
  249. />
  250. </div>
  251. <div class="col-6">
  252. <DefaultInput
  253. v-model="form.interest_rate"
  254. label="Juros (%) a.m"
  255. type="number"
  256. />
  257. </div>
  258. <div class="col-6">
  259. <DefaultInput
  260. v-model="form.late_fee"
  261. label="Multa (%)"
  262. type="number"
  263. />
  264. </div>
  265. </div>
  266. </div>
  267. <div v-if="props.contract" v-show="activeTab === 'midias'">
  268. <DefaultTable
  269. v-model:rows="medias"
  270. title="Mídias"
  271. :columns="mediaColumns"
  272. descricao="mídias"
  273. :feminino="true"
  274. no-api-call
  275. :show-search-field="false"
  276. :loading="loadingMedias"
  277. >
  278. <template #body-cell-actions="{ row }">
  279. <q-td align="center">
  280. <q-item-section class="no-wrap" style="flex-direction: row; gap: 4px">
  281. <q-btn
  282. outline
  283. icon="mdi-eye-outline"
  284. style="width: 36px"
  285. @click.prevent.stop="openFile(row.file_url)"
  286. />
  287. <q-btn
  288. outline
  289. icon="mdi-trash-can-outline"
  290. style="width: 36px"
  291. color="negative"
  292. @click.prevent.stop="handleDeleteMedia(row)"
  293. />
  294. </q-item-section>
  295. </q-td>
  296. </template>
  297. </DefaultTable>
  298. </div>
  299. </q-card-section>
  300. <q-separator />
  301. <q-card-actions align="right">
  302. <q-btn
  303. outline
  304. color="primary"
  305. label="CANCELAR"
  306. @click="onDialogCancel"
  307. />
  308. <q-btn
  309. v-if="activeTab === 'dados'"
  310. color="primary"
  311. label="SALVAR"
  312. :loading="saving"
  313. @click="handleSave"
  314. />
  315. </q-card-actions>
  316. </q-card>
  317. </q-dialog>
  318. </template>
  319. <script setup>
  320. import { computed, ref, watch, onMounted, nextTick } from "vue";
  321. import { useDialogPluginComponent, useQuasar } from "quasar";
  322. import DefaultDialogHeader from "src/components/defaults/DefaultDialogHeader.vue";
  323. import DefaultInput from "src/components/defaults/DefaultInput.vue";
  324. import DefaultSelect from "src/components/defaults/DefaultSelect.vue";
  325. import DefaultTable from "src/components/defaults/DefaultTable.vue";
  326. import DefaultCurrencyInput from "src/components/defaults/DefaultCurrencyInput.vue";
  327. import DefaultInputDatePicker from "src/components/defaults/DefaultInputDatePicker.vue";
  328. import { useSubmitHandler } from "src/composables/useSubmitHandler";
  329. import { useFormUpdateTracker } from "src/composables/useFormUpdateTracker";
  330. import { formatDateYMDtoDMY, formatDateDMYtoYMD } from "src/helpers/utils";
  331. import { getUnitPackages } from "src/api/package";
  332. import { getFinancialMe } from "src/api/unit_financial";
  333. import {
  334. createStudentContract,
  335. updateStudentContract,
  336. } from "src/api/studentContract";
  337. import { getContractMedias, deleteStudentMedia } from "src/api/student_media";
  338. const props = defineProps({
  339. student: {
  340. type: Object,
  341. required: true,
  342. },
  343. contract: {
  344. type: Object,
  345. default: null,
  346. },
  347. });
  348. defineEmits([...useDialogPluginComponent.emits]);
  349. const { dialogRef, onDialogHide, onDialogOK, onDialogCancel } =
  350. useDialogPluginComponent();
  351. const $q = useQuasar();
  352. const activeTab = ref("dados");
  353. const medias = ref([]);
  354. const loadingMedias = ref(false);
  355. const trimTime = (t) => (t ? t.slice(0, 5) : null);
  356. const { form } = useFormUpdateTracker({
  357. protocol: props.contract?.protocol ?? null,
  358. signature_date: props.contract?.signature_date ?? null,
  359. end_date: props.contract?.end_date ?? null,
  360. package_id: props.contract?.class_package_unit_id ?? null,
  361. class_quantity: props.contract?.class_quantity ?? null,
  362. weekday: props.contract?.weekday ?? null,
  363. start_time: trimTime(props.contract?.start_time),
  364. end_time: trimTime(props.contract?.end_time),
  365. second_weekday: props.contract?.second_weekday ?? null,
  366. second_start_time: trimTime(props.contract?.second_start_time),
  367. second_end_time: trimTime(props.contract?.second_end_time),
  368. due_day: props.contract?.recurring_day ?? null,
  369. enrollment_fee: props.contract?.tax_register ?? null,
  370. total_classes: props.contract?.class_quantity ?? null,
  371. installments: props.contract?.installments ?? null,
  372. enrollment_due_date: props.contract?.enrollment_due_date ?? null,
  373. package_value: props.contract?.package_value ?? null,
  374. package_installments: props.contract?.package_installments ?? null,
  375. package_due_date: props.contract?.package_due_date ?? null,
  376. early_payment_discount: props.contract?.early_payment_discount ?? null,
  377. interest_rate: props.contract?.interest_rate ?? null,
  378. payment_method: props.contract?.payment_method ?? null,
  379. late_fee: props.contract?.fine_cancelled ?? null,
  380. });
  381. const enrollmentInstallmentOptions = Array.from({ length: 12 }, (_, i) => ({
  382. value: i + 1,
  383. label: `${i + 1}x`,
  384. }));
  385. const packageInstallmentOptions = Array.from({ length: 13 }, (_, i) => ({
  386. value: i + 1,
  387. label: `${i + 1}x`,
  388. }));
  389. const enrollmentInstallmentValue = computed(() => {
  390. if (!form.enrollment_fee || !form.installments) return null;
  391. return parseFloat((form.enrollment_fee / form.installments).toFixed(2));
  392. });
  393. const packageInstallmentValue = computed(() => {
  394. if (!form.package_value || !form.package_installments) return null;
  395. return parseFloat((form.package_value / form.package_installments).toFixed(2));
  396. });
  397. // Two-way binding: due_day <-> package_due_date
  398. const _syncingFromDay = ref(false);
  399. const _syncingFromDate = ref(false);
  400. function buildDateFromDay(day) {
  401. const d = parseInt(day);
  402. if (!d || d < 1 || d > 31) return null;
  403. const now = new Date();
  404. const nextMonthIndex = (now.getMonth() + 1) % 12;
  405. const nextYear = now.getMonth() === 11 ? now.getFullYear() + 1 : now.getFullYear();
  406. const lastDayOfNextMonth = new Date(nextYear, nextMonthIndex + 1, 0).getDate();
  407. const safeDay = Math.min(d, lastDayOfNextMonth);
  408. return `${String(safeDay).padStart(2, '0')}/${String(nextMonthIndex + 1).padStart(2, '0')}/${nextYear}`;
  409. }
  410. watch(
  411. () => form.due_day,
  412. (day) => {
  413. if (_syncingFromDate.value) return;
  414. const dateStr = buildDateFromDay(day);
  415. if (!dateStr) return;
  416. _syncingFromDay.value = true;
  417. form.package_due_date = dateStr;
  418. nextTick(() => { _syncingFromDay.value = false; });
  419. },
  420. );
  421. watch(
  422. () => form.package_due_date,
  423. (dateStr) => {
  424. if (_syncingFromDay.value) return;
  425. if (!dateStr || dateStr.length < 8) return;
  426. const day = parseInt(dateStr.split('/')[0]);
  427. if (!isNaN(day) && day !== parseInt(form.due_day)) {
  428. _syncingFromDate.value = true;
  429. form.due_day = day;
  430. nextTick(() => { _syncingFromDate.value = false; });
  431. }
  432. },
  433. );
  434. const paymentMethods = [
  435. { value: "pix", label: "Pix" },
  436. { value: "credit_card", label: "Cartão de Crédito" },
  437. { value: "debit_card", label: "Cartão de Débito" },
  438. ];
  439. const weekdays = [
  440. { value: 1, label: "Segunda" },
  441. { value: 2, label: "Terça" },
  442. { value: 3, label: "Quarta" },
  443. { value: 4, label: "Quinta" },
  444. { value: 5, label: "Sexta" },
  445. { value: 6, label: "Sábado" },
  446. { value: 0, label: "Domingo" },
  447. ];
  448. const packages = ref([]);
  449. const mediaColumns = [
  450. { name: "created_at", label: "Data de Anexo", field: "created_at", align: "left" },
  451. { name: "actions", label: "Ações", field: null, align: "center" },
  452. ];
  453. async function fetchMedias() {
  454. if (!props.contract) return;
  455. loadingMedias.value = true;
  456. try {
  457. medias.value = await getContractMedias(props.contract.id);
  458. } finally {
  459. loadingMedias.value = false;
  460. }
  461. }
  462. watch(activeTab, (tab) => {
  463. if (tab === "midias") fetchMedias();
  464. });
  465. onMounted(async () => {
  466. packages.value = await getUnitPackages();
  467. // Para novo contrato, pré-preenche desconto/juros/multa com os defaults da unidade
  468. if (!props.contract) {
  469. try {
  470. const financial = await getFinancialMe();
  471. if (financial) {
  472. if (financial.default_discount != null) form.early_payment_discount = financial.default_discount;
  473. if (financial.default_interest != null) form.interest_rate = financial.default_interest;
  474. if (financial.default_fine != null) form.late_fee = financial.default_fine;
  475. }
  476. } catch (e) {
  477. console.error(e);
  478. }
  479. }
  480. });
  481. function openFile(url) {
  482. window.open(url, "_blank");
  483. }
  484. function handleDeleteMedia(media) {
  485. $q.dialog({
  486. title: "Excluir mídia",
  487. message: "Deseja excluir esta mídia permanentemente?",
  488. ok: { color: "negative", label: "Excluir" },
  489. cancel: { color: "primary", outline: true, label: "Cancelar" },
  490. }).onOk(async () => {
  491. try {
  492. await deleteStudentMedia(media.id);
  493. medias.value = medias.value.filter((m) => m.id !== media.id);
  494. } catch (e) {
  495. console.error(e);
  496. $q.notify({ type: "negative", message: "Erro ao excluir mídia." });
  497. }
  498. });
  499. }
  500. watch(
  501. () => form.package_id,
  502. (id) => {
  503. const pkg = packages.value.find((p) => p.id === id);
  504. if (!pkg) return;
  505. form.class_quantity = pkg.quantity_classes;
  506. form.total_classes = pkg.quantity_classes;
  507. form.enrollment_fee = pkg.contract_register_value;
  508. form.package_value = pkg.contract_value;
  509. },
  510. );
  511. const formattedBirthDate = computed(() =>
  512. props.student.birth_date ? formatDateYMDtoDMY(props.student.birth_date) : "",
  513. );
  514. const { loading: saving, execute } = useSubmitHandler({
  515. onSuccess: () => onDialogOK(true),
  516. });
  517. function buildPayload() {
  518. return {
  519. student_id: props.student.id,
  520. protocol: form.protocol,
  521. signature_date: form.signature_date
  522. ? formatDateDMYtoYMD(form.signature_date)
  523. : null,
  524. end_date: form.end_date ? formatDateDMYtoYMD(form.end_date) : null,
  525. class_package_unit_id: form.package_id,
  526. class_quantity: form.class_quantity,
  527. weekday: form.weekday,
  528. start_time: form.start_time,
  529. end_time: form.end_time,
  530. second_weekday: form.second_weekday,
  531. second_start_time: form.second_start_time,
  532. second_end_time: form.second_end_time,
  533. due_day: form.due_day ? parseInt(form.due_day) : null,
  534. tax_register: form.enrollment_fee,
  535. installments: form.installments,
  536. enrollment_due_date: form.enrollment_due_date
  537. ? formatDateDMYtoYMD(form.enrollment_due_date)
  538. : null,
  539. package_value: form.package_value,
  540. package_installments: form.package_installments,
  541. package_due_date: form.package_due_date
  542. ? formatDateDMYtoYMD(form.package_due_date)
  543. : null,
  544. early_payment_discount: form.early_payment_discount,
  545. interest_rate: form.interest_rate,
  546. payment_method: form.payment_method,
  547. fine_cancelled: form.late_fee,
  548. };
  549. }
  550. async function handleSave() {
  551. const payload = buildPayload();
  552. await execute(() =>
  553. props.contract
  554. ? updateStudentContract(props.contract.id, payload)
  555. : createStudentContract(payload),
  556. );
  557. }
  558. </script>