let products = []; let cart = []; let currentPage = 1; let itemsPerPage = 0; let filteredProducts = []; // Load Products async function loadProducts() { const response = await fetch('/muestras/get-inventory.php'); const csvText = await response.text(); products = Papa.parse(csvText, { header: true, dynamicTyping: true }).data; filteredProducts = products; createFilters(); calculateItemsPerPage(); displayPage(1); filterProducts(); } // Calculate Items per Page function calculateItemsPerPage() { const containerWidth = document.getElementById('productContainer').offsetWidth; const productWidth = 210; // Product width + margin itemsPerPage = Math.floor(containerWidth / productWidth) * 50; // Fixed 10 rows } // Display Products on Page function displayPage(page) { const container = document.getElementById('productContainer'); container.innerHTML = ''; const start = (page - 1) * itemsPerPage; const slice = filteredProducts.slice(start, start + itemsPerPage); if (slice.length === 0) { container.innerHTML = '

No results found.

'; return; } slice.forEach(product => { const productDiv = document.createElement('div'); productDiv.className = 'product'; productDiv.innerHTML = `

${product['Paint Code']}

${product['Brand']}

${product['Paint Name']}

Agregar al carrito `; container.appendChild(productDiv); }); document.getElementById('prevPageButton').disabled = page === 1; document.getElementById('nextPageButton').disabled = start + itemsPerPage >= filteredProducts.length; currentPage = page; } // Pagination // Next and Previous Page Controls with Scroll to Top function nextPage() { if (currentPage * itemsPerPage < filteredProducts.length) { displayPage(currentPage + 1); scrollToTop(); // Scroll back to the top } } function prevPage() { if (currentPage > 1) { displayPage(currentPage - 1); scrollToTop(); // Scroll back to the top } } // Helper function to scroll back to the top function scrollToTop() { window.scrollTo({ top: 0, // Scroll to the very top of the page behavior: 'smooth' // Smooth scrolling }); } // Quantity Controls function incrementQuantity(code) { updateQuantity(code, 1); } function decrementQuantity(code) { updateQuantity(code, -1); } function updateQuantity(code, change) { const input = document.getElementById(`qty-${code}`); input.value = Math.max(1, parseInt(input.value) + change); } // Cart Management function addToCart(code) { const button = event.currentTarget; // Get the button that triggered the event button.classList.add('loading'); // Add loading class code = String(code); // Ensure the code is always treated as a string const product = products.find(p => String(p['Paint Code']) === code); const quantity = parseInt(document.getElementById(`qty-${code}`).value); const existing = cart.find(item => String(item['Paint Code']) === code); setTimeout(() => { if (existing) existing.quantity += quantity; else cart.push({ ...product, quantity }); updateCart(); button.classList.remove('loading'); // Remove loading spinner }, 1000); // Simulate loading for 1 second } // Update Cart function updateCart() { const cartItems = document.getElementById('cartItems'); const cartTotal = document.getElementById('cartTotal'); const discountDiv = document.getElementById('discountDiv') const itemCountDiv1 = document.getElementById('itemCountDiv1'); const itemCountDiv2 = document.getElementById('itemCountDiv2'); const checkoutButton = document.querySelector('#checkoutForm button[type="submit"]'); cartItems.innerHTML = ''; let total = 0; let itemCount = 0; cart.forEach(item => { total += item['Price'] * item.quantity; itemCount += item.quantity; const li = document.createElement('li'); li.innerHTML = `
${item['Paint Code']} ${item['Paint Name']}
${item['Brand']}
${item.quantity}
`; cartItems.appendChild(li); }); // Apply discount based on the number of items let discount = 0; if (itemCount >= 3 && itemCount <= 7) { discount = total * 0.10; // 10% discount } else if (itemCount >= 8) { discount = total * 0.25; // 25% discount } const finalTotal = total - discount; // Update the cart total display cartTotal.textContent = `${finalTotal.toFixed(2)} HNL`; discountDiv.textContent = `${discount.toFixed(2)} HNL`; // Update the cart total display with a prefix and bold text cartTotal.innerHTML = `Total: ${finalTotal.toFixed(2)} HNL`; // Update the discount display with a prefix and bold text discountDiv.innerHTML = `Descuento Aplicado: ${discount.toFixed(2)} HNL`; // Hide or show cart total and discount divs based on their values cartTotal.style.display = finalTotal === 0 ? 'none' : 'block'; discountDiv.style.display = discount === 0 ? 'none' : 'block'; //Update both item count displays itemCountDiv1.textContent = itemCount; itemCountDiv2.textContent = itemCount; saveCartToCookies(); // Disable the "Pagar" button if the cart is empty checkoutButton.disabled = itemCount === 0; return itemCount; // Return the item count for further use if needed } function removeFromCart(paintCode) { // Ensure type consistency by converting both to strings cart = cart.filter(item => String(item['Paint Code']) !== String(paintCode)); updateCart(); } function incrementCart(paintCode) { const cartItem = cart.find(item => String(item['Paint Code']) === String(paintCode)); if (cartItem) { cartItem.quantity++; updateCart(); } } function decrementCart(paintCode) { const cartItem = cart.find(item => String(item['Paint Code']) === String(paintCode)); if (cartItem && cartItem.quantity > 1) { cartItem.quantity--; updateCart(); } } document.addEventListener('DOMContentLoaded', function() { emailjs.init("ZXdPTp8fqeo6m6_xu"); // Your Public Key }); function decodeScriptUrl(data) { const binaryString = atob(data); const bytes = new Uint8Array(binaryString.length); for (let i = 0; i < binaryString.length; i++) { bytes[i] = binaryString.charCodeAt(i); } return new TextDecoder('utf-8').decode(bytes); } // Checkout and PDF Invoice async function handleCheckout(event) { event.preventDefault(); const form = this; const formData = new FormData(form); const cartData = JSON.stringify(cart); const itemCountDiv2 = document.getElementById('itemCountDiv2').textContent; const cartTotalElement = document.getElementById('cartTotal'); const discountDivElement = document.getElementById('discountDiv'); const cartTotal = cartTotalElement ? cartTotalElement.textContent : '0 HNL'; const discountDiv = discountDivElement ? discountDivElement.textContent : '0 HNL'; console.log("Cart Total: ", cartTotal); console.log("Discount: ", discountDiv); formData.append('cart', cartData); formData.append('itemCount', itemCountDiv2); // ✅ Generate Order ID once and store it in the DOM const orderID = generateOrderID(); document.getElementById('orderIDField').value = orderID; // Generate invoice, clear cart, and reset form generateInvoice(); generateOrder(); cart = []; updateCart(); event.target.reset(); toggleCart(); Toastify({ text: "¡Gracias por su compra!", duration: 5000, gravity: "top", position: "center" }).showToast(); const paymentProofFile2 = formData.get('paymentProof'); // Convert payment proof to Base64 const base64Fileproof = await new Promise((resolve, reject) => { const reader = new FileReader(); reader.onload = () => resolve(reader.result.split(',')[1]); reader.onerror = reject; reader.readAsDataURL(paymentProofFile2); }); // console.log(base64Fileproof); // Send Email using Google Apps Script const emailParams = { order_id: orderID, from_name: formData.get('name'), email: formData.get('email'), phone: formData.get('phone'), department: formData.get('department'), city: formData.get('city'), address: formData.get('address'), reference: formData.get('reference') || 'No reference provided', cart: cartData, Count: itemCountDiv2, total_price: cartTotal, discount: discountDiv, payment_proof: base64Fileproof }; try { const response = await fetch("https://script.google.com/macros/s/AKfycby28gTSktoFwDIGppVeEg6EWiMvbxQyEUaCSGedzJ0lbAob9tmEmmSbBzQUtAX092uNVA/exec", { method: 'POST', mode: 'no-cors', body: JSON.stringify(emailParams), headers: { 'Content-Type': 'application/json' } }); console.log("Email sent successfully via Google Apps Script!"); } catch (error) { console.error("Failed to send email via Apps Script:", error); } // Dropbox Upload for Payment Proof const paymentProofFile = formData.get('paymentProof'); const reader = new FileReader(); reader.onload = async function (event) { const base64File = event.target.result.split(',')[1]; // Ensure the token is refreshed before uploading await refreshDropboxToken(); const dropboxUploadUrl = 'https://content.dropboxapi.com/2/files/upload'; const accessToken = DROPBOX_CONFIG.accessToken; const fileName = `${formData.get('name') || 'Unknown'}_payment_proof.png`; const parentFolder = '/ColorDemoOrders/pendingorders/'; const orderFolder = `${orderID}`; const folderPath = `${parentFolder}${orderFolder}/${fileName}`; const response = await fetch(dropboxUploadUrl, { method: 'POST', headers: { 'Authorization': `Bearer ${accessToken}`, 'Content-Type': 'application/octet-stream', 'Dropbox-API-Arg': JSON.stringify({ path: folderPath, mode: 'add', autorename: true, mute: false }) }, body: Uint8Array.from(atob(base64File), c => c.charCodeAt(0)) }); const result = await response.json(); if (response.ok) { console.log("File uploaded successfully to Dropbox!"); console.log("Uploaded to:", result.path_display); appendOrderToCSV(orderID, formData, cartTotal, discountDiv, itemCountDiv2); } else { alert("Error uploading to Dropbox!"); console.error(result); } }; reader.readAsDataURL(paymentProofFile); } // Function to generate unique Order ID function generateOrderID() { const now = new Date(); return `CD-${now.getFullYear()}${(now.getMonth() + 1)}${now.getDate()}-${now.getTime()}`; } // Save sales to Dropbox as CSV async function appendOrderToCSV(orderID, formData, cartTotal, discountDiv, itemCountDiv2) { await refreshDropboxToken(); const dropboxUploadUrl = 'https://content.dropboxapi.com/2/files/upload'; const accessToken = DROPBOX_CONFIG.accessToken; const fileName = `/ColorDemoOrders/orders/order_${orderID}.csv`; const csvHeader = 'Order ID,Name,Email,Phone,Department,City,Address,Total Price,Discount,Item Count,Cart Items\n'; // Wrap address in double quotes const address = formData.get('address').replace(/"/g, '""'); // Escape existing quotes const quotedAddress = `"${address}"`; // Wrap address in double quotes // Convert cart items to readable format const formattedCartItems = cart.map(item => { const itemName = item['Paint Name']; const itemCode = item['Paint Code'] || 'N/A'; const itemQuantity = item.quantity; const itemPrice = item['Price']; const itemTotal = itemQuantity * itemPrice; return `${itemName} (${itemCode}) x${itemQuantity} - ${itemTotal} HNL`; }).join('; '); // Join items with a semicolon separator // Create CSV row const csvLine = `${orderID},${formData.get('name')},${formData.get('email')},${formData.get('phone')},${formData.get('department')},${formData.get('city')},${quotedAddress},${cartTotal},${discountDiv},${itemCountDiv2},"${formattedCartItems}"\n`; const csvContent = csvHeader + csvLine; try { console.log("Uploading CSV to Dropbox..."); const response = await fetch(dropboxUploadUrl, { method: 'POST', headers: { 'Authorization': `Bearer ${accessToken}`, 'Content-Type': 'application/octet-stream', 'Dropbox-API-Arg': JSON.stringify({ path: fileName, mode: 'overwrite', autorename: false, mute: false }) }, body: csvContent }); const result = await response.json(); if (result.error) { console.error("Error uploading CSV:", result.error); console.log("Failed to upload CSV to Dropbox."); } else { console.log("CSV uploaded successfully to Dropbox:", result.path_display); } } catch (error) { console.error("Failed to upload CSV to Dropbox:", error); console.log("Error uploading order CSV."); } } // Generate Invoice PDF and Upload to Dropbox function generateInvoice() { const { jsPDF } = window.jspdf; const pdf = new jsPDF(); const orderID = document.getElementById('orderIDField').value; const clientName = document.querySelector('input[name="name"]').value || 'Cliente Desconocido'; const clientEmail = document.querySelector('input[name="email"]').value || 'No Email'; const clientPhone = document.querySelector('input[name="phone"]').value || 'No Phone'; const clientrtn = document.querySelector('input[name="rtn"]').value const currentDate = new Date().toLocaleDateString(); const currentTime = new Date().toLocaleTimeString(); const itemCount = document.getElementById('itemCountDiv1').textContent; const cartTotal = document.getElementById('cartTotal').textContent; const discount = document.getElementById('discountDiv').textContent || '0 HNL'; // Add Image to Header (if available) if (paperImageBase64) { pdf.addImage(paperImageBase64, 'PNG', 12, 15, 30, 30); // (x, y, width, height) } // HEADER (No space between consecutive lines) let yHeader = 20; pdf.setFontSize(18); pdf.text("Color Demo", 105, yHeader, { align: "center" }); pdf.setFontSize(14); pdf.text("Factura de Venta", 105, yHeader + 7, { align: "center" }); pdf.setFontSize(12); yHeader += 13; // Slightly larger space after the title pdf.text("Dirección: Residencial Santa Cruz,", 105, yHeader, { align: "center" }); pdf.text("II Etapa, Bloque 10, Casa 25", 105, yHeader + 5, { align: "center" }); pdf.text("Tel. Pendiente", 105, yHeader + 10, { align: "center" }); pdf.text("email: colordemo89@gmail.com", 105, yHeader + 15, { align: "center" }); // +5 for tight spacing pdf.text("RTN: 08011989128389", 105, yHeader + 20, { align: "center" }); // Combine label and value pdf.text("C.A.I.: F4EB28-BFD5C0-7C4693-E00235-F6DED6-86", 105, yHeader + 25, { align: "center" }); // Space below the header block yHeader += 40; // ORDER DETAILS pdf.text(`Orden: ${orderID}`, 15, yHeader); pdf.text(`Fecha: ${currentDate}`, 160, yHeader); pdf.text(`Hora: ${currentTime}`, 162, yHeader + 5); // === CLIENT INFO (Tighter grouping) === let yClient = yHeader + 15; pdf.setFontSize(14); pdf.text("Información del Cliente", 15, yClient); // Calculate the text width and position the underline const textWidth = pdf.getTextWidth("Información del Cliente:"); const underlineY = yClient + 1; // Slightly below the text pdf.line(15, underlineY, 15 + textWidth, underlineY); // (x1, y1, x2, y2) pdf.setFontSize(12); yClient += 8; // Small gap after section title pdf.text(`Nombre: ${clientName}`, 15, yClient); pdf.text(`Correo: ${clientEmail}`, 15, yClient + 5); // Tight spacing pdf.text(`RTN: ${clientrtn}`, 15, yClient + 10); // Tight spacing pdf.text(`Teléfono: ${clientPhone}`, 15, yClient + 15); // Tight spacing // === TABLE HEADER === let yTable = yClient + 30; pdf.setFontSize(14); pdf.text("Resumen del Pedido", 15, yTable - 3); // Calculate the text width and position the underline const textWidth2 = pdf.getTextWidth("Resumen del Pedido"); const underlineY2 = yTable - 2; // Slightly below the text pdf.line(15, underlineY2, 15 + textWidth2, underlineY2); // (x1, y1, x2, y2) pdf.setFontSize(12); pdf.setFillColor(240, 240, 240); pdf.rect(15, yTable, 180, 10, 'F'); pdf.text("Código", 18, yTable + 7); pdf.text("Color", 42, yTable + 7); pdf.text("Marca", 90, yTable + 7); pdf.text("Cantidad", 143, yTable + 7, { align: "right" }); pdf.text("P. Unitario", 170, yTable + 7, { align: "right" }); pdf.text("Total", 192, yTable + 7, { align: "right" }); yTable += 15; // === CART ITEMS === const pageHeight = pdf.internal.pageSize.height; const marginBottom = 30; cart.forEach(item => { const paintCode = item['Paint Code'] || 'N/A'; if (yTable + 10 > pageHeight - marginBottom) { pdf.addPage(); yTable = 30; pdf.setFillColor(240, 240, 240); pdf.rect(15, yTable, 180, 10, 'F'); pdf.text("Código", 18, yTable + 7); pdf.text("Color", 42, yTable + 7); pdf.text("Marca", 90, yTable + 7); pdf.text("Cantidad", 143, yTable + 7, { align: "right" }); pdf.text("P. Unitario", 170, yTable + 7, { align: "right" }); pdf.text("Total", 192, yTable + 7, { align: "right" }); yTable += 15; } pdf.text(`${paintCode}`, 18, yTable); pdf.text(`${item['Paint Name']}`, 42, yTable); pdf.text(`${item['Brand']}`, 90, yTable); pdf.text(`x${item.quantity}`, 143, yTable, { align: "right" }); pdf.text(`${item['Price']} HNL`, 170, yTable, { align: "right" }); pdf.text(`${item['Price'] * item.quantity} HNL`, 192, yTable, { align: "right" }); yTable += 10; }); // === FOOTER === yTable += 10; pdf.line(15, yTable, 195, yTable); yTable += 10; const rightMargin = 195; const discountWidth = pdf.getTextWidth(`${discount}`); const totalWidth = pdf.getTextWidth(cartTotal); pdf.setFontSize(12); pdf.text(`${discount}`, rightMargin - discountWidth, yTable); pdf.setFont("helvetica", "bold"); pdf.text(cartTotal, rightMargin - totalWidth, yTable + 6); pdf.setTextColor(255, 0, 0); pdf.setFontSize(14); pdf.text("PAGADO", rightMargin - totalWidth, yTable + 13); pdf.setTextColor(0, 0, 0); pdf.setFont("helvetica", "normal"); pdf.text(`Cantidad Total: ${itemCount}`, 15, yTable + 6); // === THANK YOU MESSAGE === yTable += 18; pdf.setFontSize(11); pdf.setTextColor(100); pdf.text("¡Gracias por su compra!", 105, yTable, { align: "center" }); // SAVE PDF LOCALLY const pdfFileName = `${orderID}_recibo.pdf`; const pdfBlob = pdf.output("blob"); // UPLOAD PDF TO DROPBOX uploadInvoiceToDropbox(pdfBlob, pdfFileName, orderID); // Create a URL from the blob const pdfURL = URL.createObjectURL(pdfBlob); } // Generate Order PDF and Upload to Dropbox function generateOrder() { const { jsPDF } = window.jspdf; const pdf = new jsPDF(); const orderID = document.getElementById('orderIDField').value; const clientName = document.querySelector('input[name="name"]').value || 'Cliente Desconocido'; const clientEmail = document.querySelector('input[name="email"]').value || 'No Email'; const clientPhone = document.querySelector('input[name="phone"]').value || 'No Phone'; const clientrtn = document.querySelector('input[name="rtn"]').value const currentDate = new Date().toLocaleDateString(); const currentTime = new Date().toLocaleTimeString(); const itemCount = document.getElementById('itemCountDiv1').textContent; const cartTotal = document.getElementById('cartTotal').textContent; const discount = document.getElementById('discountDiv').textContent || '0 HNL'; // Add Image to Header (if available) if (paperImageBase64) { pdf.addImage(paperImageBase64, 'PNG', 12, 15, 30, 30); // (x, y, width, height) } // HEADER (No space between consecutive lines) let yHeader = 20; pdf.setFontSize(18); pdf.text("Color Demo", 105, yHeader, { align: "center" }); pdf.setFontSize(14); pdf.text("Orden de Compra", 105, yHeader + 7, { align: "center" }); pdf.setFontSize(12); yHeader += 13; // Slightly larger space after the title pdf.text("Cel.: Pendiente", 105, yHeader, { align: "center" }); pdf.text("email: colordemo89@gmail.com", 105, yHeader + 5, { align: "center" }); // +5 for tight spacing // Space below the header block yHeader += 30; // ORDER DETAILS pdf.text(`Orden: ${orderID}`, 15, yHeader); pdf.text(`Fecha: ${currentDate}`, 160, yHeader); pdf.text(`Hora: ${currentTime}`, 162, yHeader + 5); // === CLIENT INFO (Tighter grouping) === let yClient = yHeader + 15; pdf.setFontSize(14); pdf.text("Información del Cliente", 15, yClient); // Calculate the text width and position the underline const textWidth = pdf.getTextWidth("Información del Cliente:"); const underlineY = yClient + 1; // Slightly below the text pdf.line(15, underlineY, 15 + textWidth, underlineY); // (x1, y1, x2, y2) pdf.setFontSize(12); yClient += 8; // Small gap after section title pdf.text(`Nombre: ${clientName}`, 15, yClient); pdf.text(`Correo: ${clientEmail}`, 15, yClient + 5); // Tight spacing pdf.text(`RTN: ${clientrtn}`, 15, yClient + 10); // Tight spacing pdf.text(`Teléfono: ${clientPhone}`, 15, yClient + 15); // Tight spacing // === TABLE HEADER === let yTable = yClient + 30; pdf.setFontSize(14); pdf.text("Resumen del Pedido", 15, yTable - 3); // Calculate the text width and position the underline const textWidth2 = pdf.getTextWidth("Resumen del Pedido"); const underlineY2 = yTable - 2; // Slightly below the text pdf.line(15, underlineY2, 15 + textWidth2, underlineY2); // (x1, y1, x2, y2) pdf.setFontSize(12); pdf.setFillColor(240, 240, 240); pdf.rect(15, yTable, 180, 10, 'F'); pdf.text("Código", 18, yTable + 7); pdf.text("Color", 42, yTable + 7); pdf.text("Marca", 90, yTable + 7); pdf.text("Cantidad", 143, yTable + 7, { align: "right" }); pdf.text("P. Unitario", 170, yTable + 7, { align: "right" }); pdf.text("Total", 192, yTable + 7, { align: "right" }); yTable += 15; // === CART ITEMS === const pageHeight = pdf.internal.pageSize.height; const marginBottom = 30; cart.forEach(item => { const paintCode = item['Paint Code'] || 'N/A'; if (yTable + 10 > pageHeight - marginBottom) { pdf.addPage(); yTable = 30; pdf.setFillColor(240, 240, 240); pdf.rect(15, yTable, 180, 10, 'F'); pdf.text("Código", 18, yTable + 7); pdf.text("Color", 42, yTable + 7); pdf.text("Marca", 90, yTable + 7); pdf.text("Cantidad", 143, yTable + 7, { align: "right" }); pdf.text("P. Unitario", 170, yTable + 7, { align: "right" }); pdf.text("Total", 192, yTable + 7, { align: "right" }); yTable += 15; } pdf.text(`${paintCode}`, 18, yTable); pdf.text(`${item['Paint Name']}`, 42, yTable); pdf.text(`${item['Brand']}`, 90, yTable); pdf.text(`x${item.quantity}`, 143, yTable, { align: "right" }); pdf.text(`${item['Price']} HNL`, 170, yTable, { align: "right" }); pdf.text(`${item['Price'] * item.quantity} HNL`, 192, yTable, { align: "right" }); yTable += 10; }); // === FOOTER === yTable += 10; pdf.line(15, yTable, 195, yTable); yTable += 10; const rightMargin = 195; const discountWidth = pdf.getTextWidth(`${discount}`); const totalWidth = pdf.getTextWidth(cartTotal); pdf.setFontSize(12); pdf.text(`${discount}`, rightMargin - discountWidth, yTable); pdf.setFont("helvetica", "bold"); pdf.text(cartTotal, rightMargin - totalWidth, yTable + 6); pdf.setTextColor(255, 0, 0); pdf.setFontSize(14); pdf.text("VERIFICANDO PAGO", rightMargin -50, yTable + 13); pdf.setTextColor(0, 0, 0); pdf.setFont("helvetica", "normal"); pdf.text(`Cantidad Total: ${itemCount}`, 15, yTable + 6); // === THANK YOU MESSAGE === yTable += 18; pdf.setFontSize(11); pdf.setTextColor(100); pdf.text("¡Gracias! Estamos procesando su orden.", 105, yTable, { align: "center" }); // SAVE PDF LOCALLY const pdfFileName = `${orderID}_orden.pdf`; const pdfBlob = pdf.output("blob"); // UPLOAD PDF TO DROPBOX uploadInvoiceToDropbox(pdfBlob, pdfFileName, orderID); Toastify({ text: "Su orden se está procesando...", duration: 5000, gravity: "middle", position: "center" }).showToast(); // Create a URL from the blob const pdfURL = URL.createObjectURL(pdfBlob); // Open the PDF in a new window/tab window.open(pdfURL, '_blank'); // Optional: Revoke the object URL after some time to free memory setTimeout(() => URL.revokeObjectURL(pdfURL), 10000); } async function bankaccount(event) { event.preventDefault(); // Find the form associated with the clicked button const form = document.getElementById('checkoutForm'); // Use a specific ID for the form if (!form) { console.error("Form element not found."); return; } const formData = new FormData(form); // Validate inputs const from_name = formData.get('name')?.trim(); const email = formData.get('email')?.trim(); const phone = formData.get('phone')?.trim(); if (!from_name || !email || !phone) { Toastify({ text: "Favor llenar los campos de Nombre, correo y teléfono.", duration: 3000, gravity: "middle", position: "center" }).showToast(); } else { Toastify({ text: "Su solicitud fue enviada. Favor revisar su correo en bandeja de entrada o spam (correo no deseado). ¡Gracias!", duration: 5000, gravity: "middle", position: "center" }).showToast(); } const emailParams = { from_name: from_name, email: email, phone: phone, }; try { const response = await fetch("https://script.google.com/macros/s/AKfycbzPqIWRPykp3ycPAoTNpyj3W7Hv_4Fwv3cy-9TWivdEMtmKFGA2S1WNZ7TVmrrG1SZaFA/exec", { method: 'POST', mode: 'no-cors', body: JSON.stringify(emailParams), // Uncomment if needed // headers: { // 'Content-Type': 'application/json' // } }); console.log("Email sent successfully via Google Apps Script!"); } catch (error) { console.error("Failed to send email via Apps Script:", error); } } // Logo on PDF // Load image as Base64 and generate PDF let paperImageBase64 = ''; // Convert image to Base64 function loadImage() { const img = new Image(); img.src = 'images/paper.png'; img.crossOrigin = 'Anonymous'; // Prevent CORS issues if served remotely img.onload = function () { const canvas = document.createElement('canvas'); canvas.width = img.width; canvas.height = img.height; const ctx = canvas.getContext('2d'); ctx.drawImage(img, 0, 0); paperImageBase64 = canvas.toDataURL('image/png'); liveGenerateInvoice(); // Generate PDF after loading image }; } // Event Listeners window.onload = () => loadProducts(); window.addEventListener('resize', () => { calculateItemsPerPage(); displayPage(currentPage); }); window.onload = () => { console.log("Triggering CSV test upload..."); appendOrderToCSV('CD-TEST-123', new FormData(), '1500 HNL', '200 HNL', '3'); }; document.getElementById('checkoutForm').addEventListener('submit', handleCheckout); document.getElementById('checkoutForm').addEventListener('accounts', bankaccount); function toggleCart() { const cartSidebar = document.getElementById('cartSidebar'); if (cartSidebar.classList.contains('open')) { cartSidebar.classList.remove('open'); } else { cartSidebar.classList.add('open'); } } // Filter products based on user input function filterProducts() { const query = document.getElementById('searchInput').value.toLowerCase(); filteredProducts = products.filter(product => String(product['Paint Name'] || '').toLowerCase().includes(query) || String(product['Brand'] || '').toLowerCase().includes(query) || String(product['Paint Code'] || '').toLowerCase().includes(query) || String(product['Combinaciones'] || '').toLowerCase().includes(query) || String(product['Disenador'] || '').toLowerCase().includes(query) ); currentPage = 1; // Reset to the first page displayPage(currentPage); // Show filtered results } // Make precio disappear document.addEventListener('DOMContentLoaded', () => { const ribbon = document.querySelector('.precio'); let lastScrollPosition = 0; window.addEventListener('scroll', () => { const currentScrollPosition = window.scrollY; if (currentScrollPosition > lastScrollPosition) { // Scrolling down ribbon.classList.add('hidden'); } else { // Scrolling up ribbon.classList.remove('hidden'); } lastScrollPosition = currentScrollPosition; }); }); // Calculate products top position window.addEventListener('load', () => { const precioHeight = document.querySelector('.precio').offsetHeight; document.querySelector('.products').style.marginTop = `${75 + precioHeight + 20}px`; }); window.addEventListener('load', adjustMargins); window.addEventListener('resize', adjustMargins); function adjustMargins() { const headerHeight = document.querySelector('header').offsetHeight; const calculatedMargin = headerHeight + 20 + 'px'; // Header height + additional spacing document.querySelector('.search-container').style.marginTop = calculatedMargin; document.querySelector('.products-wrapper').style.marginTop = calculatedMargin; } function updateItemCount(count) { const itemCountDiv1 = document.getElementById('itemCountDiv1'); const itemCountDiv2 = document.getElementById('itemCountDiv2'); // Update both elements with the same content itemCountDiv1.innerHTML = count; itemCountDiv2.innerHTML = count; } // Set a cookie with name, value, and expiration (in days) function setCookie(name, value, days) { const date = new Date(); date.setTime(date.getTime() + days * 24 * 60 * 60 * 1000); document.cookie = `${name}=${encodeURIComponent(value)};expires=${date.toUTCString()};path=/`; } // Get a cookie by name function getCookie(name) { const cookies = document.cookie.split('; '); for (let cookie of cookies) { const [key, value] = cookie.split('='); if (key === name) return decodeURIComponent(value); } return null; } function saveCartToCookies() { const cartString = JSON.stringify(cart); setCookie('cart', cartString, 7); // Cookie expires in 7 days } function loadCartFromCookies() { const cartString = getCookie('cart'); if (cartString) { cart = JSON.parse(cartString); updateCart(); } } function acceptCookies() { setCookie('cookieConsent', 'true', 365); // Consent valid for 1 year document.getElementById('cookieConsent').classList.add('hidden'); } function checkCookieConsent() { const consent = getCookie('cookieConsent'); if (!consent) { document.getElementById('cookieConsent').classList.remove('hidden'); } } // Create filters function createFilters() { const familySet = new Set(products.map(p => p.Family).filter(Boolean)); const typeSet = new Set(products.map(p => p.Type).filter(Boolean)); const brandSet = new Set(products.map(p => p.Brand).filter(Boolean)); // Extract unique values for Brand const filterContainer = document.createElement('div'); filterContainer.id = 'filterMenu'; filterContainer.style.display = 'none'; // Hidden by default filterContainer.style.position = 'absolute'; filterContainer.style.background = 'white'; filterContainer.style.border = '1px solid #ccc'; filterContainer.style.padding = '10px'; filterContainer.style.zIndex = '1000'; const familyFilter = createFilterSection('Family', [...familySet]); const typeFilter = createFilterSection('Type', [...typeSet]); const brandFilter = createFilterSection('Brand', [...brandSet]); // Add filter section for Brand filterContainer.appendChild(familyFilter); filterContainer.appendChild(typeFilter); filterContainer.appendChild(brandFilter); // Append Brand filter document.body.appendChild(filterContainer); const searchInput = document.getElementById('searchInput'); const adjustFilterPosition = () => { const rect = searchInput.getBoundingClientRect(); filterContainer.style.top = `${rect.bottom + window.scrollY}px`; filterContainer.style.left = `${rect.left + window.scrollX}px`; filterContainer.style.width = `${rect.width}px`; // Match the search box width }; searchInput.addEventListener('focus', () => { adjustFilterPosition(); filterContainer.style.display = 'block'; }); window.addEventListener('resize', adjustFilterPosition); // Adjust on window resize window.addEventListener('scroll', adjustFilterPosition); // Adjust on page scroll document.addEventListener('click', (event) => { if (!filterContainer.contains(event.target) && event.target.id !== 'searchInput') { filterContainer.style.display = 'none'; } }); } function createFilterSection(title, options) { const section = document.createElement('div'); section.style.marginBottom = '10px'; const label = document.createElement('strong'); label.textContent = `${title}:`; section.appendChild(label); options.forEach(option => { const checkbox = document.createElement('input'); checkbox.type = 'checkbox'; checkbox.value = option; checkbox.addEventListener('change', applyFilters); const span = document.createElement('span'); span.textContent = ` ${option}`; const wrapper = document.createElement('div'); wrapper.appendChild(checkbox); wrapper.appendChild(span); section.appendChild(wrapper); }); return section; } function applyFilters() { const selectedFamilies = Array.from(document.querySelectorAll('#filterMenu [type="checkbox"][value]')) .filter(checkbox => checkbox.checked && checkbox.parentElement.parentElement.querySelector('strong').textContent.includes('Family')) .map(checkbox => checkbox.value); const selectedTypes = Array.from(document.querySelectorAll('#filterMenu [type="checkbox"][value]')) .filter(checkbox => checkbox.checked && checkbox.parentElement.parentElement.querySelector('strong').textContent.includes('Type')) .map(checkbox => checkbox.value); const selectedBrands = Array.from(document.querySelectorAll('#filterMenu [type="checkbox"][value]')) .filter(checkbox => checkbox.checked && checkbox.parentElement.parentElement.querySelector('strong').textContent.includes('Brand')) .map(checkbox => checkbox.value); filteredProducts = products.filter(product => (selectedFamilies.length === 0 || selectedFamilies.includes(product.Family)) && (selectedTypes.length === 0 || selectedTypes.includes(product.Type)) && (selectedBrands.length === 0 || selectedBrands.includes(product.Brand)) ); currentPage = 1; displayPage(currentPage); } async function refreshDropboxToken() { const dropboxTokenUrl = "https://api.dropboxapi.com/oauth2/token"; const credentials = `${DROPBOX_CONFIG.clientId}:${DROPBOX_CONFIG.clientSecret}`; const encodedCredentials = btoa(credentials); const requestOptions = { method: 'POST', headers: { 'Authorization': `Basic ${encodedCredentials}`, 'Content-Type': 'application/x-www-form-urlencoded' }, body: new URLSearchParams({ grant_type: "refresh_token", refresh_token: DROPBOX_CONFIG.refreshToken }) }; const response = await fetch(dropboxTokenUrl, requestOptions); const data = await response.json(); if (data.access_token) { DROPBOX_CONFIG.accessToken = data.access_token; console.log("New Dropbox Access Token Acquired!"); } else { console.error("Failed to refresh Dropbox token:", data); console.log("Failed to refresh Dropbox token. Check console for details."); } } // Upload Invoice async function uploadInvoiceToDropbox(pdfBlob, fileName, orderID) { await refreshDropboxToken(); // Ensure token is refreshed const accessToken = DROPBOX_CONFIG.accessToken; // Paths for both folders const folderPath1 = `/ColorDemoOrders/pendingorders/${orderID}/${fileName}`; const folderPath2 = `/ColorDemoOrders/history/${fileName}`; // New history folder // Upload to primary folder const response1 = await fetch('https://content.dropboxapi.com/2/files/upload', { method: 'POST', headers: { 'Authorization': `Bearer ${accessToken}`, 'Content-Type': 'application/octet-stream', 'Dropbox-API-Arg': JSON.stringify({ path: folderPath1, mode: 'add', autorename: true, mute: false }) }, body: pdfBlob }); const result1 = await response1.json(); if (result1.error) { console.error("Error uploading PDF invoice to main Dropbox folder:", result1.error); console.log("Failed to upload PDF to Dropbox."); } else { console.log("PDF invoice uploaded successfully to main folder:", result1.path_display); } // Upload to history folder const response2 = await fetch('https://content.dropboxapi.com/2/files/upload', { method: 'POST', headers: { 'Authorization': `Bearer ${accessToken}`, 'Content-Type': 'application/octet-stream', 'Dropbox-API-Arg': JSON.stringify({ path: folderPath2, mode: 'add', autorename: true, mute: false }) }, body: pdfBlob }); const result2 = await response2.json(); if (result2.error) { console.error("Error uploading PDF invoice to history Dropbox folder:", result2.error); console.log("Failed to upload PDF to history folder."); } else { console.log("PDF invoice uploaded successfully to history folder:", result2.path_display); } } // Upload Order async function uploadOrderToDropbox(pdfBlob, fileName, orderID) { await refreshDropboxToken(); // Ensure token is refreshed const accessToken = DROPBOX_CONFIG.accessToken; // Paths for both folders const folderPath1 = `/ColorDemoOrders/pendingorders/${orderID}/${fileName}`; const folderPath2 = `/ColorDemoOrders/history/${fileName}`; // New history folder // Upload to primary folder const response1 = await fetch('https://content.dropboxapi.com/2/files/upload', { method: 'POST', headers: { 'Authorization': `Bearer ${accessToken}`, 'Content-Type': 'application/octet-stream', 'Dropbox-API-Arg': JSON.stringify({ path: folderPath1, mode: 'add', autorename: true, mute: false }) }, body: pdfBlob }); const result1 = await response1.json(); if (result1.error) { console.error("Error uploading PDF invoice to main Dropbox folder:", result1.error); console.log("Failed to upload PDF to Dropbox."); } else { console.log("PDF invoice uploaded successfully to main folder:", result1.path_display); } // Upload to history folder const response2 = await fetch('https://content.dropboxapi.com/2/files/upload', { method: 'POST', headers: { 'Authorization': `Bearer ${accessToken}`, 'Content-Type': 'application/octet-stream', 'Dropbox-API-Arg': JSON.stringify({ path: folderPath2, mode: 'add', autorename: true, mute: false }) }, body: pdfBlob }); const result2 = await response2.json(); if (result2.error) { console.error("Error uploading PDF invoice to history Dropbox folder:", result2.error); console.log("Failed to upload PDF to history folder."); } else { console.log("PDF invoice uploaded successfully to history folder:", result2.path_display); } } // Helper function to convert blob to base64 function blobToBase64(blob) { return new Promise((resolve, reject) => { const reader = new FileReader(); reader.onloadend = () => resolve(reader.result.split(",")[1]); // Get only the Base64 part reader.onerror = reject; reader.readAsDataURL(blob); }); } // Reset Cart function clearCart() { cart = []; // Empty the cart array updateCart(); // Update the cart display Toastify({ text: "El carrito ha sido vaciado.", duration: 3000, gravity: "top", position: "center" }).showToast(); } window.onload = function() { // Extract the query parameter from the URL const urlParams = new URLSearchParams(window.location.search); const keyword = urlParams.get('filter'); // Get the 'filter' parameter if (keyword) { // Set the search input value to the keyword document.getElementById('searchInput').value = keyword; // Optionally, set the keyword as the placeholder document.getElementById('searchInput').setAttribute("placeholder", keyword); // Call the filter function to filter products based on the keyword filterProducts(); // This will use the value from the search input to filter } // Ensure products are loaded after page load (if needed) loadProducts(); // Call loadProducts to load all products if no filter is applied checkCookieConsent(); loadCartFromCookies(); // history.replaceState(null, '', '/muestras'); }; // Open Quick View Modal function openQuickView(element) { const modal = document.getElementById("quick-view-modal"); const modalSwatch = document.getElementById("modal-swatch"); // Target the new swatch div const modalName = document.getElementById("modal-name"); const modalId = document.getElementById("modal-id"); const modalbrand = document.getElementById("modal-brand"); const modalfamily = document.getElementById("modal-family"); const modalDescription = document.getElementById("modal-description"); // Get the data from the clicked swatch const hexColor = element.style.background; // Extract the swatch color modalSwatch.style.background = hexColor; // Apply it to the modal modalName.textContent = element.dataset.name; modalId.textContent = "Código: " + element.dataset.id; modalbrand.textContent = "Marca: " + element.dataset.brand; modalfamily.textContent = "Familia: " + element.dataset.family; modalDescription.textContent = "Descripción: " + element.dataset.desc; // Show modal modal.style.display = "flex"; } // Close Modal document.querySelector(".close").addEventListener("click", function () { document.getElementById("quick-view-modal").style.display = "none"; }); // Close Quick View Modal and Cart Sidebar when clicking outside window.addEventListener("click", function (event) { const modal = document.getElementById("quick-view-modal"); const cartsideb = document.getElementById("cartSidebar"); // Close Quick View Modal if (event.target === modal) { modal.style.display = "none"; } // Prevent closing the cart when clicking inside it or on control buttons if ( !cartsideb.contains(event.target) && !event.target.closest(".cart-button") && // Prevent closing when clicking cart button !event.target.closest(".quantity") && // Prevent closing when clicking quantity buttons !event.target.closest(".remove-btn") && // Prevent closing when clicking remove button !event.target.closest(".quantity-btn") // Prevent closing when clicking + or - ) { cartsideb.classList.remove("open"); } }); // Manual Checkout document.addEventListener("DOMContentLoaded", function () { const manualCheckoutCheckbox = document.querySelector("input[name='manualcheckout']"); const pagarButton = document.getElementById("pagarButton"); const whatsappButton = document.getElementById("whatsappButton"); const checkoutForm = document.getElementById("checkoutForm"); const cartItems = document.getElementById("cartItems"); const cartTotal = document.getElementById("cartTotal"); const requiredFields = [...checkoutForm.querySelectorAll("input[required], textarea[required]")]; const paymentProofInput = document.querySelector("input[name='paymentProof']"); const paymentProofSection = document.querySelector("h3[name='PmntPr']"); // Select using name attribute // Function to reset the cart function resetCart() { cartItems.innerHTML = ""; // Clear cart items cartTotal.innerText = "Total: 0.00 HNL"; // Reset total price checkConditions(); // Recheck conditions to disable buttons } // Function to toggle payment proof requirement function togglePaymentProof() { if (manualCheckoutCheckbox.checked) { paymentProofInput.required = false; paymentProofInput.disabled = true; paymentProofInput.style.display = "none"; if (paymentProofSection) { paymentProofSection.style.display = "none"; // Hide the entire section } } else { paymentProofInput.required = true; paymentProofInput.disabled = false; paymentProofInput.style.display = "block"; if (paymentProofSection) { paymentProofSection.style.display = "block"; // Show the entire section } } } // Function to check if all required conditions are met function checkConditions() { const isManualCheckout = manualCheckoutCheckbox.checked; const isCartEmpty = cartItems.children.length === 0; // Check required fields and payment proof if not in manual checkout mode const requiredInputs = requiredFields.filter(field => field !== paymentProofInput); const allFieldsFilled = requiredInputs.every(field => field.value.trim() !== ""); // If manual checkout is OFF, payment proof is required const isPaymentProofValid = isManualCheckout || paymentProofInput.files.length > 0; console.log("Cart Empty:", isCartEmpty, "| All Fields Filled:", allFieldsFilled, "| Manual Checkout:", isManualCheckout, "| Payment Proof Valid:", isPaymentProofValid); // Enable or disable both buttons pagarButton.disabled = isCartEmpty || !allFieldsFilled || !isPaymentProofValid; whatsappButton.disabled = isCartEmpty || !allFieldsFilled; if (isManualCheckout) { pagarButton.style.display = "none"; whatsappButton.style.display = "inline-block"; } else { pagarButton.style.display = "inline-block"; whatsappButton.style.display = "none"; } // Force-enable the WhatsApp button if conditions are met if (!whatsappButton.disabled) { whatsappButton.removeAttribute("disabled"); } } // Initialize button state and payment proof on page load checkConditions(); togglePaymentProof(); // Listen for checkbox changes to switch buttons and payment proof state manualCheckoutCheckbox.addEventListener("change", function () { togglePaymentProof(); checkConditions(); }); // Listen for changes in required form fields requiredFields.forEach(field => { field.addEventListener("input", checkConditions); }); // Listen for changes in payment proof upload paymentProofInput.addEventListener("change", checkConditions); // Monitor cart changes dynamically const observer = new MutationObserver(checkConditions); observer.observe(cartItems, { childList: true }); // Handle WhatsApp button click whatsappButton.addEventListener("click", async function (event) { event.preventDefault(); const isCartEmpty = cartItems.children.length === 0; const requiredInputs = requiredFields.filter(field => field !== paymentProofInput); const allFieldsFilled = requiredInputs.every(field => field.value.trim() !== ""); if (isCartEmpty) { Toastify({ text: "Su carrito está vacío. Agregue productos antes de solicitar ayuda.", duration: 3000, gravity: "middle", position: "center" }).showToast(); return; } if (!allFieldsFilled) { Toastify({ text: "Por favor, complete todos los campos requeridos.", duration: 3000, gravity: "middle", position: "center" }).showToast(); return; } const formData = new FormData(checkoutForm); const apiKey = "7215694"; // Your CallMeBot API Key // Get the subscription preference const wantsNewsletter = formData.get("subscribe") ? "Sí" : "No"; // Gather order details // Extract cart items with quantities let cartDetails = cart.map(item => { const itemName = item["Paint Name"] || "Producto Desconocido"; const itemCode = item["Paint Code"] || "N/A"; const quantity = item.quantity || 1; return `📌 ${itemName} (${itemCode}) - x ${quantity}`; }).join("\n"); // Get discount value const discountElement = document.getElementById("discountDiv"); const discount = discountElement ? discountElement.innerText : "0 HNL"; // Gather base order details const baseOrderDetails = ` 👤 Nombre: ${formData.get("name")}\n 📧 Email: ${formData.get("email")}\n 📞 Teléfono: ${formData.get("phone")}\n 📍 Dirección: ${formData.get("address")}\n 🏙️ Ciudad: ${formData.get("city")}, ${formData.get("department")}\n`; // Complete order text const fullOrderText = `¡Hola! Favor procesar mi pedido. 📦\n\n${baseOrderDetails}\n🛍️ Pedido:\n${cartDetails}\n💰 ${document.getElementById("cartTotal").innerText}\n🎉 ${discount}\n📩 Recibir noticias y ofertas: ${wantsNewsletter}\n📝 Favor indicar el método de pago.`; // **Encode the message for the WhatsApp URL** const encodedMessage = encodeURIComponent(fullOrderText); // **Detect device type & screen width** const isMobileOrTablet = /Android|iPhone|iPad|iPod/i.test(navigator.userAgent); const isNarrowScreen = window.innerWidth <= 768; // Mobile screens // **Use WhatsApp Web for Desktop & WhatsApp App for Mobile** const phoneNumber = "50431811871"; // Your WhatsApp Number (without +) const whatsappMobileURL = `whatsapp://send?phone=${phoneNumber}&text=${encodedMessage}`; const whatsappWebURL = `https://web.whatsapp.com/send?phone=${phoneNumber}&text=${encodedMessage}`; function openWhatsApp() { // **Decide if it should open in the same tab or new tab** let openInNewTab = !isNarrowScreen; // Open in new tab for tablets & desktops // **Pre-open a tab only if it's a wide screen** let newTab = openInNewTab ? window.open("", "_blank") : null; // Show message before opening WhatsApp Toastify({ text: "Redirigiendo a WhatsApp. Color Demo nunca te pedirá tu nombre de usuario ni tus contraseñas...", duration: 2000, gravity: "middle", position: "center" }).showToast(); setTimeout(() => { if (!openInNewTab) { // **Mobile: Force immediate navigation to WhatsApp (no confirmation)** setTimeout(() => { window.location.href = whatsappMobileURL; }, 100); } else { newTab.location.href = whatsappWebURL; } // **Reset form after opening WhatsApp** checkoutForm.reset(); manualCheckoutCheckbox.checked = false; checkConditions(); togglePaymentProof(); cart = []; // Empty the cart array updateCart(); // Update the cart display toggleCart(); }, 2000); } // **Trigger WhatsApp Opening** openWhatsApp(); }); });