Skip to content

Commit

Permalink
make auctions list filterable
Browse files Browse the repository at this point in the history
  • Loading branch information
= committed Oct 5, 2024
1 parent 9933b78 commit a654049
Show file tree
Hide file tree
Showing 7 changed files with 145 additions and 57 deletions.
31 changes: 31 additions & 0 deletions auctions/filters.py
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,37 @@
)


class AuctionFilter(django_filters.FilterSet):
"""Filter for the main auctions list"""

query = django_filters.CharFilter(
method="auction_search",
label="",
widget=TextInput(
attrs={
"placeholder": "Filter by auction name, or type a number to see nearby auctions",
"hx-get": "",
"hx-target": "div.table-container",
"hx-trigger": "keyup changed delay:300ms",
"hx-swap": "outerHTML",
"hx-indicator": ".progress",
}
),
)

class Meta:
model = Auction
fields = [] # nothing here so no buttons show up

def auction_search(self, queryset, name, value):
if value == "joined":
return queryset.exclude(joined=False).exclude(joined=0)
if value.isnumeric():
return queryset.filter(distance__lte=int(value))
else:
return queryset.filter(title__icontains=value)


class AuctionTOSFilter(django_filters.FilterSet):
"""This filter is used on any admin views that allow adding users to an auction and on lot creation/winner screens"""

Expand Down
4 changes: 2 additions & 2 deletions auctions/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -1073,12 +1073,12 @@ def template_lot_link(self):
@property
def template_lot_link_first_column(self):
"""Shown on small screens only"""
return mark_safe(f'<span class="d-md-none"><br>{self.template_lot_link}</span>')
return mark_safe(f'<small><span class="d-md-none"><br>{self.template_lot_link}</span></small>')

@property
def template_lot_link_seperate_column(self):
"""Shown on big screens only"""
return mark_safe(f'<td class="d-none d-md-table-cell">{self.template_lot_link}</td>')
return mark_safe(f'<span class="d-none d-md-inline">{self.template_lot_link}</span>')

@property
def can_submit_lots(self):
Expand Down
53 changes: 52 additions & 1 deletion auctions/tables.py
Original file line number Diff line number Diff line change
@@ -1,8 +1,9 @@
import django_tables2 as tables
from django.urls import reverse
from django.utils import formats
from django.utils.safestring import mark_safe

from .models import AuctionTOS, Lot
from .models import Auction, AuctionTOS, Lot


class AuctionTOSHTMxTable(tables.Table):
Expand Down Expand Up @@ -159,6 +160,56 @@ class Meta:
# }


class AuctionHTMxTable(tables.Table):
hide_string = "d-md-table-cell d-none"

auction = tables.Column(accessor="title", verbose_name="Auction")
date = tables.Column(accessor="date_start", verbose_name="Status")
lots = tables.Column(
accessor="template_lot_link_seperate_column",
verbose_name="Lots",
orderable=False,
attrs={"th": {"class": hide_string}, "cell": {"class": hide_string}},
)

def render_date(self, value, record):
localized_date = formats.date_format(record.template_date_timestamp, use_l10n=True)
return mark_safe(f"{record.template_status}{localized_date}{record.ended_badge}")

def render_auction(self, value, record):
auction = record
result = f"<a href='{auction.get_absolute_url()}'>{auction.title}</a><br class='d-md-none'>"
if auction.is_last_used:
result += " <span class='ms-1 badge bg-light text-black'>Your last auction</span>"
if auction.is_online:
result += " <span class='badge bg-info'>Online</span>"
if not auction.promote_this_auction:
result += " <span class='badge bg-dark'>Not promoted</span>"
if auction.distance:
result += f" <span class='badge bg-primary'>{int(auction.distance)} miles from you</span>"
if auction.joined:
result += " <span class='badge bg-success text-black'>Joined</span>"
result += auction.template_lot_link_first_column + auction.template_promo_info
return mark_safe(result)

