Розробка конфігуратора комплектації товару на сайті

Наша компанія займається розробкою, підтримкою та обслуговуванням сайтів будь-якої складності. Від простих односторінкових сайтів до масштабних кластерних систем, побудованих на мікро сервісах. Досвід розробників підтверджено сертифікатами від вендорів.

Розробка та обслуговування будь-яких видів сайтів:

Інформаційні сайти або веб-програми
Сайти візитки, landing page, корпоративні сайти, онлайн каталоги, квіз, промо-сайти, блоги, ресурси новин, інформаційні портали, форуми, агрегатори
Сайти або веб-програми електронної комерції
Інтернет-магазини, B2B-портали, маркетплейси, онлайн-обмінники, кешбек-сайти, біржі, дропшиппінг-платформи, парсери товарів
Веб-програми для управління бізнес-процесами
CRM-системи, ERP-системи, корпоративні портали, системи управління виробництвом, парсери інформації
Сайти або веб-програми електронних послуг
Дошки оголошень, онлайн-школи, онлайн-кінотеатри, конструктори сайтів, портали надання електронних послуг, відеохостинги, тематичні портали

Це лише деякі з технічних типів сайтів, з якими ми працюємо, і кожен із них може мати свої специфічні особливості та функціональність, а також бути адаптованим під конкретні потреби та цілі клієнта.

Пропоновані послуги
Показано 1 з 1 послугУсі 2065 послуг
Розробка конфігуратора комплектації товару на сайті
Складна
~5 робочих днів
Часті питання

Наші компетенції:

Етапи розробки

Останні роботи

  • image_website-b2b-advance_0.png
    Розробка сайту компанії B2B ADVANCE
    1262
  • image_web-applications_feedme_466_0.webp
    Розробка веб-додатків для компанії FEEDME
    1171
  • image_websites_belfingroup_462_0.webp
    Розробка веб-сайту для компанії БЕЛФІНГРУП
    874
  • image_ecommerce_furnoro_435_0.webp
    Розробка інтернет магазину для компанії FURNORO
    1094
  • image_crm_enviok_479_0.webp
    Розробка веб-додатків для компанії Enviok
    831
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Розробка веб-сайту для компанії ФІКСПЕР
    851

Розроблення конфігуратора комплектації товара на сайті

Конфігуратор дозволяє покупцеві зібрати товар під свої вимоги: вибрати процесор, RAM, накопичувач та колір корпусу ноутбука; обрати розмір, матеріал та гравіювання біжутерії; скомплектувати меблевий гарнітур з сумісних елементів. Результат — унікальна комплектація з розраховною їтоговою ціною та, опціонально, візуалізацією.

Типи конфігураторів

Тип Приклади Особливості
Лінійний Ноутбук: CPU → RAM → SSD Кожен вибір незалежний, ціна сумується
Залежний ПК: материнська плата → сумісні CPU Наступний крок залежить від попереднього
Візуальний Кухня, меблі, автомобіль Зображення змінюється при виборі
Модульний Шафа-купе: ширина + секції + наповнення Довільні комбінації в межах обмежень

Схема даних

CREATE TABLE configurators (
    id              BIGSERIAL PRIMARY KEY,
    product_id      BIGINT REFERENCES products(id),
    name            VARCHAR(255),
    base_price      NUMERIC(12,2) DEFAULT 0,
    image_base_url  TEXT,
    is_active       BOOLEAN DEFAULT TRUE
);

CREATE TABLE config_groups (
    id              BIGSERIAL PRIMARY KEY,
    configurator_id BIGINT REFERENCES configurators(id) ON DELETE CASCADE,
    name            VARCHAR(255) NOT NULL,
    slug            VARCHAR(100) NOT NULL,
    type            VARCHAR(20) NOT NULL,    -- 'radio', 'checkbox', 'quantity', 'text'
    is_required     BOOLEAN DEFAULT TRUE,
    sort_order      SMALLINT DEFAULT 0,
    depends_on_group_id  BIGINT REFERENCES config_groups(id),
    depends_on_option_id BIGINT
);

CREATE TABLE config_options (
    id              BIGSERIAL PRIMARY KEY,
    group_id        BIGINT REFERENCES config_groups(id) ON DELETE CASCADE,
    name            VARCHAR(255) NOT NULL,
    sku_suffix      VARCHAR(100),
    price_modifier  NUMERIC(12,2) DEFAULT 0,
    weight_modifier INT DEFAULT 0,
    image_layer     VARCHAR(500),
    stock           INT DEFAULT 9999,
    is_default      BOOLEAN DEFAULT FALSE,
    sort_order      SMALLINT DEFAULT 0
);

CREATE TABLE config_compatibility (
    id              BIGSERIAL PRIMARY KEY,
    option_a_id     BIGINT REFERENCES config_options(id),
    option_b_id     BIGINT REFERENCES config_options(id),
    type            VARCHAR(20) NOT NULL,    -- 'required', 'forbidden', 'recommended'
    message         TEXT
);

Бекенд: обчислення ціни та валідація

