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:
@@ -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()})"
|
||||
|
||||
Reference in New Issue
Block a user