const $ = (id) => document.getElementById(id); const money = (v) => new Intl.NumberFormat('pl-PL', { style: 'currency', currency: 'PLN', maximumFractionDigits: 0 }).format(v || 0); const num = (id) => Number($(id).value || 0); let lineChart, pieChart, barChart; let socket; let debounceTimer; let lastRequest = null; window.lastSimulationData = null; function setTheme(theme) { document.body.dataset.theme = theme; localStorage.setItem('mortgage-theme', theme); const btn = $('themeToggle'); if (btn) btn.textContent = theme === 'dark' ? '☀️ Jasny' : '🌙 Ciemny'; } function initTheme() { const saved = localStorage.getItem('mortgage-theme') || 'dark'; setTheme(saved); $('themeToggle').onclick = () => setTheme(document.body.dataset.theme === 'dark' ? 'light' : 'dark'); } function connectWs() { const proto = location.protocol === 'https:' ? 'wss' : 'ws'; socket = new WebSocket(`${proto}://${location.host}/ws/simulate`); socket.onopen = () => recalc(); socket.onmessage = (event) => { const data = JSON.parse(event.data); if (data.error) return console.error(data.error); render(data); }; socket.onclose = () => setTimeout(connectWs, 1200); } function buildRequest() { const rateChanges = [...document.querySelectorAll('#rateChanges .dynamic-row')].map(row => ({ month: Number(row.querySelector('[data-field="month"]').value || 1), annual_rate: Number(row.querySelector('[data-field="rate"]').value || 0) })).filter(x => x.month > 0 && x.annual_rate >= 0); const overpayments = [...document.querySelectorAll('#overpayments .dynamic-row')].map(row => ({ month: Number(row.querySelector('[data-field="month"]').value || 1), amount: Number(row.querySelector('[data-field="amount"]').value || 0), repeat: row.querySelector('[data-field="repeat"]').value, until_month: Number(row.querySelector('[data-field="until"]').value || 0) || null })).filter(x => x.month > 0 && x.amount > 0); return { principal: num('principal'), years: num('years'), margin: num('margin'), base_rate: num('baseRate'), installment_type: $('installmentType').value, overpayment_effect: $('overpaymentEffect').value, rate_changes: rateChanges, overpayments: overpayments }; } function recalc() { clearTimeout(debounceTimer); debounceTimer = setTimeout(() => { lastRequest = buildRequest(); if (socket?.readyState === WebSocket.OPEN) socket.send(JSON.stringify(lastRequest)); }, 120); } function addRateRow(month = 13, rate = 7.0) { const div = document.createElement('div'); div.className = 'dynamic-row'; div.innerHTML = `
`; div.querySelector('button').onclick = () => { div.remove(); recalc(); }; div.querySelectorAll('input').forEach(x => x.addEventListener('input', recalc)); $('rateChanges').appendChild(div); recalc(); } function addOverpaymentRow(month = 12, amount = 10000, repeat = 'once', until = '') { const div = document.createElement('div'); div.className = 'dynamic-row overpay'; div.innerHTML = ` `; div.querySelector('[data-field="repeat"]').value = repeat; div.querySelector('button').onclick = () => { div.remove(); recalc(); }; div.querySelectorAll('input,select').forEach(x => x.addEventListener('input', recalc)); $('overpayments').appendChild(div); recalc(); } function render(data) { window.lastSimulationData = data; const s = data.summary; $('summaryCards').innerHTML = [ ['Odsetki', money(s.total_interest)], ['Oszczędność', money(s.interest_saved)], ['Nadpłaty', money(s.total_overpayment)], ['Okres', `${s.months} mies. / ${Math.ceil(s.months / 12)} lat`], ['Skrócenie', `${s.months_saved} mies.`], ['Średnia rata', money(s.average_payment)] ].map(([label, value]) => `