fix: Улучшения системы ценообразования комплектов

Исправлены 4 проблемы:
1. Расчёт цены первого товара - улучшена валидация в getProductPrice и calculateFinalPrice
2. Отображение actual_price в Select2 вместо обычной цены
3. Количество по умолчанию = 1 для новых форм компонентов
4. Auto-select текста при клике на поле количества для удобства редактирования

Изменённые файлы:
- products/forms.py: добавлен __init__ в KitItemForm для quantity.initial = 1
- products/templates/includes/select2-product-init.html: обновлена formatSelectResult
- products/templates/productkit_create.html: добавлен focus handler для auto-select
- products/templates/productkit_edit.html: добавлен focus handler для auto-select

🤖 Generated with Claude Code
Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
2025-11-02 19:04:03 +03:00
parent c84a372f98
commit 6c8af5ab2c
120 changed files with 9035 additions and 3036 deletions

View File

@@ -83,7 +83,7 @@
<!-- Колонка "Цена" -->
<td>
{% if item.price %}
{{ item.price|floatformat:0 }}
{{ item.price|floatformat:0 }} руб.
{% else %}
<span class="text-muted"></span>
{% endif %}

View File

@@ -1,4 +1,5 @@
<!-- КОМПОНЕНТЫ КОМПЛЕКТА - Shared include для создания и редактирования -->
{% load inventory_filters %}
<div class="card border-0 shadow-sm mb-3">
<div class="card-body p-3">
<h6 class="mb-3 text-muted"><i class="bi bi-boxes me-1"></i>Состав комплекта</h6>
@@ -7,7 +8,9 @@
<div id="kititem-forms">
{% for kititem_form in kititem_formset %}
<div class="card mb-2 kititem-form border" data-form-index="{{ forloop.counter0 }}">
<div class="card mb-2 kititem-form border"
data-form-index="{{ forloop.counter0 }}"
data-product-price="{% if kititem_form.instance.product %}{{ kititem_form.instance.product.actual_price|default:0 }}{% else %}0{% endif %}">
{{ kititem_form.id }}
<div class="card-body p-2">
{% if kititem_form.non_field_errors %}
@@ -17,13 +20,27 @@
{% endif %}
<div class="row g-2 align-items-end">
<div class="col-md-5">
<!-- ТОВАР -->
<div class="col-md-4">
<label class="form-label small text-muted mb-1">Товар</label>
{{ kititem_form.product }}
{% if kititem_form.product.errors %}
<div class="text-danger small">{{ kititem_form.product.errors }}</div>
{% endif %}
</div>
<!-- РАЗДЕЛИТЕЛЬ ИЛИ -->
<div class="col-md-1 d-flex justify-content-center align-items-center">
<div class="kit-item-separator">
<span class="separator-text">ИЛИ</span>
<i class="bi bi-info-circle separator-help"
data-bs-toggle="tooltip"
data-bs-placement="top"
title="Вы можете выбрать что-то одно: либо товар, либо группу вариантов"></i>
</div>
</div>
<!-- ГРУППА ВАРИАНТОВ -->
<div class="col-md-4">
<label class="form-label small text-muted mb-1">Группа вариантов</label>
{{ kititem_form.variant_group }}
@@ -31,13 +48,17 @@
<div class="text-danger small">{{ kititem_form.variant_group.errors }}</div>
{% endif %}
</div>
<!-- КОЛИЧЕСТВО -->
<div class="col-md-2">
<label class="form-label small text-muted mb-1">Кол-во</label>
{{ kititem_form.quantity }}
{{ kititem_form.quantity|smart_quantity }}
{% if kititem_form.quantity.errors %}
<div class="text-danger small">{{ kititem_form.quantity.errors }}</div>
{% endif %}
</div>
<!-- УДАЛЕНИЕ -->
<div class="col-md-1 text-end">
{% if kititem_form.DELETE %}
<button type="button" class="btn btn-sm btn-link text-danger p-0" onclick="this.previousElementSibling.checked = true; this.closest('.kititem-form').style.display='none';" title="Удалить">

View File

