Разработка калькулятора ипотеки/кредита на сайте
Кредитный калькулятор — математически строгий инструмент. Здесь нельзя ошибиться в формулах: пользователи сравнивают результат с банковскими расчётами, и расхождение в рублях подрывает доверие к сайту.
Математика аннуитетного платежа
Большинство ипотек и потребительских кредитов используют аннуитетную схему — равные ежемесячные платежи:
M = P × [r(1+r)^n] / [(1+r)^n - 1]
Где:
-
P— сумма кредита -
r— месячная процентная ставка = годовая / 12 / 100 -
n— количество месяцев -
M— ежемесячный платёж
function calcAnnuity(principal, annualRate, months) {
if (annualRate === 0) {
return { monthly: principal / months, total: principal, overpayment: 0 };
}
const r = annualRate / 12 / 100;
const factor = Math.pow(1 + r, months);
const monthly = principal * (r * factor) / (factor - 1);
const total = monthly * months;
const overpayment = total - principal;
return {
monthly: Math.round(monthly * 100) / 100,
total: Math.round(total),
overpayment: Math.round(overpayment),
};
}
Дифференцированный платёж
Первые платежи больше, последние — меньше. Часть банков (особенно по ипотеке) предлагает этот вариант:
function calcDifferentiated(principal, annualRate, months) {
const r = annualRate / 12 / 100;
const principalPart = principal / months;
const schedule = [];
let balance = principal;
let totalInterest = 0;
for (let i = 1; i <= months; i++) {
const interestPart = balance * r;
const payment = principalPart + interestPart;
totalInterest += interestPart;
balance -= principalPart;
schedule.push({
month: i,
payment: Math.round(payment * 100) / 100,
principal: Math.round(principalPart * 100) / 100,
interest: Math.round(interestPart * 100) / 100,
balance: Math.max(0, Math.round(balance * 100) / 100),
});
}
return {
schedule,
total: Math.round((principal + totalInterest) * 100) / 100,
overpayment: Math.round(totalInterest * 100) / 100,
firstPayment: schedule[0].payment,
lastPayment: schedule[months - 1].payment,
};
}
График платежей
Таблица амортизации — обязательный элемент ипотечного калькулятора:
function PaymentSchedule({ schedule }) {
const [expanded, setExpanded] = useState(false);
const visible = expanded ? schedule : schedule.slice(0, 12);
const fmt = (n) => new Intl.NumberFormat('ru-RU', {
minimumFractionDigits: 2, maximumFractionDigits: 2,
}).format(n);
return (
<div className="schedule">
<table>
<thead>
<tr>
<th>Месяц</th>
<th>Платёж</th>
<th>Основной долг</th>
<th>Проценты</th>
<th>Остаток долга</th>
</tr>
</thead>
<tbody>
{visible.map(row => (
<tr key={row.month}>
<td>{row.month}</td>
<td>{fmt(row.payment)} ₽</td>
<td>{fmt(row.principal)} ₽</td>
<td>{fmt(row.interest)} ₽</td>
<td>{fmt(row.balance)} ₽</td>
</tr>
))}
</tbody>
</table>
{schedule.length > 12 && (
<button onClick={() => setExpanded(e => !e)}>
{expanded ? 'Свернуть' : `Показать все ${schedule.length} месяцев`}
</button>
)}
</div>
);
}
Досрочное погашение
Реальная ипотека — это серия пересчётов при частичном досрочном погашении:
function applyEarlyRepayment(schedule, month, amount, mode = 'reduce_term') {
// mode: 'reduce_term' — сокращаем срок, 'reduce_payment' — уменьшаем платёж
let balance = schedule[month - 1].balance - amount;
if (balance <= 0) return { paidOff: true };
const remainingMonths = schedule.length - month;
if (mode === 'reduce_term') {
// Пересчитываем, сколько месяцев нужно при той же ставке
// (обратная задача — итерационно)
return recalculateWithNewBalance(balance, currentRate, 'minimize_term');
}
if (mode === 'reduce_payment') {
return recalculateWithNewBalance(balance, currentRate, remainingMonths);
}
}
Визуализация
Круговая или столбчатая диаграмма «основной долг / переплата» — обязательна. Реализация через Chart.js (лёгкий) или Recharts (если уже используется в проекте):
import { Doughnut } from 'react-chartjs-2';
function OverpaymentChart({ principal, overpayment }) {
return (
<Doughnut
data={{
labels: ['Основной долг', 'Переплата'],
datasets: [{
data: [principal, overpayment],
backgroundColor: ['#3b82f6', '#f87171'],
borderWidth: 0,
}],
}}
options={{ plugins: { legend: { position: 'bottom' } } }}
/>
);
}
Параметры ипотечного калькулятора
| Параметр | Тип | Типовые значения |
|---|---|---|
| Стоимость недвижимости | Число | 1–50 млн ₽ |
| Первоначальный взнос | % или сумма | 10–50% |
| Процентная ставка | % | 6–25% годовых |
| Срок | Годы/месяцы | 1–30 лет |
| Тип платежа | Radio | Аннуитет / Дифференцированный |
| Страховка | % от остатка | 0.1–1% в год |
Страховку добавляем к ежемесячному платежу: insurance = balance * insuranceRate / 100 / 12.
Обмен данными с формой заявки
function prefillApplicationForm(calcResult) {
const data = {
loan_amount: calcResult.principal,
monthly_payment: calcResult.monthly,
loan_term_months: calcResult.months,
estimated_rate: calcResult.annualRate,
};
// Можно передать через URL на страницу заявки
const qs = new URLSearchParams(data).toString();
window.open(`/mortgage-application?${qs}`, '_blank');
}
Сроки
Аннуитетный калькулятор с графиком платежей и диаграммой — 3–4 рабочих дня. Полный вариант с дифференцированным расчётом, досрочным погашением, страховкой, экспортом в PDF и интеграцией с формой заявки — 7–10 дней.







