Evgenii Legotckoi
Evgenii Legotckoi04 липня 2018 р. 02:38

Django - Урок 034. Як зробити пошук по декількох моделях даних

У попередніх статтях ми розглядали, як зробити пошук на сайті на сайті. А саме:

Але що якщо у вас більше, ніж один тип контенту. У вас можуть бути статті, коментарі, форум і повідомлення на форумі. Як тоді бути?

Якщо Ви хочете зробити все самостійно, без застосування сторонніх бібліотек, то тоді потрібно буде зробити пошук по всіх необхідних моделям і об'єднати результат. У мене зроблено точно також на сайті.


urls.py

Вам буде потрібен один View клас, який буде обробляти запит на пошук.

Важливим моментів тут є те, що ми будемо обробляти get запити, щоб користувачі могли ділитися посиланнями з пошуковою видачею.

У файлі urls.py пропишемо маршрут для пошукової видачі

app_name = 'home'
urlpatterns = [
    path('search/', views.SearchView.as_view(), name='search'),
]

views.py

Припустимо у нас є кілька типів контенту:

  • Article
  • Comment
  • Topic
  • Post

Потрібно в поданні виконати пошук по всім видам контенту і об'єднати їх в один QuerySet і підготувати для пагінацію і видачі

from itertools import chain

from django.shortcuts import render
from django.core.paginator import Paginator, PageNotAnInteger, EmptyPage
from django.views import View

from .models import Article, Comment, Topic, Post


class ESearchView(View):
    template_name = 'search/index.html'

    def get(self, request, *args, **kwargs):
        context = {}

        q = request.GET.get('q')
        if q:
            query_sets = []  # Загальний QuerySet

            # Шукаємо по всіх моделях
            query_sets.append(Article.objects.search(query=q))
            query_sets.append(Comment.objects.search(query=q))
            query_sets.append(Topic.objects.search(query=q))
            query_sets.append(Post.objects.search(query=q))

            # і об'єднуємо видачу
            final_set = list(chain(*query_sets))
            final_set.sort(key=lambda x: x.pub_date, reverse=True)  # виконуємо сортування

            context['last_question'] = '?q=%s' % q

            current_page = Paginator(final_set, 10)

            page = request.GET.get('page')
            try:
                context['object_list'] = current_page.page(page)
            except PageNotAnInteger:
                context['object_list'] = current_page.page(1)
            except EmptyPage:
                context['object_list'] = current_page.page(current_page.num_pages)

        return render(request=request, template_name=self.template_name, context=context)

Нюанс вище наведеного коду в тому, що ми об'єднуємо всі моделі даних, а також виконуємо їх сортування за датою. Щоб це було можливим, ми скористаємося можливостями мови програмування Python , а саме качиної типізацією. Сортування за датою стала можливою тому, що всі моделі даних мають поле дати публікації з однаковою назвою pub_date .

Взагалі це дуже важливо, коли ви намагаєтеся називати поля моделей даних однаково для різних моделей даних. Це дозволяє розробляти сайт на Django дуже гнучко і використовуючи качину типізацію писати шаблони для відображення даних, які не залежать від конкретного типу даних, а скоріше, які залежать від інтерфейсу, який підтримують ваші моделі даних.

Також цікаво те, що objects всіх моделей даних, які представлені в даному коді мають один і той же метод search. Даний метод не є стандартним. Для його реалізації необхідно написати власний менеджер моделі і привласнити його полю objects .

ArticleManager

Розглянемо менеджер моделі статей. Успадковуємо його від базового менеджера моделей і визначимо метод search з логікою пошуку. Аналогічним чином потрібно буде прописати даний метод для всіх моделей, в яких буде виконуватися пошук.

from django.db import models
from django.db.models import Q

class ArticleManager(models.Manager):
    use_for_related_fields = True

    def search(self, query=None):
        qs = self.get_queryset()
        if query:
            or_lookup = (Q(title__icontains=query) | Q(content__icontains=query))
            qs = qs.filter(or_lookup)

        return qs

Установка менеджера в модель

class Article(models.Model):
    objects = ArticleManager()

search/index.html

А для шаблону пошуку можна використовувати трохи модифікований шаблон з однією з перших статей з організації пошуку.

