Feat: Автоматическая себестоимость товара (read-only)

- Удалено ручное редактирование себестоимости из формы товара
- Себестоимость теперь рассчитывается автоматически из партий (FIFO)
- Добавлена модель CostPriceHistory для логирования изменений
- Добавлен signal для автоматического логирования изменений cost_price
- Админ-панель: себестоимость read-only с детальной информацией о партиях
- Фронтенд: цены перемещены под название, теги под категории
- Поле cost_price сделано опциональным (default=0) для создания товаров

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

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
2025-11-23 23:22:45 +03:00
parent 493b6c212d
commit addc5e0962
9 changed files with 374 additions and 71 deletions

View File

@@ -63,6 +63,9 @@ class Product(BaseProductEntity):
cost_price = models.DecimalField(
max_digits=10,
decimal_places=2,
default=0,
null=True,
blank=True,
verbose_name="Себестоимость",
help_text="Автоматически вычисляется из партий (средневзвешенная стоимость по FIFO)"
)
@@ -149,3 +152,79 @@ class Product(BaseProductEntity):
return Product.objects.filter(
variant_groups__in=self.variant_groups.all()
).exclude(id=self.id).distinct()
class CostPriceHistory(models.Model):
"""
История изменений себестоимости товара.
Логирует все изменения себестоимости, их причины и дата/время.
"""
REASON_CHOICES = [
('incoming', 'Поступление товара'),
('batch_edit', 'Редактирование партии'),
('batch_delete', 'Удаление партии'),
('recalculation', 'Пересчет себестоимости'),
('system', 'Системная корректировка'),
]
product = models.ForeignKey(
Product,
on_delete=models.CASCADE,
related_name='cost_price_history',
verbose_name="Товар"
)
old_cost_price = models.DecimalField(
max_digits=10,
decimal_places=2,
verbose_name="Старая себестоимость"
)
new_cost_price = models.DecimalField(
max_digits=10,
decimal_places=2,
verbose_name="Новая себестоимость"
)
reason = models.CharField(
max_length=20,
choices=REASON_CHOICES,
verbose_name="Причина изменения"
)
related_object_id = models.IntegerField(
null=True,
blank=True,
verbose_name="ID связанного объекта",
help_text="Например, ID партии (StockBatch) для поступлений"
)
related_object_type = models.CharField(
max_length=50,
blank=True,
verbose_name="Тип связанного объекта",
help_text="Например, 'StockBatch' для партий"
)
notes = models.TextField(
blank=True,
verbose_name="Примечания",
help_text="Дополнительная информация об изменении"
)
created_at = models.DateTimeField(
auto_now_add=True,
verbose_name="Дата и время изменения"
)
class Meta:
verbose_name = "История себестоимости"
verbose_name_plural = "Истории себестоимости"
ordering = ['-created_at']
indexes = [
models.Index(fields=['product', '-created_at']),
models.Index(fields=['reason']),
]
def __str__(self):
return f"{self.product.name}: {self.old_cost_price}{self.new_cost_price} ({self.get_reason_display()})"