2021-03-08 16:49:10 +00:00
|
|
|
""" book list views"""
|
2021-04-18 18:46:28 +00:00
|
|
|
from typing import Optional
|
|
|
|
|
2021-01-31 05:33:41 +00:00
|
|
|
from django.contrib.auth.decorators import login_required
|
2022-01-26 18:37:50 +00:00
|
|
|
from django.core.exceptions import PermissionDenied
|
2021-02-01 01:34:06 +00:00
|
|
|
from django.core.paginator import Paginator
|
2022-02-28 18:29:58 +00:00
|
|
|
from django.db import transaction
|
2021-11-16 17:21:12 +00:00
|
|
|
from django.db.models import Avg, DecimalField, Q, Max
|
2021-04-08 16:05:21 +00:00
|
|
|
from django.db.models.functions import Coalesce
|
2022-01-25 21:37:57 +00:00
|
|
|
from django.http import HttpResponseBadRequest, HttpResponse
|
2021-01-31 05:33:41 +00:00
|
|
|
from django.shortcuts import get_object_or_404, redirect
|
|
|
|
from django.template.response import TemplateResponse
|
2021-04-26 14:24:03 +00:00
|
|
|
from django.urls import reverse
|
2021-01-31 05:33:41 +00:00
|
|
|
from django.utils.decorators import method_decorator
|
|
|
|
from django.views import View
|
2021-01-31 18:34:25 +00:00
|
|
|
from django.views.decorators.http import require_POST
|
2021-01-31 05:33:41 +00:00
|
|
|
|
2021-09-16 17:55:23 +00:00
|
|
|
from bookwyrm import book_search, forms, models
|
2021-01-31 05:33:41 +00:00
|
|
|
from bookwyrm.activitypub import ActivitypubResponse
|
2021-05-03 21:47:27 +00:00
|
|
|
from bookwyrm.settings import PAGE_LENGTH
|
2022-01-25 01:41:21 +00:00
|
|
|
from bookwyrm.views.helpers import is_api_request
|
2021-01-31 05:33:41 +00:00
|
|
|
|
2021-04-08 16:05:21 +00:00
|
|
|
|
2021-01-31 05:33:41 +00:00
|
|
|
# pylint: disable=no-self-use
|
|
|
|
class List(View):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""book list page"""
|
2021-03-08 16:49:10 +00:00
|
|
|
|
2022-03-02 08:21:23 +00:00
|
|
|
def get(self, request, list_id, **kwargs):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""display a book list"""
|
2022-03-02 08:21:23 +00:00
|
|
|
add_failed = kwargs.get("add_failed", False)
|
|
|
|
add_succeeded = kwargs.get("add_succeeded", False)
|
|
|
|
|
2021-01-31 05:33:41 +00:00
|
|
|
book_list = get_object_or_404(models.List, id=list_id)
|
2021-09-27 23:04:40 +00:00
|
|
|
book_list.raise_visible_to_user(request.user)
|
2021-01-31 05:33:41 +00:00
|
|
|
|
|
|
|
if is_api_request(request):
|
2021-02-01 20:03:11 +00:00
|
|
|
return ActivitypubResponse(book_list.to_activity(**request.GET))
|
2021-01-31 17:08:06 +00:00
|
|
|
|
2021-03-08 16:49:10 +00:00
|
|
|
query = request.GET.get("q")
|
2021-01-31 21:38:26 +00:00
|
|
|
suggestions = None
|
2021-04-08 16:05:21 +00:00
|
|
|
|
2022-02-28 18:41:40 +00:00
|
|
|
items = book_list.listitem_set.filter(approved=True).prefetch_related(
|
|
|
|
"user", "book", "book__authors"
|
|
|
|
)
|
|
|
|
items = sort_list(request, items)
|
2021-04-18 01:31:38 +00:00
|
|
|
|
2021-05-03 21:47:27 +00:00
|
|
|
paginated = Paginator(items, PAGE_LENGTH)
|
2021-04-08 16:05:21 +00:00
|
|
|
|
2021-01-31 21:38:26 +00:00
|
|
|
if query and request.user.is_authenticated:
|
2021-01-31 19:11:26 +00:00
|
|
|
# search for books
|
2021-09-16 17:55:23 +00:00
|
|
|
suggestions = book_search.search(
|
2021-04-26 15:02:30 +00:00
|
|
|
query,
|
|
|
|
filters=[~Q(parent_work__editions__in=book_list.books.all())],
|
|
|
|
)
|
2021-01-31 21:38:26 +00:00
|
|
|
elif request.user.is_authenticated:
|
2021-01-31 19:11:26 +00:00
|
|
|
# just suggest whatever books are nearby
|
|
|
|
suggestions = request.user.shelfbook_set.filter(
|
|
|
|
~Q(book__in=book_list.books.all())
|
|
|
|
)
|
|
|
|
suggestions = [s.book for s in suggestions[:5]]
|
|
|
|
if len(suggestions) < 5:
|
2021-01-31 19:21:50 +00:00
|
|
|
suggestions += [
|
2021-03-08 16:49:10 +00:00
|
|
|
s.default_edition
|
|
|
|
for s in models.Work.objects.filter(
|
|
|
|
~Q(editions__in=book_list.books.all()),
|
|
|
|
).order_by("-updated_date")
|
|
|
|
][: 5 - len(suggestions)]
|
2021-01-31 17:08:06 +00:00
|
|
|
|
2021-05-01 23:30:43 +00:00
|
|
|
page = paginated.get_page(request.GET.get("page"))
|
2021-12-04 15:17:21 +00:00
|
|
|
|
|
|
|
embed_key = str(book_list.embed_key.hex)
|
|
|
|
embed_url = reverse("embed-list", args=[book_list.id, embed_key])
|
|
|
|
embed_url = request.build_absolute_uri(embed_url)
|
|
|
|
|
|
|
|
if request.GET:
|
2021-12-04 16:33:28 +00:00
|
|
|
embed_url = f"{embed_url}?{request.GET.urlencode()}"
|
2021-12-04 15:17:21 +00:00
|
|
|
|
2021-01-31 05:33:41 +00:00
|
|
|
data = {
|
2021-03-08 16:49:10 +00:00
|
|
|
"list": book_list,
|
2021-05-01 23:30:43 +00:00
|
|
|
"items": page,
|
|
|
|
"page_range": paginated.get_elided_page_range(
|
|
|
|
page.number, on_each_side=2, on_ends=1
|
|
|
|
),
|
2021-03-08 16:49:10 +00:00
|
|
|
"pending_count": book_list.listitem_set.filter(approved=False).count(),
|
|
|
|
"suggested_books": suggestions,
|
|
|
|
"list_form": forms.ListForm(instance=book_list),
|
|
|
|
"query": query or "",
|
2022-02-28 18:41:40 +00:00
|
|
|
"sort_form": forms.SortListForm(request.GET),
|
2021-12-04 15:17:21 +00:00
|
|
|
"embed_url": embed_url,
|
2022-02-28 18:29:58 +00:00
|
|
|
"add_failed": add_failed,
|
|
|
|
"add_succeeded": add_succeeded,
|
2021-01-31 05:33:41 +00:00
|
|
|
}
|
2021-03-08 16:49:10 +00:00
|
|
|
return TemplateResponse(request, "lists/list.html", data)
|
2021-01-31 05:33:41 +00:00
|
|
|
|
2021-03-08 16:49:10 +00:00
|
|
|
@method_decorator(login_required, name="dispatch")
|
2021-01-31 05:33:41 +00:00
|
|
|
def post(self, request, list_id):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""edit a list"""
|
2021-01-31 05:33:41 +00:00
|
|
|
book_list = get_object_or_404(models.List, id=list_id)
|
2021-09-27 23:04:40 +00:00
|
|
|
book_list.raise_not_editable(request.user)
|
|
|
|
|
2021-01-31 18:34:25 +00:00
|
|
|
form = forms.ListForm(request.POST, instance=book_list)
|
|
|
|
if not form.is_valid():
|
2022-01-26 19:46:42 +00:00
|
|
|
# this shouldn't happen
|
|
|
|
raise Exception(form.errors)
|
2021-01-31 18:34:25 +00:00
|
|
|
book_list = form.save()
|
2021-09-26 08:28:16 +00:00
|
|
|
if not book_list.curation == "group":
|
2021-10-16 05:38:02 +00:00
|
|
|
book_list.group = None
|
|
|
|
book_list.save(broadcast=False)
|
|
|
|
|
2021-01-31 05:33:41 +00:00
|
|
|
return redirect(book_list.local_path)
|
2021-01-31 18:34:25 +00:00
|
|
|
|
|
|
|
|
2022-02-28 18:41:40 +00:00
|
|
|
def sort_list(request, items):
|
|
|
|
"""helper to handle the surprisngly involved sorting"""
|
|
|
|
# sort_by shall be "order" unless a valid alternative is given
|
|
|
|
sort_by = request.GET.get("sort_by", "order")
|
|
|
|
if sort_by not in ("order", "title", "rating"):
|
|
|
|
sort_by = "order"
|
|
|
|
|
|
|
|
# direction shall be "ascending" unless a valid alternative is given
|
|
|
|
direction = request.GET.get("direction", "ascending")
|
|
|
|
if direction not in ("ascending", "descending"):
|
|
|
|
direction = "ascending"
|
|
|
|
|
|
|
|
directional_sort_by = {
|
|
|
|
"order": "order",
|
|
|
|
"title": "book__title",
|
|
|
|
"rating": "average_rating",
|
|
|
|
}[sort_by]
|
|
|
|
if direction == "descending":
|
|
|
|
directional_sort_by = "-" + directional_sort_by
|
|
|
|
|
|
|
|
if sort_by == "rating":
|
|
|
|
items = items.annotate(
|
|
|
|
average_rating=Avg(
|
|
|
|
Coalesce("book__review__rating", 0.0),
|
|
|
|
output_field=DecimalField(),
|
|
|
|
)
|
|
|
|
)
|
|
|
|
return items.order_by(directional_sort_by)
|
|
|
|
|
|
|
|
|
2021-01-31 18:34:25 +00:00
|
|
|
@require_POST
|
2021-08-23 20:15:35 +00:00
|
|
|
@login_required
|
2021-08-23 22:33:49 +00:00
|
|
|
def save_list(request, list_id):
|
2021-08-23 22:07:38 +00:00
|
|
|
"""save a list"""
|
|
|
|
book_list = get_object_or_404(models.List, id=list_id)
|
2021-08-23 20:15:35 +00:00
|
|
|
request.user.saved_lists.add(book_list)
|
|
|
|
return redirect("list", list_id)
|
|
|
|
|
|
|
|
|
|
|
|
@require_POST
|
|
|
|
@login_required
|
2021-08-23 22:33:49 +00:00
|
|
|
def unsave_list(request, list_id):
|
2021-08-23 20:15:35 +00:00
|
|
|
"""unsave a list"""
|
2021-08-23 22:07:38 +00:00
|
|
|
book_list = get_object_or_404(models.List, id=list_id)
|
2021-08-23 20:15:35 +00:00
|
|
|
request.user.saved_lists.remove(book_list)
|
|
|
|
return redirect("list", list_id)
|
|
|
|
|
|
|
|
|
2021-09-06 17:38:37 +00:00
|
|
|
@require_POST
|
|
|
|
@login_required
|
|
|
|
def delete_list(request, list_id):
|
|
|
|
"""delete a list"""
|
|
|
|
book_list = get_object_or_404(models.List, id=list_id)
|
|
|
|
|
|
|
|
# only the owner or a moderator can delete a list
|
2021-09-27 23:04:40 +00:00
|
|
|
book_list.raise_not_deletable(request.user)
|
2021-09-06 17:38:37 +00:00
|
|
|
|
|
|
|
book_list.delete()
|
|
|
|
return redirect("lists")
|
|
|
|
|
|
|
|
|
2021-08-23 20:15:35 +00:00
|
|
|
@require_POST
|
|
|
|
@login_required
|
2022-01-25 20:10:58 +00:00
|
|
|
@transaction.atomic
|
2021-03-15 21:44:03 +00:00
|
|
|
def add_book(request):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""put a book on a list"""
|
2022-01-25 20:10:58 +00:00
|
|
|
book_list = get_object_or_404(models.List, id=request.POST.get("book_list"))
|
|
|
|
# make sure the user is allowed to submit to this list
|
2022-01-26 18:37:50 +00:00
|
|
|
book_list.raise_visible_to_user(request.user)
|
|
|
|
if request.user != book_list.user and book_list.curation == "closed":
|
|
|
|
raise PermissionDenied()
|
2022-01-26 19:46:42 +00:00
|
|
|
|
2022-01-26 18:37:50 +00:00
|
|
|
is_group_member = models.GroupMember.objects.filter(
|
|
|
|
group=book_list.group, user=request.user
|
|
|
|
).exists()
|
2022-01-25 20:10:58 +00:00
|
|
|
|
|
|
|
form = forms.ListItemForm(request.POST)
|
|
|
|
if not form.is_valid():
|
2022-02-28 18:29:58 +00:00
|
|
|
return List().get(request, book_list.id, add_failed=True)
|
|
|
|
|
2022-01-25 20:10:58 +00:00
|
|
|
item = form.save(commit=False)
|
|
|
|
|
|
|
|
if book_list.curation == "curated":
|
|
|
|
# make a pending entry at the end of the list
|
|
|
|
order_max = (book_list.listitem_set.aggregate(Max("order"))["order__max"]) or 0
|
2022-01-26 18:37:50 +00:00
|
|
|
item.approved = is_group_member or request.user == book_list.user
|
2022-01-25 20:10:58 +00:00
|
|
|
else:
|
|
|
|
# add the book at the latest order of approved books, before pending books
|
|
|
|
order_max = (
|
|
|
|
book_list.listitem_set.filter(approved=True).aggregate(Max("order"))[
|
|
|
|
"order__max"
|
|
|
|
]
|
|
|
|
) or 0
|
|
|
|
increment_order_in_reverse(book_list.id, order_max + 1)
|
|
|
|
item.order = order_max + 1
|
2022-02-28 18:29:58 +00:00
|
|
|
item.save()
|
2021-10-02 23:49:38 +00:00
|
|
|
|
2022-02-28 18:29:58 +00:00
|
|
|
return List().get(request, book_list.id, add_succeeded=True)
|
2021-01-31 20:15:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
@require_POST
|
2021-08-23 20:15:35 +00:00
|
|
|
@login_required
|
2021-01-31 20:15:38 +00:00
|
|
|
def remove_book(request, list_id):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""remove a book from a list"""
|
2021-01-31 20:15:38 +00:00
|
|
|
|
2021-09-28 00:52:27 +00:00
|
|
|
book_list = get_object_or_404(models.List, id=list_id)
|
|
|
|
item = get_object_or_404(models.ListItem, id=request.POST.get("item"))
|
2021-10-02 23:49:38 +00:00
|
|
|
|
2021-09-28 00:52:27 +00:00
|
|
|
item.raise_not_deletable(request.user)
|
2021-01-31 20:15:38 +00:00
|
|
|
|
2021-09-28 00:52:27 +00:00
|
|
|
with transaction.atomic():
|
2021-04-08 16:05:21 +00:00
|
|
|
deleted_order = item.order
|
|
|
|
item.delete()
|
2021-09-27 21:03:06 +00:00
|
|
|
normalize_book_list_ordering(book_list.id, start=deleted_order)
|
2021-10-02 23:49:38 +00:00
|
|
|
|
2021-03-08 16:49:10 +00:00
|
|
|
return redirect("list", list_id)
|
2021-04-08 16:05:21 +00:00
|
|
|
|
|
|
|
|
|
|
|
@require_POST
|
2021-08-23 20:15:35 +00:00
|
|
|
@login_required
|
2021-04-08 16:05:21 +00:00
|
|
|
def set_book_position(request, list_item_id):
|
|
|
|
"""
|
2021-04-18 18:46:28 +00:00
|
|
|
Action for when the list user manually specifies a list position, takes
|
|
|
|
special care with the unique ordering per list.
|
2021-04-08 16:05:21 +00:00
|
|
|
"""
|
2021-09-27 21:03:06 +00:00
|
|
|
list_item = get_object_or_404(models.ListItem, id=list_item_id)
|
2021-09-28 00:52:27 +00:00
|
|
|
list_item.book_list.raise_not_editable(request.user)
|
2021-09-27 21:03:06 +00:00
|
|
|
try:
|
|
|
|
int_position = int(request.POST.get("position"))
|
|
|
|
except ValueError:
|
2021-09-27 23:08:52 +00:00
|
|
|
return HttpResponseBadRequest("bad value for position. should be an integer")
|
2021-04-08 16:05:21 +00:00
|
|
|
|
2021-09-27 21:03:06 +00:00
|
|
|
if int_position < 1:
|
|
|
|
return HttpResponseBadRequest("position cannot be less than 1")
|
2021-04-08 16:05:21 +00:00
|
|
|
|
2021-09-27 21:03:06 +00:00
|
|
|
book_list = list_item.book_list
|
2021-04-18 18:46:28 +00:00
|
|
|
|
2021-09-27 21:03:06 +00:00
|
|
|
# the max position to which a book may be set is the highest order for
|
|
|
|
# books which are approved
|
2021-09-27 23:08:52 +00:00
|
|
|
order_max = book_list.listitem_set.filter(approved=True).aggregate(Max("order"))[
|
|
|
|
"order__max"
|
|
|
|
]
|
2021-04-08 16:05:21 +00:00
|
|
|
|
2021-09-27 21:03:06 +00:00
|
|
|
int_position = min(int_position, order_max)
|
2021-04-08 16:05:21 +00:00
|
|
|
|
2021-09-27 21:03:06 +00:00
|
|
|
original_order = list_item.order
|
|
|
|
if original_order == int_position:
|
|
|
|
# no change
|
|
|
|
return HttpResponse(status=204)
|
2021-04-08 16:05:21 +00:00
|
|
|
|
2021-09-27 21:03:06 +00:00
|
|
|
with transaction.atomic():
|
2021-04-19 22:01:20 +00:00
|
|
|
if original_order > int_position:
|
2021-04-08 16:05:21 +00:00
|
|
|
list_item.order = -1
|
|
|
|
list_item.save()
|
|
|
|
increment_order_in_reverse(book_list.id, int_position, original_order)
|
|
|
|
else:
|
|
|
|
list_item.order = -1
|
|
|
|
list_item.save()
|
|
|
|
decrement_order(book_list.id, original_order, int_position)
|
|
|
|
|
|
|
|
list_item.order = int_position
|
|
|
|
list_item.save()
|
|
|
|
|
|
|
|
return redirect("list", book_list.id)
|
|
|
|
|
|
|
|
|
|
|
|
@transaction.atomic
|
2021-04-18 18:46:28 +00:00
|
|
|
def increment_order_in_reverse(
|
|
|
|
book_list_id: int, start: int, end: Optional[int] = None
|
|
|
|
):
|
2021-09-18 18:32:00 +00:00
|
|
|
"""increase the order number for every item in a list"""
|
2021-04-08 16:05:21 +00:00
|
|
|
try:
|
|
|
|
book_list = models.List.objects.get(id=book_list_id)
|
|
|
|
except models.List.DoesNotExist:
|
|
|
|
return
|
2021-04-18 18:46:28 +00:00
|
|
|
items = book_list.listitem_set.filter(order__gte=start)
|
|
|
|
if end is not None:
|
|
|
|
items = items.filter(order__lt=end)
|
|
|
|
items = items.order_by("-order")
|
2021-04-08 16:05:21 +00:00
|
|
|
for item in items:
|
|
|
|
item.order += 1
|
|
|
|
item.save()
|
|
|
|
|
|
|
|
|
|
|
|
@transaction.atomic
|
|
|
|
def decrement_order(book_list_id, start, end):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""decrement the order value for every item in a list"""
|
2021-04-08 16:05:21 +00:00
|
|
|
try:
|
|
|
|
book_list = models.List.objects.get(id=book_list_id)
|
|
|
|
except models.List.DoesNotExist:
|
|
|
|
return
|
|
|
|
items = book_list.listitem_set.filter(order__gt=start, order__lte=end).order_by(
|
|
|
|
"order"
|
|
|
|
)
|
|
|
|
for item in items:
|
|
|
|
item.order -= 1
|
|
|
|
item.save()
|
|
|
|
|
|
|
|
|
|
|
|
@transaction.atomic
|
|
|
|
def normalize_book_list_ordering(book_list_id, start=0, add_offset=0):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""gives each book in a list the proper sequential order number"""
|
2021-04-08 16:05:21 +00:00
|
|
|
try:
|
|
|
|
book_list = models.List.objects.get(id=book_list_id)
|
|
|
|
except models.List.DoesNotExist:
|
2021-04-19 22:01:20 +00:00
|
|
|
return
|
2021-04-08 16:05:21 +00:00
|
|
|
items = book_list.listitem_set.filter(order__gt=start).order_by("order")
|
|
|
|
for i, item in enumerate(items, start):
|
|
|
|
effective_order = i + add_offset
|
|
|
|
if item.order != effective_order:
|
|
|
|
item.order = effective_order
|
|
|
|
item.save()
|