- Add /cfdi/emisores and /cfdi/receptores API endpoints - Search by RFC or nombre with ILIKE - Show suggestions dropdown while typing (min 2 chars) - Click suggestion to select and populate filter input - Show loading state while searching Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
118 lines
3.5 KiB
TypeScript
118 lines
3.5 KiB
TypeScript
import { apiClient } from './client';
|
|
import type { CfdiListResponse, CfdiFilters, Cfdi } from '@horux/shared';
|
|
|
|
export async function getCfdis(filters: CfdiFilters): Promise<CfdiListResponse> {
|
|
const params = new URLSearchParams();
|
|
|
|
if (filters.tipo) params.set('tipo', filters.tipo);
|
|
if (filters.estado) params.set('estado', filters.estado);
|
|
if (filters.fechaInicio) params.set('fechaInicio', filters.fechaInicio);
|
|
if (filters.fechaFin) params.set('fechaFin', filters.fechaFin);
|
|
if (filters.rfc) params.set('rfc', filters.rfc);
|
|
if (filters.emisor) params.set('emisor', filters.emisor);
|
|
if (filters.receptor) params.set('receptor', filters.receptor);
|
|
if (filters.search) params.set('search', filters.search);
|
|
if (filters.page) params.set('page', filters.page.toString());
|
|
if (filters.limit) params.set('limit', filters.limit.toString());
|
|
|
|
const response = await apiClient.get<CfdiListResponse>(`/cfdi?${params}`);
|
|
return response.data;
|
|
}
|
|
|
|
export async function getCfdiById(id: string): Promise<Cfdi> {
|
|
const response = await apiClient.get<Cfdi>(`/cfdi/${id}`);
|
|
return response.data;
|
|
}
|
|
|
|
export async function getResumenCfdi(año?: number, mes?: number) {
|
|
const params = new URLSearchParams();
|
|
if (año) params.set('año', año.toString());
|
|
if (mes) params.set('mes', mes.toString());
|
|
|
|
const response = await apiClient.get(`/cfdi/resumen?${params}`);
|
|
return response.data;
|
|
}
|
|
|
|
export interface CreateCfdiData {
|
|
uuidFiscal: string;
|
|
tipo: 'ingreso' | 'egreso' | 'traslado' | 'nomina' | 'pago';
|
|
serie?: string;
|
|
folio?: string;
|
|
fechaEmision: string;
|
|
fechaTimbrado: string;
|
|
rfcEmisor: string;
|
|
nombreEmisor: string;
|
|
rfcReceptor: string;
|
|
nombreReceptor: string;
|
|
subtotal: number;
|
|
descuento?: number;
|
|
iva?: number;
|
|
isrRetenido?: number;
|
|
ivaRetenido?: number;
|
|
total: number;
|
|
moneda?: string;
|
|
tipoCambio?: number;
|
|
metodoPago?: string;
|
|
formaPago?: string;
|
|
usoCfdi?: string;
|
|
estado?: string;
|
|
}
|
|
|
|
export async function createCfdi(data: CreateCfdiData): Promise<Cfdi> {
|
|
const response = await apiClient.post<Cfdi>('/cfdi', data);
|
|
return response.data;
|
|
}
|
|
|
|
export interface BatchUploadResult {
|
|
message: string;
|
|
batchNumber: number;
|
|
totalBatches: number;
|
|
inserted: number;
|
|
duplicates: number;
|
|
errors: number;
|
|
errorMessages?: string[];
|
|
}
|
|
|
|
export async function createManyCfdis(
|
|
cfdis: CreateCfdiData[],
|
|
batchNumber?: number,
|
|
totalBatches?: number,
|
|
totalFiles?: number
|
|
): Promise<BatchUploadResult> {
|
|
const response = await apiClient.post<BatchUploadResult>('/cfdi/bulk', {
|
|
cfdis,
|
|
batchNumber: batchNumber || 1,
|
|
totalBatches: totalBatches || 1,
|
|
totalFiles: totalFiles || cfdis.length
|
|
});
|
|
return response.data;
|
|
}
|
|
|
|
export async function deleteCfdi(id: string): Promise<void> {
|
|
await apiClient.delete(`/cfdi/${id}`);
|
|
}
|
|
|
|
export async function getCfdiXml(id: string): Promise<string> {
|
|
const response = await apiClient.get<string>(`/cfdi/${id}/xml`, {
|
|
responseType: 'text'
|
|
});
|
|
return response.data;
|
|
}
|
|
|
|
export interface EmisorReceptor {
|
|
rfc: string;
|
|
nombre: string;
|
|
}
|
|
|
|
export async function searchEmisores(search: string): Promise<EmisorReceptor[]> {
|
|
if (search.length < 2) return [];
|
|
const response = await apiClient.get<EmisorReceptor[]>(`/cfdi/emisores?search=${encodeURIComponent(search)}`);
|
|
return response.data;
|
|
}
|
|
|
|
export async function searchReceptores(search: string): Promise<EmisorReceptor[]> {
|
|
if (search.length < 2) return [];
|
|
const response = await apiClient.get<EmisorReceptor[]>(`/cfdi/receptores?search=${encodeURIComponent(search)}`);
|
|
return response.data;
|
|
}
|