class ConfiguratorEngine
{
    public function calculate(int $configuratorId, array $selectedOptions): ConfigResult
    {
        $configurator = Configurator::with([
            'groups.options',
            'compatibilityRules',
        ])->findOrFail($configuratorId);

        $errors      = [];
        $totalPrice  = $configurator->base_price;
        $totalWeight = 0;
        $skuParts    = [];
        $imageLayers = [];

        foreach ($configurator->groups as $group) {
            $selected = collect($selectedOptions)->where('group_id', $group->id)->first();

            if ($group->is_required && !$selected) {
                $errors[] = "Не вибрано: {$group->name}";
                continue;
            }

            if (!$selected) continue;

            $option = $group->options->find($selected['option_id']);
            if (!$option) {
                $errors[] = "Неверный вариант для групи {$group->name}";
                continue;
            }

            $totalPrice  += $option->price_modifier;
            $totalWeight += $option->weight_modifier;

            if ($option->sku_suffix)  $skuParts[]    = $option->sku_suffix;
            if ($option->image_layer) $imageLayers[]  = $option->image_layer;
        }

        // Перевірка сумісності
        $compatErrors = $this->checkCompatibility($selectedOptions, $configurator->compatibilityRules);
        $errors = array_merge($errors, $compatErrors);

        return new ConfigResult(
            isValid:     empty($errors),
            errors:      $errors,
            totalPrice:  $totalPrice,
            totalWeight: $totalWeight,
            configSku:   implode('-', $skuParts),
            imageLayers: $imageLayers,
        );
    }

    private function checkCompatibility(array $selected, Collection $rules): array
    {
        $errors       = [];
        $selectedIds  = array_column($selected, 'option_id');

        foreach ($rules as $rule) {
            $hasA = in_array($rule->option_a_id, $selectedIds);
            $hasB = in_array($rule->option_b_id, $selectedIds);

            if ($rule->type === 'forbidden' && $hasA && $hasB) {
                $errors[] = $rule->message ?? 'Несумісні компоненти';
            }

            if ($rule->type === 'required' && $hasA && !$hasB) {
                $option = ConfigOption::find($rule->option_b_id);
                $errors[] = $rule->message ?? "Для цієї опції потрібно: {$option->name}";
            }
        }

        return $errors;
    }
}

API-ендпоінти

Route::get('/configurators/{id}', [ConfiguratorController::class, 'show']);
Route::post('/configurators/{id}/calculate', [ConfiguratorController::class, 'calculate']);
Route::post('/configurators/{id}/add-to-cart', [ConfiguratorController::class, 'addToCart']);

class ConfiguratorController extends Controller
{
    public function calculate(Request $request, int $id): JsonResponse
    {
        $data = $request->validate([
            'options'            => 'required|array',
            'options.*.group_id' => 'required|integer',
            'options.*.option_id' => 'required|integer',
        ]);

        $result = $this->engine->calculate($id, $data['options']);

        return response()->json([
            'valid'        => $result->isValid,
            'errors'       => $result->errors,
            'total_price'  => $result->totalPrice,
            'total_weight' => $result->totalWeight,
            'config_sku'   => $result->configSku,
            'image_layers' => $result->imageLayers,
        ]);
    }
}

Фронтенд-компонент

interface ConfigGroup {
  id: number;
  name: string;
  type: 'radio' | 'checkbox';
  options: ConfigOption[];
  depends_on_group_id?: number;
  depends_on_option_id?: number;
}

const Configurator: React.FC<{ configuratorId: number }> = ({ configuratorId }) => {
  const { data: config }  = useQuery(['configurator', configuratorId], fetchConfigurator);
  const [selections, setSelections] = useState<Record<number, number>>({});
  const [result, setResult] = useState<CalcResult | null>(null);

  const updateSelection = async (groupId: number, optionId: number) => {
    const newSelections = { ...selections, [groupId]: optionId };
    setSelections(newSelections);

    const options = Object.entries(newSelections).map(([gId, oId]) => ({
      group_id: Number(gId), option_id: oId,
    }));
    const res = await api.post(`/configurators/${configuratorId}/calculate`, { options });
    setResult(res.data);
  };

  const visibleGroups = config?.groups.filter(g => {
    if (!g.depends_on_group_id) return true;
    return selections[g.depends_on_group_id] === g.depends_on_option_id;
  });

  return (
    <div className="space-y-6">
      {visibleGroups?.map(group => (
        <ConfigGroupWidget
          key={group.id}
          group={group}
          selected={selections[group.id]}
          onSelect={(optId) => updateSelection(group.id, optId)}
        />
      ))}

      {result && (
        <div className="border-t pt-4">
          <p className="text-2xl font-bold">{formatPrice(result.total_price)}</p>
          {result.errors.map((e, i) => (
            <p key={i} className="text-red-500 text-sm">{e}</p>
          ))}
          <button
            disabled={!result.valid}
            onClick={() => addToCart(configuratorId, selections)}
            className="btn-primary mt-3 disabled:opacity-50"
          >
            Додати в корзину
          </button>
        </div>
      )}
    </div>
  );
};

Візуальний конфігуратор (шари зображень)

const ProductVisualizer: React.FC<{ baseSrc: string; layers: string[] }> = ({ baseSrc, layers }) => (
  <div className="relative w-full aspect-square">
    <img src={baseSrc} className="absolute inset-0 w-full h-full object-contain" alt="base" />
    {layers.map((src, i) => (
      <img
        key={i}
        src={src}
        className="absolute inset-0 w-full h-full object-contain"
        alt={`layer-${i}`}
      />
    ))}
  </div>
);

Збереження конфігурації в корзину

CartItem::create([
    'cart_id'       => $cart->id,
    'product_id'    => $configurator->product_id,
    'configurator_id' => $configurator->id,
    'config_options'  => json_encode($selectedOptions),
    'config_sku'      => $result->configSku,
    'unit_price'      => $result->totalPrice,
    'quantity'        => 1,
]);

Графік реалізації

  • Схема даних + ConfiguratorEngine (без залежностей): 2 дні
  • Залежні групи + правила сумісності: +1 день
  • API-ендпоінти: 0.5 дня
  • Фронтенд radio/checkbox конфігуратор: 2 дні
  • Візуальний конфігуратор (шари): +1–2 дні
  • Інтерфейс створення конфігураторів в админці: 2 дні

Разом без візуалізації: 6–7 днів. З візуалізацією: 8–9 днів.