Фильтрация и упорядочение по нескольким параметрам request.GET

python django sorting

59 просмотра

1 ответ

В настоящее время я застрял в попытке найти решение моей проблемы. Итак, у меня есть URL, который выглядит так:

https://www.domain.com/forum/topic/

В моем шаблонном виде у меня есть форма и входные данные, которые отвечают за поиск сообщений:

<form method="GET" action="">
    <div class="input-group">
        <input type="text" name="q" placeholder="Search..." value="{{ request.GET.q }}" class="form-control">
        <span class="input-group-btn">
            <input class="btn btn-secondary" type="submit" value="Search">
        </span>
    </div>
</form>

В моем Views.py поиск работает следующим образом:

def discussion(request, discussion):
    topics_list = Topic.objects.all().filter(discussion__url=discussion)
    discussion = Discussion.objects.get(url=discussion)

    search_query = request.GET.get('q')

    if search_query:
        topics_list = topics_list.filter(
            Q(title__icontains=search_query) |
            Q(user__username__icontains=search_query)
            )

    paginator = Paginator(topics_list, 10)
    page = request.GET.get('page')
    try:
        topics = paginator.page(page)
    except PageNotAnInteger:
        topics = paginator.page(1)
    except EmptyPage:
        topics = paginator.page(paginator.num_pages)


    context = {'topics': topics, 'discussion': discussion,}
    return render(request, 'forum/forum_show_posts.html', context)

Теперь, когда я запускаю поиск, он работает нормально, фактически фильтрует объекты на основе моего запроса, таким образом, URL выглядит так:

https://www.domain.com/forum/topic/?q=test

Теперь я хочу поработать с заказом для моих объектов, поэтому я перешел к обсуждению:

def discussion(request, discussion):
    topics_list = Topic.objects.all().filter(discussion__url=discussion)
    discussion = Discussion.objects.get(url=discussion)

    search_query = request.GET.get('q')
    sort_query = request.GET.get('sort')

    if search_query:
        topics_list = topics_list.filter(
            Q(title__icontains=search_query) |
            Q(user__username__icontains=search_query)
            )

    elif sort_query:
        if sort_query == "newest":
            topics_list = topics_list.order_by('-timestamp')
        if sort_query == "oldest":
            topics_list = topics_list.order_by('timestamp')
        if sort_query == "name":
            topics_list = topics_list.order_by('title')

        # sort_query = sort_query.title()

    paginator = Paginator(topics_list, 10)
    page = request.GET.get('page')
    try:
        topics = paginator.page(page)
    except PageNotAnInteger:
        topics = paginator.page(1)
    except EmptyPage:
        topics = paginator.page(paginator.num_pages)


    context = {'topics': topics, 'discussion': discussion, 'sort_value':sort_query,}
    return render(request, 'forum/forum_show_posts.html', context)

и мой шаблон, чтобы иметь соответствующие ссылки для каждого метода заказа:

<div class="dropdown-menu">
    <a class="dropdown-item disabled" href="#">Sort...</a>
        <form method="GET" action="">
            <div class="input-group">
                <button class="dropdown-item" type="submit" name="sort" value="newest">Newest</button>
                <button class="dropdown-item" type="submit" name="sort" value="oldest">Oldest</button>
                <button class="dropdown-item" type="submit" name="sort" value="views">Views</button>
                <button class="dropdown-item" type="submit" name="sort" value="comments">Comments</button>
                <button class="dropdown-item" type="submit" name="sort" value="replies">Replies</button>
                <button class="dropdown-item" type="submit" name="sort" value="name">Name</button>
            </div>
        </form>
</div>

Теперь, когда я на самом деле иду вперед и выбираю порядок по Новейшему или Старейшему, он сортирует их так, что URL выглядит так:

https://www.domain.com/forum/topic/?sort=newest

Моя проблема в том, что, скажем, я хотел найти «тест» для создания URL

https://www.domain.com/forum/topic/?q=test

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

С https://www.domain.com/forum/topic/?q=testна по https://www.domain.com/forum/topic/?q=test&sort=newestТак он показывает самый новый из списка сообщений с запросом «тест».

Автор: Zer0 Источник Размещён: 08.11.2019 11:10

Ответы (1)


1 плюс

Решение

Вы должны следить за вашими GETпараметрами, обновите ваше представление так:

def discussion(request, discussion): # <<- view name and var name both are same which might cause issues
    search_query = request.GET.get('q', '')
    sort = request.GET.get('sort', '')
    direction = request.GET.get('dir', 'asc')
    if direction not in ['asc', 'desc']:
        direction = 'asc'

    topics_list = Topic.objects.all().filter(discussion__url=discussion)
    discussion = Discussion.objects.get(url=discussion)

    if search_query:
        topics_list = topics_list.filter(
            Q(title__icontains=search_query) |
            Q(user__username__icontains=search_query)
        )

    if sort:
        order_by = '{0}{1}'.format('-' if direction == 'desc' else '', sort)
        topics_list = topics_list.order_by(order_by)

    # rest of code
    # pass search_query, sort and direction in context
    context = {
        'topics': topics,
        'discussion': discussion,
        'sort': sort,
        'direction': direction,
        'search_query': search_query,
    }

    return render(request, 'forum/forum_show_posts.html', context)

Теперь в шаблоне отслеживайте эти параметры в обеих формах:

Форма поиска:

<form method="GET" action="">
    <div class="input-group">
        <input type="text" name="q" placeholder="Search..." value="{{ search_query }}" class="form-control">
        <span class="input-group-btn">
            <input class="btn btn-secondary" type="submit" value="Search">
        </span>
    </div>
    <input type="hidden" name="sort" value="{{ sort }}" />
    <input type="hidden" name="direction" value="{{ direction }}" />
</form>

Форма сортировки:

<div class="dropdown-menu">
    <a class="dropdown-item disabled" href="#">Sort...</a>
        <form method="GET" action="">
            <div class="input-group">
                <button class="dropdown-item" type="submit" name="sort" value="newest">Newest</button>
                <button class="dropdown-item" type="submit" name="sort" value="oldest">Oldest</button>
               <button class="dropdown-item" type="submit" name="sort" value="views">Views</button>
               <button class="dropdown-item" type="submit" name="sort" value="comments">Comments</button>
               <button class="dropdown-item" type="submit" name="sort" value="replies">Replies</button>
               <button class="dropdown-item" type="submit" name="sort" value="name">Name</button>
            </div>

            <input type="hidden" name="search_query" value="{{ search_query }}" />
            <input type="hidden" name="direction" value="{{ direction }}" />

        </form>
</div>
Автор: Aamir Adnan Размещён: 20.08.2016 11:03
Вопросы из категории :
32x32