add: модели (draft 07)

This commit is contained in:
2026-05-31 14:18:07 +03:00
parent eec127912c
commit dbb6103378

View File

@@ -7,23 +7,14 @@ import datetime
# ИЗОБРАЖЕНИЯ
# ============================================================================
class TbImage(models.Model):
# Типы изображений
class ImageType(models.TextChoices):
COVER = 'cover', 'Обложка (лицевая сторона)'
BACK = 'back', 'Задняя сторона'
OBI = 'obi', 'OBI полоска (японская)'
MATRIX = 'matrix', 'Матричный номер (на виниле)'
SPINE = 'spine', 'Корешок'
CONDITION = 'condition', 'Фото состояния товара'
BOOKLET = 'booklet', 'Буклет/Постер'
OTHER = 'other', 'Другое'
"""
Изображение, связанное с релизом, оффером, исполнителем и т.д.
"""
# Источник изображения
class ImageSource(models.TextChoices):
DISCOGS = 'discogs', 'Discogs (внешний)'
PARSER_UPLOAD = 'parser', 'Загружено парсером (из Discogs, Meshok или другого сайта)'
MANUAL_UPLOAD = 'manual', 'Ручная загрузка пользователем'
VENDOR = 'vendor', 'От продавца'
API = 'api', 'API (автоматически)'
OTHER = 'other', 'Другое'
# Тип изображения (реальное или абстрактное)
@@ -31,25 +22,14 @@ class TbImage(models.Model):
REAL_PHOTO = 'real', 'Реальная фотография товара'
ABSTRACT = 'abstract', 'Абстрактное (из внешнего источника)'
# Файл через django_filer
file = FilerImageField(
# Файл через django_filer
null=True,
blank=True,
on_delete=models.SET_NULL,
related_name='images',
)
# Связи (через GenericForeignKey?)
# или отдельные FK к Product/Offer/Artist?
# Основные поля
l_image_type = models.CharField(
max_length=10,
choices=ImageType.choices,
default=ImageType.OTHER,
verbose_name='Тип изображения',
)
l_source = models.CharField(
max_length=10,
choices=ImageSource.choices,
@@ -57,7 +37,6 @@ class TbImage(models.Model):
verbose_name='Источник',
)
# КЛЮЧЕВОЕ: реальная или абстрактная?
l_reality = models.CharField(
max_length=10,
choices=ImageReality.choices,
@@ -66,7 +45,6 @@ class TbImage(models.Model):
help_text='Реальная фотография товара или картинка из внешнего источника?',
)
# URL источника (для внешних изображений)
s_source_url = models.URLField(
blank=True,
null=True,
@@ -78,15 +56,9 @@ class TbImage(models.Model):
i_sort_order = models.IntegerField(
default=0,
db_index=True,
verbose_name='Порядок сортировки',
)
# Основная картинка?
b_is_primary = models.BooleanField(
default=False,
db_index=True,
verbose_name='Основное изображение',
help_text='Показывать как обложку в списке товаров',
verbose_name='Cортировка',
help_text='Порядок отображения изображений. Чем меньше число, тем выше в списке. Можно использовать'
' для указания обложки (0), задника (1) и т.д.',
)
# Доверие данным (для парсеров и API)
@@ -97,15 +69,6 @@ class TbImage(models.Model):
verbose_name='Уверенность (для автоматических данных)',
help_text='0.0 - 1.0, насколько уверены, что это правильное изображение',
)
# Описание
s_description = models.TextField(
blank=True,
default='',
verbose_name='Описание',
help_text='Например: "Состояние пластинки", "Царапина на обложке"',
)
# Авторские права
s_copyright = models.CharField(
max_length=255,
@@ -115,14 +78,6 @@ class TbImage(models.Model):
help_text='Например: "© 2024 User" или "CC-BY"',
)
# Метаданные (размеры, EXIF и т.д.)
j_metadata = models.JSONField(
default=dict,
blank=True,
verbose_name='Метаданные',
help_text='Разрешение, размер файла, EXIF и т.д.',
)
# Timestamps
t_created = models.DateTimeField(
auto_now_add=True,
@@ -153,6 +108,16 @@ class TbArtist(models.Model):
verbose_name='Исполнитель (типографированно в HTML)',
help_text='С сохранением типографирования и спецсимволов.',
)
s_artist_image = models.OneToOneField(
TbImage,
on_delete=models.SET_NULL,
related_name='artist_image',
blank=True,
null=True,
verbose_name='Изображение для исполнителя',
help_text='Изображение исполнителя, например, логотип группы или фотография. Если указано, будет'
' отображаться рядом с именем исполнителя.',
)
s_slug = models.SlugField(
max_length=64,
verbose_name='Слаг',
@@ -182,7 +147,7 @@ class TbArtist(models.Model):
ordering = ('s_artist',)
class TbProduct(models.Model):
class TbCatalogItem(models.Model):
"""
Абстрактный релиз (альбом, сингл, компиляция).
Может быть представлен в виде одного или нескольких предложений от разных продавцов.
@@ -198,7 +163,7 @@ class TbProduct(models.Model):
help_text="Один или несколько для коллабораций",
)
s_title = models.CharField(
s_item_title = models.CharField(
max_length=255, blank=False, null=False, default='',
# db_index=True,
verbose_name='Название товара (релиза)',
@@ -206,7 +171,7 @@ class TbProduct(models.Model):
' или <tt>TDK, CDing I, 90</tt>.',
)
s_title_html = models.TextField(
s_item_title_html = models.TextField(
blank=True,
null=True,
default='',
@@ -214,7 +179,17 @@ class TbProduct(models.Model):
help_text='С HTML-тегами для типографирования',
)
s_title_slug = models.SlugField(
s_item_image = models.OneToOneField(
TbImage,
on_delete=models.SET_NULL,
related_name='product_image',
blank=True,
null=True,
verbose_name='Изображение',
help_text='Изображение альбома/продукта, например, обложка альбома. Если указано, будет отображаться'
' рядом с названием релиза.',
)
s_item_slug = models.SlugField(
blank=True,
null=True,
default='',
@@ -223,18 +198,19 @@ class TbProduct(models.Model):
)
# Год выпуска (важно для коллекционеров)
s_release_date = models.CharField(
s_item_date = models.CharField(
max_length=10,
blank=True,
null=True,
default='XXXX-XX-XX',
verbose_name='Дата релиза (str)',
help_text='Например: 1969-05-25, или 1969-05-XX (если день неизвестен), или 1969-XX-XX (если известен только'
' год, или XXXX-XX-XX (если дата релиза неизвестна). Более приоритетное поле для отображения даты'
' год, или XXXX-XX-XX (если дата релиза неизвестна). Менее приоритетное поле для отображения даты'
' релиза, чем t_release_date, так как может содержать неполную дату и/или текстовую информацию.'
' Срабатывает только если t_release_date не указано.'
)
t_release_date = models.DateField(
t_item_date = models.DateField(
blank=True,
null=True,
verbose_name='Дата релиза',
@@ -242,23 +218,25 @@ class TbProduct(models.Model):
)
i_discogs_master_id = models.IntegerField(
blank=True, null=True, default=None,
blank=True,
null=True,
default=None,
verbose_name='ID на мастер-релиз Discogs',
help_text='Уникальный идентификатор мастер-релиза на Discogs, если он там есть. Например: <tt>306323</tt>',
)
# Метаданные релиза (страна, жанр, количество треков и т.д.)
j_product_metadata = models.JSONField(
j_item_metadata = models.JSONField(
default=dict, blank=True, null=True,
verbose_name='Дополнительные данные',
help_text='Дополнительные данные о релизе в виде JSON-словаря. Сюда же включены варианты написания релиза в источниках',
)
t_created = models.DateTimeField(
t_item_created = models.DateTimeField(
auto_now_add=True,
verbose_name="Дата создания",
)
t_updated = models.DateTimeField(
t_item_updated = models.DateTimeField(
auto_now=True,
verbose_name="Дата обновления",
)
@@ -427,8 +405,8 @@ class TbOffer(models.Model):
OTHER = '++', 'Other'
# Связи
k_product = models.ForeignKey(
TbProduct,
k_offer_to_item = models.ForeignKey(
TbCatalogItem,
blank=True,
null=True,
default=None,
@@ -437,7 +415,7 @@ class TbOffer(models.Model):
verbose_name='Релиз (товар)',
)
k_label = models.ForeignKey(
k_offer_to_label = models.ForeignKey(
TbLabel,
null=True,
default=None,
@@ -447,7 +425,7 @@ class TbOffer(models.Model):
help_text='Лейбл, на котором был выпущен релиз, если он известен. Например: <tt>Atlantic</tt> или <tt>Мелодия</tt>',
)
k_source = models.ForeignKey(
k_offer_to_source = models.ForeignKey(
to='TbSource',
null=True,
default=None,
@@ -458,15 +436,26 @@ class TbOffer(models.Model):
' продавца `offer.k_source.k_seller`.',
)
# Изображения товара (одна картинка может быть у многих офферов, а офер иметь много картинок)
# M2M связь для удобства в админке (filter_horizontal)
# Порядок картинок определяется полем i_sort_order в TbImage
k_offer_images = models.ManyToManyField(
TbImage,
blank=True,
related_name='offers',
verbose_name='Изображения',
help_text='Картинки этого товара. Порядок определяется полем i_sort_order в ка<D0BA><D0B0>тинке.',
)
# Характеристики
s_catalog_num = models.TextField(
s_offer_catalog_num = models.TextField(
blank=True,
default='',
verbose_name='Каталожный номер / Barcode',
help_text='Например: "SD 16023" или "5099923452355"',
)
l_primary_media = models.CharField(
l_offer_primary_media = models.CharField(
max_length=2,
choices=Format.choices,
default=Format.LP,
@@ -474,13 +463,13 @@ class TbOffer(models.Model):
help_text='Основной формат носителя (пластинка, CD, кассета и т.п.)'
)
i_discogs_id = models.IntegerField(
i_offer_discogs_id = models.IntegerField(
blank=True,default=0,
verbose_name='ID на релиз Discogs',
help_text='Уникальный идентификатор релиза на Discogs, если он там есть. Например: <tt>306323</tt>',
)
l_condition_media = models.CharField(
l_offer_condition_media = models.CharField(
max_length=2,
choices=Condition.choices,
default=Condition.S,
@@ -488,7 +477,7 @@ class TbOffer(models.Model):
help_text='Состояние носителя (пластинки, CD и т.п.) по шкале от "Still Sealed" (запечатано) до "Poor" (плохое).',
)
l_condition_sleeve = models.CharField(
l_offer_condition_sleeve = models.CharField(
max_length=2,
choices=Condition.choices,
default=Condition.S,
@@ -497,7 +486,7 @@ class TbOffer(models.Model):
)
# Цена и наличие
f_price = models.DecimalField(
f_offer_price = models.DecimalField(
max_digits=10,
decimal_places=2,
blank=True,
@@ -505,7 +494,7 @@ class TbOffer(models.Model):
verbose_name='Цена',
)
l_currency = models.CharField(
l_offer_currency = models.CharField(
max_length=3,
choices=Currency.choices,
default=Currency.RUB,