Some checks failed
continuous-integration/drone/push Build is failing
- 📊 Создана ContactInfo модель с полями компании, контактов и описания - 🎨 Полностью переработана страница about.html с современными карточками - 🔗 Админ-панель для управления контактной информацией - 💎 CSS анимации и градиенты для улучшения UI/UX - 🗄️ Миграция 0012_contactinfo.py для создания таблицы - 🔧 Обновлены views для использования данных из БД
215 lines
11 KiB
Python
215 lines
11 KiB
Python
from django.db import models
|
||
from django.contrib.auth.models import AbstractUser, User
|
||
import uuid
|
||
from django.urls import reverse
|
||
|
||
class ContactInfo(models.Model):
|
||
"""Модель для контактной информации компании"""
|
||
company_name = models.CharField(max_length=200, default="SmartSolTech", verbose_name="Название компании")
|
||
email = models.EmailField(default="info@smartsoltech.kr", verbose_name="Email")
|
||
phone = models.CharField(max_length=20, default="+82-10-5693-6103", verbose_name="Телефон")
|
||
telegram = models.CharField(max_length=100, default="@smartsoltech", verbose_name="Telegram")
|
||
address = models.TextField(default="Чолланамдо, Кванджу", verbose_name="Адрес")
|
||
working_hours = models.CharField(max_length=100, default="Пн-Пт 9:00-18:00", verbose_name="Часы работы")
|
||
description = models.TextField(default="Мы - команда профессионалов в сфере IT-решений", verbose_name="Описание")
|
||
call_to_action = models.CharField(max_length=200, default="Начнем сотрудничество?", verbose_name="Призыв к действию")
|
||
subtitle = models.CharField(max_length=200, default="Свяжитесь с нами для обсуждения вашего проекта", verbose_name="Подзаголовок")
|
||
is_active = models.BooleanField(default=True, verbose_name="Активно")
|
||
|
||
class Meta:
|
||
verbose_name = 'Контактная информация'
|
||
verbose_name_plural = 'Контактная информация'
|
||
|
||
def __str__(self):
|
||
return f"Контакты - {self.company_name}"
|
||
|
||
@classmethod
|
||
def get_active(cls):
|
||
"""Получить активную контактную информацию"""
|
||
return cls.objects.filter(is_active=True).first() or cls.objects.create()
|
||
|
||
class HeroBanner(models.Model):
|
||
"""Модель для главного баннера на сайте"""
|
||
title = models.CharField(max_length=200, verbose_name="Заголовок")
|
||
subtitle = models.TextField(blank=True, verbose_name="Подзаголовок")
|
||
description = models.TextField(blank=True, verbose_name="Описание")
|
||
image = models.ImageField(upload_to='static/img/hero/', blank=True, null=True, verbose_name="Фоновое изображение")
|
||
video = models.FileField(upload_to='static/video/hero/', blank=True, null=True,
|
||
help_text='Фоновое видео для баннера (MP4, WebM)', verbose_name="Фоновое видео")
|
||
video_poster = models.ImageField(upload_to='static/img/hero/posters/', blank=True, null=True,
|
||
help_text='Превью изображение для видео', verbose_name="Превью видео")
|
||
button_text = models.CharField(max_length=100, blank=True, verbose_name="Текст кнопки")
|
||
button_link = models.URLField(blank=True, verbose_name="Ссылка кнопки")
|
||
is_active = models.BooleanField(default=True, verbose_name="Активен")
|
||
order = models.PositiveIntegerField(default=0, verbose_name="Порядок отображения")
|
||
created_at = models.DateTimeField(auto_now_add=True)
|
||
|
||
class Meta:
|
||
verbose_name = 'Hero Баннер'
|
||
verbose_name_plural = 'Hero Баннеры'
|
||
ordering = ['order', '-created_at']
|
||
|
||
def __str__(self):
|
||
return self.title
|
||
|
||
class Category(models.Model):
|
||
name = models.CharField(max_length=100)
|
||
description = models.TextField(default='Описание категории')
|
||
|
||
class Meta:
|
||
verbose_name = 'Категория'
|
||
verbose_name_plural = 'Категории'
|
||
ordering = ['name']
|
||
|
||
def __str__(self):
|
||
return self.name
|
||
|
||
class Service(models.Model):
|
||
name = models.CharField(max_length=200)
|
||
description = models.TextField(default='Описание услуги')
|
||
price = models.DecimalField(max_digits=10, decimal_places=2)
|
||
category = models.ForeignKey(Category, on_delete=models.SET_NULL, null=True, related_name='services')
|
||
image = models.ImageField(upload_to='static/img/services/', blank=True, null=True)
|
||
video = models.FileField(upload_to='static/video/services/', blank=True, null=True, help_text='Видео файл для услуги (MP4, WebM, AVI)')
|
||
video_poster = models.ImageField(upload_to='static/img/services/posters/', blank=True, null=True, help_text='Превью изображение для видео')
|
||
video = models.FileField(upload_to='static/video/services/', blank=True, null=True, help_text='Видео файл для услуги (MP4, WebM, AVI)')
|
||
video_poster = models.ImageField(upload_to='static/img/services/posters/', blank=True, null=True, help_text='Превью изображение для видео')
|
||
|
||
class Meta:
|
||
verbose_name = 'Услуга'
|
||
verbose_name_plural = 'Услуги'
|
||
ordering = ['name']
|
||
|
||
def __str__(self):
|
||
return self.name
|
||
|
||
def average_rating(self):
|
||
reviews = self.reviews.all()
|
||
if reviews:
|
||
return sum(review.rating for review in reviews) / reviews.count()
|
||
return 0
|
||
|
||
def review_count(self):
|
||
return self.reviews.count()
|
||
|
||
class Client(models.Model):
|
||
user = models.OneToOneField(User, on_delete=models.CASCADE, related_name='client_profile', null=True, blank=True)
|
||
first_name = models.CharField(max_length=100)
|
||
last_name = models.CharField(max_length=100)
|
||
email = models.EmailField(unique=True)
|
||
phone_number = models.CharField(max_length=15, unique=True)
|
||
image = models.ImageField(upload_to='static/img/customer/', blank=True, null=True)
|
||
chat_id = models.CharField(max_length=100, blank=True, null=True) # Telegram chat ID
|
||
|
||
class Meta:
|
||
verbose_name = 'Клиент'
|
||
verbose_name_plural = 'Клиенты'
|
||
ordering = ['last_name', 'first_name']
|
||
|
||
def __str__(self):
|
||
return f"{self.first_name} {self.last_name} {self.chat_id}"
|
||
|
||
class BlogPost(models.Model):
|
||
title = models.CharField(max_length=200)
|
||
content = models.TextField()
|
||
published_date = models.DateTimeField(auto_now_add=True)
|
||
image = models.ImageField(upload_to='static/img/blog/', blank=True, null=True)
|
||
video = models.FileField(upload_to='static/video/blog/', blank=True, null=True, help_text='Видео файл для блог поста')
|
||
video_poster = models.ImageField(upload_to='static/img/blog/posters/', blank=True, null=True, help_text='Превью изображение для видео')
|
||
|
||
class Meta:
|
||
verbose_name = 'Блог'
|
||
verbose_name_plural = 'Блоги'
|
||
ordering = ['-published_date']
|
||
|
||
def __str__(self):
|
||
return self.title
|
||
|
||
class ServiceRequest(models.Model):
|
||
service = models.ForeignKey(Service, on_delete=models.CASCADE)
|
||
client = models.ForeignKey(Client, on_delete=models.CASCADE, related_name='related_service_requests', null=True, blank=True)
|
||
created_at = models.DateTimeField(auto_now_add=True)
|
||
token = models.UUIDField(default=uuid.uuid4, unique=True) # Генерация уникального токена
|
||
chat_id = models.CharField(max_length=100, blank=True, null=True) # Telegram chat ID
|
||
is_verified = models.BooleanField(default=False)
|
||
|
||
class Meta:
|
||
verbose_name = 'Заявка на услугу'
|
||
verbose_name_plural = 'Заявки на услуги'
|
||
ordering = ['-is_verified', '-created_at']
|
||
|
||
def __str__(self):
|
||
return f"Request for {self.service.name} by {self.client.first_name}"
|
||
|
||
class Order(models.Model):
|
||
service_request = models.OneToOneField(ServiceRequest, on_delete=models.CASCADE, related_name='related_order')
|
||
client = models.ForeignKey(Client, on_delete=models.CASCADE, related_name='related_orders')
|
||
service = models.ForeignKey(Service, on_delete=models.CASCADE, related_name='related_orders')
|
||
message = models.TextField(blank=True, null=True)
|
||
order_date = models.DateTimeField(auto_now_add=True)
|
||
status = models.CharField(
|
||
max_length=50,
|
||
choices=[
|
||
('pending', 'Ожидание'),
|
||
('in_progress', 'В процессе'),
|
||
('completed', 'Завершен'),
|
||
('cancelled', 'Отменён')
|
||
],
|
||
default='pending'
|
||
)
|
||
|
||
class Meta:
|
||
ordering = ['-order_date']
|
||
verbose_name = 'Заказ'
|
||
verbose_name_plural = 'Заказы'
|
||
|
||
def __str__(self):
|
||
return f"Order #{self.id} by {self.client.first_name}"
|
||
|
||
def is_completed(self):
|
||
return self.status == 'completed'
|
||
|
||
def get_absolute_url(self):
|
||
return reverse('order_detail', kwargs={'pk': self.pk})
|
||
|
||
class Project(models.Model):
|
||
name = models.CharField(max_length=200)
|
||
description = models.TextField(default='Описание проекта')
|
||
completion_date = models.DateField(blank=True, null=True)
|
||
client = models.ForeignKey(Client, on_delete=models.CASCADE, related_name='projects')
|
||
service = models.ForeignKey(Service, on_delete=models.CASCADE, related_name='projects')
|
||
order = models.OneToOneField(Order, on_delete=models.CASCADE, related_name='project', null=True, blank=True)
|
||
category = models.ForeignKey(Category, on_delete=models.SET_NULL, null=True, blank=True)
|
||
image = models.ImageField(upload_to='static/img/project/', blank=True, null=True)
|
||
video = models.FileField(upload_to='static/video/project/', blank=True, null=True, help_text='Видео презентация проекта')
|
||
video_poster = models.ImageField(upload_to='static/img/project/posters/', blank=True, null=True, help_text='Превью изображение для видео проекта')
|
||
status = models.CharField(max_length=50, choices=[('in_progress', 'В процессе'), ('completed', 'Завершен')], default='in_progress')
|
||
|
||
class Meta:
|
||
verbose_name = 'Проект'
|
||
verbose_name_plural = 'Проекты'
|
||
ordering = ['-completion_date']
|
||
|
||
def __str__(self):
|
||
return self.name
|
||
|
||
class Review(models.Model):
|
||
client = models.ForeignKey(Client, on_delete=models.CASCADE, related_name='reviews')
|
||
service = models.ForeignKey(Service, on_delete=models.CASCADE, related_name='reviews')
|
||
project = models.ForeignKey(Project, on_delete=models.CASCADE, related_name='reviews', blank=True, null=True)
|
||
rating = models.IntegerField()
|
||
comment = models.TextField()
|
||
review_date = models.DateTimeField(auto_now_add=True)
|
||
image = models.ImageField(upload_to='static/img/review/', blank=True, null=True)
|
||
video = models.FileField(upload_to='static/video/review/', blank=True, null=True, help_text='Видео отзыв о работе')
|
||
video_poster = models.ImageField(upload_to='static/img/review/posters/', blank=True, null=True, help_text='Превью для видео отзыва')
|
||
|
||
class Meta:
|
||
verbose_name = 'Отзыв'
|
||
verbose_name_plural = 'Отзывы'
|
||
ordering = ['-review_date']
|
||
|
||
def __str__(self):
|
||
return f"Отзыв от {self.client.first_name} {self.client.last_name} for {self.service.name}"
|
||
|