Total Estimated Appreciation
$${totalAppreciation.toLocaleString(undefined, {maximumFractionDigits: 0})}
`;
// Chart
html += `
`;
html += `
`;
outputDiv.innerHTML = html;
// Render Chart
const ctx = document.getElementById('appreciationChart').getContext('2d');
new Chart(ctx, {
type: 'line',
data: {
labels: results.forecast.map(f => `Year ${f.year}`),
datasets: [{
label: 'Projected Property Value',
data: results.forecast.map(f => f.value),
borderColor: '#14B8A6',
backgroundColor: 'rgba(20, 184, 166, 0.1)',
fill: true,
tension: 0.1
}]
},
options: {
responsive: true,
plugins: {
legend: { position: 'top' },
title: { display: true, text: 'Property Value Over Time' }
},
scales: {
y: {
beginAtZero: false,
ticks: {
callback: function(value) {
return '$' + value.toLocaleString();
}
}
}
}
}
});
}
// --- PDF Download ---
function downloadPDF() {
const { jsPDF } = window.jspdf;
const loader = document.getElementById('loader');
loader.style.display = 'block';
const content = document.getElementById('pdf-content');
html2canvas(content, { scale: 2 }).then(canvas => {
const imgData = canvas.toDataURL('image/png');
const pdf = new jsPDF({ orientation: 'p', unit: 'mm', format: 'a4' });
const pdfWidth = pdf.internal.pageSize.getWidth();
const imgWidth = pdfWidth - 20;
const imgHeight = canvas.height * imgWidth / canvas.width;
let heightLeft = imgHeight;
let position = 10;
pdf.addImage(imgData, 'PNG', 10, position, imgWidth, imgHeight);
heightLeft -= (pdf.internal.pageSize.getHeight() - 20);
while (heightLeft > 0) {
position = heightLeft - imgHeight + 10;
pdf.addPage();
pdf.addImage(imgData, 'PNG', 10, position, imgWidth, imgHeight);
heightLeft -= (pdf.internal.pageSize.getHeight() - 20);
}
pdf.save('Price_Appreciation_Forecast.pdf');
loader.style.display = 'none';
});
}