Preficicação, Convenio, Ajustes Agenda, Configurações Excessões

This commit is contained in:
Leonardo
2026-03-13 16:03:08 -03:00
parent f4b185ae17
commit 06fb369beb
30 changed files with 24851 additions and 307 deletions
@@ -0,0 +1,343 @@
<!-- src/layout/configuracoes/ConfiguracoesConveniosPage.vue -->
<script setup>
import { ref, onMounted } from 'vue'
import { supabase } from '@/lib/supabase/client'
import { useTenantStore } from '@/stores/tenantStore'
import { useToast } from 'primevue/usetoast'
const toast = useToast()
const tenantStore = useTenantStore()
const ownerId = ref(null)
const tenantId = ref(null)
const pageLoading = ref(true)
const plans = ref([])
// ── Formulário ───────────────────────────────────────────────────────
const emptyForm = () => ({ name: '', notes: '', default_value: null })
const newForm = ref(emptyForm())
const addingNew = ref(false)
const savingNew = ref(false)
const editingId = ref(null)
const editForm = ref({})
const savingEdit = ref(false)
// ── Load ─────────────────────────────────────────────────────────────
async function load (uid) {
const { data, error } = await supabase
.from('insurance_plans')
.select('*')
.eq('owner_id', uid)
.eq('active', true)
.order('name')
if (error) throw error
plans.value = data || []
}
// ── Save ─────────────────────────────────────────────────────────────
async function savePlan (payload) {
if (payload.id) {
const { error } = await supabase
.from('insurance_plans')
.update({
name: payload.name.trim(),
notes: payload.notes?.trim() || null,
default_value: payload.default_value ?? null,
updated_at: new Date().toISOString(),
})
.eq('id', payload.id)
if (error) throw error
} else {
const { error } = await supabase
.from('insurance_plans')
.insert({
owner_id: ownerId.value,
tenant_id: tenantId.value,
name: payload.name.trim(),
notes: payload.notes?.trim() || null,
default_value: payload.default_value ?? null,
})
if (error) throw error
}
}
// ── Remove (soft-delete) ─────────────────────────────────────────────
async function removePlan (id) {
const { error } = await supabase
.from('insurance_plans')
.update({ active: false })
.eq('id', id)
if (error) throw error
plans.value = plans.value.filter(p => p.id !== id)
toast.add({ severity: 'success', summary: 'Removido', detail: 'Convênio desativado.', life: 3000 })
}
// ── Edit helpers ──────────────────────────────────────────────────────
function startEdit (plan) {
editingId.value = plan.id
editForm.value = {
id: plan.id,
name: plan.name,
notes: plan.notes ?? '',
default_value: plan.default_value != null ? Number(plan.default_value) : null,
}
}
function cancelEdit () {
editingId.value = null
editForm.value = {}
}
async function saveEdit () {
if (!editForm.value.name?.trim()) {
toast.add({ severity: 'warn', summary: 'Campo obrigatório', detail: 'Nome é obrigatório.', life: 3000 })
return
}
savingEdit.value = true
try {
await savePlan(editForm.value)
await load(ownerId.value)
cancelEdit()
toast.add({ severity: 'success', summary: 'Salvo', detail: 'Convênio atualizado.', life: 3000 })
} catch (e) {
toast.add({ severity: 'error', summary: 'Erro', detail: e?.message || 'Falha ao salvar.', life: 4000 })
} finally {
savingEdit.value = false
}
}
async function saveNew () {
if (!newForm.value.name?.trim()) {
toast.add({ severity: 'warn', summary: 'Campo obrigatório', detail: 'Nome é obrigatório.', life: 3000 })
return
}
savingNew.value = true
try {
await savePlan(newForm.value)
await load(ownerId.value)
newForm.value = emptyForm()
addingNew.value = false
toast.add({ severity: 'success', summary: 'Adicionado', detail: 'Convênio criado com sucesso.', life: 3000 })
} catch (e) {
toast.add({ severity: 'error', summary: 'Erro', detail: e?.message || 'Falha ao criar.', life: 4000 })
} finally {
savingNew.value = false
}
}
function fmtBRL (v) {
if (v == null || v === '') return '—'
return Number(v).toLocaleString('pt-BR', { style: 'currency', currency: 'BRL' })
}
// ── Mount ─────────────────────────────────────────────────────────────
onMounted(async () => {
try {
const uid = tenantStore.user?.id || (await supabase.auth.getUser()).data?.user?.id
if (!uid) return
ownerId.value = uid
tenantId.value = tenantStore.activeTenantId || null
await load(uid)
} catch (e) {
toast.add({ severity: 'error', summary: 'Erro', detail: e?.message || 'Falha ao carregar.', life: 4000 })
} finally {
pageLoading.value = false
}
})
</script>
<template>
<Toast />
<div class="flex flex-col gap-4">
<!-- Header -->
<Card>
<template #content>
<div class="flex items-center justify-between gap-3 flex-wrap">
<div class="flex items-center gap-3">
<div class="cfg-icon-box">
<i class="pi pi-id-card text-lg" />
</div>
<div>
<div class="text-900 font-semibold text-lg">Convênios</div>
<div class="text-600 text-sm">
Cadastre os convênios que você atende e seus valores de tabela.
</div>
</div>
</div>
<Button
label="Novo convênio"
icon="pi pi-plus"
:disabled="pageLoading || addingNew"
@click="addingNew = true; cancelEdit()"
/>
</div>
</template>
</Card>
<!-- Loading -->
<div v-if="pageLoading" class="flex justify-center py-10">
<ProgressSpinner style="width:40px;height:40px" />
</div>
<template v-else>
<!-- Formulário novo convênio -->
<Card v-if="addingNew">
<template #title>
<div class="flex items-center gap-2">
<i class="pi pi-plus-circle text-primary-500" />
<span>Novo convênio</span>
</div>
</template>
<template #content>
<div class="grid grid-cols-12 gap-3">
<div class="col-span-12 sm:col-span-4">
<FloatLabel variant="on">
<InputText v-model="newForm.name" inputId="new-name" class="w-full" />
<label for="new-name">Nome *</label>
</FloatLabel>
</div>
<div class="col-span-12 sm:col-span-4">
<FloatLabel variant="on">
<InputNumber
v-model="newForm.default_value"
inputId="new-value"
mode="currency"
currency="BRL"
locale="pt-BR"
:min="0"
:minFractionDigits="2"
fluid
/>
<label for="new-value">Valor padrão da tabela (R$)</label>
</FloatLabel>
</div>
<div class="col-span-12 sm:col-span-4">
<FloatLabel variant="on">
<InputText v-model="newForm.notes" inputId="new-notes" class="w-full" />
<label for="new-notes">Observações (opcional)</label>
</FloatLabel>
</div>
</div>
<div class="flex gap-2 justify-end mt-4">
<Button label="Cancelar" severity="secondary" outlined @click="addingNew = false; newForm = emptyForm()" />
<Button label="Salvar" icon="pi pi-check" :loading="savingNew" @click="saveNew" />
</div>
</template>
</Card>
<!-- Lista vazia -->
<Card v-if="!plans.length && !addingNew">
<template #content>
<div class="text-center py-6 text-color-secondary">
<i class="pi pi-id-card text-4xl opacity-30 mb-3 block" />
<div class="font-medium mb-1">Nenhum convênio cadastrado</div>
<div class="text-sm">Clique em "Novo convênio" para começar.</div>
</div>
</template>
</Card>
<!-- Lista de convênios -->
<Card v-for="plan in plans" :key="plan.id">
<template #content>
<!-- Modo edição -->
<template v-if="editingId === plan.id">
<div class="grid grid-cols-12 gap-3">
<div class="col-span-12 sm:col-span-4">
<FloatLabel variant="on">
<InputText v-model="editForm.name" :inputId="`edit-name-${plan.id}`" class="w-full" />
<label :for="`edit-name-${plan.id}`">Nome *</label>
</FloatLabel>
</div>
<div class="col-span-12 sm:col-span-4">
<FloatLabel variant="on">
<InputNumber
v-model="editForm.default_value"
:inputId="`edit-value-${plan.id}`"
mode="currency"
currency="BRL"
locale="pt-BR"
:min="0"
:minFractionDigits="2"
fluid
/>
<label :for="`edit-value-${plan.id}`">Valor padrão da tabela (R$)</label>
</FloatLabel>
</div>
<div class="col-span-12 sm:col-span-4">
<FloatLabel variant="on">
<InputText v-model="editForm.notes" :inputId="`edit-notes-${plan.id}`" class="w-full" />
<label :for="`edit-notes-${plan.id}`">Observações (opcional)</label>
</FloatLabel>
</div>
</div>
<div class="flex gap-2 justify-end mt-4">
<Button label="Cancelar" severity="secondary" outlined @click="cancelEdit" />
<Button label="Salvar" icon="pi pi-check" :loading="savingEdit" @click="saveEdit" />
</div>
</template>
<!-- Modo leitura -->
<template v-else>
<div class="flex items-center justify-between gap-3 flex-wrap">
<div class="flex items-center gap-3">
<div class="cfg-icon-box-sm">
<i class="pi pi-id-card" />
</div>
<div>
<div class="font-semibold text-900">{{ plan.name }}</div>
<div class="text-sm text-color-secondary flex flex-wrap gap-x-3 gap-y-0.5 mt-0.5">
<span><b class="text-primary-500">{{ fmtBRL(plan.default_value) }}</b></span>
<span v-if="plan.notes" class="italic">{{ plan.notes }}</span>
</div>
</div>
</div>
<div class="flex items-center gap-2">
<Tag value="Ativo" severity="success" />
<Button icon="pi pi-pencil" severity="secondary" outlined size="small" v-tooltip.top="'Editar'" @click="startEdit(plan)" />
<Button icon="pi pi-trash" severity="danger" outlined size="small" v-tooltip.top="'Desativar'" @click="removePlan(plan.id)" />
</div>
</div>
</template>
</template>
</Card>
<Message severity="info" :closable="false">
<span class="text-sm">
O convênio selecionado em um evento preenche automaticamente o valor da sessão com o valor de tabela cadastrado aqui.
</span>
</Message>
</template>
</div>
</template>
<style scoped>
.cfg-icon-box {
display: grid;
place-items: center;
width: 2.5rem;
height: 2.5rem;
border-radius: 0.875rem;
background: color-mix(in srgb, var(--p-primary-500, #6366f1) 12%, transparent);
color: var(--p-primary-500, #6366f1);
flex-shrink: 0;
}
.cfg-icon-box-sm {
display: grid;
place-items: center;
width: 2rem;
height: 2rem;
border-radius: 0.625rem;
background: color-mix(in srgb, var(--p-primary-500, #6366f1) 10%, transparent);
color: var(--p-primary-500, #6366f1);
flex-shrink: 0;
}
</style>