mirror of
https://github.com/marcogll/ap_pos.git
synced 2026-01-13 05:15:14 +00:00
- Added comprehensive discount detection logic in hasAnyDiscount() - Created extractDiscountInfo() to handle multiple data sources - Updated all discount rendering functions to use new extraction logic - Enhanced support for manual anticipos and applied discounts - Improved fallback detection using subtotal vs monto differences - Added Material Symbols icons to action buttons in table - Fixed discount display issues in PNG receipt generation 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
348 lines
14 KiB
JavaScript
348 lines
14 KiB
JavaScript
/**
|
|
* Escapa caracteres HTML para prevenir XSS.
|
|
* @param {string} str El string a escapar.
|
|
* @returns {string} El string escapado.
|
|
*/
|
|
function esc(str) {
|
|
return String(str || '').replace(/[&<>"']/g, c => ({
|
|
"&": "&",
|
|
"<": "<",
|
|
">": ">",
|
|
'"': '"',
|
|
"'": "'"
|
|
}[c]));
|
|
}
|
|
|
|
|
|
/**
|
|
* Genera el HTML para un ticket de movimiento.
|
|
* @param {object} mov El objeto del movimiento.
|
|
* @param {object} settings El objeto de configuración.
|
|
* @returns {string} El HTML del ticket.
|
|
*/
|
|
function templateTicket(mov, settings) {
|
|
// FUNCIÓN DE FECHA DEFINITIVA - NO MAS UNDEFINED NUNCA MAS
|
|
function generarFechaTicketFINAL() {
|
|
// Crear nueva fecha DIRECTAMENTE
|
|
const fecha = new Date();
|
|
|
|
// Obtener componentes de fecha SIN VARIABLES INTERMEDIAS
|
|
const dia = fecha.getDate().toString().padStart(2, '0');
|
|
const mes = (fecha.getMonth() + 1).toString().padStart(2, '0');
|
|
const año = fecha.getFullYear().toString();
|
|
const hora = fecha.getHours().toString().padStart(2, '0');
|
|
const minutos = fecha.getMinutes().toString().padStart(2, '0');
|
|
|
|
// Construir fecha final DIRECTAMENTE
|
|
const resultado = `${dia}/${mes}/${año} ${hora}:${minutos}`;
|
|
console.log("FECHA FINAL DEFINITIVA GENERADA:", resultado);
|
|
return resultado;
|
|
}
|
|
|
|
const montoFormateado = Number(mov.monto).toFixed(2);
|
|
const tipoServicio = mov.subtipo === 'Retoque' ? `Retoque de ${mov.tipo}` :
|
|
(mov.tipo.charAt(0).toUpperCase() + mov.tipo.slice(1).toLowerCase());
|
|
|
|
const lines = [];
|
|
lines.push('<div class="ticket">');
|
|
lines.push('<img src="src/logo.png" alt="Logo" class="t-logo">');
|
|
|
|
// Información del negocio - verificar estructura de settings
|
|
// Extraer datos desde settings o settings.settings (doble anidación)
|
|
const businessData = settings?.settings || settings || {};
|
|
|
|
const negocioNombre = businessData?.negocio || settings?.negocio || 'Ale Ponce';
|
|
const negocioTagline = businessData?.tagline || settings?.tagline || 'beauty expert';
|
|
const negocioCalle = businessData?.calle || settings?.calle;
|
|
const negocioColonia = businessData?.colonia || settings?.colonia;
|
|
const negocioCP = businessData?.cp || settings?.cp;
|
|
const negocioRFC = businessData?.rfc || settings?.rfc;
|
|
const negocioTel = businessData?.tel || settings?.tel || '8443555108';
|
|
|
|
lines.push(`<div class="t-center t-bold t-business-name">${esc(negocioNombre)}</div>`);
|
|
lines.push(`<div class="t-center t-tagline">${esc(negocioTagline)}</div>`);
|
|
lines.push('<div class="t-spacer"></div>');
|
|
if (negocioCalle) lines.push(`<div class="t-center t-small">${esc(negocioCalle)}</div>`);
|
|
if (negocioColonia && negocioCP) lines.push(`<div class="t-center t-small">${esc(negocioColonia)}, ${esc(negocioCP)}</div>`);
|
|
if (negocioRFC) lines.push(`<div class="t-center t-small">RFC: ${esc(negocioRFC)}</div>`);
|
|
lines.push(`<div class="t-center t-small">Tel: ${esc(negocioTel)}</div>`);
|
|
|
|
lines.push('<div class="t-divider"></div>');
|
|
|
|
|
|
// INFORMACIÓN COMPACTA DEL CLIENTE Y CITA
|
|
if (mov.client) lines.push(`<div class="t-left t-small"><b>Cliente:</b> ${esc(mov.client.nombre)}</div>`);
|
|
|
|
lines.push('<div class="t-spacer"></div>');
|
|
|
|
// INFO EN LÍNEAS COMPACTAS
|
|
if (mov.fechaCita && mov.horaCita) {
|
|
lines.push(`<div class="t-left t-small"><b>Cita:</b> ${esc(mov.fechaCita)} - ${esc(mov.horaCita)}</div>`);
|
|
} else {
|
|
if (mov.fechaCita) lines.push(`<div class="t-left t-small"><b>Fecha de Cita:</b> ${esc(mov.fechaCita)}</div>`);
|
|
if (mov.horaCita) lines.push(`<div class="t-left t-small"><b>Hora de Cita:</b> ${esc(mov.horaCita)}</div>`);
|
|
}
|
|
|
|
lines.push('<div class="t-spacer"></div>');
|
|
lines.push(`<div class="t-left t-small"><b>Folio:</b> ${esc(mov.folio)}</div>`);
|
|
|
|
|
|
// Fecha de Venta
|
|
const fechaMovimiento = new Date(mov.fechaISO);
|
|
const dia = fechaMovimiento.getDate().toString().padStart(2, '0');
|
|
const mes = (fechaMovimiento.getMonth() + 1).toString().padStart(2, '0');
|
|
const año = fechaMovimiento.getFullYear().toString();
|
|
const hora = fechaMovimiento.getHours().toString().padStart(2, '0');
|
|
const minutos = fechaMovimiento.getMinutes().toString().padStart(2, '0');
|
|
const fechaFinal = `${dia}/${mes}/${año} ${hora}:${minutos}`;
|
|
lines.push(`<div class="t-left t-small"><b>Fecha de Venta:</b> ${esc(fechaFinal)}</div>`);
|
|
|
|
lines.push('<div class="t-divider"></div>');
|
|
lines.push(`<div class="t-left t-service-title t-bold">${esc(tipoServicio)}</div>`);
|
|
|
|
// CONCEPTO CON PRECIO
|
|
if (mov.concepto) {
|
|
lines.push(`<div class="t-left t-small t-service-detail"><b>Concepto:</b></div>`);
|
|
lines.push(`<div class="t-row t-small"><span>${esc(mov.concepto)}</span><span>$${montoFormateado}</span></div>`);
|
|
}
|
|
|
|
// DESCUENTOS SI EXISTEN
|
|
if (mov.descuento && mov.descuento > 0) {
|
|
if (mov.discountInfo) {
|
|
// Mostrar información detallada del descuento
|
|
const discountInfo = mov.discountInfo;
|
|
let descriptionText = '';
|
|
let amountText = `-$${Number(mov.descuento).toFixed(2)}`;
|
|
|
|
if (discountInfo.type === 'anticipo') {
|
|
// Distinguir entre anticipo registrado y manual
|
|
if (discountInfo.anticipo && discountInfo.anticipo.manual) {
|
|
descriptionText = `Anticipo manual aplicado (${discountInfo.anticipo.comentario})`;
|
|
} else {
|
|
descriptionText = `Anticipo aplicado ${discountInfo.anticipo ? `(${discountInfo.anticipo.folio})` : ''}`;
|
|
}
|
|
amountText = `-$${Number(mov.descuento).toFixed(2)} (${discountInfo.percentage.toFixed(1)}%)`;
|
|
} else if (discountInfo.type === 'warrior') {
|
|
descriptionText = 'Descuento Vanity (100%)';
|
|
amountText = `-$${Number(mov.descuento).toFixed(2)} (${discountInfo.percentage.toFixed(1)}%)`;
|
|
} else if (discountInfo.type === 'percentage') {
|
|
descriptionText = `Descuento (${discountInfo.value}%)`;
|
|
amountText = `-$${Number(mov.descuento).toFixed(2)} (${discountInfo.percentage.toFixed(1)}%)`;
|
|
} else if (discountInfo.type === 'amount') {
|
|
descriptionText = `Descuento ($${discountInfo.value})`;
|
|
amountText = `-$${Number(mov.descuento).toFixed(2)} (${discountInfo.percentage.toFixed(1)}%)`;
|
|
}
|
|
|
|
lines.push(`<div class="t-row t-small"><span>${descriptionText}:</span><span>${amountText}</span></div>`);
|
|
|
|
// Mostrar comentario del descuento si existe
|
|
if (discountInfo.reason && discountInfo.reason.trim()) {
|
|
lines.push(`<div class="t-left t-small t-service-detail"><b>Motivo:</b> ${esc(discountInfo.reason)}</div>`);
|
|
}
|
|
} else {
|
|
// Fallback para formato anterior
|
|
lines.push(`<div class="t-row t-small"><span>Descuento ${mov.motivoDescuento ? '(' + esc(mov.motivoDescuento) + ')' : ''}:</span><span>-$${Number(mov.descuento).toFixed(2)}</span></div>`);
|
|
}
|
|
}
|
|
|
|
// Add special notes for anticipo tickets
|
|
const isAnticipo = mov.tipo === 'Anticipo' || (mov.concepto && mov.concepto.toLowerCase().includes('anticipo'));
|
|
if (isAnticipo) {
|
|
lines.push('<div class="t-spacer"></div>');
|
|
lines.push('<div class="t-left t-small">Al dejar tu anticipo, te agradecemos tu compromiso con nuestro tiempo, de la misma forma en que nosotros respetamos el tuyo.</div>');
|
|
lines.push('<div class="t-spacer"></div>');
|
|
lines.push('<div class="t-left t-small">Las cancelaciones con menos de 48 horas no son reembolsables.</div>');
|
|
lines.push('<div class="t-spacer"></div>');
|
|
}
|
|
|
|
if (mov.notas) lines.push(`<div class="t-left t-small t-service-detail"><b>Notas:</b> ${esc(mov.notas)}</div>`);
|
|
|
|
lines.push('<div class="t-divider"></div>');
|
|
|
|
// TOTAL ALINEADO A LA IZQUIERDA
|
|
lines.push(`<div class="t-left t-bold t-total-large"><b>Total: $${montoFormateado}</b></div>`);
|
|
|
|
// MÉTODO DE PAGO DEBAJO DEL TOTAL - ALINEADO A LA IZQUIERDA
|
|
if (mov.metodo) lines.push(`<div class="t-left t-small"><b>Método:</b> ${esc(mov.metodo)}</div>`);
|
|
|
|
// PEQUEÑO ESPACIO ANTES DE TE ATENDIÓ
|
|
if (mov.staff) lines.push('<div class="t-spacer"></div>');
|
|
|
|
// TE ATENDIÓ DEBAJO DEL MÉTODO DE PAGO
|
|
if (mov.staff) lines.push(`<div class="t-left t-small"><b>Te atendió:</b> ${esc(mov.staff)}</div>`);
|
|
|
|
if (mov.client && (mov.client.esOncologico || mov.client.consentimiento)) {
|
|
lines.push('<div class="t-divider"></div>');
|
|
if (mov.client.esOncologico) {
|
|
lines.push('<div class="t-section t-bold t-center">Consentimiento Oncológico</div>');
|
|
lines.push(`<div class="t-small">El cliente declara ser paciente oncológico y que la información de su médico es veraz.</div>`);
|
|
if (mov.client.nombreMedico) lines.push(`<div class="t-small">Médico: ${esc(mov.client.nombreMedico)}</div>`);
|
|
if (mov.client.telefonoMedico) lines.push(`<div class="t-small">Tel. Médico: ${esc(mov.client.telefonoMedico)}</div>`);
|
|
if (mov.client.cedulaMedico) lines.push(`<div class="t-small">Cédula: ${esc(mov.client.cedulaMedico)}</div>`);
|
|
}
|
|
lines.push('<div class="t-divider"></div>');
|
|
const consentText = mov.tipo === 'Curso' || tipoServicio.toLowerCase().includes('curso')
|
|
? 'Al inscribirse al curso, acepta los términos y condiciones del programa educativo.'
|
|
: 'Al consentir el servicio, declara que la información médica proporcionada es veraz.';
|
|
lines.push(`<div class="t-small t-center">${consentText}</div>`);
|
|
}
|
|
|
|
lines.push('<div class="t-qr-section">');
|
|
lines.push('<div class="t-small t-bold t-center">¡Tu opinión es muy importante!</div>');
|
|
lines.push('<div class="t-small t-center">Escanea el código QR para darnos tu feedback.</div>');
|
|
lines.push('<canvas id="qr-canvas"></canvas>');
|
|
lines.push('</div>');
|
|
|
|
const negocioLeyenda = businessData?.leyenda || settings?.leyenda;
|
|
if (negocioLeyenda) lines.push(`<div class="t-footer t-center t-small">${esc(negocioLeyenda)}</div>`);
|
|
|
|
lines.push('</div>');
|
|
return lines.join('');
|
|
}
|
|
|
|
/**
|
|
* Renderiza el ticket en el DOM, genera el QR y llama a la función de impresión.
|
|
* @param {object} mov El objeto del movimiento.
|
|
* @param {object} settings El objeto de configuración.
|
|
*/
|
|
export async function renderTicketAndPrint(mov, settings, options = {}) {
|
|
const printArea = document.getElementById('printArea');
|
|
if (!printArea) {
|
|
console.error("El área de impresión #printArea no se encontró.");
|
|
alert("Error: No se encontró el área de impresión. Contacte al soporte.");
|
|
return;
|
|
}
|
|
|
|
try {
|
|
printArea.innerHTML = templateTicket(mov, settings);
|
|
|
|
const canvas = document.getElementById('qr-canvas');
|
|
if (!canvas) {
|
|
console.error("El canvas del QR #qr-canvas no se encontró. Se imprimirá sin QR.");
|
|
if (options.saveAsPDF) {
|
|
await generatePDF(mov);
|
|
} else {
|
|
window.print();
|
|
}
|
|
return;
|
|
}
|
|
|
|
const qrUrl = 'http://vanityexperience.mx/qr';
|
|
await QRCode.toCanvas(canvas, qrUrl, { width: 140, margin: 1 });
|
|
|
|
if (options.saveAsPDF) {
|
|
// Generate PDF without printing
|
|
await generatePDF(mov);
|
|
} else {
|
|
requestAnimationFrame(() => window.print());
|
|
}
|
|
|
|
} catch (error) {
|
|
console.error("Error al intentar imprimir:", error);
|
|
alert(`Ocurrió un error al preparar la impresión: ${error.message}. Revise la consola para más detalles.`);
|
|
}
|
|
}
|
|
|
|
|
|
document.addEventListener('DOMContentLoaded', () => {
|
|
const btnTestTicket = document.getElementById('btnTestTicket');
|
|
if (btnTestTicket) {
|
|
btnTestTicket.addEventListener('click', () => {
|
|
const demoMovement = {
|
|
id: 'demo',
|
|
folio: 'DEMO-000001',
|
|
fechaISO: new Date().toISOString(),
|
|
client: {
|
|
nombre: 'Cliente de Prueba',
|
|
esOncologico: true,
|
|
nombreMedico: 'Dr. Juan Pérez',
|
|
telefonoMedico: '5512345678',
|
|
cedulaMedico: '1234567'
|
|
},
|
|
tipo: 'Pago',
|
|
monto: 123.45,
|
|
metodo: 'Efectivo',
|
|
concepto: 'Producto de demostración',
|
|
staff: 'Admin',
|
|
notas: 'Esta es una impresión de prueba.'
|
|
};
|
|
renderTicketAndPrint(demoMovement, window.settings || {});
|
|
});
|
|
}
|
|
});
|
|
|
|
// PDF Generation function
|
|
async function generatePDF(mov) {
|
|
try {
|
|
console.log('Generating PDF for movement:', mov.folio);
|
|
|
|
// Get the print area that contains the rendered ticket
|
|
const printArea = document.getElementById('printArea');
|
|
if (!printArea) {
|
|
throw new Error('Print area not found');
|
|
}
|
|
|
|
// Create a temporary container for PDF generation
|
|
const tempContainer = document.createElement('div');
|
|
tempContainer.innerHTML = printArea.innerHTML;
|
|
tempContainer.style.position = 'absolute';
|
|
tempContainer.style.left = '-9999px';
|
|
tempContainer.style.top = '-9999px';
|
|
tempContainer.style.width = '58mm';
|
|
tempContainer.style.background = 'white';
|
|
document.body.appendChild(tempContainer);
|
|
|
|
// Use html2canvas to convert to image, then create PDF
|
|
const canvas = await html2canvas(tempContainer, {
|
|
scale: 2,
|
|
width: 220, // ~58mm in pixels
|
|
height: 'auto',
|
|
backgroundColor: 'white',
|
|
useCORS: true,
|
|
allowTaint: true
|
|
});
|
|
|
|
console.log('PDF Canvas created, dimensions:', canvas.width, 'x', canvas.height);
|
|
|
|
// Clean up temporary container
|
|
document.body.removeChild(tempContainer);
|
|
|
|
// Validate canvas before creating blob
|
|
if (canvas.width === 0 || canvas.height === 0) {
|
|
throw new Error('Canvas has invalid dimensions: ' + canvas.width + 'x' + canvas.height);
|
|
}
|
|
|
|
// Convert canvas to blob and download
|
|
canvas.toBlob(blob => {
|
|
if (!blob) {
|
|
console.error('Failed to create blob from canvas');
|
|
alert('Error al generar el PDF del ticket - blob creation failed');
|
|
return;
|
|
}
|
|
|
|
console.log('Blob created successfully:', blob.size, 'bytes');
|
|
|
|
try {
|
|
// Create a link and trigger download
|
|
const link = document.createElement('a');
|
|
link.href = URL.createObjectURL(blob);
|
|
link.download = `Ticket_${mov.folio}_thermal.png`;
|
|
document.body.appendChild(link);
|
|
link.click();
|
|
document.body.removeChild(link);
|
|
URL.revokeObjectURL(link.href);
|
|
|
|
console.log('PDF (PNG) ticket downloaded successfully');
|
|
} catch (urlError) {
|
|
console.error('Error creating object URL:', urlError);
|
|
alert('Error al generar el enlace de descarga: ' + urlError.message);
|
|
}
|
|
}, 'image/png');
|
|
|
|
} catch (error) {
|
|
console.error('Error generating PDF:', error);
|
|
alert('Error al generar el PDF del ticket: ' + error.message);
|
|
}
|
|
}
|
|
|
|
// FORZAR RECARGA - 2025-09-09T21:33:00 - TODO ALINEADO A LA IZQUIERDA
|