feat: Реализовать систему поступления товаров с партиями (IncomingBatch)

Основные изменения:
- Создана модель IncomingBatch для группировки товаров по документам
- Каждое поступление (Incoming) связано с одной батчем поступления
- Автоматическое создание StockBatch для каждого товара в приходе
- Реализована система нумерации партий (IN-XXXX-XXXX) с поиском максимума в БД
- Обновлены все представления (views) для работы с новой архитектурой
- Добавлены детальные страницы просмотра партий поступлений
- Обновлены шаблоны для отображения информации о партиях и их товарах
- Исправлена логика сигналов для создания StockBatch при приходе товара
- Обновлены формы для работы с новой структурой IncomingBatch

Архитектура FIFO:
- IncomingBatch: одна партия поступления (номер IN-XXXX-XXXX)
- Incoming: товар в партии поступления
- StockBatch: одна партия товара на складе (создается для каждого товара)

Это позволяет системе правильно применять FIFO при продаже товаров.

🤖 Generated with Claude Code

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
2025-10-29 03:26:06 +03:00
parent 097d4ea304
commit 6735be9b08
73 changed files with 6536 additions and 122 deletions

View File

@@ -0,0 +1,54 @@
# -*- coding: utf-8 -*-
"""
WriteOff (Списание товара) views
GROUP 2: MEDIUM PRIORITY
"""
from django.views.generic import ListView, CreateView, UpdateView, DeleteView
from django.urls import reverse_lazy
from django.contrib.auth.mixins import LoginRequiredMixin
from django.contrib import messages
from ..models import WriteOff
from ..forms import WriteOffForm
class WriteOffListView(LoginRequiredMixin, ListView):
model = WriteOff
template_name = 'inventory/writeoff/writeoff_list.html'
context_object_name = 'writeoffs'
paginate_by = 20
def get_queryset(self):
return WriteOff.objects.select_related('batch', 'batch__product').order_by('-date')
class WriteOffCreateView(LoginRequiredMixin, CreateView):
model = WriteOff
form_class = WriteOffForm
template_name = 'inventory/writeoff/writeoff_form.html'
success_url = reverse_lazy('inventory:writeoff-list')
def form_valid(self, form):
messages.success(self.request, f'Списание товара успешно создано.')
return super().form_valid(form)
class WriteOffUpdateView(LoginRequiredMixin, UpdateView):
model = WriteOff
form_class = WriteOffForm
template_name = 'inventory/writeoff/writeoff_form.html'
success_url = reverse_lazy('inventory:writeoff-list')
def form_valid(self, form):
messages.success(self.request, f'Списание товара обновлено.')
return super().form_valid(form)
class WriteOffDeleteView(LoginRequiredMixin, DeleteView):
model = WriteOff
template_name = 'inventory/writeoff/writeoff_confirm_delete.html'
success_url = reverse_lazy('inventory:writeoff-list')
def form_valid(self, form):
writeoff = self.get_object()
messages.success(self.request, f'Списание товара отменено.')
return super().form_valid(form)