Розробка калькулятора вартості послуг на сайті
Калькулятор вартості — один із найбільш конвертуючих елементів сайту послуг. Людина хоче зрозуміти ціну до розмови з менеджером. Якщо калькулятор дає реалістичну оцінку — він знижує бар'єр входу та фільтрує нецільові звернення.
Моделі ціноутворення в калькуляторах
Адитивна — остаточна ціна = сума компонентів. Підходить для конструкторів послуг: «базовий тариф + опція А + опція Б».
Мультиплікативна — базова ціна множиться на коефіцієнти. Типово для обсяжних послуг: ціна за одиницю × кількість × коефіцієнт корекції.
Формульна — довільна формула з параметрів. Вимагає парсера виразів або явного коду на JS.
Матрична — ціна береться з таблиці за перетином параметрів (наприклад, місто × тип доставки).
Архітектура: конфігурація окремо від коду
const pricingConfig = {
base: 5000,
options: {
hosting: { label: 'Хостинг на рік', price: 1200 },
ssl: { label: 'SSL-сертифікат', price: 0, note: 'безплатно' },
backup: { label: 'Щоденні бекапи', price: 800 },
support: { label: 'Технічна підтримка 1 рік', price: 3600 },
},
multipliers: {
pages: { label: 'Кількість сторінок', perUnit: 500, freeUnits: 5 },
languages: { label: 'Мовні версії', perUnit: 2000, freeUnits: 1 },
},
discounts: [
{ minTotal: 20000, percent: 5 },
{ minTotal: 50000, percent: 10 },
],
};
function calculateTotal(selected, counts) {
let total = pricingConfig.base;
// Опції
for (const key of selected) {
total += pricingConfig.options[key]?.price ?? 0;
}
// Множники
for (const [key, count] of Object.entries(counts)) {
const m = pricingConfig.multipliers[key];
if (!m) continue;
const billable = Math.max(0, count - m.freeUnits);
total += billable * m.perUnit;
}
// Знижки
const discount = [...pricingConfig.discounts]
.reverse()
.find(d => total >= d.minTotal);
if (discount) {
total = total * (1 - discount.percent / 100);
}
return Math.round(total);
}
React-реалізація
function ServiceCalculator({ config }) {
const [selectedOptions, setSelectedOptions] = useState(new Set());
const [counts, setCounts] = useState({ pages: 5, languages: 1 });
const total = calculateTotal([...selectedOptions], counts);
const formatted = new Intl.NumberFormat('uk-UA', {
style: 'currency', currency: 'UAH', maximumFractionDigits: 0,
}).format(total);
const toggleOption = (key) => {
setSelectedOptions(prev => {
const next = new Set(prev);
next.has(key) ? next.delete(key) : next.add(key);
return next;
});
};
return (
<div className="calculator">
<section className="calculator__options">
{Object.entries(config.options).map(([key, opt]) => (
<label key={key} className="calculator__option">
<input
type="checkbox"
checked={selectedOptions.has(key)}
onChange={() => toggleOption(key)}
/>
<span>{opt.label}</span>
<span className="price">
{opt.price > 0 ? `+${opt.price.toLocaleString('uk')}` : opt.note}
</span>
</label>
))}
</section>
<section className="calculator__counters">
{Object.entries(config.multipliers).map(([key, m]) => (
<div key={key} className="calculator__counter">
<label>{m.label}</label>
<input
type="range"
min={m.freeUnits}
max={50}
value={counts[key] ?? m.freeUnits}
onChange={e => setCounts(p => ({ ...p, [key]: +e.target.value }))}
/>
<output>{counts[key] ?? m.freeUnits}</output>
</div>
))}
</section>
<div className="calculator__total">
<span>Усього</span>
<strong>{formatted}</strong>
</div>
<button className="calculator__cta" onClick={() => openRequestForm(total)}>
Залишити заявку
</button>
</div>
);
}
Передача суми в форму заявки
Користувач натиснув «Залишити заявку» — форма повинна отримати вибрані параметри та розраховану суму:
function openRequestForm(total, selectedOptions, counts) {
// Варіант 1: параметри в URL
const params = new URLSearchParams({
estimated_cost: total,
options: [...selectedOptions].join(','),
});
window.location.href = `/contact?${params}`;
// Варіант 2: приховані поля форми на тій же сторінці
document.getElementById('field-estimated-cost').value = total;
document.getElementById('field-options').value = [...selectedOptions].join(', ');
document.getElementById('contact-modal').showModal();
}
Зберігання конфігурації цін на сервері
Якщо ціни часто змінюються — не варто жорстко кодувати їх у JS. Ендпоінт /api/pricing повертає актуальну конфігурацію:
// Laravel
public function getPricingConfig()
{
return response()->json(
Cache::remember('pricing_config', 3600, fn() =>
PricingConfig::where('active', true)->first()?->config ?? []
)
);
}
На фронтенді завантажуємо один раз при монтуванні компонента, кешуємо в sessionStorage.
Терміни
Статичний калькулятор з опціями та слайдерами, передачею даних у форму — 2–3 робочі дні. З динамічною конфігурацією з CMS, знижуючими правилами, кількома валютами та аналітикою вибраних параметрів — 5–7 днів.







