Customer Behavior Dashboard
Key Customer Behavior Indicators
Behavioral Insights
No insights generated yet. Please configure your data.
Data Configuration
Define your customer behavior metrics, targets, and actuals. Click 'Add Metric' to add more rows.
Customer Behavior Dashboard Report
Key Customer Behavior Indicators
Behavioral Insights
Areas for Improvement: Metrics currently below target include: ${directBelow}. Focus efforts on these to bridge the gap.
`; } if (inverseBelow) { insightsHtml += `Potential Risks: Metrics showing unfavorable trends (e.g., churn rate above target) include: ${inverseBelow}. Investigate root causes and implement corrective actions.
`; } } if (insightsHtml === '') { insightsHtml = 'No specific insights generated. Ensure your metrics have valid target and actual values.
'; } } behavioralInsights.innerHTML = insightsHtml; } // --- Event Handlers (Globally accessible for onclick attributes) --- /** * Switches between dashboard and data configuration tabs. * @param {string} tabName - 'dashboard' or 'dataConfig'. */ window.switchTab = function(tabName) { // Guard clauses for elements if (!dashboardTabBtn || !dataConfigTabBtn || !dashboardTabContent || !dataConfigTabContent || !prevTabBtn || !nextTabBtn) { console.error("One or more tab elements not found."); return; } currentTab = tabName; // Update button styles dashboardTabBtn.classList.remove('active'); dataConfigTabBtn.classList.remove('active'); dashboardTabContent.classList.add('hidden'); dataConfigTabContent.classList.add('hidden'); if (tabName === 'dashboard') { dashboardTabBtn.classList.add('active'); dashboardTabContent.classList.remove('hidden'); prevTabBtn.disabled = true; nextTabBtn.disabled = false; renderAllKpiCards(); // Re-render dashboard content when tab is active generateBehavioralInsights(); // Re-generate insights } else if (tabName === 'dataConfig') { dataConfigTabBtn.classList.add('active'); dataConfigTabContent.classList.remove('hidden'); prevTabBtn.disabled = false; nextTabBtn.disabled = true; renderAllMetricInputRows(); // Re-render input rows when tab is active } }; /** * Navigates between tabs using 'Next' and 'Previous' buttons. * @param {string} direction - 'prev' or 'next'. */ window.navigateTabs = function(direction) { if (direction === 'next') { if (currentTab === 'dashboard') { switchTab('dataConfig'); } } else if (direction === 'prev') { if (currentTab === 'dataConfig') { switchTab('dashboard'); } } }; /** * Adds a new empty metric row to the data configuration. */ if (addMetricBtn) { // Null check for addMetricBtn addMetricBtn.onclick = function() { const newMetricId = generateUniqueId(); const newMetric = { id: newMetricId, name: '', target: null, actual: null, unit: '' }; customerMetrics.push(newMetric); renderMetricInputRow(newMetric); showMessage('New metric row added.', 'info'); }; } /** * Updates a specific property of a metric in the customerMetrics array. * This function is called directly from input fields' oninput. * @param {string} id - The ID of the metric to update. * @param {string} property - The property to update ('name', 'target', 'actual', 'unit'). * @param {*} value - The new value for the property. */ window.updateMetricValue = function(id, property, value) { const metricIndex = customerMetrics.findIndex(m => m.id === id); if (metricIndex !== -1) { customerMetrics[metricIndex][property] = value; } }; /** * Removes a metric from the customerMetrics array and its corresponding input row. * @param {string} id - The ID of the metric to remove. */ window.removeMetric = function(id) { customerMetrics = customerMetrics.filter(m => m.id !== id); const metricRow = document.getElementById(`metric-row-${id}`); if (metricRow) { metricRow.remove(); showMessage('Metric removed successfully.', 'success'); } else { console.warn(`Metric row with ID ${id} not found for removal.`); } }; /** * Saves the data (which is already updated via oninput) and refreshes the dashboard. */ if (saveDataBtn) { // Null check for saveDataBtn saveDataBtn.onclick = function() { // Filter out any metrics that are completely empty or invalid customerMetrics = customerMetrics.filter(metric => metric.name.trim() !== '' || (typeof metric.target === 'number' && !isNaN(metric.target)) || (typeof metric.actual === 'number' && !isNaN(metric.actual)) ); // Ensure all numerical values are indeed numbers customerMetrics.forEach(metric => { metric.target = parseFloat(metric.target) || 0; metric.actual = parseFloat(metric.actual) || 0; }); renderAllKpiCards(); // Re-render dashboard generateBehavioralInsights(); // Re-generate insights showMessage('Data saved and dashboard updated!', 'success'); }; } /** * Handles the PDF download functionality. */ if (downloadPdfBtn) { // Null check for downloadPdfBtn downloadPdfBtn.onclick = async function() { if (!pdfContent || !pdfKpiMetrics || !pdfBehavioralInsights) { console.error("PDF content elements not found."); showMessage('Error: PDF content elements missing.', 'error'); return; } // Populate the hidden PDF content div with current dashboard data pdfKpiMetrics.innerHTML = `Key Customer Behavior Indicators
`; if (customerMetrics.length === 0) { pdfKpiMetrics.innerHTML += 'No KPI data available.
'; } else { let kpiTableHtml = `| Metric Name | Target Value | Actual Value | Unit | Progress (%) |
|---|---|---|---|---|
| ${metric.name || 'N/A'} | ${formatValue(metric.target, metric.unit)} | ${formatValue(metric.actual, metric.unit)} | ${metric.unit || 'N/A'} | ${progress.toFixed(2)}% |
No insights generated. Please configure your data.
'; } else { let overallProgress = 0; let metricsBelowTarget = []; let metricsAboveTarget = []; customerMetrics.forEach(metric => { const progress = calculateProgress(metric.actual, metric.target, metric.name); overallProgress += progress; if (metric.name.toLowerCase().includes('churn rate')) { if (metric.actual > metric.target) { metricsBelowTarget.push({ name: metric.name, progress: progress.toFixed(2), type: 'inverse' }); } else { metricsAboveTarget.push({ name: metric.name, progress: progress.toFixed(2), type: 'inverse' }); } } else { if (progress < 100 && metric.actual < metric.target) { metricsBelowTarget.push({ name: metric.name, progress: progress.toFixed(2), type: 'direct' }); } else if (progress >= 100 && metric.actual >= metric.target) { metricsAboveTarget.push({ name: metric.name, progress: progress.toFixed(2), type: 'direct' }); } } }); const avgProgress = customerMetrics.length > 0 ? (overallProgress / customerMetrics.length).toFixed(2) : 0; insightsForPdf += `Overall Performance: The average progress across all defined metrics is ${avgProgress}%.
`; if (metricsAboveTarget.length > 0) { const directAbove = metricsAboveTarget.filter(m => m.type === 'direct').map(m => `${m.name} (${m.progress}%)`).join(', '); const inverseAbove = metricsAboveTarget.filter(m => m.type === 'inverse').map(m => `${m.name} (Target: ${formatValue(customerMetrics.find(cm => cm.name === m.name).target, customerMetrics.find(cm => cm.name === m.name).unit)}, Actual: ${formatValue(customerMetrics.find(cm => cm.name === m.name).actual, customerMetrics.find(cm => cm.name === m.name).unit)})`).join(', '); if (directAbove) { insightsForPdf += `Strengths: Excellent performance observed in: ${directAbove}. Continue to leverage these areas.
`; } if (inverseAbove) { insightsForPdf += `Positive Trend: Metrics showing favorable trends (e.g., churn rate below target) include: ${inverseAbove}. Good job managing these risks.
`; } } if (metricsBelowTarget.length > 0) { const directBelow = metricsBelowTarget.filter(m => m.type === 'direct').map(m => `${m.name} (${m.progress}%)`).join(', '); const inverseBelow = metricsBelowTarget.filter(m => m.type === 'inverse').map(m => `${m.name} (Target: ${formatValue(customerMetrics.find(cm => cm.name === m.name).target, customerMetrics.find(cm => cm.name === m.name).unit)}, Actual: ${formatValue(customerMetrics.find(cm => cm.name === m.name).actual, customerMetrics.find(cm => cm.name === m.name).unit)})`).join(', '); if (directBelow) { insightsForPdf += `Areas for Improvement: Metrics currently below target include: ${directBelow}. Focus efforts on these to bridge the gap.
`; } if (inverseBelow) { insightsForPdf += `Potential Risks: Metrics showing unfavorable trends (e.g., churn rate above target) include: ${inverseBelow}. Investigate root causes and implement corrective actions.
`; } } insightsForPdf += `