Evgenii Legotckoi
Evgenii Legotckoi28 лютого 2018 р. 04:57

Django - Підручник 033. Передача списку аргументів методу order_by для сортування QuerySet

Для покращення зручності використання розділів статей було впроваджено сортування статей за датою, заголовком та кількістю переглядів. Крім цього, була додана можливість пошуку інформації за статтями розділу. Реалізується ця можливість через кілька чекбоксів, які додають назви колонок для сортування URL сторінки, відповідно сторінка перезавантажується.

Наприклад, у моделі даних є кілька колонок

  • назва
  • дата_публікування
  • перегляди

За ними і будемо робити сортування, яке у звичайному варіанті запиту виглядало б так

Article.objects.all().order_by('title', 'pub_date', 'views')

Але оскільки ми використовуємо чекбокси, то варіанти сортування можуть бути присутніми, а можу і відсутніми. Але не будемо ми писати if else блоки на кожну комбінацію чекбоксів? Звичайно, ні.


Подивимося для початку, як може бути написана форма для реалізації сортування. Відразу зазначу, що наведу варіант форми без стилізації, яка застосована у мене на сайті. Справа в тому, що для цього використовується Bootstrap 4 Material Design, що дещо ускладнюємо варіант верстки і додасть ряду зайвих елементів у приклад.

<form method="get">
    <button type="submit" class="btn btn-sm btn-primary btn-raised mr-3">{% trans 'Сортировать' %}</button>
    <input name="sort" type="checkbox" value="title" {{ by_title }}>{% trans "по заголовку" %}
    <input name="sort" type="checkbox" value="pub_date" {{ by_date }}>{% trans "по дате" %}
    <input name="sort" type="checkbox" value="views" {{ by_views }}>{% trans "по просмотрам" %}
</form>

Як бачите, всі чекбокси в коді мають ім'я sort, а значення value дорівнюватиме імені колонки, за якою можна буде включити сортування.

Таким чином в URL фігуруватимуть такі аргументи

?sort=title&sort=pub_date&sort=views

Django дозволяє вилучити всі аргументи із запиту як список, який ми можемо передати в метод order_by, щоб виконати сортування.

І фактично подання для рендерингу розділу зі статтями може виглядати так

class SectionView(View):

    def get(self, request, slug):
        section = get_object_or_404(Section, slug=slug)
        sort = request.GET.getlist('sort')
        articles = section.article_set.all().order_by(*sort)

        return render(
            request=request,
            template_name='knowledge/section.html',
            context={
                'section': section,
                'articles': articles
            }
        )

Зверніть увагу, що замість методу get використовується метод getlist , який повертає список значень аргументів, якщо в запиті фігурувало одне й теж ім'я аргументу кілька разів.

sort = request.GET.getlist('sort')

А далі за допомогою покажчика передаємо список як аргументи в метод order_by

articles = section.article_set.all().order_by(*sort)

Для Django рекомендую VDS-сервера хостера Timeweb .

Рекомендуємо хостинг TIMEWEB
Рекомендуємо хостинг TIMEWEB
Стабільний хостинг, на якому розміщується соціальна мережа EVILEG. Для проектів на Django радимо VDS хостинг.

Вам це подобається? Поділіться в соціальних мережах!

bernar92
  • 16 травня 2018 р. 23:53
можно же это реализовать через django-filter!
Evgenii Legotckoi
  • 17 травня 2018 р. 02:05

Ну. Массово пока не использую фильтрации, поэтому не искал батареек. Так получилось, что даже не знал про django-filters.

Спасибо. Можете черкануть заметку, как аналог этой статье ;-)
bernar92
  • 17 травня 2018 р. 11:16

хорошие статьи я много чего нашел тут интересного и нового... мне нравиться!
если вдруг интересно будет по фильтрам вот примерный код)

import django_filters
from .models import Product

CHOICES =[
        ["name", "по алфавиту"],
        ["price", "дешевые сверху"],
        ["-price", "дорогие сверху"]
]


class ProductFilter(django_filters.FilterSet):
    name = django_filters.CharFilter(name='name', lookup_expr='icontains')
    category__slug = django_filters.CharFilter()
    price__gt = django_filters.NumberFilter(name='price', lookup_expr='gt')
    price__lt = django_filters.NumberFilter(name='price', lookup_expr='lt')
    ordering = django_filters.OrderingFilter(choices=CHOICES, required=True, empty_label=None,)

    class Meta:
        model = Product
        exclude = [field.name for field in Product._meta.fields]
        order_by_field = 'name'


from django_filters.views import FilterView
class EnumerationListView(FilterView):
    template_name = '.html'
    model = Lot
    paginate_by = 50
    filterset_class = ProductFilter
    context_object_name = 'product_list'

Evgenii Legotckoi
  • 17 травня 2018 р. 16:23

Спасибо за пример кода.
Когда буду внедрять больше поисковых виджетов на сайт, в первую очередь воспользуюсь вашим примером кода. Благо уже есть некоторые целевые места, где это можно применить.

Коментарі

Only authorized users can post comments.
Please, Log in or Sign up
AD

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат:50бали,
  • Рейтинг балів-4
m
  • molni99
  • 26 жовтня 2024 р. 01:37

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат:80бали,
  • Рейтинг балів4
m
  • molni99
  • 26 жовтня 2024 р. 01:29

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат:20бали,
  • Рейтинг балів-10
Останні коментарі
i
innorwall11 листопада 2024 р. 22:12
Django - Урок 055. Як написати функціонал auto populate field Freckles because of several brand names retin a, atralin buy generic priligy
i
innorwall11 листопада 2024 р. 18:23
QML - Підручник 035. Використання перерахувань в QML без C++ priligy cvs 24 Together with antibiotics such as amphotericin B 10, griseofulvin 11 and streptomycin 12, chloramphenicol 9 is in the World Health Organisation s List of Essential Medici…
i
innorwall11 листопада 2024 р. 15:50
Qt/C++ - Урок 052. Налаштування Qt Audio player в стилі AIMP It decreases stress, supports hormone balance, and regulates and increases blood flow to the reproductive organs buy priligy online safe Promising data were reported in a PDX model re…
i
innorwall11 листопада 2024 р. 14:19
Алгоритм сортування купою The role of raloxifene in preventing breast cancer priligy precio
i
innorwall11 листопада 2024 р. 13:55
PyQt5 - Урок 006. Робота з QTableWidget buy priligy 60 mg 53 have been reported by Javanovic Santa et al
Тепер обговоріть на форумі
i
innorwall11 листопада 2024 р. 20:56
добавить qlineseries в функции buy priligy senior brother Chu He, whom he had known for many years
i
innorwall11 листопада 2024 р. 10:55
Всё ещё разбираюсь с кешем. priligy walgreens levitra dulcolax carbs The third ring was found to be made up of ultra relativistic electrons, which are also present in both the outer and inner rings
9
9Anonim25 жовтня 2024 р. 09:10
Машина тьюринга // Начальное состояние 0 0, ,<,1 // Переход в состояние 1 при пустом символе 0,0,>,0 // Остаемся в состоянии 0, двигаясь вправо при встрече 0 0,1,>…
ИМ
Игорь Максимов03 жовтня 2024 р. 04:05
Реализация навигации по разделам Спасибо Евгений!

Слідкуйте за нами в соціальних мережах