agenda: Fase 5 (status change/edit cobrada) + indicadores visuais + UX convenio

DB
- drop agenda_excecoes (substituida por financial_exceptions + lock-edit
  baseado em financial_records)
- financial_records.payment_link (Asaas + link compartilhavel)
- financial_exceptions.consume_on_miss (rotular nao-show consome ou nao)
- billing_contracts.charging_style (upfront/saldo/per_session)

Payment refactor
- paymentSettlement -> paymentMethod (string) + markPaidNow (bool).
  Handler aplica payment_method sempre; status='paid'+paid_at apenas
  quando markPaidNow=true && method != 'link'. Asaas (link) sempre
  liquida via webhook, nunca nasce paid.
- create_financial_record_for_session com pos-RPC patch pra payment_method
  e (opcional) status='paid' quando user marca "ja recebi".

Indicadores visuais (3 canais distintos por estado)
- Paid: barra esquerda emerald-500 4px na agenda (MelissaAgenda),
  pi-check-circle no popover/Resumo.
- Pending: badge \$ amber canto direito (mantido); linha amber no popover/
  Resumo "A receber R\$ X (cobranca pendente)".
- Neutro: sem badge nem barra (compromisso pessoal, bloqueio, ou
  ocorrencia virtual de pacote upfront/saldo).
- Bulk-load de paymentState em _reloadRange etapa 4 (1 query unica em
  financial_records mapeada por agenda_evento_id).
- AgendaEventDialog Resumo lateral ganha linha entre pi-clock e
  pi-map-marker via novo sessionPaymentRecord (sem guard de
  occurrenceMode, contrario ao occFinancialRecord que continua so pra
  Rail/Clinica). 5 estados: paid+paid_at, overdue+venceu, pending+vence,
  sem cobranca c/ valor, sem cobranca s/ valor.

UX de convenio
- InsurancePlanServiceQuickCreateDialog novo: cadastra procedimento
  POR CIMA do AgendaEventDialog sem sair da agenda. Auto-seleciona
  novo procedimento so quando nada estava selecionado antes.
- Caixa cinza "Cadastrar procedimento" sempre visivel quando convenio
  selecionado, com copy variavel (0 procedimentos: chamada urgente;
  1+: "se quiser adicionar mais").
- "+ Novo convenio" toolbar em ConfiguracoesConveniosPage (botao
  estava faltando, empty state mandava clicar em botao inexistente).
- Hint contextual abaixo do card Sessao/Honorarios: convenio = "N da
  guia eh opcional", gratuito = "sem cobranca", particular = sem hint.
  Label "N da Guia" tambem ganhou "(opcional)" no service-picker dialog.

Bug fixes
- pickDbFields whitelist faltando 'modalidade' (useMelissaAgenda.js:74)
  — sessoes avulsas eram salvas como presencial independente da
  escolha visual. Adicionado.
