- Упрощена модель Shop: только name обязательное поле - Удалены поля: district, режим работы, координаты, инструкции - Description перенесено после name - Все поля кроме name теперь опциональные - Создан полный CRUD для магазинов: * ShopListView - список магазинов с пагинацией * ShopCreateView - создание нового магазина * ShopUpdateView - редактирование магазина * ShopDeleteView - удаление с подтверждением - Создана форма ShopForm с Bootstrap стилями - Поле "Название магазина" помечено как обязательное (*) - Настроена обработка PhoneNumberField - Созданы шаблоны: * shop_list.html - таблица со списком магазинов * shop_form.html - форма создания/редактирования * shop_confirm_delete.html - подтверждение удаления - Настроены URLs для приложения shops - Добавлена ссылка "Магазины" в главную навигацию - Обновлена админ-панель shops 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
94 lines
2.6 KiB
Python
94 lines
2.6 KiB
Python
from django.db import models
|
||
from phonenumber_field.modelfields import PhoneNumberField
|
||
|
||
|
||
class Shop(models.Model):
|
||
"""
|
||
Модель магазина/пункта самовывоза для цветочного магазина.
|
||
"""
|
||
name = models.CharField(
|
||
max_length=200,
|
||
verbose_name="Название магазина"
|
||
)
|
||
|
||
description = models.TextField(
|
||
blank=True,
|
||
null=True,
|
||
verbose_name="Описание",
|
||
help_text="Дополнительная информация о магазине"
|
||
)
|
||
|
||
# Адрес магазина
|
||
street = models.CharField(
|
||
max_length=255,
|
||
blank=True,
|
||
null=True,
|
||
verbose_name="Улица"
|
||
)
|
||
|
||
building_number = models.CharField(
|
||
max_length=20,
|
||
blank=True,
|
||
null=True,
|
||
verbose_name="Номер здания"
|
||
)
|
||
|
||
# Контактная информация
|
||
phone = PhoneNumberField(
|
||
blank=True,
|
||
null=True,
|
||
verbose_name="Телефон",
|
||
help_text="Контактный телефон магазина"
|
||
)
|
||
|
||
email = models.EmailField(
|
||
blank=True,
|
||
null=True,
|
||
verbose_name="Email"
|
||
)
|
||
|
||
# Статусы и настройки
|
||
is_active = models.BooleanField(
|
||
default=True,
|
||
verbose_name="Активен",
|
||
help_text="Работает ли магазин в данный момент"
|
||
)
|
||
|
||
is_pickup_point = models.BooleanField(
|
||
default=True,
|
||
verbose_name="Пункт самовывоза",
|
||
help_text="Доступен ли магазин для самовывоза заказов"
|
||
)
|
||
|
||
# Временные метки
|
||
created_at = models.DateTimeField(
|
||
auto_now_add=True,
|
||
verbose_name="Дата создания"
|
||
)
|
||
|
||
updated_at = models.DateTimeField(
|
||
auto_now=True,
|
||
verbose_name="Дата обновления"
|
||
)
|
||
|
||
class Meta:
|
||
verbose_name = "Магазин"
|
||
verbose_name_plural = "Магазины"
|
||
indexes = [
|
||
models.Index(fields=['is_active']),
|
||
models.Index(fields=['is_pickup_point']),
|
||
]
|
||
ordering = ['name']
|
||
|
||
def __str__(self):
|
||
if self.street and self.building_number:
|
||
return f"{self.name} ({self.full_address})"
|
||
return self.name
|
||
|
||
@property
|
||
def full_address(self):
|
||
"""Полный адрес магазина"""
|
||
if self.street and self.building_number:
|
||
return f"{self.street}, {self.building_number}"
|
||
return ""
|