feat: enhance notification bell component by refactoring notification fetching logic, improving type safety, and updating UI elements for better user experience

This commit is contained in:
2025-12-15 11:33:51 -03:00
parent c272ca05e8
commit f3288b9639
5 changed files with 589 additions and 575 deletions

View File

@@ -2,8 +2,9 @@
import { api } from '@sgse-app/backend/convex/_generated/api';
import type { Doc, Id } from '@sgse-app/backend/convex/_generated/dataModel';
import { useConvexClient, useQuery } from 'convex-svelte';
import { Pencil, Plus, Trash2, X } from 'lucide-svelte';
import { Pencil, Plus, Trash2, X, Package } from 'lucide-svelte';
import { maskCurrencyBRL } from '$lib/utils/masks';
import { resolve } from '$app/paths';
const client = useConvexClient();
@@ -116,150 +117,187 @@
}
</script>
<div class="container mx-auto p-6">
<div class="mb-6 flex items-center justify-between">
<h1 class="text-2xl font-bold">Objetos</h1>
<button
onclick={() => openModal()}
class="flex items-center gap-2 rounded-md bg-blue-600 px-4 py-2 text-white hover:bg-blue-700"
>
<Plus size={20} />
Novo Objeto
</button>
<main class="container mx-auto flex max-w-7xl flex-col px-4 py-4">
<div class="breadcrumbs mb-4 text-sm">
<ul>
<li><a href={resolve('/')} class="text-primary hover:underline">Dashboard</a></li>
<li><a href={resolve('/compras')} class="text-primary hover:underline">Compras</a></li>
<li>Objetos</li>
</ul>
</div>
<div class="mb-6">
<div class="flex flex-col items-start justify-between gap-4 sm:flex-row sm:items-center">
<div class="flex items-center gap-4">
<div class="bg-primary/10 rounded-xl p-3">
<Package class="text-primary h-8 w-8" strokeWidth={2} />
</div>
<div>
<h1 class="text-primary text-3xl font-bold">Objetos</h1>
<p class="text-base-content/70">Cadastro e gestão de objetos e serviços</p>
</div>
</div>
<button
class="btn btn-primary gap-2 shadow-md transition-all hover:shadow-lg"
onclick={() => openModal()}
>
<Plus class="h-5 w-5" strokeWidth={2} />
Novo Objeto
</button>
</div>
</div>
{#if loading}
<p>Carregando...</p>
<div class="flex items-center justify-center py-10">
<span class="loading loading-spinner loading-lg"></span>
</div>
{:else if error}
<p class="text-red-600">{error}</p>
<div class="alert alert-error">
<span>{error}</span>
</div>
{:else}
<div class="overflow-hidden rounded-lg bg-white shadow-md">
<table class="min-w-full divide-y divide-gray-200">
<thead class="bg-gray-50">
<tr>
<th
class="px-6 py-3 text-left text-xs font-medium tracking-wider text-gray-500 uppercase"
>Nome</th
>
<th
class="px-6 py-3 text-left text-xs font-medium tracking-wider text-gray-500 uppercase"
>Tipo</th
>
<th
class="px-6 py-3 text-left text-xs font-medium tracking-wider text-gray-500 uppercase"
>Unidade</th
>
<th
class="px-6 py-3 text-left text-xs font-medium tracking-wider text-gray-500 uppercase"
>Valor Estimado</th
>
<th
class="px-6 py-3 text-right text-xs font-medium tracking-wider text-gray-500 uppercase"
>Ações</th
>
</tr>
</thead>
<tbody class="divide-y divide-gray-200 bg-white">
{#each objetos as objeto (objeto._id)}
<tr>
<td class="px-6 py-4 whitespace-nowrap">
<div class="flex flex-col">
<span class="font-medium">{objeto.nome}</span>
<span class="text-xs text-gray-500">
Efisco: {objeto.codigoEfisco}
{#if objeto.codigoCatmat}
| Catmat: {objeto.codigoCatmat}{/if}
{#if objeto.codigoCatserv}
| Catserv: {objeto.codigoCatserv}{/if}
</span>
</div>
</td>
<td class="px-6 py-4 whitespace-nowrap">
<span
class="inline-flex rounded-full px-2 text-xs leading-5 font-semibold
{objeto.tipo === 'servico'
? 'bg-green-100 text-green-800'
: 'bg-blue-100 text-blue-800'}"
<div class="card bg-base-100/90 border-base-300 border shadow-xl backdrop-blur-sm">
<div class="card-body p-0">
<div class="overflow-x-auto">
<table class="table-zebra table w-full">
<thead class="from-base-300 to-base-200 sticky top-0 z-10 bg-linear-to-r shadow-md">
<tr>
<th class="text-base-content border-base-400 border-b font-bold whitespace-nowrap"
>Nome</th
>
{objeto.tipo === 'material' ? 'Material' : 'Serviço'}
</span>
</td>
<td class="px-6 py-4 whitespace-nowrap">{objeto.unidade}</td>
<td class="px-6 py-4 whitespace-nowrap">
{maskCurrencyBRL(objeto.valorEstimado) || 'R$ 0,00'}
</td>
<td class="px-6 py-4 text-right text-sm font-medium whitespace-nowrap">
<button
onclick={() => openModal(objeto)}
class="mr-4 text-indigo-600 hover:text-indigo-900"
<th class="text-base-content border-base-400 border-b font-bold whitespace-nowrap"
>Tipo</th
>
<Pencil size={18} />
</button>
<button
onclick={() => handleDelete(objeto._id)}
class="text-red-600 hover:text-red-900"
<th class="text-base-content border-base-400 border-b font-bold whitespace-nowrap"
>Unidade</th
>
<Trash2 size={18} />
</button>
</td>
</tr>
{/each}
{#if objetos.length === 0}
<tr>
<td colspan="5" class="px-6 py-4 text-center text-gray-500"
>Nenhum objeto cadastrado.</td
>
</tr>
{/if}
</tbody>
</table>
<th class="text-base-content border-base-400 border-b font-bold whitespace-nowrap"
>Valor Estimado</th
>
<th
class="text-base-content border-base-400 border-b text-right font-bold whitespace-nowrap"
>Ações</th
>
</tr>
</thead>
<tbody>
{#if objetos.length === 0}
<tr>
<td colspan="5" class="py-12 text-center">
<div class="text-base-content/60 flex flex-col items-center gap-2">
<p class="text-lg font-semibold">Nenhum objeto cadastrado</p>
<p class="text-sm">Clique em “Novo Objeto” para cadastrar.</p>
</div>
</td>
</tr>
{:else}
{#each objetos as objeto (objeto._id)}
<tr class="hover:bg-base-200/50 transition-colors">
<td>
<div class="flex flex-col">
<span class="font-medium">{objeto.nome}</span>
<span class="text-base-content/60 text-xs">
Efisco: {objeto.codigoEfisco}
{#if objeto.codigoCatmat}
| Catmat: {objeto.codigoCatmat}{/if}
{#if objeto.codigoCatserv}
| Catserv: {objeto.codigoCatserv}{/if}
</span>
</div>
</td>
<td class="whitespace-nowrap">
<span
class="badge badge-sm {objeto.tipo === 'servico'
? 'badge-success'
: 'badge-info'}"
>
{objeto.tipo === 'material' ? 'Material' : 'Serviço'}
</span>
</td>
<td class="whitespace-nowrap">{objeto.unidade}</td>
<td class="whitespace-nowrap">
{maskCurrencyBRL(objeto.valorEstimado) || 'R$ 0,00'}
</td>
<td class="text-right whitespace-nowrap">
<div class="flex items-center justify-end gap-1">
<button
type="button"
class="btn btn-ghost btn-sm"
aria-label="Editar objeto"
onclick={() => openModal(objeto)}
>
<Pencil size={18} />
</button>
<button
type="button"
class="btn btn-ghost btn-sm text-error"
aria-label="Excluir objeto"
onclick={() => handleDelete(objeto._id)}
>
<Trash2 size={18} />
</button>
</div>
</td>
</tr>
{/each}
{/if}
</tbody>
</table>
</div>
</div>
</div>
{/if}
{#if showModal}
<div
class="fixed inset-0 z-50 flex h-full w-full items-center justify-center overflow-y-auto bg-black/40"
>
<div class="relative w-full max-w-md rounded-lg bg-white p-8 shadow-xl">
<div class="modal modal-open">
<div class="modal-box max-w-2xl">
<button
type="button"
class="btn btn-sm btn-circle absolute top-2 right-2"
onclick={closeModal}
class="absolute top-4 right-4 text-gray-400 hover:text-gray-600"
aria-label="Fechar modal"
>
<X size={24} />
<X class="h-5 w-5" />
</button>
<h2 class="mb-6 text-xl font-bold">{editingId ? 'Editar' : 'Novo'} Objeto</h2>
<form onsubmit={handleSubmit}>
<div class="mb-4">
<label class="mb-2 block text-sm font-bold text-gray-700" for="nome"> Nome </label>
<h3 class="text-lg font-bold">{editingId ? 'Editar' : 'Novo'} Objeto</h3>
<form class="mt-6 space-y-4" onsubmit={handleSubmit}>
<div class="form-control w-full">
<label class="label" for="nome">
<span class="label-text font-semibold">Nome</span>
</label>
<input
class="focus:shadow-outline w-full appearance-none rounded border px-3 py-2 leading-tight text-gray-700 shadow focus:outline-none"
id="nome"
class="input input-bordered focus:input-primary w-full"
type="text"
bind:value={formData.nome}
required
/>
</div>
<div class="mb-4 grid grid-cols-2 gap-4">
<div>
<label class="mb-2 block text-sm font-bold text-gray-700" for="tipo"> Tipo </label>
<div class="grid grid-cols-1 gap-4 md:grid-cols-2">
<div class="form-control w-full">
<label class="label" for="tipo">
<span class="label-text font-semibold">Tipo</span>
</label>
<select
class="focus:shadow-outline w-full rounded border px-3 py-2 leading-tight text-gray-700 shadow focus:outline-none"
id="tipo"
class="select select-bordered focus:select-primary w-full"
bind:value={formData.tipo}
>
<option value="material">Material</option>
<option value="servico">Serviço</option>
</select>
</div>
<div>
<label class="mb-2 block text-sm font-bold text-gray-700" for="unidade">
Unidade
<div class="form-control w-full">
<label class="label" for="unidade">
<span class="label-text font-semibold">Unidade</span>
</label>
<input
class="focus:shadow-outline w-full appearance-none rounded border px-3 py-2 leading-tight text-gray-700 shadow focus:outline-none"
id="unidade"
class="input input-bordered focus:input-primary w-full"
type="text"
bind:value={formData.unidade}
required
@@ -267,13 +305,13 @@
</div>
</div>
<div class="mb-4">
<label class="mb-2 block text-sm font-bold text-gray-700" for="codigoEfisco">
Código Efisco
<div class="form-control w-full">
<label class="label" for="codigoEfisco">
<span class="label-text font-semibold">Código Efisco</span>
</label>
<input
class="focus:shadow-outline w-full appearance-none rounded border px-3 py-2 leading-tight text-gray-700 shadow focus:outline-none"
id="codigoEfisco"
class="input input-bordered focus:input-primary w-full"
type="text"
bind:value={formData.codigoEfisco}
required
@@ -281,88 +319,86 @@
</div>
{#if formData.tipo === 'material'}
<div class="mb-4">
<label class="mb-2 block text-sm font-bold text-gray-700" for="codigoCatmat">
Código Catmat
<div class="form-control w-full">
<label class="label" for="codigoCatmat">
<span class="label-text font-semibold">Código Catmat</span>
</label>
<input
class="focus:shadow-outline w-full appearance-none rounded border px-3 py-2 leading-tight text-gray-700 shadow focus:outline-none"
id="codigoCatmat"
class="input input-bordered focus:input-primary w-full"
type="text"
bind:value={formData.codigoCatmat}
/>
</div>
{:else}
<div class="mb-4">
<label class="mb-2 block text-sm font-bold text-gray-700" for="codigoCatserv">
Código Catserv
<div class="form-control w-full">
<label class="label" for="codigoCatserv">
<span class="label-text font-semibold">Código Catserv</span>
</label>
<input
class="focus:shadow-outline w-full appearance-none rounded border px-3 py-2 leading-tight text-gray-700 shadow focus:outline-none"
id="codigoCatserv"
class="input input-bordered focus:input-primary w-full"
type="text"
bind:value={formData.codigoCatserv}
/>
</div>
{/if}
<div class="mb-6">
<label class="mb-2 block text-sm font-bold text-gray-700" for="valor">
Valor Estimado
<div class="form-control w-full">
<label class="label" for="valor">
<span class="label-text font-semibold">Valor Estimado</span>
</label>
<input
class="focus:shadow-outline w-full appearance-none rounded border px-3 py-2 leading-tight text-gray-700 shadow focus:outline-none"
id="valor"
class="input input-bordered focus:input-primary w-full"
type="text"
placeholder="R$ 0,00"
bind:value={formData.valorEstimado}
oninput={(e) => (formData.valorEstimado = maskCurrencyBRL(e.currentTarget.value))}
placeholder="R$ 0,00"
/>
</div>
<div class="mb-6">
<label class="mb-2 block text-sm font-bold text-gray-700" for="atas">
Vincular Atas
<div class="form-control w-full">
<label class="label" for="atas">
<span class="label-text font-semibold">Vincular Atas</span>
</label>
<div class="max-h-40 overflow-y-auto rounded border p-2">
{#each atas as ata (ata._id)}
<div class="mb-2 flex items-center">
<input
type="checkbox"
id={`ata-${ata._id}`}
checked={formData.atas.includes(ata._id)}
onchange={() => toggleAtaSelection(ata._id)}
class="mr-2 h-4 w-4 rounded border-gray-300 text-blue-600 focus:ring-blue-500"
/>
<label for={`ata-${ata._id}`} class="text-sm text-gray-700">
{ata.numero} ({ata.numeroSei})
</label>
</div>
{/each}
<div class="border-base-300 max-h-48 overflow-y-auto rounded-lg border p-2">
{#if atas.length === 0}
<p class="text-sm text-gray-500">Nenhuma ata disponível.</p>
<p class="text-base-content/60 px-2 py-3 text-sm">Nenhuma ata disponível.</p>
{:else}
{#each atas as ata (ata._id)}
<label
class="hover:bg-base-200/50 flex cursor-pointer items-center gap-3 rounded-md px-2 py-2"
>
<input
type="checkbox"
class="checkbox checkbox-primary checkbox-sm"
checked={formData.atas.includes(ata._id)}
onchange={() => toggleAtaSelection(ata._id)}
aria-label="Vincular ata {ata.numero}"
/>
<span class="text-sm">{ata.numero} ({ata.numeroSei})</span>
</label>
{/each}
{/if}
</div>
</div>
<div class="flex items-center justify-end">
<button
type="button"
onclick={closeModal}
class="mr-2 rounded bg-gray-300 px-4 py-2 font-bold text-gray-800 hover:bg-gray-400"
>
<div class="modal-action">
<button type="button" class="btn" onclick={closeModal} disabled={saving}>
Cancelar
</button>
<button
type="submit"
disabled={saving}
class="focus:shadow-outline rounded bg-blue-600 px-4 py-2 font-bold text-white hover:bg-blue-700 focus:outline-none disabled:opacity-50"
>
<button type="submit" class="btn btn-primary" disabled={saving}>
{#if saving}
<span class="loading loading-spinner loading-sm"></span>
{/if}
{saving ? 'Salvando...' : 'Salvar'}
</button>
</div>
</form>
</div>
<button type="button" class="modal-backdrop" onclick={closeModal} aria-label="Fechar modal"
></button>
</div>
{/if}
</div>
</main>