- goToConveniosConfig removida — fazia router.push("/therapist/
  configuracoes/convenios") mas /configuracoes/* eh rota raiz, nao
  filha. Substituida pelo quick-create inline (#1).
- bloqueioCobrindo + dialogBlockOverlap passados via deps em
  _buildHandlers (refs do useMelissaAgenda nao sao acessiveis no
  escopo de _buildHandlers).

Fase 5 (status change + edit cobrada)
- AgendaStatusChangeConfirmDialog: confirm dialog quando user muda
  status pra realizada/faltou/cancelado, com opcoes de markPaid ou
  gerar cobranca conforme o caso.
- useAgendaBloqueios novo composable: extrai logica de bloqueios
  cinza (background events) do MelissaAgenda.

Doc viva
- src/docs/agenda-compromisso-financeiro-cenarios.html: 13 cenarios
  de teste manual. C1-C4 ja validados. Cada teste validado vira parte
  da doc final pra area de ajuda (pos-Fase 9).

Wiki/handoff
- agenda-compromisso-fluxo e agenda-billing-pesquisa-mercado (decisoes
  arquiteturais sobre billing).
- HANDOFF.md atualizado.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
Leonardo
2026-05-19 08:31:18 -03:00
parent 41c44272a3
commit e95ed9b585
41 changed files with 8715 additions and 852 deletions
@@ -159,9 +159,10 @@ describe('isFirstOccurrence', () => {
});
describe('editScopeOptions', () => {
it('retorna 4 opções', () => {
it('retorna 3 opções (todos_sem_excecao removido da UI em 2026-05-12)', () => {
const { composer } = setup();
expect(composer.editScopeOptions.value).toHaveLength(4);
expect(composer.editScopeOptions.value).toHaveLength(3);
expect(composer.editScopeOptions.value.map((o) => o.value)).toEqual(['somente_este', 'este_e_seguintes', 'todos']);
});
it('"este_e_seguintes" disabled quando isFirstOccurrence', () => {
const serieEvents = ref([{ recurrence_date: '2026-05-15' }, { recurrence_date: '2026-05-22' }]);
@@ -0,0 +1,98 @@
/*
|--------------------------------------------------------------------------
| Agência PSI
|--------------------------------------------------------------------------
| Criado e desenvolvido por Leonardo Nohama
|
| Tecnologia aplicada à escuta.
| Estrutura para o cuidado.
|
| Arquivo: src/features/agenda/composables/useAgendaBloqueios.js
| Data: 2026
| Local: São Carlos/SP — Brasil
|--------------------------------------------------------------------------
| © 2026 — Todos os direitos reservados
|--------------------------------------------------------------------------
*/
// useAgendaBloqueios
// Carrega e expõe rows de public.agenda_bloqueios + computed de events
// background pra renderizar no FullCalendar (cinza). Usado pelos 3 layouts
// da agenda (Melissa, Terapeuta/Rail, Clínica).
//
// Contrato:
// - load(ownerId, rangeStart, rangeEnd)
// carrega bloqueios cujo data_inicio esteja dentro do range OU que
// sejam recorrentes (data_inicio pode estar em qualquer ponto, mas
// o build de events filtra pra emitir só dentro do range visível)
// - bloqueios ref(Array)
// - loading ref(Bool)
// - error ref(String)
// - buildEventsForRange(rangeStart, rangeEnd)
// wrapper sobre agendaMappers.buildBloqueioBackgroundEvents
import { ref } from 'vue';
import { supabase } from '@/lib/supabase/client';
import { buildBloqueioBackgroundEvents } from '@/features/agenda/services/agendaMappers';
export function useAgendaBloqueios() {
const bloqueios = ref([]);
const loading = ref(false);
const error = ref('');
// ownerIdOrIds: string (1 owner) ou Array<string> (multi-owner, Clínica).
async function load(ownerIdOrIds, rangeStart, rangeEnd) {
if (!ownerIdOrIds) return;
const ids = Array.isArray(ownerIdOrIds)
? ownerIdOrIds.filter(Boolean)
: [ownerIdOrIds];
if (!ids.length) return;
loading.value = true;
error.value = '';
try {
const isoStart = _toISODate(rangeStart);
const isoEnd = _toISODate(rangeEnd);
// Query: recorrentes (qualquer data) OU não-recorrentes com
// data_inicio <= isoEnd e (data_fim ?? data_inicio) >= isoStart.
// 2 queries simples + merge pra evitar string-building frágil.
const baseNonRec = supabase
.from('agenda_bloqueios')
.select('*')
.eq('recorrente', false)
.lte('data_inicio', isoEnd)
.or(`data_fim.gte.${isoStart},and(data_fim.is.null,data_inicio.gte.${isoStart})`);
const baseRec = supabase
.from('agenda_bloqueios')
.select('*')
.eq('recorrente', true);
const [{ data: nonRec, error: e1 }, { data: rec, error: e2 }] = await Promise.all([
ids.length === 1 ? baseNonRec.eq('owner_id', ids[0]) : baseNonRec.in('owner_id', ids),
ids.length === 1 ? baseRec.eq('owner_id', ids[0]) : baseRec.in('owner_id', ids)
]);
if (e1) throw e1;
if (e2) throw e2;
bloqueios.value = [...(nonRec || []), ...(rec || [])];
} catch (e) {
error.value = e?.message || 'Falha ao carregar bloqueios.';
bloqueios.value = [];
} finally {
loading.value = false;
}
}
function buildEventsForRange(rangeStart, rangeEnd) {
return buildBloqueioBackgroundEvents(bloqueios.value, rangeStart, rangeEnd);
}
return { bloqueios, loading, error, load, buildEventsForRange };
}
function _toISODate(d) {
if (!d) return null;
const dt = d instanceof Date ? d : new Date(d);
const y = dt.getFullYear();
const m = String(dt.getMonth() + 1).padStart(2, '0');
const day = String(dt.getDate()).padStart(2, '0');
return `${y}-${m}-${day}`;
}
@@ -48,6 +48,25 @@ export function useAgendaEventActions({
servicePickerSel,
selectedPlanService,
saveCommitmentItems,
// chargeMode (Opção C1, 2026-05-13): ref string com modo de cobrança.
// Valores: 'none' | 'session' (avulsa) | 'package' | 'per_session' (recorrente).
// O emit('save') leva chargeMode no payload; handler em useMelissaAgenda
// decide o que criar (financial_record | billing_contract | N events+records).
// Substituiu o boolean gerarCobrancaAoSalvar.
chargeMode,
// packageStyle (2026-05-14): só relevante em chargeMode='package'.
// Valores: 'upfront' (cria 1 financial_record total + materializa 1ª ocorrência)
// | 'saldo' (só billing_contract, sem financial_record imediato — Cliniko).
packageStyle,
// paymentMethod (refatorado 2026-05-16): forma de recebimento quando
// avulsa+session OU pacote+upfront. Valores: 'link' (Asaas, status pending)
// | 'pix' | 'dinheiro' | 'deposito' | 'cartao_maquininha'. Status do
// record é controlado pelo markPaidNow abaixo, não pela forma.
paymentMethod,
// markPaidNow (refatorado 2026-05-16): boolean. Quando true E método !== 'link',
// handler marca o financial_record como paid (paciente pagou na hora).
// Quando false, record nasce pending independente do método.
markPaidNow,
props,
emit
}) {
@@ -62,88 +81,88 @@ export function useAgendaEventActions({
const samePatientConflict = ref(null);
// ────────────────────────────────────────────────────────────────────
// 1. Watcher do form.status — confirma cancelar/remarcar via dialog
// e persiste no banco IMEDIATAMENTE. Reverte se cancelar.
// Antes vivia no .vue; testado em isolamento agora.
// 1. Watcher do form.status
// Fase 5 (2026-05-14): pra realizado/faltou/cancelado, emit
// `updateSeriesEvent` pro parent abrir o AgendaStatusChangeConfirmDialog
// (com regras de exceção, saldo de pacote, etc). Sem confirm.require
// aqui — o dialog do parent é a fonte canônica.
// Pra remarcado mantém path antigo (confirm.require simples).
// Se user cancelar o dialog: parent chama onReject pra reverter o form.
// ────────────────────────────────────────────────────────────────────
watch(
() => composer.form.value?.status,
async (newVal, oldVal) => {
if (_skipStatusWatch.value) return;
if (!composer.isEdit.value || !composer.form.value?.id) return;
if (newVal !== 'cancelado' && newVal !== 'remarcado') return;
const isStatusComDialog = ['realizado', 'faltou', 'cancelado'].includes(newVal);
const isRemarcado = newVal === 'remarcado';
if (!isStatusComDialog && !isRemarcado) return;
_prevStatus.value = oldVal;
const isCancelar = newVal === 'cancelado';
// Fase 5: emit pro parent abrir AgendaStatusChangeConfirmDialog.
// Parent decide o que fazer e chama onReject() se user cancelar.
if (isStatusComDialog) {
const formId = composer.form.value.id;
const isVirtual = !!composer.form.value.is_occurrence || (typeof formId === 'string' && formId.startsWith('rec::'));
emit('updateSeriesEvent', {
id: isVirtual ? null : formId,
status: newVal,
recurrence_date:
composer.form.value.recurrence_date ||
composer.form.value.original_date ||
String(composer.form.value.inicio_em || '').slice(0, 10),
inicio_em: composer.form.value.inicio_em,
fim_em: composer.form.value.fim_em,
is_virtual: isVirtual,
// Form completo — handler usa pra resolver recurrence_id, billing_contract_id, etc
row: { ...composer.form.value },
// Callback pra reverter status no form se user cancelar o dialog do parent.
// _skipStatusWatch evita loop recursivo no watcher.
onReject: () => {
_skipStatusWatch.value = true;
composer.form.value.status = _prevStatus.value;
Promise.resolve().then(() => {
_skipStatusWatch.value = false;
});
}
});
return;
}
// Path legacy pra 'remarcado': confirm.require simples + UPDATE direto.
confirm.require({
header: isCancelar ? 'Cancelar sessão' : 'Remarcar sessão',
message: isCancelar
? 'Tem certeza que deseja cancelar esta sessão? O status será salvo imediatamente.'
: 'Tem certeza que deseja marcar esta sessão para remarcação? O status será salvo imediatamente.',
icon: isCancelar ? 'pi pi-times-circle' : 'pi pi-refresh',
header: 'Remarcar sessão',
message: 'Tem certeza que deseja marcar esta sessão para remarcação? O status será salvo imediatamente.',
icon: 'pi pi-refresh',
acceptLabel: 'Sim, confirmar',
rejectLabel: 'Não',
acceptSeverity: isCancelar ? 'danger' : 'warn',
acceptSeverity: 'warn',
accept: async () => {
try {
// Se o evento é ocorrência VIRTUAL de recorrência
// (id "rec::..." sem row real em agenda_eventos),
// delega pro parent — useMelissaAgenda.onUpdateSeriesEvent
// e AgendaTerapeutaPage.onUpdateSeriesEvent materializam
// a linha antes de aplicar status. Sem essa delegação,
// UPDATE direto em id virtual quebra com PostgreSQL
// "invalid input syntax for type uuid".
const formId = composer.form.value.id;
const isVirtual =
!!composer.form.value.is_occurrence ||
(typeof formId === 'string' && formId.startsWith('rec::'));
const isVirtual = !!composer.form.value.is_occurrence || (typeof formId === 'string' && formId.startsWith('rec::'));
if (isVirtual) {
emit('updateSeriesEvent', {
id: null, // sem row real
id: null,
status: newVal,
recurrence_date:
composer.form.value.recurrence_date ||
composer.form.value.original_date ||
String(composer.form.value.inicio_em || '').slice(0, 10),
recurrence_date: composer.form.value.recurrence_date || composer.form.value.original_date || String(composer.form.value.inicio_em || '').slice(0, 10),
inicio_em: composer.form.value.inicio_em,
fim_em: composer.form.value.fim_em,
is_virtual: true,
// Form completo do dialog — handler usa pra resolver
// recurrence_id/patient_id sem depender de dialogEventRow.
row: { ...composer.form.value }
});
toast.add({
severity: 'success',
summary: 'Status atualizado',
detail: `Sessão marcada como ${labelStatusSessao(newVal)}.`,
life: 3000
});
toast.add({ severity: 'success', summary: 'Status atualizado', detail: `Sessão marcada como ${labelStatusSessao(newVal)}.`, life: 3000 });
return;
}
const { data, error } = await supabase
.from('agenda_eventos')
.update({ status: newVal })
.eq('id', formId)
.select()
.single();
const { data, error } = await supabase.from('agenda_eventos').update({ status: newVal }).eq('id', formId).select().single();
if (error) throw error;
toast.add({
severity: 'success',
summary: 'Status atualizado',
detail: `Sessão marcada como ${labelStatusSessao(newVal)}.`,
life: 3000
});
toast.add({ severity: 'success', summary: 'Status atualizado', detail: `Sessão marcada como ${labelStatusSessao(newVal)}.`, life: 3000 });
emit('updated', data);
} catch (e) {
toast.add({
severity: 'error',
summary: 'Erro',
detail: e?.message || 'Não foi possível atualizar o status.',
life: 4000
});
toast.add({ severity: 'error', summary: 'Erro', detail: e?.message || 'Não foi possível atualizar o status.', life: 4000 });
composer.form.value.status = _prevStatus.value;
}
},
@@ -327,6 +346,24 @@ export function useAgendaEventActions({
editMode: emitEditMode,
recurrence_id: emitRecurrenceId,
original_date: emitOriginalDate,
// _occurrenceMode: flag pra distinguir save do 2o dialog empilhado
// (editar UMA ocorrencia) do save do dialog pai. Handler decide qual
// dialog fechar — sem isso, fechava sempre o pai. 2026-05-12.
_occurrenceMode: !!props.occurrenceMode,
// chargeMode (Opção C1, 2026-05-13): handler decide entre criar
// financial_record (avulsa+session), billing_contract (recorrente+package)
// ou materializar N ocorrências + N records (recorrente+per_session).
// UI no .vue garante valores válidos por modo.
chargeMode: chargeMode?.value ?? 'none',
// packageStyle (2026-05-14): handler em useMelissaAgenda usa pra
// decidir entre upfront (1 record total + materializa 1ª) ou
// saldo (só contrato).
packageStyle: packageStyle?.value ?? 'upfront',
// paymentMethod + markPaidNow (refatorado 2026-05-16): substituem
// o antigo paymentSettlement. Handler aplica payment_method (sempre)
// e status=paid+paid_at apenas quando markPaidNow=true && method!='link'.
paymentMethod: paymentMethod?.value ?? 'link',
markPaidNow: markPaidNow?.value === true,
// legado — mantido para compatibilidade
serie_id: props.eventRow?.serie_id ?? null,
serviceItems: composer.isSessionEvent.value ? commitmentItems.value.slice() : null,
@@ -88,11 +88,14 @@ export function useAgendaEventComposer(props, emit, extras = {}) {
return false;
});
// 'todos_sem_excecao' removido da UI em 2026-05-12 — padrao SimplePractice
// nao expoe override de customizacoes (e destrutivo e raro). Backend ainda
// suporta caso outro fluxo precise, mas dialog so oferece os 3 escopos
// padrao do mercado.
const editScopeOptions = computed(() => [
{ value: 'somente_este', label: 'Somente esta sessão' },
{ value: 'este_e_seguintes', label: 'Esta e as seguintes', disabled: isFirstOccurrence.value },
{ value: 'todos', label: 'Todas da série' },
{ value: 'todos_sem_excecao', label: 'Todas sem exceção' }
{ value: 'todos', label: 'Todas da série' }
]);
// ── 4. Recorrência (criação) ───────────────────────────────────
@@ -96,6 +96,21 @@ export function useAgendaEventLifecycle({
const sendingReminder = ref(false);
const serviceQuickDlgOpen = ref(false);
const insuranceQuickDlgOpen = ref(false);
const planServiceQuickDlgOpen = ref(false);
// occurrenceMode: financial_record da ocorrencia atual (se existir).
// Usado pra travar edicao de tipo/servicos quando ja ha cobranca emitida
// (padrao SimplePractice — cobranca emitida e imutavel; ajustes via fluxo
// do Financeiro, nao via dialog). 2026-05-12.
const occFinancialRecord = ref(null);
const occFinancialLoading = ref(false);
// sessionPaymentRecord (2026-05-18): financial_record da sessão (mesmo
// shape do occFinancialRecord) mas SEM o guard de occurrenceMode.
// Carregado em qualquer edit de sessão pra alimentar a linha "Cobrança"
// do Resumo lateral do AgendaEventDialog. Não dispara lock — esse
// continua via occFinancialRecord (território da Fase 6/C13).
const sessionPaymentRecord = ref(null);
// ── computeds locais ───────────────────────────────────────
const serieCountByStatus = computed(() => {
@@ -192,6 +207,56 @@ export function useAgendaEventLifecycle({
}
}
// ── occurrence financial record loader ────────────────────
async function loadOccFinancialRecord() {
occFinancialRecord.value = null;
if (!props.occurrenceMode) return;
const evId = props.eventRow?.id;
if (!evId) return;
occFinancialLoading.value = true;
try {
const { data, error } = await supabase
.from('financial_records')
.select('id, amount, final_amount, status, due_date, paid_at, payment_method')
.eq('agenda_evento_id', evId)
.in('status', ['pending', 'paid', 'overdue'])
.order('created_at', { ascending: false })
.limit(1)
.maybeSingle();
if (error) throw error;
occFinancialRecord.value = data ?? null;
} catch (e) {
console.warn('[occurrence] erro ao carregar financial_record:', e?.message);
occFinancialRecord.value = null;
} finally {
occFinancialLoading.value = false;
}
}
// sessionPaymentRecord loader (2026-05-18): mesma query, sem guard
// de occurrenceMode. Alimenta a linha "Cobrança" do Resumo do dialog
// em qualquer edit de sessão (Melissa/Rail/Clínica) com eventRow.id.
async function loadSessionPaymentRecord() {
sessionPaymentRecord.value = null;
const evId = props.eventRow?.id;
if (!evId) return;
try {
const { data, error } = await supabase
.from('financial_records')
.select('id, amount, final_amount, status, due_date, paid_at, payment_method')
.eq('agenda_evento_id', evId)
.in('status', ['pending', 'paid', 'overdue'])
.order('created_at', { ascending: false })
.limit(1)
.maybeSingle();
if (error) throw error;
sessionPaymentRecord.value = data ?? null;
} catch (e) {
console.warn('[session-payment] erro ao carregar financial_record:', e?.message);
sessionPaymentRecord.value = null;
}
}
function onPillEditClick(ev) {
emit('editSeriesOccurrence', {
id: ev.id,
@@ -278,6 +343,26 @@ export function useAgendaEventLifecycle({
}
}
// Quick-create de procedimento (insurance_plan_services) — inline,
// sem sair do dialog. Trigger no card Sessao/Honorarios quando o
// convenio selecionado nao tem procedimentos ou quando user quer
// adicionar mais. Apos criar, recarrega os planos pra refletir no
// computed planServices.
function openPlanServiceQuickCreate() {
if (!composer.form.value.insurance_plan_id) return;
planServiceQuickDlgOpen.value = true;
}
async function onPlanServiceCreated(service) {
await loadInsurancePlans(props.planOwnerId || props.ownerId);
// Auto-seleciona o procedimento recem-criado se o user nao
// tinha nenhum selecionado ainda (caso comum: convenio sem
// procedimentos -> cadastra o primeiro -> ja entra selecionado).
if (service?.id && !pickerBilling.selectedPlanService.value) {
pickerBilling.selectedPlanService.value = service.id;
pickerBilling.onProcedureSelect(service.id);
}
}
// ── lembrete WhatsApp manual (8.2) ─────────────────────────
async function onSendManualReminder() {
if (!composer.form.value?.id) return;
@@ -349,7 +434,21 @@ export function useAgendaEventLifecycle({
if (composer.hasSerie.value) loadSerieEvents();
else serieEvents.value = [];
if (composer.isEdit.value) {
// occurrenceMode: carrega financial_record desta ocorrencia
// pra decidir se o card Sessao/Honorarios fica locked (cobranca
// ja emitida) ou unlocked (sem cobranca, edicao livre).
loadOccFinancialRecord();
// sessionPaymentRecord: carrega em qualquer edit (Melissa
// tambem) pra alimentar a linha "Cobrança" do Resumo lateral.
loadSessionPaymentRecord();
// occurrenceMode: editando UMA ocorrencia de serie ja existente —
// tipo de compromisso ja foi escolhido (paciente + sessao). Pular
// step 1 incondicionalmente. Defesa em camadas: useMelissaAgenda
// ja seta is_occurrence=true na row (faz isEdit=true), mas se outro
// call site esquecer essa flag o guard aqui salva.
if (props.occurrenceMode || composer.isEdit.value) {
composer.step.value = 2;
} else {
const preset = props.presetCommitmentId;
@@ -452,11 +551,17 @@ export function useAgendaEventLifecycle({
sendingReminder,
serviceQuickDlgOpen,
insuranceQuickDlgOpen,
planServiceQuickDlgOpen,
occFinancialRecord,
occFinancialLoading,
sessionPaymentRecord,
// computeds
serieCountByStatus,
pillDeleteMenuItems,
// series
loadSerieEvents,
loadOccFinancialRecord,
loadSessionPaymentRecord,
onPillEditClick,
onPillStatusChange,
onPillDeleteClick,
@@ -468,6 +573,8 @@ export function useAgendaEventLifecycle({
onServiceCreated,
openInsuranceQuickCreate,
onInsuranceCreated,
openPlanServiceQuickCreate,
onPlanServiceCreated,
// reminder
onSendManualReminder
};
@@ -157,8 +157,16 @@ export function useCommitmentServices() {
// Atualiza commitment_services nos agenda_eventos com recurrence_id = ruleId
// onde services_customized = false (não foram editados individualmente).
//
// Invariante adicional (2026-05-12, padrão SimplePractice): NUNCA propaga
// para eventos que já têm financial_record emitido (status pending/paid/
// overdue). Cobranças emitidas são imutáveis — ajustes só via fluxo do
// Financeiro. Sem isso, mudar o template da regra mudaria silenciosamente
// o valor referenciado por uma cobrança já entregue ao paciente.
//
// opts.fromDate: string ISO 'YYYY-MM-DD' — limita a ocorrências a partir
// dessa data inclusive (escopo 'este_e_seguintes'). null = todas da série.
// opts.ignoreCustomized: bypass do filtro services_customized=false
// (escopo 'todos_sem_excecao' operacional — NÃO afeta filtro financeiro).
async function propagateToSerie(ruleId, items, { fromDate = null, ignoreCustomized = false } = {}) {
if (!ruleId) return;
@@ -177,8 +185,23 @@ export function useCommitmentServices() {
if (queryError) throw queryError;
if (!events?.length) return;
// Filtra OUT eventos que já têm financial_record emitido. Uma query
// em batch evita N round-trips. Status considerados imutáveis: pending,
// paid, overdue. cancelled é ok propagar (record foi descartado).
const eventIds = events.map((e) => e.id);
const { data: lockedEvents, error: frErr } = await supabase
.from('financial_records')
.select('agenda_evento_id')
.in('agenda_evento_id', eventIds)
.in('status', ['pending', 'paid', 'overdue']);
if (frErr) throw frErr;
const lockedSet = new Set((lockedEvents || []).map((r) => r.agenda_evento_id));
const eligibleEvents = events.filter((ev) => !lockedSet.has(ev.id));
if (!eligibleEvents.length) return;
// Para cada evento elegível: delete + insert (padrão idempotente)
for (const ev of events) {
for (const ev of eligibleEvents) {
const { error: delErr } = await supabase.from('commitment_services').delete().eq('commitment_id', ev.id);
if (delErr) throw delErr;
@@ -66,7 +66,8 @@ export function useFinancialExceptions() {
charge_mode: payload.charge_mode,
charge_value: payload.charge_value ?? null,
charge_pct: payload.charge_pct ?? null,
min_hours_notice: payload.min_hours_notice ?? null
min_hours_notice: payload.min_hours_notice ?? null,
default_consume_on_miss: !!payload.default_consume_on_miss
})
.eq('id', payload.id);
if (err) throw err;
@@ -78,7 +79,8 @@ export function useFinancialExceptions() {
charge_mode: payload.charge_mode,
charge_value: payload.charge_value ?? null,
charge_pct: payload.charge_pct ?? null,
min_hours_notice: payload.min_hours_notice ?? null
min_hours_notice: payload.min_hours_notice ?? null,
default_consume_on_miss: !!payload.default_consume_on_miss
});
if (err) throw err;
}