class Meta:
model = Auction
template_name = "tables/bootstrap_htmx.html"
fields = (
"auction",
"date",
"lots",
)
row_attrs = {
# 'class': lambda record: str(record.table_class),
# 'style':'cursor:pointer;',
# 'hx-get': lambda record: "/api/lot/" + str(record.pk),
# 'hx-target':"#modals-here",
# 'hx-trigger':"click",
# '_':"on htmx:afterOnLoad wait 10ms then add .show to #modal then add .show to #modal-backdrop"
}


class LotHTMxTableForUsers(tables.Table):
hide_string = "d-md-table-cell d-none"
# seller = tables.Column(accessor='auctiontos_seller', verbose_name="Seller")
Expand Down
54 changes: 15 additions & 39 deletions auctions/templates/all_auctions.html
Original file line number Diff line number Diff line change
@@ -1,51 +1,27 @@
{% extends "base.html" %}
{% load render_table from django_tables2 %}
{% load crispy_forms_tags %}

{% block title %}Auctions
{% endblock %}
{% load static %}
{% block content %}
<h3>Auctions</h3>
<div class='mb-3'>This is a list of club auctions which have been created on this site.</div>
<a href="/auctions/new/" class='btn bg-info mb-3 text-white'><i class="bi bi-plus-circle"></i> Create a new auction</a><br>

<form class="col-sm-12 mt-2">
{% crispy filter.form %}
</form>
<div class="progress d-none"> <div class="indeterminate"></div> </div>
{% render_table table %}

<h3>Auctions</h3>
<div class='mb-3'>This is a listing of club auctions which have been created on this site.</div>
<a href="/auctions/new/" class='btn bg-info mb-3 text-white'><i class="bi bi-plus-circle"></i> Create a new auction</a><br>
<table class="table table-responsive">
<thead>
<tr>
<th scope="col">Auction</th>
<th scope="col">Date</th>
<th scope="col" class="d-none d-md-block">Lots</th>
</tr>
</thead>
<tbody>
{% for auction in object_list %}
<tr>
<td><a href='/auctions/{{auction.slug}}'>{{ auction.title }}</a><br class="d-md-none">
{% if auction.is_last_used %}<span class='ms-1 badge bg-light text-black'>Your last auction</span>{% endif %}
{% if auction.is_online %}<span class="badge bg-info">Online</span>{% endif %}
{% if not auction.promote_this_auction %}<span class='badge bg-dark'>Not promoted</span>{% endif %}
{% if not location_message and auction.number_of_locations %}<span class='badge bg-primary'>{{ auction.distance | floatformat:0 }} miles from you</span>{% endif %}
{{ auction.template_lot_link_first_column }}
{{ auction.template_promo_info }}
</td>
<td>
{{ auction.template_status }}
{{ auction.template_date_timestamp }}
{{ auction.ended_badge }}
</td>
{{ auction.template_lot_link_seperate_column }}
</tr>
{% endfor %}
</tbody>
</table>
<small class="text-muted">Note: Auctions you haven't joined won't appear in this list if they:
<small class="text-muted">Auctions you've joined will always show up here. Other auctions will only be listed here if they:
<ul>
<li>aren't related to the fish hobby</li>
<li>are set to "do not promote"</li>
<li>are starting more than 90 days from today</li>
<li>were created more than 2 years ago</li>
<li>are related to the fish hobby</li>
<li>have "promote this auction" checked</li>
<li>are starting less than 90 days from today</li>
<li>were created less than 2 years ago</li>
</ul>
Auctions you've joined will always show up here.</small></span>
</small>
{% endblock %}
{% block extra_js %}<script type='text/javascript'>pageView();</script>{% endblock %}
7 changes: 6 additions & 1 deletion auctions/templates/tables/table_generic.html
Original file line number Diff line number Diff line change
@@ -1,3 +1,8 @@
{% load render_table from django_tables2 %}