{% load bootstrap34%}
{% block page %}
    <h1>Поиск</h1>
    {% if object_list %}
        {% for object in object_list %}
            <div>
                <a href="{{ object.get_absolute_url }}">
                    <h2>{{ object.itle }}</h2>
                </a>
                {{ object.content|safe }}
                <p><a class="btn btn-default btn-sm" href="{{ object.get_absolute_url }}">Читать далее</a></p>
            </div>
        {% endfor %}
        {% bootstrap_pagination object_list url=last_question %}
    {% else %}
        <p>Не найдено публикаций по вашему запросу<br>Попробуйте повторить запрос с другой формулировкой</p>
    {% endif %}
{% endblock %}

Спеціально використовую в даному випадку знеособлений параметр object, який не має вказівки на конкретний тип контенту, щоб було ясно, що там може знаходитися будь-якій об'єкт контента. Головне, щоб були реалізовані всі методи у всіх моделей, які застосовуються в даному шаблоні.

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

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

g
  • 09 липня 2018 р. 05:00

Исправьте код в views.py, пропущен импорт chain из itertools.

Evgenii Legotckoi
  • 10 липня 2018 р. 12:49

Спасибо! Исправил.

S
  • 13 травня 2020 р. 06:36

Добрый день, небольшая проблемка с пагинацией, отображается информация только на первой странице при переходе на другую - пустая страница, views, models.Manager, шаблон, все сделал как в данном уроке, подставил только свои значения. Подскажите, в чем может быть проблема?
Спасибо.

Evgenii Legotckoi
  • 13 травня 2020 р. 06:41

Добрый день.
Покажите ваш код... Невозможно понять, где вы допустили ошибку, не видя, что вы делали.
Когда будете добавлять код в сообщение, используйте диалог вставки программного кода, одна из кнопок в тулбаре редактора комментариев. Тогда будет добавлена соответствующая разметка для блоков кода.

S
  • 13 травня 2020 р. 07:28
Models, остальные похожие:

class PostManager(models.Manager):
    use_for_related_fields = True
    def search(self, query=None):
        qs = self.get_queryset()
        if query:
            or_lookup = (Q(title__icontains=query) | Q(text__icontains=query))
            qs = qs.filter(or_lookup)
        return qs

class Post(models.Model):
    title = models.CharField(max_length=40, verbose_name='Название')
    text = RichTextUploadingField(blank=True, null=True, verbose_name='Текст')
    image = models.ImageField(blank=True, upload_to=get_timestamp_path, verbose_name='Изображение')
    is_active = models.BooleanField(default=True, db_index=True, verbose_name='Выводить в списке?')
    created_at = models.DateTimeField(auto_now_add=True, verbose_name='Создано')
    published_date = models.DateTimeField(blank=True, null=True, verbose_name='Опубликовано')
    slug = models.SlugField(max_length=255, blank=True)
    objects = PostManager()

    def delete(self, *args, **kwargs):
        for ap in self.postImage_set.all():
            ap.delete()
        super().delete(*args, **kwargs)

    def publish(self):
        self.published_date = timezone.now()
        self.save()

    def __str__(self):
        return self.title

    def get_absolute_url(self):
        return '/post_list/%s/' % self.pk

    class Meta:
        verbose_name_plural = 'Публикации'
        verbose_name = 'Публикация'


Views:
class SearchView(View):
    template_name = 'main/search.html'

    def get(self, request, *args, **kwargs):
        context = {}

        q = request.GET.get('q')
        if q:
            query_sets = []  # Общий QuerySet

            # Ищем по всем моделям
            query_sets.append(Bb.objects.search(query=q))
            query_sets.append(Article.objects.search(query=q))
            query_sets.append(Post.objects.search(query=q))


            # и объединяем выдачу
            final_set = list(chain(*query_sets))
            final_set.sort(key=lambda x: x.created_at, reverse=True)  # Выполняем сортировку

            context['last_question'] = '?q=%s' % query_sets

            paginator = Paginator( final_set, 2)

            page = request.GET.get('page')

            try:
                context['object_list'] = paginator.page(page)
            except PageNotAnInteger:
                context['object_list'] = paginator.page(1)
            except EmptyPage:
                context['object_list'] = paginator.page(paginator.num_pages)
        return render(request=request, template_name=self.template_name, context=context )




Шаблон:
{% extends "layout/basic.html" %}
{% load thumbnail %}
{% load static %}
{% load bootstrap4 %}

{% block content %}

 <h1>Поиск</h1>
    {% if object_list %}
        {% for object in object_list %}

            <div>
                <a href="{{ object.get_absolute_url }}">
                    <h2>{{ object.title }}</h2>
                </a>
                {{ object.text|safe }}
                <p><a class="btn btn-default btn-sm" href="{{ object.get_absolute_url }}">Читать далее</a></p>
            </div>

        {% endfor %}

        {% bootstrap_pagination object_list  url=last_question %}

    {% else %}
        <p>Не найдено публикаций по вашему запросу<br>Попробуйте повторить запрос с другой формулировкой</p>
    {% endif %}
{% endblock %}


