feat: Add confirmation modal for item actions and enhance user feedback with toast notifications in pedidos management.
This commit is contained in:
133
apps/web/src/lib/components/ConfirmationModal.svelte
Normal file
133
apps/web/src/lib/components/ConfirmationModal.svelte
Normal file
@@ -0,0 +1,133 @@
|
||||
<script lang="ts">
|
||||
import { AlertTriangle, X } from 'lucide-svelte';
|
||||
|
||||
interface Props {
|
||||
open: boolean;
|
||||
title?: string;
|
||||
message: string;
|
||||
confirmText?: string;
|
||||
cancelText?: string;
|
||||
isDestructive?: boolean;
|
||||
onConfirm: () => void;
|
||||
onClose: () => void;
|
||||
}
|
||||
|
||||
let {
|
||||
open = $bindable(false),
|
||||
title = 'Confirmar Ação',
|
||||
message,
|
||||
confirmText = 'Confirmar',
|
||||
cancelText = 'Cancelar',
|
||||
isDestructive = false,
|
||||
onConfirm,
|
||||
onClose
|
||||
}: Props = $props();
|
||||
|
||||
// Tenta centralizar, mas se tiver um contexto específico pode ser ajustado
|
||||
// Por padrão, centralizado.
|
||||
function getModalStyle() {
|
||||
return 'position: fixed; top: 50%; left: 50%; transform: translate(-50%, -50%); width: 100%; max-width: 500px;';
|
||||
}
|
||||
|
||||
function handleClose() {
|
||||
open = false;
|
||||
onClose();
|
||||
}
|
||||
|
||||
function handleConfirm() {
|
||||
open = false;
|
||||
onConfirm();
|
||||
}
|
||||
</script>
|
||||
|
||||
{#if open}
|
||||
<div
|
||||
class="pointer-events-none fixed inset-0 z-50"
|
||||
style="animation: fadeIn 0.2s ease-out;"
|
||||
role="dialog"
|
||||
aria-modal="true"
|
||||
aria-labelledby="modal-confirm-title"
|
||||
>
|
||||
<!-- Backdrop leve -->
|
||||
<div
|
||||
class="pointer-events-auto absolute inset-0 bg-black/20 transition-opacity duration-200"
|
||||
onclick={handleClose}
|
||||
aria-hidden="true"
|
||||
></div>
|
||||
|
||||
<!-- Modal Box -->
|
||||
<div
|
||||
class="pointer-events-auto absolute z-10 flex w-full max-w-lg flex-col overflow-hidden rounded-2xl bg-white shadow-2xl transition-all duration-300"
|
||||
style="animation: slideUp 0.3s cubic-bezier(0.34, 1.56, 0.64, 1); {getModalStyle()}"
|
||||
onclick={(e) => e.stopPropagation()}
|
||||
>
|
||||
<!-- Header -->
|
||||
<div class="flex shrink-0 items-center justify-between border-b border-gray-100 px-6 py-4">
|
||||
<h2
|
||||
id="modal-confirm-title"
|
||||
class="flex items-center gap-2 text-xl font-bold {isDestructive
|
||||
? 'text-red-600'
|
||||
: 'text-gray-900'}"
|
||||
>
|
||||
{#if isDestructive}
|
||||
<AlertTriangle class="h-6 w-6" strokeWidth={2.5} />
|
||||
{/if}
|
||||
{title}
|
||||
</h2>
|
||||
<button
|
||||
type="button"
|
||||
class="rounded-full p-1 text-gray-400 hover:bg-gray-100 hover:text-gray-600"
|
||||
onclick={handleClose}
|
||||
aria-label="Fechar"
|
||||
>
|
||||
<X class="h-5 w-5" />
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<!-- Content -->
|
||||
<div class="flex-1 overflow-y-auto px-6 py-6">
|
||||
<p class="text-base leading-relaxed font-medium text-gray-700">{message}</p>
|
||||
</div>
|
||||
|
||||
<!-- Footer -->
|
||||
<div class="flex shrink-0 justify-end gap-3 border-t border-gray-100 bg-gray-50 px-6 py-4">
|
||||
<button
|
||||
class="rounded-lg border border-gray-300 bg-white px-4 py-2 text-sm font-medium text-gray-700 shadow-sm hover:bg-gray-200"
|
||||
onclick={handleClose}
|
||||
>
|
||||
{cancelText}
|
||||
</button>
|
||||
<button
|
||||
class="rounded-lg px-4 py-2 text-sm font-medium text-white shadow-sm {isDestructive
|
||||
? 'bg-red-600 hover:bg-red-700 focus:ring-red-500'
|
||||
: 'bg-blue-600 hover:bg-blue-700 focus:ring-blue-500'}"
|
||||
onclick={handleConfirm}
|
||||
>
|
||||
{confirmText}
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{/if}
|
||||
|
||||
<style>
|
||||
@keyframes fadeIn {
|
||||
from {
|
||||
opacity: 0;
|
||||
}
|
||||
to {
|
||||
opacity: 1;
|
||||
}
|
||||
}
|
||||
|
||||
@keyframes slideUp {
|
||||
from {
|
||||
opacity: 0;
|
||||
transform: translateY(20px) scale(0.95);
|
||||
}
|
||||
to {
|
||||
opacity: 1;
|
||||
transform: translateY(0) scale(1);
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -3,6 +3,8 @@
|
||||
import type { Doc, Id } from '@sgse-app/backend/convex/_generated/dataModel';
|
||||
import { useConvexClient, useQuery } from 'convex-svelte';
|
||||
import { SvelteSet } from 'svelte/reactivity';
|
||||
import { toast } from 'svelte-sonner';
|
||||
import ConfirmationModal from '$lib/components/ConfirmationModal.svelte';
|
||||
import {
|
||||
AlertTriangle,
|
||||
CheckCircle,
|
||||
@@ -156,6 +158,43 @@
|
||||
let showDetailsModal = $state(false);
|
||||
let selectedObjeto = $state<Doc<'objetos'> | null>(null);
|
||||
|
||||
// Confirmation Modal State
|
||||
let confirmModal = $state({
|
||||
open: false,
|
||||
title: '',
|
||||
message: '',
|
||||
confirmText: 'Confirmar',
|
||||
cancelText: 'Cancelar',
|
||||
isDestructive: false,
|
||||
onConfirm: async () => {}
|
||||
});
|
||||
|
||||
function openConfirm(
|
||||
title: string,
|
||||
message: string,
|
||||
onConfirm: () => Promise<void> | void,
|
||||
options: {
|
||||
confirmText?: string;
|
||||
cancelText?: string;
|
||||
isDestructive?: boolean;
|
||||
} = {}
|
||||
) {
|
||||
confirmModal.title = title;
|
||||
confirmModal.message = message;
|
||||
confirmModal.confirmText = options.confirmText || 'Confirmar';
|
||||
confirmModal.cancelText = options.cancelText || 'Cancelar';
|
||||
confirmModal.isDestructive = options.isDestructive || false;
|
||||
confirmModal.onConfirm = async () => {
|
||||
try {
|
||||
await onConfirm();
|
||||
} catch (e) {
|
||||
console.error('Error in confirmation action:', e);
|
||||
toast.error('Erro ao executar ação: ' + (e as Error).message);
|
||||
}
|
||||
};
|
||||
confirmModal.open = true;
|
||||
}
|
||||
|
||||
function openDetails(objetoId: string) {
|
||||
const obj = objetos.find((o) => o._id === objetoId);
|
||||
if (obj) {
|
||||
@@ -232,11 +271,13 @@
|
||||
ataId: ''
|
||||
};
|
||||
showAddItem = false;
|
||||
if (pedido.status === 'em_analise') {
|
||||
alert('Solicitação de adição enviada para análise.');
|
||||
if (pedido.status === 'em_analise' || pedido.status === 'aguardando_aceite') {
|
||||
toast.success('Solicitação de adição enviada para análise.');
|
||||
} else {
|
||||
toast.success('Item adicionado com sucesso!');
|
||||
}
|
||||
} catch (e) {
|
||||
alert('Erro ao adicionar item: ' + (e as Error).message);
|
||||
toast.error('Erro ao adicionar item: ' + (e as Error).message);
|
||||
} finally {
|
||||
addingItem = false;
|
||||
}
|
||||
@@ -246,7 +287,7 @@
|
||||
if (!pedido) return;
|
||||
|
||||
if (novaQuantidade < 1) {
|
||||
alert('Quantidade deve ser pelo menos 1.');
|
||||
toast.error('Quantidade deve ser pelo menos 1.');
|
||||
return;
|
||||
}
|
||||
try {
|
||||
@@ -254,61 +295,79 @@
|
||||
itemId,
|
||||
novaQuantidade
|
||||
});
|
||||
if (pedido.status === 'em_analise') {
|
||||
alert('Solicitação de alteração de quantidade enviada para análise.');
|
||||
if (pedido.status === 'em_analise' || pedido.status === 'aguardando_aceite') {
|
||||
toast.success('Solicitação de alteração de quantidade enviada para análise.');
|
||||
} else {
|
||||
toast.success('Quantidade atualizada com sucesso!');
|
||||
}
|
||||
} catch (e) {
|
||||
alert('Erro ao atualizar quantidade: ' + (e as Error).message);
|
||||
toast.error('Erro ao atualizar quantidade: ' + (e as Error).message);
|
||||
}
|
||||
}
|
||||
|
||||
async function handleRemoveItem(itemId: Id<'objetoItems'>) {
|
||||
function handleRemoveItem(itemId: Id<'objetoItems'>) {
|
||||
if (!pedido) return;
|
||||
if (!confirm('Remover este item?')) return;
|
||||
try {
|
||||
await client.mutation(api.pedidos.removeItem, { itemId });
|
||||
if (pedido.status === 'em_analise') {
|
||||
alert('Solicitação de remoção enviada para análise.');
|
||||
}
|
||||
} catch (e) {
|
||||
alert('Erro ao remover item: ' + (e as Error).message);
|
||||
}
|
||||
openConfirm(
|
||||
'Remover Item',
|
||||
'Tem certeza que deseja remover este item do pedido?',
|
||||
async () => {
|
||||
await client.mutation(api.pedidos.removeItem, { itemId });
|
||||
if (pedido.status === 'em_analise' || pedido.status === 'aguardando_aceite') {
|
||||
toast.success('Solicitação de remoção enviada para análise.');
|
||||
} else {
|
||||
toast.success('Item removido com sucesso!');
|
||||
}
|
||||
},
|
||||
{ isDestructive: true, confirmText: 'Remover' }
|
||||
);
|
||||
}
|
||||
|
||||
async function handleEnviarParaAceite() {
|
||||
if (!confirm('Enviar para aceite?')) return;
|
||||
try {
|
||||
await client.mutation(api.pedidos.enviarParaAceite, { pedidoId });
|
||||
} catch (e) {
|
||||
alert('Erro: ' + (e as Error).message);
|
||||
}
|
||||
function handleEnviarParaAceite() {
|
||||
openConfirm(
|
||||
'Enviar para Aceite',
|
||||
'Tem certeza que deseja enviar este pedido para aceite? Ele ficará aguardando análise do setor responsável.',
|
||||
async () => {
|
||||
await client.mutation(api.pedidos.enviarParaAceite, { pedidoId });
|
||||
toast.success('Pedido enviado para aceite com sucesso!');
|
||||
},
|
||||
{ confirmText: 'Enviar' }
|
||||
);
|
||||
}
|
||||
|
||||
async function handleIniciarAnalise() {
|
||||
if (!confirm('Iniciar análise?')) return;
|
||||
try {
|
||||
await client.mutation(api.pedidos.iniciarAnalise, { pedidoId });
|
||||
} catch (e) {
|
||||
alert('Erro: ' + (e as Error).message);
|
||||
}
|
||||
function handleIniciarAnalise() {
|
||||
openConfirm(
|
||||
'Iniciar Análise',
|
||||
'Tem certeza que deseja iniciar a análise deste pedido? O status será alterado para "Em Análise".',
|
||||
async () => {
|
||||
await client.mutation(api.pedidos.iniciarAnalise, { pedidoId });
|
||||
toast.success('Análise iniciada com sucesso!');
|
||||
},
|
||||
{ confirmText: 'Iniciar Análise' }
|
||||
);
|
||||
}
|
||||
|
||||
async function handleConcluir() {
|
||||
if (!confirm('Concluir pedido?')) return;
|
||||
try {
|
||||
await client.mutation(api.pedidos.concluirPedido, { pedidoId });
|
||||
} catch (e) {
|
||||
alert('Erro: ' + (e as Error).message);
|
||||
}
|
||||
function handleConcluir() {
|
||||
openConfirm(
|
||||
'Concluir Pedido',
|
||||
'Tem certeza que deseja concluir este pedido? Esta ação não pode ser desfeita.',
|
||||
async () => {
|
||||
await client.mutation(api.pedidos.concluirPedido, { pedidoId });
|
||||
toast.success('Pedido concluído com sucesso!');
|
||||
},
|
||||
{ confirmText: 'Concluir Pedido' }
|
||||
);
|
||||
}
|
||||
|
||||
async function handleCancelar() {
|
||||
if (!confirm('Cancelar pedido?')) return;
|
||||
try {
|
||||
await client.mutation(api.pedidos.cancelarPedido, { pedidoId });
|
||||
} catch (e) {
|
||||
alert('Erro: ' + (e as Error).message);
|
||||
}
|
||||
function handleCancelar() {
|
||||
openConfirm(
|
||||
'Cancelar Pedido',
|
||||
'Tem certeza que deseja cancelar este pedido? Esta ação não pode ser desfeita e o pedido ficará inutilizável.',
|
||||
async () => {
|
||||
await client.mutation(api.pedidos.cancelarPedido, { pedidoId });
|
||||
toast.success('Pedido cancelado com sucesso!');
|
||||
},
|
||||
{ isDestructive: true, confirmText: 'Cancelar Pedido' }
|
||||
);
|
||||
}
|
||||
|
||||
function getObjetoName(id: string) {
|
||||
@@ -411,8 +470,14 @@
|
||||
acaoId: current.acaoId ? (current.acaoId as Id<'acoes'>) : undefined,
|
||||
ataId: current.ataId ? (current.ataId as Id<'atas'>) : undefined
|
||||
});
|
||||
delete editingItems[item._id]; // Clear editing state
|
||||
if (pedido?.status === 'em_analise' || pedido?.status === 'aguardando_aceite') {
|
||||
toast.success('Solicitação de alteração enviada para análise.');
|
||||
} else {
|
||||
toast.success('Item atualizado com sucesso!');
|
||||
}
|
||||
} catch (e) {
|
||||
alert('Erro ao atualizar item: ' + (e as Error).message);
|
||||
toast.error('Erro ao atualizar item: ' + (e as Error).message);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -437,18 +502,19 @@
|
||||
|
||||
async function handleUpdateSei() {
|
||||
if (!seiValue.trim()) {
|
||||
alert('O número SEI não pode estar vazio.');
|
||||
toast.error('O número SEI não pode estar vazio.');
|
||||
return;
|
||||
}
|
||||
updatingSei = true;
|
||||
try {
|
||||
await client.mutation(api.pedidos.updateSeiNumber, {
|
||||
pedidoId,
|
||||
numeroSei: seiValue.trim()
|
||||
numeroSei: seiValue
|
||||
});
|
||||
editingSei = false;
|
||||
toast.success('Número SEI atualizado com sucesso!');
|
||||
} catch (e) {
|
||||
alert('Erro ao atualizar número SEI: ' + (e as Error).message);
|
||||
toast.error('Erro ao atualizar número SEI: ' + (e as Error).message);
|
||||
} finally {
|
||||
updatingSei = false;
|
||||
}
|
||||
@@ -553,26 +619,22 @@
|
||||
}
|
||||
}
|
||||
|
||||
async function handleRemoveSelectedItems() {
|
||||
function handleRemoveSelectedItems() {
|
||||
if (!hasSelection) return;
|
||||
if (
|
||||
!confirm(
|
||||
selectedCount === 1
|
||||
? 'Remover o item selecionado deste pedido?'
|
||||
: `Remover os ${selectedCount} itens selecionados deste pedido?`
|
||||
)
|
||||
)
|
||||
return;
|
||||
|
||||
try {
|
||||
const itemIds = Array.from(selectedItemIds) as Id<'objetoItems'>[];
|
||||
await client.mutation(api.pedidos.removeItemsBatch, {
|
||||
itemIds
|
||||
});
|
||||
clearSelection();
|
||||
} catch (e) {
|
||||
alert('Erro ao remover itens selecionados: ' + (e as Error).message);
|
||||
}
|
||||
openConfirm(
|
||||
'Remover Itens Selecionados',
|
||||
selectedCount === 1
|
||||
? 'Tem certeza que deseja remover o item selecionado deste pedido?'
|
||||
: `Tem certeza que deseja remover os ${selectedCount} itens selecionados deste pedido?`,
|
||||
async () => {
|
||||
await client.mutation(api.pedidos.removeItemsBatch, {
|
||||
itemIds: Array.from(selectedItemIds) as Id<'objetoItems'>[]
|
||||
});
|
||||
clearSelection();
|
||||
toast.success('Itens removidos com sucesso!');
|
||||
},
|
||||
{ isDestructive: true, confirmText: 'Remover Selecionados' }
|
||||
);
|
||||
}
|
||||
|
||||
let showSplitResultModal = $state(false);
|
||||
@@ -603,8 +665,9 @@
|
||||
showSplitConfirmationModal = false;
|
||||
showSplitResultModal = true;
|
||||
clearSelection();
|
||||
toast.success('Pedido dividido com sucesso!');
|
||||
} catch (e) {
|
||||
alert('Erro ao dividir pedido: ' + (e as Error).message);
|
||||
toast.error('Erro ao dividir pedido: ' + (e as Error).message);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -619,7 +682,7 @@
|
||||
|
||||
async function confirmRequestAdjustments() {
|
||||
if (!adjustmentDescription.trim()) {
|
||||
alert('Por favor, informe a descrição dos ajustes necessários.');
|
||||
toast.error('Por favor, informe a descrição dos ajustes necessários.');
|
||||
return;
|
||||
}
|
||||
try {
|
||||
@@ -628,36 +691,98 @@
|
||||
descricao: adjustmentDescription
|
||||
});
|
||||
showRequestAdjustmentsModal = false;
|
||||
toast.success('Solicitação de ajustes enviada com sucesso!');
|
||||
} catch (e) {
|
||||
alert('Erro: ' + (e as Error).message);
|
||||
toast.error('Erro ao solicitar ajustes: ' + (e as Error).message);
|
||||
}
|
||||
}
|
||||
|
||||
async function handleConcluirAjustes() {
|
||||
if (!confirm('Concluir ajustes e retornar para análise?')) return;
|
||||
try {
|
||||
await client.mutation(api.pedidos.concluirAjustes, { pedidoId });
|
||||
} catch (e) {
|
||||
alert('Erro: ' + (e as Error).message);
|
||||
}
|
||||
function handleConcluirAjustes() {
|
||||
openConfirm(
|
||||
'Concluir Ajustes',
|
||||
'Tem certeza que deseja concluir os ajustes e retornar o pedido para análise?',
|
||||
async () => {
|
||||
await client.mutation(api.pedidos.concluirAjustes, { pedidoId });
|
||||
toast.success('Ajustes concluídos com sucesso!');
|
||||
},
|
||||
{ confirmText: 'Concluir Ajustes' }
|
||||
);
|
||||
}
|
||||
|
||||
async function handleApproveRequest(requestId: Id<'solicitacoesItens'>) {
|
||||
if (!confirm('Aprovar esta solicitação?')) return;
|
||||
try {
|
||||
await client.mutation(api.pedidos.approveItemRequest, { requestId });
|
||||
} catch (e) {
|
||||
alert('Erro ao aprovar: ' + (e as Error).message);
|
||||
}
|
||||
function handleApproveRequest(requestId: Id<'solicitacoesItens'>) {
|
||||
openConfirm(
|
||||
'Aprovar Solicitação',
|
||||
'Tem certeza que deseja aprovar esta solicitação?',
|
||||
async () => {
|
||||
await client.mutation(api.pedidos.approveItemRequest, { requestId });
|
||||
toast.success('Solicitação aprovada com sucesso!');
|
||||
},
|
||||
{ confirmText: 'Aprovar' }
|
||||
);
|
||||
}
|
||||
|
||||
async function handleRejectRequest(requestId: Id<'solicitacoesItens'>) {
|
||||
if (!confirm('Rejeitar esta solicitação?')) return;
|
||||
try {
|
||||
await client.mutation(api.pedidos.rejectItemRequest, { requestId });
|
||||
} catch (e) {
|
||||
alert('Erro ao rejeitar: ' + (e as Error).message);
|
||||
function handleRejectRequest(requestId: Id<'solicitacoesItens'>) {
|
||||
openConfirm(
|
||||
'Rejeitar Solicitação',
|
||||
'Tem certeza que deseja rejeitar esta solicitação?',
|
||||
async () => {
|
||||
await client.mutation(api.pedidos.rejectItemRequest, { requestId });
|
||||
toast.success('Solicitação rejeitada com sucesso!');
|
||||
},
|
||||
{ isDestructive: true, confirmText: 'Rejeitar' }
|
||||
);
|
||||
}
|
||||
|
||||
function describeChangedDetails(data: unknown): string {
|
||||
if (!data || typeof data !== 'object') {
|
||||
return 'Alteração de detalhes do item';
|
||||
}
|
||||
|
||||
const { de, para } = data as {
|
||||
de?: {
|
||||
valorEstimado?: string;
|
||||
modalidade?: Modalidade;
|
||||
acaoId?: string | null;
|
||||
ataId?: string | null;
|
||||
};
|
||||
para?: {
|
||||
valorEstimado?: string;
|
||||
modalidade?: Modalidade;
|
||||
acaoId?: string | null;
|
||||
ataId?: string | null;
|
||||
};
|
||||
};
|
||||
|
||||
if (!de || !para) {
|
||||
return 'Alteração de detalhes do item';
|
||||
}
|
||||
|
||||
const changed: string[] = [];
|
||||
|
||||
if (de.valorEstimado !== para.valorEstimado) {
|
||||
changed.push('valor estimado');
|
||||
}
|
||||
if (de.modalidade !== para.modalidade) {
|
||||
changed.push('modalidade');
|
||||
}
|
||||
if ((de.acaoId ?? null) !== (para.acaoId ?? null)) {
|
||||
changed.push('ação');
|
||||
}
|
||||
if ((de.ataId ?? null) !== (para.ataId ?? null)) {
|
||||
changed.push('ata');
|
||||
}
|
||||
|
||||
if (changed.length === 0) {
|
||||
return 'Alteração de detalhes do item';
|
||||
}
|
||||
|
||||
if (changed.length === 1) {
|
||||
return `Alteração de ${changed[0]}`;
|
||||
}
|
||||
|
||||
const last = changed.pop()!;
|
||||
const prefix = changed.join(', ');
|
||||
return `Alteração de ${prefix} e ${last}`;
|
||||
}
|
||||
|
||||
function parseRequestData(json: string) {
|
||||
@@ -824,6 +949,8 @@
|
||||
<span class="text-blue-600">Alteração Qtd</span>
|
||||
{:else if req.tipo === 'exclusao'}
|
||||
<span class="text-red-600">Exclusão</span>
|
||||
{:else if req.tipo === 'alteracao_detalhes'}
|
||||
<span class="text-purple-600">{describeChangedDetails(data)}</span>
|
||||
{/if}
|
||||
</td>
|
||||
<td class="px-4 py-2">{req.solicitadoPorNome}</td>
|
||||
@@ -844,6 +971,14 @@
|
||||
{:else}
|
||||
Remover Item
|
||||
{/if}
|
||||
{:else if req.tipo === 'alteracao_detalhes'}
|
||||
{#if data.itemId}
|
||||
{@const item = items.find((i) => i._id === data.itemId)}
|
||||
Alterar detalhes de:
|
||||
{item ? getObjetoName(item.objetoId) : 'Item desconhecido'}
|
||||
{:else}
|
||||
Alteração de detalhes do item
|
||||
{/if}
|
||||
{/if}
|
||||
</td>
|
||||
<td class="px-4 py-2 text-right">
|
||||
@@ -879,7 +1014,7 @@
|
||||
<div class="mb-6 overflow-hidden rounded-lg bg-white shadow-md">
|
||||
<div class="flex items-center justify-between border-b border-gray-200 px-6 py-4">
|
||||
<h2 class="text-lg font-semibold">Itens do Pedido</h2>
|
||||
{#if pedido.status === 'em_rascunho' || pedido.status === 'precisa_ajustes' || pedido.status === 'em_analise'}
|
||||
{#if pedido.status === 'em_rascunho' || pedido.status === 'precisa_ajustes' || pedido.status === 'em_analise' || pedido.status === 'aguardando_aceite'}
|
||||
<button
|
||||
onclick={() => (showAddItem = true)}
|
||||
class="flex items-center gap-1 text-sm font-medium text-blue-600 hover:text-blue-800"
|
||||
@@ -1121,7 +1256,7 @@
|
||||
{/if}
|
||||
<td class="px-6 py-4 whitespace-nowrap">{getObjetoName(item.objetoId)}</td>
|
||||
<td class="px-6 py-4 whitespace-nowrap">
|
||||
{#if pedido.status === 'em_rascunho' || pedido.status === 'precisa_ajustes' || pedido.status === 'em_analise'}
|
||||
{#if pedido.status === 'em_rascunho' || pedido.status === 'precisa_ajustes' || pedido.status === 'em_analise' || pedido.status === 'aguardando_aceite'}
|
||||
<input
|
||||
type="number"
|
||||
min="1"
|
||||
@@ -1135,7 +1270,7 @@
|
||||
{/if}
|
||||
</td>
|
||||
<td class="px-6 py-4 whitespace-nowrap">
|
||||
{#if pedido.status === 'em_rascunho' || pedido.status === 'precisa_ajustes'}
|
||||
{#if pedido.status === 'em_rascunho' || pedido.status === 'precisa_ajustes' || pedido.status === 'em_analise' || pedido.status === 'aguardando_aceite'}
|
||||
<input
|
||||
type="text"
|
||||
class="w-28 rounded border px-2 py-1 text-sm"
|
||||
@@ -1154,7 +1289,7 @@
|
||||
{/if}
|
||||
</td>
|
||||
<td class="px-6 py-4 text-sm whitespace-nowrap text-gray-600">
|
||||
{#if pedido.status === 'em_rascunho' || pedido.status === 'precisa_ajustes'}
|
||||
{#if pedido.status === 'em_rascunho' || pedido.status === 'precisa_ajustes' || pedido.status === 'em_analise' || pedido.status === 'aguardando_aceite'}
|
||||
<select
|
||||
class="rounded border px-2 py-1 text-xs"
|
||||
value={ensureEditingItem(item).modalidade}
|
||||
@@ -1177,7 +1312,7 @@
|
||||
{/if}
|
||||
</td>
|
||||
<td class="px-6 py-4 text-sm whitespace-nowrap text-gray-600">
|
||||
{#if pedido.status === 'em_rascunho' || pedido.status === 'precisa_ajustes'}
|
||||
{#if pedido.status === 'em_rascunho' || pedido.status === 'precisa_ajustes' || pedido.status === 'em_analise' || pedido.status === 'aguardando_aceite'}
|
||||
<select
|
||||
class="rounded border px-2 py-1 text-xs"
|
||||
value={ensureEditingItem(item).acaoId}
|
||||
@@ -1196,7 +1331,7 @@
|
||||
{/if}
|
||||
</td>
|
||||
<td class="px-6 py-4 text-sm whitespace-nowrap text-gray-600">
|
||||
{#if pedido.status === 'em_rascunho' || pedido.status === 'precisa_ajustes'}
|
||||
{#if pedido.status === 'em_rascunho' || pedido.status === 'precisa_ajustes' || pedido.status === 'em_analise' || pedido.status === 'aguardando_aceite'}
|
||||
<select
|
||||
class="rounded border px-2 py-1 text-xs"
|
||||
value={ensureEditingItem(item).ataId}
|
||||
@@ -1233,7 +1368,7 @@
|
||||
>
|
||||
<Eye size={18} />
|
||||
</button>
|
||||
{#if pedido.status === 'em_rascunho' || pedido.status === 'precisa_ajustes' || pedido.status === 'em_analise'}
|
||||
{#if pedido.status === 'em_rascunho' || pedido.status === 'precisa_ajustes' || pedido.status === 'em_analise' || pedido.status === 'aguardando_aceite'}
|
||||
<button
|
||||
onclick={() => handleRemoveItem(item._id)}
|
||||
class="text-red-600 hover:text-red-900"
|
||||
@@ -1512,4 +1647,14 @@
|
||||
</div>
|
||||
</div>
|
||||
{/if}
|
||||
|
||||
<ConfirmationModal
|
||||
bind:open={confirmModal.open}
|
||||
title={confirmModal.title}
|
||||
message={confirmModal.message}
|
||||
confirmText={confirmModal.confirmText}
|
||||
isDestructive={confirmModal.isDestructive}
|
||||
onConfirm={confirmModal.onConfirm}
|
||||
onClose={() => (confirmModal.open = false)}
|
||||
/>
|
||||
</div>
|
||||
|
||||
Reference in New Issue
Block a user