feat(products): реализована система единиц продажи на фронтенде

Добавлена полноценная интеграция единиц измерения (UoM) для продажи
товаров в разных единицах с автоматическим пересчётом цен и остатков.

## Основные изменения:

### Backend
- Расширен API поиска товаров (api_views.py): добавлена сериализация sales_units
- Создан новый endpoint get_product_sales_units_api для загрузки единиц с остатками
- Добавлено поле sales_unit в OrderItemForm и SaleForm с валидацией
- Созданы CRUD views для управления единицами продажи (uom_views.py)
- Обновлена ProductForm: использует base_unit вместо устаревшего unit

### Frontend
- Создан модуль sales-units.js с функциями для работы с единицами
- Интегрирован в select2-product-search.js: автозагрузка единиц при выборе товара
- Добавлены контейнеры для единиц в order_form.html и sale_form.html
- Реализовано автоматическое обновление цены при смене единицы продажи
- При выборе базовой единицы цена возвращается к базовой цене товара

### UI
- Добавлены страницы управления единицами продажи в навбар
- Созданы шаблоны: sales_unit_list.html, sales_unit_form.html, sales_unit_delete.html
- Добавлены фильтры по товару, единице, активности и дефолтности

## Исправленные ошибки:
- Порядок инициализации: обработчики устанавливаются ДО триггера события change
- Цена корректно обновляется при выборе единицы продажи
- При выборе "Базовая единица" возвращается базовая цена товара

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
This commit is contained in:
2026-01-02 12:35:01 +03:00
parent 5b68f14bb4
commit e831c4fb6e
19 changed files with 1574 additions and 52 deletions

View File

@@ -456,13 +456,14 @@ class OrderItemForm(forms.ModelForm):
class Meta:
model = OrderItem
fields = ['product', 'product_kit', 'quantity', 'price', 'is_custom_price']
fields = ['product', 'product_kit', 'sales_unit', 'quantity', 'price', 'is_custom_price']
# ВАЖНО: НЕ включаем 'id' в fields - это предотвращает ошибку валидации
widgets = {
'quantity': forms.NumberInput(attrs={'min': 1, 'value': 1}),
# Скрываем поля product и product_kit - они будут заполняться через JS
'product': forms.HiddenInput(),
'product_kit': forms.HiddenInput(),
'sales_unit': forms.HiddenInput(), # Управляется через JS
'is_custom_price': forms.HiddenInput(),
}
@@ -478,6 +479,9 @@ class OrderItemForm(forms.ModelForm):
self.fields['product'].required = False
self.fields['product_kit'].required = False
# Поле sales_unit опционально (управляется через JS)
self.fields['sales_unit'].required = False
# Поле цены заполняется автоматически, но можно редактировать вручную
self.fields['price'].widget.attrs.update({
'placeholder': 'Цена',
@@ -506,6 +510,7 @@ class OrderItemForm(forms.ModelForm):
cleaned_data = super().clean()
product = cleaned_data.get('product')
product_kit = cleaned_data.get('product_kit')
sales_unit = cleaned_data.get('sales_unit')
quantity = cleaned_data.get('quantity')
# Пустая форма - это нормально (будет удалена)
@@ -525,6 +530,17 @@ class OrderItemForm(forms.ModelForm):
if not quantity or quantity <= 0:
raise forms.ValidationError('Необходимо указать количество больше 0')
# Валидация единицы продажи
if sales_unit:
if product and sales_unit.product_id != product.id:
raise forms.ValidationError('Единица продажи не принадлежит товару')
if quantity:
try:
sales_unit.validate_quantity(quantity)
except ValidationError as e:
raise forms.ValidationError(str(e))
return cleaned_data