urls:
path('search/', views.SearchView.as_view(), name='search'),


кнопка:

<form class="navbar-form navbar-left m-2 p-4" action="{% url 'main:search' %}">
            <div class="input-group">
                <input id="search" name="q" type="text" class="form-control" placeholder="Найти">
                <div class="input-group-btn">
                    <button class="btn btn-success" type="submit">
                        <i class="fas fa-search"></i>
                    </button>
                </div>
            </div>
        </form>
Evgenii Legotckoi
  • 13 травня 2020 р. 15:50

Опечатка в статье. Напишите так

context['last_question'] = '?q=%s' % q
Evgenii Legotckoi
  • 13 травня 2020 р. 16:13
  • (відредаговано)

И на старуху бывает проруха. Спасибо за вопрос.
Если бы не просмотрел ваш код, то скорее всего не заметил бы опечатку

S
  • 13 травня 2020 р. 16:18

Спасибо, Евгений, все заработало. Хороший сайт.

Evgenii Legotckoi
  • 13 травня 2020 р. 16:26

Спасибо за отзыв. Не стесняйтесь задавать вопросы. Только у меня убедительная просьба, все вопросы или на форуме сайта, или в комментариях к статьям. Чтобы потом у других пользователей было больше шансов найти ответ. Также на форуме присутствуют и другие опытные специалисты, они вполне могут тоже что-то подсказать, так что шансы получить ответ будут выше.

S
  • 14 травня 2020 р. 12:58

Здравствуйте, небольшой вопрос,
при добавлении формы пагинатора левая крайняя кнопка назад "<<" постоянно смещена вверх, относительно ряда цифр, не могу ее никак выровнять, в чем может быть проблема?
{% bootstrap_pagination object_list url=last_question %}
Хотел установить пагинатор прописав код, но немного запутался с адресами между страниц.
Подскажите, как правильно прописать адреса.
Спасибо.

<ul class="pagination">
            {% if object_list.has_previous %}
            <li class="page-item"><a class="page-link" href="?q={{ object_list.previous_page_number }}">&lt;</a></li>
            <li class="page-item"><a class="page-link" href="?q={{ object_list.previous_page_number }}">{{ object_list.previous_page_number }}</a></li>
            {% endif %}

            <li class="page-item active"><a class="page-link" href="?q={{ object_list.number }}">{{ object_list.number }}</a>
            </li>

            {% if object_list.has_next %}
            <li class="page-item"><a class="page-link" href="?q={{ object_list.next_page_number }}">{{ object_list.next_page_number }}</a></li>
            <li class="page-item"><a class="page-link" href="?q={{ object_list.next_page_number }}">&gt;</a></li>
            {% endif %}
        </ul>
Evgenii Legotckoi
  • 15 травня 2020 р. 03:08

У вас есть ещё какие-то кастомные стили, которые могут влиять на "<<", поскольку у меня такой проблемы не наблюдалось.

Ну по идее, это как-то тако должно выглядеть

href="?q={{q}}&page={{ object_list.previous_page_number }}"
S
  • 15 травня 2020 р. 05:03

Добрый день, Евгений,
сработал вот такой вариант:
href="?q={{ question }}&page={{ object_list.previous_page_number }}"
и во views добавил:
context['question'] = q
По поводу формы пагинатора, пробовал отключать все дополнительные стили, все также вылезает ошибка. Подумаю еще над этим, главное что есть рабочий вариант.
Спасибо за подсказку.

Коментарі

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
innorwall13 листопада 2024 р. 23:03
Як зробити гру за допомогою Qt - Урок 3. Взаємодія з іншими об&#39;єктами what is priligy tablets What happens during the LASIK surgery process
i
innorwall13 листопада 2024 р. 20:09
Використання змінних оголошених в CMakeLists.txt всередині C++ файлів where can i buy priligy online safely Tom Platz How about things like we read about in the magazines like roid rage and does that really
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
innorwall14 листопада 2024 р. 00:39
добавить qlineseries в функции Listen intently to what Jerry says about Conditional Acceptance because that s the bargaining chip in the song and dance you will have to engage in to protect yourself and your family from AMI S…
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
Реализация навигации по разделам Спасибо Евгений!

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