diff --git a/myproject/myproject/settings.py b/myproject/myproject/settings.py index 9ab9c7a..99d4ade 100644 --- a/myproject/myproject/settings.py +++ b/myproject/myproject/settings.py @@ -66,6 +66,7 @@ TENANT_APPS = [ # Приложения с бизнес-логикой (изолированные для каждого магазина) 'simple_history', # История изменений для каждого тенанта 'nested_admin', + 'django_filters', # Фильтрация данных 'customers', # Клиенты магазина 'shops', # Точки магазина/самовывоза 'products', # Товары и категории diff --git a/myproject/orders/filters.py b/myproject/orders/filters.py new file mode 100644 index 0000000..078e7bb --- /dev/null +++ b/myproject/orders/filters.py @@ -0,0 +1,136 @@ +# -*- coding: utf-8 -*- +""" +Фильтры для заказов с использованием django-filter +""" + +import django_filters +from django import forms +from django.db.models import Q +from .models import Order + + +class OrderFilter(django_filters.FilterSet): + """ + Фильтр для списка заказов + Поддерживает фильтрацию по: + - Поиску (номер, клиент, телефон, email) + - Дате доставки (диапазон) + - Дате создания (диапазон) + - Статусу заказа + - Типу доставки + - Статусу оплаты + """ + + # Поиск по нескольким полям + search = django_filters.CharFilter( + method='filter_search', + label='Поиск', + widget=forms.TextInput(attrs={ + 'class': 'form-control', + 'placeholder': 'Номер заказа, клиент, телефон...' + }) + ) + + # Фильтр по диапазону дат доставки + delivery_date_after = django_filters.DateFilter( + field_name='delivery_date', + lookup_expr='gte', + label='Дата доставки от', + widget=forms.DateInput(attrs={ + 'class': 'form-control date-input', + 'type': 'date' + }) + ) + + delivery_date_before = django_filters.DateFilter( + field_name='delivery_date', + lookup_expr='lte', + label='Дата доставки до', + widget=forms.DateInput(attrs={ + 'class': 'form-control date-input', + 'type': 'date' + }) + ) + + # Фильтр по диапазону дат создания + created_at_after = django_filters.DateFilter( + field_name='created_at', + lookup_expr='gte', + label='Дата создания от', + widget=forms.DateInput(attrs={ + 'class': 'form-control date-input', + 'type': 'date' + }) + ) + + created_at_before = django_filters.DateFilter( + field_name='created_at', + lookup_expr='lte', + label='Дата создания до', + widget=forms.DateInput(attrs={ + 'class': 'form-control date-input', + 'type': 'date' + }) + ) + + # Фильтр по статусу + status = django_filters.ChoiceFilter( + choices=Order.STATUS_CHOICES, + empty_label='Все статусы', + label='Статус', + widget=forms.Select(attrs={'class': 'form-select'}) + ) + + # Фильтр по типу доставки + delivery_type = django_filters.ChoiceFilter( + method='filter_delivery_type', + choices=[ + ('delivery', 'Доставка'), + ('pickup', 'Самовывоз') + ], + empty_label='Все типы', + label='Тип', + widget=forms.Select(attrs={'class': 'form-select'}) + ) + + # Фильтр по статусу оплаты + payment_status = django_filters.ChoiceFilter( + choices=Order.PAYMENT_STATUS_CHOICES, + empty_label='Все статусы оплаты', + label='Оплата', + widget=forms.Select(attrs={'class': 'form-select'}) + ) + + class Meta: + model = Order + fields = ['search', 'status', 'delivery_type', 'payment_status', + 'delivery_date_after', 'delivery_date_before', + 'created_at_after', 'created_at_before'] + + def filter_search(self, queryset, name, value): + """ + Кастомный метод для поиска по нескольким полям: + - Номер заказа + - Имя клиента + - Телефон клиента + - Email клиента + """ + if not value: + return queryset + + return queryset.filter( + Q(order_number__icontains=value) | + Q(customer__name__icontains=value) | + Q(customer__phone__icontains=value) | + Q(customer__email__icontains=value) + ) + + def filter_delivery_type(self, queryset, name, value): + """ + Кастомный фильтр для типа доставки + """ + if value == 'delivery': + return queryset.filter(is_delivery=True) + elif value == 'pickup': + return queryset.filter(is_delivery=False) + return queryset diff --git a/myproject/orders/static/orders/css/date_filter.css b/myproject/orders/static/orders/css/date_filter.css new file mode 100644 index 0000000..6fe3f23 --- /dev/null +++ b/myproject/orders/static/orders/css/date_filter.css @@ -0,0 +1,112 @@ +/** + * Стили для календарного фильтра по датам + * Используется в компоненте date_range_filter.html + */ + +.date-range-filter { + padding: 1rem; + background: #f8f9fa; + border-radius: 8px; + border: 1px solid #dee2e6; +} + +.date-range-filter .form-label { + font-weight: 500; + color: #495057; + margin-bottom: 0.75rem; + display: flex; + align-items: center; + gap: 0.5rem; +} + +.date-range-filter .form-label i { + color: #0d6efd; +} + +.date-range-filter .date-input { + cursor: pointer; + transition: border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out; +} + +.date-range-filter .date-input:focus { + border-color: #0d6efd; + box-shadow: 0 0 0 0.2rem rgba(13, 110, 253, 0.15); +} + +.date-range-filter .text-muted.small { + font-size: 0.75rem; + margin-bottom: 0.25rem; +} + +/* Быстрые кнопки фильтров */ +.quick-filters { + margin-top: 0.75rem; +} + +.quick-filters .btn-group { + display: flex; + gap: 0.25rem; +} + +.quick-date-btn { + font-size: 0.75rem; + padding: 0.375rem 0.5rem; + border-radius: 4px !important; + transition: all 0.2s ease; + flex: 1; + white-space: nowrap; +} + +.quick-date-btn:hover { + background-color: #0d6efd; + color: white; + border-color: #0d6efd; + transform: translateY(-1px); + box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); +} + +.quick-date-btn:active { + transform: translateY(0); + box-shadow: none; +} + +.quick-date-btn.active { + background-color: #0d6efd; + color: white; + border-color: #0d6efd; +} + +/* Адаптивность для мобильных устройств */ +@media (max-width: 576px) { + .date-range-filter { + padding: 0.75rem; + } + + .quick-filters .btn-group { + flex-wrap: wrap; + } + + .quick-date-btn { + flex: 1 1 calc(33.333% - 0.25rem); + min-width: 0; + font-size: 0.7rem; + padding: 0.35rem 0.4rem; + } +} + +/* Анимация для визуальной обратной связи */ +@keyframes pulse { + 0% { + transform: scale(1); + } + 50% { + transform: scale(1.05); + } + 100% { + transform: scale(1); + } +} + +.quick-date-btn.clicked { + animation: pulse 0.3s ease; +} diff --git a/myproject/orders/static/orders/js/date_filter.js b/myproject/orders/static/orders/js/date_filter.js new file mode 100644 index 0000000..2983cba --- /dev/null +++ b/myproject/orders/static/orders/js/date_filter.js @@ -0,0 +1,131 @@ +/** + * Календарный фильтр для выбора диапазона дат + * Поддерживает быстрые фильтры (сегодня, завтра, неделя) + * + * Использование: + * Подключить этот файл в шаблоне после компонента date_range_filter.html + */ + +document.addEventListener('DOMContentLoaded', function() { + console.log('Date filter initialized'); + + const quickDateButtons = document.querySelectorAll('.quick-date-btn'); + + quickDateButtons.forEach(button => { + button.addEventListener('click', function(e) { + e.preventDefault(); + + const period = this.getAttribute('data-period'); + const minInputId = this.getAttribute('data-min-input'); + const maxInputId = this.getAttribute('data-max-input'); + + const minInput = document.getElementById(minInputId); + const maxInput = document.getElementById(maxInputId); + + if (!minInput || !maxInput) { + console.error('Date inputs not found:', minInputId, maxInputId); + return; + } + + const dates = getDateRange(period); + + minInput.value = dates.min; + maxInput.value = dates.max; + + // Визуальная обратная связь + this.classList.add('clicked'); + setTimeout(() => this.classList.remove('clicked'), 300); + + console.log(`Set date range: ${dates.min} - ${dates.max}`); + }); + }); + + /** + * Вычисляет диапазон дат для выбранного периода + * @param {string} period - период (today, tomorrow, week) + * @returns {Object} объект с min и max датами в формате YYYY-MM-DD + */ + function getDateRange(period) { + const today = new Date(); + const tomorrow = new Date(today); + tomorrow.setDate(tomorrow.getDate() + 1); + + let minDate, maxDate; + + switch(period) { + case 'today': + minDate = maxDate = formatDate(today); + break; + case 'tomorrow': + minDate = maxDate = formatDate(tomorrow); + break; + case 'week': + minDate = formatDate(today); + const weekEnd = new Date(today); + weekEnd.setDate(weekEnd.getDate() + 6); + maxDate = formatDate(weekEnd); + break; + case 'month': + minDate = formatDate(today); + const monthEnd = new Date(today); + monthEnd.setMonth(monthEnd.getMonth() + 1); + maxDate = formatDate(monthEnd); + break; + default: + minDate = maxDate = ''; + } + + return { min: minDate, max: maxDate }; + } + + /** + * Форматирует дату в формат YYYY-MM-DD для input[type="date"] + * @param {Date} date - объект даты + * @returns {string} дата в формате YYYY-MM-DD + */ + function formatDate(date) { + const year = date.getFullYear(); + const month = String(date.getMonth() + 1).padStart(2, '0'); + const day = String(date.getDate()).padStart(2, '0'); + return `${year}-${month}-${day}`; + } + + /** + * Валидация диапазона дат (начало <= конец) + */ + const dateInputs = document.querySelectorAll('.date-input'); + dateInputs.forEach(input => { + input.addEventListener('change', function() { + const container = this.closest('.date-range-filter'); + if (!container) return; + + const minInput = container.querySelector('.date-input[id$="_after"]'); + const maxInput = container.querySelector('.date-input[id$="_before"]'); + + if (!minInput || !maxInput) return; + + if (minInput.value && maxInput.value) { + const minDate = new Date(minInput.value); + const maxDate = new Date(maxInput.value); + + if (minDate > maxDate) { + alert('Дата начала не может быть позже даты окончания'); + this.value = ''; + } + } + }); + }); + + /** + * Сброс дат при клике на кнопку "Сбросить" формы + */ + const resetButtons = document.querySelectorAll('a[href*="order-list"]:not([href*="?"])'); + resetButtons.forEach(button => { + button.addEventListener('click', function() { + // Очищаем все date inputs + dateInputs.forEach(input => { + input.value = ''; + }); + }); + }); +}); diff --git a/myproject/orders/templates/orders/components/date_range_filter.html b/myproject/orders/templates/orders/components/date_range_filter.html new file mode 100644 index 0000000..599fcef --- /dev/null +++ b/myproject/orders/templates/orders/components/date_range_filter.html @@ -0,0 +1,57 @@ +{% comment %} +Переиспользуемый компонент для фильтрации по диапазону дат + +Параметры: +- field_after: поле фильтра "от" (например, filter.form.delivery_date_after) +- field_before: поле фильтра "до" (например, filter.form.delivery_date_before) +- label: заголовок фильтра (например, "Дата доставки") +- icon: иконка Bootstrap Icons (default: calendar-range) + +Пример использования: +{% include 'orders/components/date_range_filter.html' with field_after=filter.form.delivery_date_after field_before=filter.form.delivery_date_before label="Дата доставки" icon="truck" %} +{% endcomment %} + +{% load static %} + +