Разработка калькулятора стоимости проекта на 1С-Битрикс
Калькулятор стоимости проекта — это инструмент первичной квалификации для IT-компаний, агентств, проектных организаций и студий. Он не даёт точную цену — это невозможно без погружения в задачу — но помогает клиенту понять порядок цифр и сегментирует входящий поток: отсекает нецелевые запросы и подогревает серьёзные.
Специфика расчёта проектных работ
Проект — не товар с фиксированной ценой. Стоимость зависит от:
- Набора функциональных блоков (что входит в объём)
- Сложности каждого блока (от стандартного до кастомного)
- Технологического стека (скорость разработки, стоимость лицензий)
- Сроков (срочный проект стоит дороже)
- Типа отношений (разовая разработка, поддержка, аутсорс)
Калькулятор структурирует эти переменные в управляемый опросник.
Модель данных: блоки и их стоимость
Функциональные блоки хранятся в HL-блоке ProjectBlocks:
| Поле | Тип | Описание |
|---|---|---|
UF_CATEGORY |
Enum | Категория (Дизайн, Разработка, Интеграция, SEO...) |
UF_BLOCK_NAME |
String | Название блока |
UF_COMPLEXITY |
Enum | simple / standard / complex |
UF_HOURS_MIN |
Int | Минимум часов |
UF_HOURS_MAX |
Int | Максимум часов |
UF_HOURLY_RATE |
Float | Ставка часа для этого типа работ |
UF_IS_REQUIRED |
Bool | Обязательный блок (включён всегда) |
UF_DEPENDS_ON |
String | ID блока-зависимости (нельзя выбрать без него) |
PHP-калькулятор с учётом зависимостей
namespace MyProject\Services\Calculators;
class ProjectCostCalculator
{
private array $blocks;
private array $selectedIds;
private float $urgencyCoeff;
private float $marginPercent;
public function __construct(
array $allBlocks,
array $selectedIds,
string $urgency = 'normal',
float $marginPercent = 30
) {
$this->blocks = $allBlocks;
$this->selectedIds = $this->resolveDependencies($selectedIds, $allBlocks);
$this->urgencyCoeff = match ($urgency) {
'urgent' => 1.5,
'fast' => 1.25,
'normal' => 1.0,
'flexible' => 0.9,
default => 1.0,
};
$this->marginPercent = $marginPercent;
}
public function calculate(): array
{
$breakdown = [];
$totalHoursMin = 0;
$totalHoursMax = 0;
$totalCost = 0;
foreach ($this->selectedIds as $blockId) {
$block = $this->findBlock($blockId);
if (!$block) continue;
$hoursMin = $block['UF_HOURS_MIN'];
$hoursMax = $block['UF_HOURS_MAX'];
$rate = $block['UF_HOURLY_RATE'];
$costMin = $hoursMin * $rate * $this->urgencyCoeff;
$costMax = $hoursMax * $rate * $this->urgencyCoeff;
$totalHoursMin += $hoursMin;
$totalHoursMax += $hoursMax;
$totalCost += ($costMin + $costMax) / 2;
$breakdown[] = [
'id' => $blockId,
'name' => $block['UF_BLOCK_NAME'],
'category' => $block['UF_CATEGORY'],
'hours' => "{$hoursMin}–{$hoursMax}",
'cost_min' => round($costMin),
'cost_max' => round($costMax),
];
}
// Добавляем маржу
$margin = $totalCost * ($this->marginPercent / 100);
$finalCost = $totalCost + $margin;
return [
'breakdown' => $breakdown,
'hours_min' => $totalHoursMin,
'hours_max' => $totalHoursMax,
'cost_min' => round($finalCost * 0.85),
'cost_max' => round($finalCost * 1.2),
'cost_avg' => round($finalCost),
'urgency_coeff' => $this->urgencyCoeff,
'weeks_min' => ceil($totalHoursMin / 40),
'weeks_max' => ceil($totalHoursMax / 40),
];
}
private function resolveDependencies(array $selectedIds, array $blocks): array
{
$resolved = $selectedIds;
foreach ($blocks as $block) {
if (in_array($block['ID'], $selectedIds, true) && !empty($block['UF_DEPENDS_ON'])) {
$depId = (int)$block['UF_DEPENDS_ON'];
if (!in_array($depId, $resolved, true)) {
$resolved[] = $depId;
}
}
// Обязательные блоки всегда включены
if ($block['UF_IS_REQUIRED'] && !in_array($block['ID'], $resolved, true)) {
$resolved[] = $block['ID'];
}
}
return array_unique($resolved);
}
}
UX: чекбоксы с подсказками
Интерфейс калькулятора — набор групп с чекбоксами. Каждый чекбок сопровождается кратким описанием: клиент понимает, что входит в блок.
// Обновление итога при каждом изменении
document.querySelectorAll('.block-checkbox').forEach(cb => {
cb.addEventListener('change', async () => {
const selected = [...document.querySelectorAll('.block-checkbox:checked')]
.map(el => parseInt(el.value));
const urgency = document.querySelector('[name="urgency"]:checked').value;
const resp = await fetch('/ajax/calculator/project/', {
method: 'POST',
body: new URLSearchParams({
selected: JSON.stringify(selected),
urgency,
sessid: BX.bitrix_sessid(),
}),
});
const data = await resp.json();
updateResultPanel(data);
});
});
Передача брифа в CRM
Результат расчёта формирует структурированный бриф:
// Создание сделки в Битрикс24 или лида в CRM сайта
$comments = "РАСЧЁТ СТОИМОСТИ ПРОЕКТА\n\n";
$comments .= "Выбранные блоки:\n";
foreach ($calcResult['breakdown'] as $item) {
$comments .= "— {$item['category']}: {$item['name']} ({$item['hours']} ч.)\n";
}
$comments .= "\nИтого: {$calcResult['cost_min']} – {$calcResult['cost_max']} руб.\n";
$comments .= "Срок: {$calcResult['weeks_min']} – {$calcResult['weeks_max']} недель\n";
$comments .= "Срочность: {$urgency}\n";
Сроки разработки
| Задача | Срок |
|---|---|
| Базовый калькулятор (10–15 блоков, простая сумма, форма заявки) | 5–8 дней |
| Калькулятор с категориями, зависимостями, коэффициентами, передачей в CRM | 2–3 недели |
| Калькулятор с PDF-брифом, историей расчётов, A/B-тестами | 4–6 недель |
Ключевое правило: калькулятор стоимости проекта не должен давать цену ниже реальной. Лучше заложить запас 20–30% сверху и объяснить это «предварительной оценкой», чем получить клиента с завышенными ожиданиями.







