🚀 Initial commit: Versão atual do TrackSteel APP
This commit is contained in:
746
src/components/apontamento/ApontamentoFormCore.tsx
Normal file
746
src/components/apontamento/ApontamentoFormCore.tsx
Normal file
@@ -0,0 +1,746 @@
|
||||
import React, { useState, useEffect, useMemo, useCallback } from 'react';
|
||||
import { Button } from '@/components/ui/button';
|
||||
import { Input } from '@/components/ui/input';
|
||||
import { Label } from '@/components/ui/label';
|
||||
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select';
|
||||
import { Textarea } from '@/components/ui/textarea';
|
||||
import { Card, CardContent } from '@/components/ui/card';
|
||||
import { Checkbox } from '@/components/ui/checkbox';
|
||||
import { Alert, AlertDescription } from '@/components/ui/alert';
|
||||
import { Info, Package, RefreshCw, Loader2 } from 'lucide-react';
|
||||
import { SeletorPecasSimples } from './SeletorPecasSimples';
|
||||
import { toast } from 'sonner';
|
||||
import { useAuth } from '@/hooks/useAuth';
|
||||
import { supabase } from '@/integrations/supabase/client';
|
||||
|
||||
interface ItemDisponivel {
|
||||
id: string;
|
||||
marca: string;
|
||||
descricao: string;
|
||||
tipo: 'peca' | 'componente';
|
||||
quantidade_disponivel: number;
|
||||
processo_atual_permitido: number;
|
||||
}
|
||||
|
||||
interface ApontamentoFormCoreProps {
|
||||
pecas: any[];
|
||||
ofs: any[];
|
||||
componentesAgrupados: any[];
|
||||
processosOrdenados: any[];
|
||||
onCarregarItensDisponiveis: (ofNumber: string, processoId: string, filteredPecas: any[], componentesAgrupados: any[]) => Promise<{
|
||||
pecasDisponiveis: ItemDisponivel[];
|
||||
componentesDisponiveis: ItemDisponivel[];
|
||||
}>;
|
||||
onValidarSequencia: (ofNumber: string, marca: string, processoId: string, quantidade: number, fase?: string) => Promise<{ valido: boolean; erro?: string }>;
|
||||
onCriarApontamento: (data: any) => Promise<{ success: boolean; error?: any }>;
|
||||
onRefetch: () => Promise<any>;
|
||||
onInvalidateCache: (ofNumber: string, fase?: string) => void;
|
||||
loading: boolean;
|
||||
validacaoLoading: boolean;
|
||||
}
|
||||
|
||||
const getStoredCache = () => {
|
||||
try {
|
||||
const stored = localStorage.getItem('apontamento_form_cache');
|
||||
return stored ? JSON.parse(stored) : {
|
||||
of_number: '',
|
||||
fase: '',
|
||||
processo_id: '',
|
||||
data_apontamento: new Date().toISOString().split('T')[0]
|
||||
};
|
||||
} catch {
|
||||
return {
|
||||
of_number: '',
|
||||
fase: '',
|
||||
processo_id: '',
|
||||
data_apontamento: new Date().toISOString().split('T')[0]
|
||||
};
|
||||
}
|
||||
};
|
||||
|
||||
export const ApontamentoFormCore: React.FC<ApontamentoFormCoreProps> = ({
|
||||
pecas = [],
|
||||
ofs = [],
|
||||
componentesAgrupados = [],
|
||||
processosOrdenados = [],
|
||||
onCarregarItensDisponiveis,
|
||||
onValidarSequencia,
|
||||
onCriarApontamento,
|
||||
onRefetch,
|
||||
onInvalidateCache,
|
||||
loading,
|
||||
validacaoLoading
|
||||
}) => {
|
||||
const { user } = useAuth();
|
||||
|
||||
const [formData, setFormData] = useState(() => ({
|
||||
...getStoredCache(),
|
||||
quantidade_produzida: '',
|
||||
observacoes: '',
|
||||
todas_disponiveis: false
|
||||
}));
|
||||
|
||||
const [itemSelecionado, setItemSelecionado] = useState<ItemDisponivel | null>(null);
|
||||
const [itensDisponiveis, setItensDisponiveis] = useState<{
|
||||
pecasDisponiveis: ItemDisponivel[];
|
||||
componentesDisponiveis: ItemDisponivel[];
|
||||
}>({ pecasDisponiveis: [], componentesDisponiveis: [] });
|
||||
const [saving, setSaving] = useState(false);
|
||||
const [loadingItens, setLoadingItens] = useState(false);
|
||||
const [lastApontamentoId, setLastApontamentoId] = useState<string | null>(null);
|
||||
const [canUndo, setCanUndo] = useState(false);
|
||||
const [undoLoading, setUndoLoading] = useState(false);
|
||||
|
||||
const fasesDisponiveis = useMemo(() => {
|
||||
if (!pecas || !Array.isArray(pecas) || !formData.of_number) return [];
|
||||
|
||||
return pecas
|
||||
.filter(peca => peca && peca.of_number === formData.of_number)
|
||||
.map(peca => peca.etapa_fase)
|
||||
.filter((fase, index, array) => fase && array.indexOf(fase) === index)
|
||||
.sort();
|
||||
}, [pecas, formData.of_number]);
|
||||
|
||||
const filteredPecas = useMemo(() => {
|
||||
if (!pecas || !Array.isArray(pecas)) return [];
|
||||
|
||||
return pecas.filter(peca =>
|
||||
peca &&
|
||||
peca.of_number === formData.of_number &&
|
||||
peca.etapa_fase === formData.fase
|
||||
);
|
||||
}, [pecas, formData.of_number, formData.fase]);
|
||||
|
||||
const updateCache = useCallback((updates: Partial<typeof formData>) => {
|
||||
const newCache = { ...getStoredCache(), ...updates };
|
||||
localStorage.setItem('apontamento_form_cache', JSON.stringify(newCache));
|
||||
}, []);
|
||||
|
||||
const carregarItensDisponiveis = useCallback(async () => {
|
||||
if (!formData.of_number || !formData.fase || !formData.processo_id) {
|
||||
console.log('⚠️ Dados insuficientes para carregar itens:', {
|
||||
of: formData.of_number,
|
||||
fase: formData.fase,
|
||||
processo: formData.processo_id
|
||||
});
|
||||
setItensDisponiveis({ pecasDisponiveis: [], componentesDisponiveis: [] });
|
||||
return;
|
||||
}
|
||||
|
||||
if (!filteredPecas || (filteredPecas.length === 0 && (!componentesAgrupados || componentesAgrupados.length === 0))) {
|
||||
console.log('⚠️ Sem peças ou componentes para processar');
|
||||
setItensDisponiveis({ pecasDisponiveis: [], componentesDisponiveis: [] });
|
||||
return;
|
||||
}
|
||||
|
||||
setLoadingItens(true);
|
||||
console.log('🔄 Carregando itens disponíveis...', {
|
||||
of: formData.of_number,
|
||||
fase: formData.fase,
|
||||
processo: formData.processo_id,
|
||||
pecasCount: filteredPecas.length,
|
||||
componentesCount: componentesAgrupados?.length || 0
|
||||
});
|
||||
|
||||
try {
|
||||
const itens = await onCarregarItensDisponiveis(
|
||||
formData.of_number,
|
||||
formData.processo_id,
|
||||
filteredPecas || [],
|
||||
componentesAgrupados || []
|
||||
);
|
||||
|
||||
console.log('✅ Itens carregados:', {
|
||||
pecasDisponiveis: itens?.pecasDisponiveis?.length || 0,
|
||||
componentesDisponiveis: itens?.componentesDisponiveis?.length || 0
|
||||
});
|
||||
|
||||
setItensDisponiveis(itens || { pecasDisponiveis: [], componentesDisponiveis: [] });
|
||||
} catch (error) {
|
||||
console.error('❌ Erro ao carregar itens:', error);
|
||||
setItensDisponiveis({ pecasDisponiveis: [], componentesDisponiveis: [] });
|
||||
} finally {
|
||||
setLoadingItens(false);
|
||||
}
|
||||
}, [
|
||||
formData.of_number,
|
||||
formData.fase,
|
||||
formData.processo_id,
|
||||
filteredPecas,
|
||||
componentesAgrupados,
|
||||
onCarregarItensDisponiveis
|
||||
]);
|
||||
|
||||
useEffect(() => {
|
||||
const timeoutId = setTimeout(() => {
|
||||
carregarItensDisponiveis();
|
||||
}, 300);
|
||||
|
||||
return () => clearTimeout(timeoutId);
|
||||
}, [carregarItensDisponiveis]);
|
||||
|
||||
useEffect(() => {
|
||||
setItemSelecionado(null);
|
||||
setFormData(prev => ({
|
||||
...prev,
|
||||
quantidade_produzida: '',
|
||||
todas_disponiveis: false
|
||||
}));
|
||||
}, [formData.of_number, formData.fase, formData.processo_id]);
|
||||
|
||||
useEffect(() => {
|
||||
if (formData.todas_disponiveis && itemSelecionado) {
|
||||
setFormData(prev => ({
|
||||
...prev,
|
||||
quantidade_produzida: itemSelecionado.quantidade_disponivel.toString()
|
||||
}));
|
||||
}
|
||||
}, [formData.todas_disponiveis, itemSelecionado]);
|
||||
|
||||
const handleOFChange = (ofNumber: string) => {
|
||||
const updates = {
|
||||
of_number: ofNumber,
|
||||
fase: '',
|
||||
processo_id: '',
|
||||
quantidade_produzida: '',
|
||||
todas_disponiveis: false
|
||||
};
|
||||
setFormData(prev => ({ ...prev, ...updates }));
|
||||
updateCache({ of_number: ofNumber, fase: '', processo_id: '' });
|
||||
setItemSelecionado(null);
|
||||
};
|
||||
|
||||
const handleFaseChange = (fase: string) => {
|
||||
const updates = {
|
||||
fase: fase,
|
||||
processo_id: '',
|
||||
quantidade_produzida: '',
|
||||
todas_disponiveis: false
|
||||
};
|
||||
setFormData(prev => ({ ...prev, ...updates }));
|
||||
updateCache({ fase: fase, processo_id: '' });
|
||||
setItemSelecionado(null);
|
||||
};
|
||||
|
||||
const handleProcessoChange = (processoId: string) => {
|
||||
console.log('🔄 Mudança de processo detectada, limpando cache específico...');
|
||||
|
||||
// LIMPAR CACHE ESPECÍFICO ANTES DE ALTERAR O PROCESSO
|
||||
if (formData.of_number && formData.fase) {
|
||||
console.log(`🗑️ Invalidando cache para OF: ${formData.of_number}, Fase: ${formData.fase}`);
|
||||
onInvalidateCache(formData.of_number, formData.fase);
|
||||
}
|
||||
|
||||
const updates = {
|
||||
processo_id: processoId,
|
||||
quantidade_produzida: '',
|
||||
todas_disponiveis: false
|
||||
};
|
||||
setFormData(prev => ({ ...prev, ...updates }));
|
||||
updateCache({ processo_id: processoId });
|
||||
setItemSelecionado(null);
|
||||
|
||||
// Limpar itens disponíveis imediatamente para forçar nova consulta
|
||||
setItensDisponiveis({ pecasDisponiveis: [], componentesDisponiveis: [] });
|
||||
|
||||
console.log('✅ Cache limpo, nova consulta será realizada automaticamente');
|
||||
};
|
||||
|
||||
const handleItemSelect = useCallback((item: ItemDisponivel) => {
|
||||
console.log('🎯 Item selecionado:', {
|
||||
marca: item.marca,
|
||||
tipo: item.tipo,
|
||||
quantidade_disponivel: item.quantidade_disponivel
|
||||
});
|
||||
setItemSelecionado(item);
|
||||
setFormData(prev => ({
|
||||
...prev,
|
||||
quantidade_produzida: '',
|
||||
todas_disponiveis: false
|
||||
}));
|
||||
}, []);
|
||||
|
||||
const handleQuantidadeChange = (value: string) => {
|
||||
const quantidade = parseInt(value);
|
||||
|
||||
if (itemSelecionado && quantidade > itemSelecionado.quantidade_disponivel) {
|
||||
toast.error(`Quantidade não pode ser maior que ${itemSelecionado.quantidade_disponivel} unidades disponíveis`);
|
||||
return;
|
||||
}
|
||||
|
||||
setFormData(prev => ({
|
||||
...prev,
|
||||
quantidade_produzida: value,
|
||||
todas_disponiveis: false
|
||||
}));
|
||||
};
|
||||
|
||||
const handleTodasDisponiveisChange = (checked: boolean) => {
|
||||
console.log('🔄 Checkbox "Todas" alterado:', {
|
||||
checked,
|
||||
itemSelecionado: itemSelecionado?.marca,
|
||||
quantidade_disponivel: itemSelecionado?.quantidade_disponivel
|
||||
});
|
||||
|
||||
setFormData(prev => ({
|
||||
...prev,
|
||||
todas_disponiveis: checked,
|
||||
quantidade_produzida: checked && itemSelecionado ? itemSelecionado.quantidade_disponivel.toString() : prev.quantidade_produzida
|
||||
}));
|
||||
};
|
||||
|
||||
const resetFormForNewEntry = async () => {
|
||||
console.log('🔄 Resetando formulário para nova entrada...');
|
||||
setItemSelecionado(null);
|
||||
setFormData(prev => ({
|
||||
...prev,
|
||||
quantidade_produzida: '',
|
||||
observacoes: '',
|
||||
todas_disponiveis: false
|
||||
}));
|
||||
|
||||
console.log('📋 Recarregando itens disponíveis após reset...');
|
||||
await carregarItensDisponiveis();
|
||||
};
|
||||
|
||||
const handleBatchSelect = async (items: ItemDisponivel[], tipo: 'peca' | 'componente') => {
|
||||
setSaving(true);
|
||||
let sucessos = 0;
|
||||
let erros = 0;
|
||||
|
||||
try {
|
||||
for (const item of items) {
|
||||
try {
|
||||
const validacao = await onValidarSequencia(
|
||||
formData.of_number,
|
||||
item.marca,
|
||||
formData.processo_id,
|
||||
item.quantidade_disponivel,
|
||||
formData.fase
|
||||
);
|
||||
|
||||
if (!validacao.valido) {
|
||||
erros++;
|
||||
continue;
|
||||
}
|
||||
|
||||
const apontamentoData: any = {
|
||||
of_number: formData.of_number,
|
||||
tipo_apontamento: item.tipo,
|
||||
processo_id: formData.processo_id,
|
||||
quantidade_produzida: item.quantidade_disponivel,
|
||||
data_apontamento: formData.data_apontamento,
|
||||
observacoes: `Apontamento em lote - ${tipo}`
|
||||
};
|
||||
|
||||
if (item.tipo === 'componente') {
|
||||
apontamentoData.componente_id = item.id;
|
||||
} else {
|
||||
apontamentoData.peca_id = item.id;
|
||||
}
|
||||
|
||||
const result = await onCriarApontamento(apontamentoData);
|
||||
|
||||
if (result.success) {
|
||||
sucessos++;
|
||||
} else {
|
||||
erros++;
|
||||
}
|
||||
} catch (error) {
|
||||
erros++;
|
||||
}
|
||||
}
|
||||
|
||||
if (sucessos > 0) {
|
||||
toast.success(`${sucessos} apontamentos realizados com sucesso!`);
|
||||
await Promise.all([
|
||||
onRefetch(),
|
||||
resetFormForNewEntry()
|
||||
]);
|
||||
}
|
||||
|
||||
if (erros > 0) {
|
||||
toast.error(`${erros} apontamentos falharam.`);
|
||||
}
|
||||
|
||||
} catch (error) {
|
||||
toast.error('Erro ao realizar apontamento em lote');
|
||||
} finally {
|
||||
setSaving(false);
|
||||
}
|
||||
};
|
||||
|
||||
const handleSubmit = async (e: React.FormEvent) => {
|
||||
e.preventDefault();
|
||||
|
||||
if (!itemSelecionado || !formData.processo_id || !formData.quantidade_produzida) {
|
||||
toast.error('Preencha todos os campos obrigatórios');
|
||||
return;
|
||||
}
|
||||
|
||||
const quantidade = parseInt(formData.quantidade_produzida);
|
||||
|
||||
if (quantidade <= 0 || quantidade > itemSelecionado.quantidade_disponivel) {
|
||||
toast.error('Quantidade inválida');
|
||||
return;
|
||||
}
|
||||
|
||||
setSaving(true);
|
||||
|
||||
try {
|
||||
console.log('🔍 Iniciando validação de sequência:', {
|
||||
of: formData.of_number,
|
||||
marca: itemSelecionado.marca,
|
||||
processo: formData.processo_id,
|
||||
quantidade,
|
||||
metodo: formData.todas_disponiveis ? 'CHECKBOX_TODAS' : 'MANUAL'
|
||||
});
|
||||
|
||||
const validacao = await onValidarSequencia(
|
||||
formData.of_number,
|
||||
itemSelecionado.marca,
|
||||
formData.processo_id,
|
||||
quantidade,
|
||||
formData.fase
|
||||
);
|
||||
|
||||
if (!validacao.valido) {
|
||||
toast.error(validacao.erro);
|
||||
return;
|
||||
}
|
||||
|
||||
console.log('✅ Validação aprovada, criando apontamento...');
|
||||
|
||||
const apontamentoData: any = {
|
||||
of_number: formData.of_number,
|
||||
tipo_apontamento: itemSelecionado.tipo,
|
||||
processo_id: formData.processo_id,
|
||||
quantidade_produzida: quantidade,
|
||||
data_apontamento: formData.data_apontamento,
|
||||
observacoes: formData.observacoes || null
|
||||
};
|
||||
|
||||
if (itemSelecionado.tipo === 'componente') {
|
||||
apontamentoData.componente_id = itemSelecionado.id;
|
||||
} else {
|
||||
apontamentoData.peca_id = itemSelecionado.id;
|
||||
}
|
||||
|
||||
console.log('📝 Dados do apontamento sendo criado:', {
|
||||
...apontamentoData,
|
||||
metodo_utilizado: formData.todas_disponiveis ? 'CHECKBOX_TODAS' : 'DIGITACAO_MANUAL',
|
||||
quantidade_original_disponivel: itemSelecionado.quantidade_disponivel,
|
||||
quantidade_sendo_apontada: quantidade
|
||||
});
|
||||
|
||||
const result = await onCriarApontamento(apontamentoData);
|
||||
|
||||
if (result.success) {
|
||||
console.log('✅ Apontamento criado com sucesso!');
|
||||
toast.success('Apontamento registrado com sucesso!');
|
||||
|
||||
// Atualizar estado do último apontamento
|
||||
if (result.success) {
|
||||
// Para desfazer, buscar o último apontamento criado
|
||||
const { data: lastApontamento } = await supabase
|
||||
.from('apontamentos_producao')
|
||||
.select('id')
|
||||
.eq('created_by', user.id)
|
||||
.order('created_at', { ascending: false })
|
||||
.limit(1)
|
||||
.single();
|
||||
|
||||
if (lastApontamento) {
|
||||
setLastApontamentoId(lastApontamento.id);
|
||||
setCanUndo(true);
|
||||
}
|
||||
}
|
||||
|
||||
// INVALIDAR CACHE ESPECÍFICO DA OF E FASE ANTES DE RECARREGAR
|
||||
console.log('🗑️ Invalidando cache após apontamento bem-sucedido...');
|
||||
onInvalidateCache(formData.of_number, formData.fase);
|
||||
|
||||
// Aguardar atualização e reset
|
||||
await Promise.all([
|
||||
onRefetch(),
|
||||
resetFormForNewEntry()
|
||||
]);
|
||||
|
||||
console.log('✅ Formulário resetado e dados atualizados após apontamento');
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('❌ Erro ao registrar apontamento:', error);
|
||||
toast.error('Erro ao registrar apontamento');
|
||||
} finally {
|
||||
setSaving(false);
|
||||
}
|
||||
};
|
||||
|
||||
const limparCacheCompleto = () => {
|
||||
localStorage.removeItem('apontamento_form_cache');
|
||||
setFormData({
|
||||
of_number: '',
|
||||
fase: '',
|
||||
data_apontamento: new Date().toISOString().split('T')[0],
|
||||
processo_id: '',
|
||||
quantidade_produzida: '',
|
||||
observacoes: '',
|
||||
todas_disponiveis: false
|
||||
});
|
||||
setItemSelecionado(null);
|
||||
setItensDisponiveis({ pecasDisponiveis: [], componentesDisponiveis: [] });
|
||||
toast.success('Cache limpo completamente!');
|
||||
};
|
||||
|
||||
// Função para desfazer último apontamento
|
||||
const handleUndo = useCallback(async () => {
|
||||
if (!lastApontamentoId || !user?.id) return;
|
||||
|
||||
setUndoLoading(true);
|
||||
try {
|
||||
const { error } = await supabase
|
||||
.from('apontamentos_producao')
|
||||
.delete()
|
||||
.eq('id', lastApontamentoId)
|
||||
.eq('created_by', user.id);
|
||||
|
||||
if (error) {
|
||||
console.error('Erro ao desfazer apontamento:', error);
|
||||
toast.error('Erro ao desfazer apontamento');
|
||||
return;
|
||||
}
|
||||
|
||||
// Resetar estado do botão de desfazer
|
||||
setLastApontamentoId(null);
|
||||
setCanUndo(false);
|
||||
|
||||
// Recarregar itens disponíveis se necessário
|
||||
if (formData.of_number && formData.fase && formData.processo_id) {
|
||||
await carregarItensDisponiveis();
|
||||
}
|
||||
|
||||
toast.success('Apontamento desfeito com sucesso!');
|
||||
} catch (error) {
|
||||
console.error('Erro inesperado ao desfazer:', error);
|
||||
toast.error('Erro inesperado ao desfazer apontamento');
|
||||
} finally {
|
||||
setUndoLoading(false);
|
||||
}
|
||||
}, [lastApontamentoId, user?.id, supabase, formData.of_number, formData.fase, formData.processo_id, carregarItensDisponiveis]);
|
||||
|
||||
const processoSelecionado = processosOrdenados?.find(p => p.id === formData.processo_id);
|
||||
|
||||
if (loading) {
|
||||
return (
|
||||
<div className="flex items-center justify-center p-8 space-y-4">
|
||||
<Loader2 className="h-6 w-6 animate-spin" />
|
||||
<span className="text-muted-foreground">Carregando dados...</span>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<form onSubmit={handleSubmit} className="space-y-6">
|
||||
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
|
||||
<div className="space-y-4">
|
||||
{/* Cache status */}
|
||||
{(formData.of_number || formData.fase || formData.processo_id) && (
|
||||
<Alert>
|
||||
<Info className="h-4 w-4" />
|
||||
<AlertDescription className="flex items-center justify-between">
|
||||
<span>Formulário restaurado do cache</span>
|
||||
<Button
|
||||
type="button"
|
||||
variant="outline"
|
||||
size="sm"
|
||||
onClick={limparCacheCompleto}
|
||||
className="ml-2"
|
||||
>
|
||||
<RefreshCw className="h-3 w-3 mr-1" />
|
||||
Limpar Cache
|
||||
</Button>
|
||||
</AlertDescription>
|
||||
</Alert>
|
||||
)}
|
||||
|
||||
{/* Campos de seleção básicos */}
|
||||
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
|
||||
<div>
|
||||
<Label htmlFor="of">Ordem de Fabricação *</Label>
|
||||
<Select value={formData.of_number} onValueChange={handleOFChange}>
|
||||
<SelectTrigger>
|
||||
<SelectValue placeholder="Selecione a OF" />
|
||||
</SelectTrigger>
|
||||
<SelectContent>
|
||||
{(ofs || []).map((of) => (
|
||||
<SelectItem key={of.id} value={of.num_of}>
|
||||
{of.num_of} - {of.descritivo}
|
||||
</SelectItem>
|
||||
))}
|
||||
</SelectContent>
|
||||
</Select>
|
||||
</div>
|
||||
|
||||
{formData.of_number && (
|
||||
<div>
|
||||
<Label htmlFor="fase">Fase *</Label>
|
||||
<Select value={formData.fase} onValueChange={handleFaseChange}>
|
||||
<SelectTrigger>
|
||||
<SelectValue placeholder="Selecione a fase" />
|
||||
</SelectTrigger>
|
||||
<SelectContent>
|
||||
{fasesDisponiveis.map((fase) => (
|
||||
<SelectItem key={fase} value={fase}>
|
||||
{fase}
|
||||
</SelectItem>
|
||||
))}
|
||||
</SelectContent>
|
||||
</Select>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
|
||||
{formData.fase && (
|
||||
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
|
||||
<div>
|
||||
<Label htmlFor="data">Data do Apontamento *</Label>
|
||||
<Input
|
||||
id="data"
|
||||
type="date"
|
||||
value={formData.data_apontamento}
|
||||
onChange={(e) => {
|
||||
const newDate = e.target.value;
|
||||
setFormData(prev => ({ ...prev, data_apontamento: newDate }));
|
||||
updateCache({ data_apontamento: newDate });
|
||||
}}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
<Label>Processo *</Label>
|
||||
<Select value={formData.processo_id} onValueChange={handleProcessoChange}>
|
||||
<SelectTrigger>
|
||||
<SelectValue placeholder="Selecione o processo" />
|
||||
</SelectTrigger>
|
||||
<SelectContent>
|
||||
{(processosOrdenados || []).map((processo) => (
|
||||
<SelectItem key={processo.id} value={processo.id}>
|
||||
{processo.ordem}. {processo.nome}
|
||||
</SelectItem>
|
||||
))}
|
||||
</SelectContent>
|
||||
</Select>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{/* Informação sobre o processo */}
|
||||
{processoSelecionado && (
|
||||
<Alert>
|
||||
<Info className="h-4 w-4" />
|
||||
<AlertDescription>
|
||||
{processoSelecionado.ordem === 1
|
||||
? `Processo inicial: ${processoSelecionado.nome}. Todos os itens estão disponíveis.`
|
||||
: `Processo ${processoSelecionado.ordem}: ${processoSelecionado.nome}. Apenas itens que passaram pelos processos anteriores estão disponíveis.`
|
||||
}
|
||||
</AlertDescription>
|
||||
</Alert>
|
||||
)}
|
||||
|
||||
{/* Seletor de Itens */}
|
||||
{formData.processo_id && (
|
||||
<SeletorPecasSimples
|
||||
pecasDisponiveis={itensDisponiveis.pecasDisponiveis || []}
|
||||
componentesDisponiveis={itensDisponiveis.componentesDisponiveis || []}
|
||||
onItemSelect={handleItemSelect}
|
||||
onBatchSelect={handleBatchSelect}
|
||||
loading={loadingItens || validacaoLoading}
|
||||
onSubmit={() => handleSubmit({} as React.FormEvent)}
|
||||
submitDisabled={saving || !itemSelecionado || !formData.processo_id || !formData.quantidade_produzida || loadingItens}
|
||||
submitLoading={saving}
|
||||
canUndo={canUndo}
|
||||
onUndo={handleUndo}
|
||||
undoLoading={undoLoading}
|
||||
lastApontamentoId={lastApontamentoId}
|
||||
/>
|
||||
)}
|
||||
|
||||
{/* Campos de Quantidade */}
|
||||
{itemSelecionado && (
|
||||
<div className="space-y-4 p-4 bg-slate-800/50 rounded-lg border border-slate-700">
|
||||
<h4 className="font-medium text-slate-200">Quantidade a Apontar</h4>
|
||||
<div className="space-y-3">
|
||||
<Input
|
||||
type="number"
|
||||
min="1"
|
||||
max={itemSelecionado.quantidade_disponivel}
|
||||
placeholder="Digite a quantidade..."
|
||||
value={formData.quantidade_produzida}
|
||||
onChange={(e) => handleQuantidadeChange(e.target.value)}
|
||||
disabled={formData.todas_disponiveis}
|
||||
className="w-full text-lg h-12 bg-slate-800 border-slate-600"
|
||||
/>
|
||||
<div className="flex items-center space-x-2">
|
||||
<Checkbox
|
||||
id="todas-disponiveis"
|
||||
checked={formData.todas_disponiveis}
|
||||
onCheckedChange={handleTodasDisponiveisChange}
|
||||
/>
|
||||
<Label htmlFor="todas-disponiveis" className="text-sm cursor-pointer">
|
||||
Todas ({itemSelecionado.quantidade_disponivel})
|
||||
</Label>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
|
||||
<div>
|
||||
<Label htmlFor="observacoes">Observações</Label>
|
||||
<Textarea
|
||||
id="observacoes"
|
||||
placeholder="Observações sobre o apontamento..."
|
||||
value={formData.observacoes}
|
||||
onChange={(e) => setFormData(prev => ({ ...prev, observacoes: e.target.value }))}
|
||||
rows={3}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* Card de informações do item selecionado */}
|
||||
<div>
|
||||
<Card className="bg-muted/50 h-fit">
|
||||
<CardContent className="p-4">
|
||||
<h4 className="font-medium flex items-center gap-2 mb-3">
|
||||
<Package className="h-4 w-4" />
|
||||
Informações do Item Selecionado
|
||||
</h4>
|
||||
{!itemSelecionado ? (
|
||||
<div className="text-sm text-muted-foreground">
|
||||
Selecione um item para ver as informações
|
||||
</div>
|
||||
) : (
|
||||
<div className="space-y-2 text-sm">
|
||||
<div><strong>Tipo:</strong> {itemSelecionado.tipo === 'componente' ? 'Componente' : 'Peça'}</div>
|
||||
<div><strong>Marca:</strong> {itemSelecionado.marca}</div>
|
||||
<div><strong>OF:</strong> {formData.of_number}</div>
|
||||
<div><strong>Fase:</strong> {formData.fase}</div>
|
||||
<div><strong>Processo:</strong> {processoSelecionado?.nome || 'N/A'}</div>
|
||||
<div><strong>Descrição:</strong> {itemSelecionado.descricao || 'N/A'}</div>
|
||||
<div><strong>Quantidade Disponível:</strong> {itemSelecionado.quantidade_disponivel} unidades</div>
|
||||
{formData.quantidade_produzida && (
|
||||
<div className="pt-2 border-t">
|
||||
<div><strong>Quantidade a Apontar:</strong> {formData.quantidade_produzida} unidades</div>
|
||||
<div><strong>Método:</strong> {formData.todas_disponiveis ? 'Checkbox "Todas"' : 'Digitação Manual'}</div>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
</CardContent>
|
||||
</Card>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</form>
|
||||
);
|
||||
};
|
||||
Reference in New Issue
Block a user