photum/frontend/components/EventFiltersBar.tsx
NANDO9322 788e0dca70 feat: melhorias no dashboard e correções no perfil
- Implementa filtros de Empresa e Instituição no Dashboard.
- Adiciona barra de estatísticas de equipe (fotógrafos, cinegrafistas, recepcionistas) na modal de Gerenciar Equipe.
- Corrige bug de atualização da interface após editar evento (mapeamento snake_case).
- Adiciona máscaras de input (CPF/CNPJ, Telefone) na página de Perfil.
- Corrige ordenação e persistência da listagem de eventos por FOT.
- Corrige crash e corrupção de dados na página de Perfil.

fix: permite reenviar notificação de logística

- Remove bloqueio do botão de notificação de logística quando já enviada.
- Altera texto do botão para "Reenviar Notificação" quando aplicável.

feat: melhorias no dashboard, perfil e logística

- Implementa filtros de Empresa e Instituição no Dashboard.
- Adiciona barra de estatísticas de equipe na modal de Gerenciar Equipe.
- Desacopla notificação de logística da aprovação do evento (agora apenas manual).
- Permite reenviar notificação de logística e remove exibição redundante de data.
- Adiciona máscaras de input (CPF/CNPJ, Telefone) no Perfil.
- Corrige atualização da interface pós-edição de evento.
- Corrige crash, ordenação e persistência na listagem de eventos e perfil.
2026-02-08 12:54:41 -03:00

228 lines
8.1 KiB
TypeScript

