Реализована трёхуровневая система статусов товаров и комплектов: - active (Активный) - товар доступен для продажи - archived (Архивный) - скрыт, можно восстановить в следующем сезоне - discontinued (Снят) - морально устарел, готов к удалению Изменения: 1. Модели (BaseProductEntity, Product, ProductKit): - Заменено поле is_deleted (Boolean) на status (CharField) - Добавлены архивные метаданные (archived_at, archived_by) - Обновлены методы: archive(), restore(), discontinue(), delete() - Уникальное ограничение изменено на conditional (status='active') 2. Менеджеры (ActiveManager, SoftDeleteQuerySet): - Полиморфная поддержка обеих систем (status и is_active) - Использует hasattr() для совместимости с наследниками - Методы: archive(), restore(), discontinue(), archived_only(), active_only() 3. Формы (ProductForm, ProductKitForm): - Включены поле status в формы - Валидация уникальности по status='active' - CSS классы для статус-селектора 4. Admin панель: - DeletedFilter переименован в StatusFilter с тремя опциями - get_status_display() с цветным отображением статуса - Actions: restore_items, hard_delete_selected, delete_selected - Readonly поля для архивирования 5. Представления: - ProductListView: фильтр status вместо is_active - CombinedProductListView: поддержка фильтра status для товаров и комплектов - API views обновлены для работы со статусом 6. Шаблоны: - product_form.html: form.status вместо form.is_active - productkit_create.html: form.status вместо form.is_active - productkit_edit.html: form.status вместо form.is_active 7. Миграции: - Удалены все старые миграции (чистый перезапуск по требованию пользователя) - Создана новая миграция 0001_initial с полной структурой status-системы - Удален старый код преобразования is_deleted -> status Проведённые проверки: - Django system check passed ✓ - Полиморфные менеджеры работают с обеими системами - Уникальные ограничения корректно работают с условиями - История заказов сохраняется даже после архивирования товара (django-simple-history) 🤖 Generated with Claude Code Co-Authored-By: Claude <noreply@anthropic.com>
166 lines
8.1 KiB
Python
166 lines
8.1 KiB
Python
# Generated by Django 5.0.10 on 2025-11-15 11:57
|
|
|
|
import django.db.models.deletion
|
|
from django.conf import settings
|
|
from django.db import migrations, models
|
|
|
|
|
|
class Migration(migrations.Migration):
|
|
|
|
initial = True
|
|
|
|
dependencies = [
|
|
('customers', '0001_initial'),
|
|
('inventory', '0002_initial'),
|
|
('orders', '0001_initial'),
|
|
('products', '0001_initial'),
|
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
|
]
|
|
|
|
operations = [
|
|
migrations.AddField(
|
|
model_name='historicalorderitem',
|
|
name='product',
|
|
field=models.ForeignKey(blank=True, db_constraint=False, null=True, on_delete=django.db.models.deletion.DO_NOTHING, related_name='+', to='products.product', verbose_name='Товар'),
|
|
),
|
|
migrations.AddField(
|
|
model_name='historicalorderitem',
|
|
name='product_kit',
|
|
field=models.ForeignKey(blank=True, db_constraint=False, null=True, on_delete=django.db.models.deletion.DO_NOTHING, related_name='+', to='products.productkit', verbose_name='Комплект товаров'),
|
|
),
|
|
migrations.AddField(
|
|
model_name='order',
|
|
name='customer',
|
|
field=models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, related_name='orders', to='customers.customer', verbose_name='Клиент'),
|
|
),
|
|
migrations.AddField(
|
|
model_name='order',
|
|
name='delivery_address',
|
|
field=models.OneToOneField(blank=True, help_text='Обязательно для курьерской доставки', null=True, on_delete=django.db.models.deletion.CASCADE, related_name='order', to='orders.address', verbose_name='Адрес доставки'),
|
|
),
|
|
migrations.AddField(
|
|
model_name='order',
|
|
name='modified_by',
|
|
field=models.ForeignKey(blank=True, help_text='Последний пользователь, изменивший заказ', null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='modified_orders', to=settings.AUTH_USER_MODEL, verbose_name='Изменен пользователем'),
|
|
),
|
|
migrations.AddField(
|
|
model_name='order',
|
|
name='pickup_warehouse',
|
|
field=models.ForeignKey(blank=True, help_text='Обязательно для самовывоза', null=True, on_delete=django.db.models.deletion.PROTECT, related_name='pickup_orders', to='inventory.warehouse', verbose_name='Склад для самовывоза'),
|
|
),
|
|
migrations.AddField(
|
|
model_name='historicalorderitem',
|
|
name='order',
|
|
field=models.ForeignKey(blank=True, db_constraint=False, null=True, on_delete=django.db.models.deletion.DO_NOTHING, related_name='+', to='orders.order', verbose_name='Заказ'),
|
|
),
|
|
migrations.AddField(
|
|
model_name='orderitem',
|
|
name='order',
|
|
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='items', to='orders.order', verbose_name='Заказ'),
|
|
),
|
|
migrations.AddField(
|
|
model_name='orderitem',
|
|
name='product',
|
|
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.PROTECT, related_name='order_items', to='products.product', verbose_name='Товар'),
|
|
),
|
|
migrations.AddField(
|
|
model_name='orderitem',
|
|
name='product_kit',
|
|
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.PROTECT, related_name='order_items', to='products.productkit', verbose_name='Комплект товаров'),
|
|
),
|
|
migrations.AddField(
|
|
model_name='orderstatus',
|
|
name='created_by',
|
|
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='created_order_statuses', to=settings.AUTH_USER_MODEL, verbose_name='Создано'),
|
|
),
|
|
migrations.AddField(
|
|
model_name='orderstatus',
|
|
name='updated_by',
|
|
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='updated_order_statuses', to=settings.AUTH_USER_MODEL, verbose_name='Последнее изменение'),
|
|
),
|
|
migrations.AddField(
|
|
model_name='order',
|
|
name='status',
|
|
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.PROTECT, related_name='orders', to='orders.orderstatus', verbose_name='Статус заказа'),
|
|
),
|
|
migrations.AddField(
|
|
model_name='historicalorder',
|
|
name='status',
|
|
field=models.ForeignKey(blank=True, db_constraint=False, null=True, on_delete=django.db.models.deletion.DO_NOTHING, related_name='+', to='orders.orderstatus', verbose_name='Статус заказа'),
|
|
),
|
|
migrations.AddField(
|
|
model_name='payment',
|
|
name='created_by',
|
|
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='payments_created', to=settings.AUTH_USER_MODEL, verbose_name='Принял платеж'),
|
|
),
|
|
migrations.AddField(
|
|
model_name='payment',
|
|
name='order',
|
|
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='payments', to='orders.order', verbose_name='Заказ'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='orderitem',
|
|
index=models.Index(fields=['order'], name='orders_orde_order_i_5d347b_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='orderitem',
|
|
index=models.Index(fields=['product'], name='orders_orde_product_32ff41_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='orderitem',
|
|
index=models.Index(fields=['product_kit'], name='orders_orde_product_925b51_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='orderstatus',
|
|
index=models.Index(fields=['code'], name='orders_orde_code_5e1ef7_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='orderstatus',
|
|
index=models.Index(fields=['is_system'], name='orders_orde_is_syst_2f5b85_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='orderstatus',
|
|
index=models.Index(fields=['order'], name='orders_orde_order_2e2930_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='order',
|
|
index=models.Index(fields=['customer'], name='orders_orde_custome_59b6fb_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='order',
|
|
index=models.Index(fields=['status'], name='orders_orde_status__eb4f00_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='order',
|
|
index=models.Index(fields=['delivery_date'], name='orders_orde_deliver_e4274f_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='order',
|
|
index=models.Index(fields=['is_delivery'], name='orders_orde_is_deli_07c9c0_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='order',
|
|
index=models.Index(fields=['payment_status'], name='orders_orde_payment_bc131d_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='order',
|
|
index=models.Index(fields=['created_at'], name='orders_orde_created_0e92de_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='order',
|
|
index=models.Index(fields=['order_number'], name='orders_orde_order_n_f3ada5_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='order',
|
|
index=models.Index(fields=['is_custom_delivery_cost'], name='orders_orde_is_cust_108e98_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='payment',
|
|
index=models.Index(fields=['order'], name='orders_paym_order_i_8c8d98_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='payment',
|
|
index=models.Index(fields=['payment_date'], name='orders_paym_payment_9e5ac0_idx'),
|
|
),
|
|
]
|