feat: Implement initial pedido (order) management, product catalog, and TI configuration features.

This commit is contained in:
2025-12-01 17:11:34 -03:00
parent 5e7de6c943
commit b8a67e0a57
18 changed files with 4429 additions and 1934 deletions

View File

@@ -367,6 +367,15 @@
palette: 'accent',
icon: 'building'
},
{
title: 'Configurações Gerais',
description:
'Configure opções gerais do sistema, incluindo setor de compras e outras configurações administrativas.',
ctaLabel: 'Configurar',
href: '/(dashboard)/ti/configuracoes',
palette: 'secondary',
icon: 'control'
},
{
title: 'Documentação',
description:

View File

@@ -0,0 +1,94 @@
<script lang="ts">
import { useQuery, useConvexClient } from 'convex-svelte';
import { api } from '@sgse-app/backend/convex/_generated/api';
import type { Id } from '@sgse-app/backend/convex/_generated/dataModel';
const client = useConvexClient();
// Reactive queries
const setoresQuery = useQuery(api.setores.list, {});
const configQuery = useQuery(api.config.getComprasSetor, {});
const setores = $derived(setoresQuery.data || []);
const config = $derived(configQuery.data);
const loading = $derived(setoresQuery.isLoading || configQuery.isLoading);
// Initialize selected setor from config - using boxed $state to avoid reactivity warning
let selectedSetorId = $state('');
// Update selectedSetorId when config changes
$effect(() => {
if (config?.comprasSetorId && !selectedSetorId) {
selectedSetorId = config.comprasSetorId;
}
});
let saving = $state(false);
let error: string | null = $state(null);
let success: string | null = $state(null);
async function saveConfig() {
saving = true;
error = null;
success = null;
try {
await client.mutation(api.config.updateComprasSetor, {
setorId: selectedSetorId as Id<'setores'>
});
success = 'Configuração salva com sucesso!';
} catch (e) {
error = (e as Error).message;
} finally {
saving = false;
}
}
</script>
<div class="container mx-auto p-6">
<h1 class="mb-6 text-2xl font-bold">Configurações Gerais</h1>
{#if loading}
<p>Carregando...</p>
{:else}
<div class="max-w-md rounded-lg bg-white p-6 shadow-md">
<h2 class="mb-4 text-xl font-semibold">Setor de Compras</h2>
<p class="mb-4 text-sm text-gray-600">
Selecione o setor responsável por receber e aprovar pedidos de compra.
</p>
{#if error}
<div class="mb-4 rounded border border-red-400 bg-red-100 px-4 py-3 text-red-700">
{error}
</div>
{/if}
{#if success}
<div class="mb-4 rounded border border-green-400 bg-green-100 px-4 py-3 text-green-700">
{success}
</div>
{/if}
<div class="mb-4">
<label for="setor" class="mb-1 block text-sm font-medium text-gray-700"> Setor </label>
<select
id="setor"
bind:value={selectedSetorId}
class="w-full rounded-md border-gray-300 shadow-sm focus:border-blue-500 focus:ring-blue-500"
>
<option value="">Selecione um setor...</option>
{#each setores as setor (setor._id)}
<option value={setor._id}>{setor.nome} ({setor.sigla})</option>
{/each}
</select>
</div>
<button
onclick={saveConfig}
disabled={saving || !selectedSetorId}
class="w-full rounded-md bg-blue-600 px-4 py-2 text-white transition-colors hover:bg-blue-700 disabled:cursor-not-allowed disabled:opacity-50"
>
{saving ? 'Salvando...' : 'Salvar Configuração'}
</button>
</div>
{/if}
</div>