{% if no_results %}
<div class="table-container mt-2 mb-2">
{{ no_results | safe }}
</div>
{% else %}
{% render_table table %}
{% endif %}
4 changes: 2 additions & 2 deletions auctions/urls.py
Original file line number Diff line number Diff line change
Expand Up @@ -133,8 +133,8 @@
),
path("images/<int:pk>/delete/", views.ImageDelete.as_view(), name="delete_image"),
path("images/<int:pk>/edit", views.ImageUpdateView.as_view(), name="edit_image"),
path("auctions/", views.allAuctions.as_view(), name="auctions"),
path("auctions/all/", views.allAuctions.as_view()),
path("auctions/", views.AllAuctions.as_view(), name="auctions"),
path("auctions/all/", views.AllAuctions.as_view()),
# path('auctions/new/', views.createAuction, name='createAuction'),
path("auctions/new/", login_required(views.AuctionCreateView.as_view())),
path("auctions/<slug:slug>/edit/", views.AuctionUpdate.as_view(), name="edit_auction"),
Expand Down
49 changes: 37 additions & 12 deletions auctions/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -86,6 +86,7 @@
from webpush.models import PushInformation

from .filters import (
AuctionFilter,
AuctionTOSFilter,
LotAdminFilter,
LotFilter,
Expand Down Expand Up @@ -154,7 +155,7 @@
median_value,
nearby_auctions,
)
from .tables import AuctionTOSHTMxTable, LotHTMxTable, LotHTMxTableForUsers
from .tables import AuctionHTMxTable, AuctionTOSHTMxTable, LotHTMxTable, LotHTMxTableForUsers

logger = logging.getLogger(__name__)

Expand Down Expand Up @@ -4177,10 +4178,19 @@ def toAccount(request):
return redirect(reverse("userpage", kwargs={"slug": request.user.username}))


class allAuctions(LocationMixin, ListView):
class AllAuctions(LocationMixin, SingleTableMixin, FilterView):
model = Auction
template_name = "all_auctions.html"
no_location_message = "Set your location to see how far away auctions are"
table_class = AuctionHTMxTable
filterset_class = AuctionFilter
paginate_by = 100

def get_template_names(self):
if self.request.htmx:
template_name = "tables/table_generic.html"
else:
template_name = "all_auctions.html"
return template_name

def get_queryset(self):
last_auction_pk = -1
Expand Down Expand Up @@ -4213,21 +4223,36 @@ def get_queryset(self):
.values("distance")[:1]
)
qs = qs.annotate(distance=Subquery(closest_pickup_location_subquery))
else:
qs = qs.annotate(distance=Value(0, output_field=FloatField()))
if not self.request.user.is_authenticated:
return qs.filter(standard_filter).distinct()
qs = qs.filter(
Q(auctiontos__user=self.request.user)
| Q(auctiontos__email=self.request.user.email)
| Q(created_by=self.request.user)
| standard_filter
).distinct()
return qs.filter(standard_filter).annotate(joined=Value(0, output_field=FloatField())).distinct()
qs = (
qs.filter(
Q(auctiontos__user=self.request.user)
| Q(auctiontos__email=self.request.user.email)
| Q(created_by=self.request.user)
| standard_filter
)
.annotate(
joined=Exists(
AuctionTOS.objects.filter(
auction=OuterRef("pk"),
user=self.request.user,
)
)
)
.distinct()
)
return qs

def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
context["hide_google_login"] = True
if self.request.user.is_authenticated:
context["last_auction_used"] = self.request.user.userdata.last_auction_used
if not self.object_list.exists():
context["no_results"] = (
"<span class='text-danger'>No auctions found.</span> This only searches club auctions, if you're looking for fish to buy, check out <a href='/lots/'>the list of lots for sale</a>"
)
return context


Expand Down

0 comments on commit a654049

Please sign in to comment.