@@ -0,0 +1,90 @@
<!-- Select2 Product Search Initialization -->
<!-- Используется для инициализации Select2 с AJAX поиском товаров -->
<!-- Требует: jQuery, Select2 CSS/JS, и переменные: apiUrl, containerSelector, fieldNamePattern -->
<script>
(function() {
// Функции форматирования для Select2
function formatSelectResult(item) {
if (item.loading) return item.text;
var $container = $('<div class="select2-result-item">');
$container.text(item.text);
// Отображаем actual_price (цену со скидкой если она есть), иначе обычную цену
var displayPrice = item.actual_price || item.price;
if (displayPrice) {
$container.append($('<div class="text-muted small">').text(displayPrice + ' руб.'));
}
return $container;
}
function formatSelectSelection(item) {
if (!item.id) return item.text;
// Показываем только текст при выборе, цена будет обновляться в JavaScript
return item.text || item.id;
}
/**
* Инициализирует Select2 для элемента с AJAX поиском товаров
* @param {Element} element - DOM элемент select
* @param {string} type - Тип поиска ('product' или 'variant')
* @param {string} apiUrl - URL API для поиска
*/
window.initProductSelect2 = function(element, type, apiUrl) {
if (!element || $(element).data('select2')) {
return; // Уже инициализирован
}
var placeholders = {
'product': 'Начните вводить название товара...',
'variant': 'Начните вводить название группы...'
};
$(element).select2({
theme: 'bootstrap-5',
placeholder: placeholders[type] || 'Выберите...',
allowClear: true,
width: '100%',
language: 'ru',
minimumInputLength: 0,
dropdownAutoWidth: false,
ajax: {
url: apiUrl,
dataType: 'json',
delay: 250,
data: function (params) {
return {
q: params.term || '',
type: type,
page: params.page || 1
};
},
processResults: function (data) {
return {
results: data.results,
pagination: {
more: data.pagination.more
}
};
},
cache: true
},
templateResult: formatSelectResult,
templateSelection: formatSelectSelection
});
};
/**
* Инициализирует Select2 для всех селектов, совпадающих с паттерном
* @param {string} fieldPattern - Паттерн name атрибута (например: 'items-', 'kititem-')
* @param {string} type - Тип поиска ('product' или 'variant')
* @param {string} apiUrl - URL API для поиска
*/
window.initAllProductSelect2 = function(fieldPattern, type, apiUrl) {
document.querySelectorAll('[name*="' + fieldPattern + '"][name*="-product"]').forEach(function(element) {
window.initProductSelect2(element, type, apiUrl);
});
};
})();
</script>

View File

@@ -0,0 +1,103 @@
/**
* Select2 Product Search Module
* Переиспользуемый модуль для инициализации Select2 с AJAX поиском товаров
*/
(function(window) {
'use strict';
// Форматирование результата в выпадающем списке
function formatSelectResult(item) {
if (item.loading) return item.text;
var $container = $('<div class="select2-result-item">');
$container.text(item.text);
if (item.price) {
$container.append($('<div class="text-muted small">').text(item.price + ' руб.'));
}
return $container;
}
// Форматирование выбранного элемента
function formatSelectSelection(item) {
return item.text || item.id;
}
/**
* Инициализирует Select2 для элемента с AJAX поиском товаров
* @param {Element|jQuery} element - DOM элемент или jQuery объект select
* @param {string} type - Тип поиска ('product' или 'variant')
* @param {string} apiUrl - URL API для поиска
* @param {Object} preloadedData - Предзагруженные данные товара
*/
window.initProductSelect2 = function(element, type, apiUrl, preloadedData) {
if (!element) return;
// Преобразуем в jQuery если нужно
var $element = $(element);
// Если уже инициализирован, пропускаем
if ($element.data('select2')) {
return;
}
var placeholders = {
'product': 'Начните вводить название товара...',
'variant': 'Начните вводить название группы...'
};
var config = {
theme: 'bootstrap-5',
placeholder: placeholders[type] || 'Выберите...',
allowClear: true,
language: 'ru',
minimumInputLength: 0,
ajax: {
url: apiUrl,
dataType: 'json',
delay: 250,
data: function (params) {
return {
q: params.term || '',
type: type,
page: params.page || 1
};
},
processResults: function (data) {
return {
results: data.results,
pagination: {
more: data.pagination.more
}
};
},
cache: true
},
templateResult: formatSelectResult,
templateSelection: formatSelectSelection
};
// Если есть предзагруженные данные, создаем option с ними
if (preloadedData) {
var option = new Option(preloadedData.text, preloadedData.id, true, true);
$element.append(option);
}
$element.select2(config);
};
/**
* Инициализирует Select2 для всех элементов с данным селектором
* @param {string} selector - CSS селектор элементов
* @param {string} type - Тип поиска ('product' или 'variant')
* @param {string} apiUrl - URL API для поиска
*/
window.initAllProductSelect2 = function(selector, type, apiUrl) {
document.querySelectorAll(selector).forEach(function(element) {
window.initProductSelect2(element, type, apiUrl);
});
};
})(window);

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@@ -77,7 +77,7 @@
<td class="fw-bold">{{ item.priority }}</td>
<td>{{ item.product.name }}</td>
<td><small class="text-muted">{{ item.product.sku }}</small></td>
<td><strong>{{ item.product.sale_price }} </strong></td>
<td><strong>{{ item.product.sale_price }} руб.</strong></td>
<td>
{% if item.product.in_stock %}
<span class="badge bg-success"><i class="bi bi-check-circle"></i> Да</span>

View File

@@ -253,7 +253,7 @@ document.addEventListener('DOMContentLoaded', function() {
if (data.results && data.results.length > 0) {
const product = data.results[0];
row.querySelector('[data-product-sku]').textContent = product.sku || sku;
row.querySelector('[data-product-price]').innerHTML = `<strong>${product.price}</strong> ` || '-';
row.querySelector('[data-product-price]').innerHTML = `<strong>${product.price}</strong> руб.` || '-';
// Отображаем статус наличия
const stockCell = row.querySelector('[data-product-stock]');