import React from "react";
import { Calendar, Hash, Filter, X, Building2 } from "lucide-react";
export interface EventFilters {
date: string;
fotId: string;
type: string;
company: string;
institution: string;
}
interface EventFiltersBarProps {
filters: EventFilters;
onFilterChange: (filters: EventFilters) => void;
availableTypes: string[];
availableCompanies: string[];
availableInstitutions: string[];
}
export const EventFiltersBar: React.FC<EventFiltersBarProps> = ({
filters,
onFilterChange,
availableTypes,
availableCompanies,
availableInstitutions,
}) => {
const handleReset = () => {
onFilterChange({
date: "",
fotId: "",
type: "",
company: "",
institution: "",
});
};
const hasActiveFilters = Object.values(filters).some((value) => value !== "");
return (
<div className="bg-white rounded-lg border border-gray-200 p-4 shadow-sm">
<div className="flex items-center justify-between mb-4">
<div className="flex items-center gap-2">
<Filter size={18} className="text-brand-gold" />
<h3 className="font-semibold text-gray-800">Filtros Avançados</h3>
</div>
{hasActiveFilters && (
<button
onClick={handleReset}
className="flex items-center gap-1 text-sm text-gray-600 hover:text-brand-gold transition-colors"
>
<X size={16} />
Limpar filtros
</button>
)}
</div>
<div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-5 gap-3">
{/* Filtro por FOT */}
<div className="flex flex-col">
<label className="text-xs font-medium text-gray-600 mb-1 flex items-center gap-1">
<Hash size={14} className="text-brand-gold" />
FOT
</label>
<input
type="text"
value={filters.fotId}
onChange={(e) => {
const value = e.target.value.replace(/\D/g, '').slice(0, 5);
onFilterChange({ ...filters, fotId: value });
}}
placeholder="Buscar FOT..."
maxLength={5}
className="px-3 py-2 border border-gray-300 rounded text-sm focus:outline-none focus:border-brand-gold transition-colors"
/>
</div>
{/* Filtro por Data */}
<div className="flex flex-col">
<label className="text-xs font-medium text-gray-600 mb-1 flex items-center gap-1">
<Calendar size={14} className="text-brand-gold" />
Data
</label>
<input
type="date"
value={filters.date}
onChange={(e) =>
onFilterChange({ ...filters, date: e.target.value })
}
className="px-3 py-2 border border-gray-300 rounded text-sm focus:outline-none focus:border-brand-gold transition-colors"
/>
</div>
{/* Filtro por Tipo */}
<div className="flex flex-col">
<label className="text-xs font-medium text-gray-600 mb-1 flex items-center gap-1">
<Filter size={14} className="text-brand-gold" />
Tipo de Evento
</label>
<select
value={filters.type}
onChange={(e) =>
onFilterChange({ ...filters, type: e.target.value })
}
className="px-3 py-2 border border-gray-300 rounded text-sm focus:outline-none focus:border-brand-gold transition-colors bg-white"
>
<option value="">Todos os tipos</option>
{availableTypes.map((type) => (
<option key={type} value={type}>
{type}
</option>
))}
</select>
</div>
{/* Filtro por Empresa */}
<div className="flex flex-col">
<label className="text-xs font-medium text-gray-600 mb-1 flex items-center gap-1">
<Building2 size={14} className="text-brand-gold" />
Empresa
</label>
<select
value={filters.company}
onChange={(e) =>
onFilterChange({ ...filters, company: e.target.value })
}
className="px-3 py-2 border border-gray-300 rounded text-sm focus:outline-none focus:border-brand-gold transition-colors bg-white"
>
<option value="">Todas as empresas</option>
{availableCompanies.map((comp) => (
<option key={comp} value={comp}>
{comp}
</option>
))}
</select>
</div>
{/* Filtro por Instituição */}
<div className="flex flex-col">
<label className="text-xs font-medium text-gray-600 mb-1 flex items-center gap-1">
<Building2 size={14} className="text-brand-gold" />
Instituição
</label>
<select
value={filters.institution}
onChange={(e) =>
onFilterChange({ ...filters, institution: e.target.value })
}
className="px-3 py-2 border border-gray-300 rounded text-sm focus:outline-none focus:border-brand-gold transition-colors bg-white"
>
<option value="">Todas as instituições</option>
{availableInstitutions.map((inst) => (
<option key={inst} value={inst}>
{inst}
</option>
))}
</select>
</div>
</div>
{/* Active Filters Display */}
{hasActiveFilters && (
<div className="mt-3 pt-3 border-t border-gray-100">
<div className="flex flex-wrap gap-2">
<span className="text-xs text-gray-500">Filtros ativos:</span>
{filters.date && (
<span className="inline-flex items-center gap-1 px-2 py-1 bg-brand-gold/10 text-brand-gold text-xs rounded">
Data:{" "}
{new Date(filters.date + "T00:00:00").toLocaleDateString(
"pt-BR"
)}
<button
onClick={() => onFilterChange({ ...filters, date: "" })}
className="hover:text-brand-black"
>
<X size={12} />
</button>
</span>
)}
{filters.fotId && (
<span className="inline-flex items-center gap-1 px-2 py-1 bg-brand-gold/10 text-brand-gold text-xs rounded">
FOT: {filters.fotId}
<button
onClick={() => onFilterChange({ ...filters, fotId: "" })}
className="hover:text-brand-black"
>
<X size={12} />
</button>
</span>
)}
{filters.type && (
<span className="inline-flex items-center gap-1 px-2 py-1 bg-brand-gold/10 text-brand-gold text-xs rounded">
Tipo: {filters.type}
<button
onClick={() => onFilterChange({ ...filters, type: "" })}
className="hover:text-brand-black"
>
<X size={12} />
</button>
</span>
)}
{filters.company && (
<span className="inline-flex items-center gap-1 px-2 py-1 bg-brand-gold/10 text-brand-gold text-xs rounded">
Empresa: {filters.company}
<button
onClick={() => onFilterChange({ ...filters, company: "" })}
className="hover:text-brand-black"
>
<X size={12} />
</button>
</span>
)}
{filters.institution && (
<span className="inline-flex items-center gap-1 px-2 py-1 bg-brand-gold/10 text-brand-gold text-xs rounded">
Inst: {filters.institution}
<button
onClick={() => onFilterChange({ ...filters, institution: "" })}
className="hover:text-brand-black"
>
<X size={12} />
</button>
</span>
)}
</div>
</div>
)}
</div>
);
};