Merge branch 'main' into production

This commit is contained in:
Mouse Reeve 2021-03-29 13:21:56 -07:00
commit 211bf318c4
11 changed files with 133 additions and 43 deletions

4
.gitignore vendored
View file

@ -3,6 +3,7 @@
*.pyc *.pyc
*.swp *.swp
**/__pycache__ **/__pycache__
.local
# VSCode # VSCode
/.vscode /.vscode
@ -17,3 +18,6 @@
# Testing # Testing
.coverage .coverage
#PyCharm
.idea

View file

@ -131,6 +131,8 @@ class HomeStream(ActivityStream):
def stream_users(self, status): def stream_users(self, status):
audience = super().stream_users(status) audience = super().stream_users(status)
if not audience:
return []
return audience.filter( return audience.filter(
Q(id=status.user.id) # if the user is the post's author Q(id=status.user.id) # if the user is the post's author
| Q(following=status.user) # if the user is following the author | Q(following=status.user) # if the user is following the author

View file

@ -11,6 +11,12 @@ from . import fields
class Shelf(OrderedCollectionMixin, BookWyrmModel): class Shelf(OrderedCollectionMixin, BookWyrmModel):
""" a list of books owned by a user """ """ a list of books owned by a user """
TO_READ = "to-read"
READING = "reading"
READ_FINISHED = "read"
READ_STATUS_IDENTIFIERS = (TO_READ, READING, READ_FINISHED)
name = fields.CharField(max_length=100) name = fields.CharField(max_length=100)
identifier = models.CharField(max_length=100) identifier = models.CharField(max_length=100)
user = fields.ForeignKey( user = fields.ForeignKey(

View file

@ -33,4 +33,8 @@
</div> </div>
{% endfor %} {% endfor %}
</div> </div>
<div>
{% include 'snippets/pagination.html' with page=editions path=request.path %}
</div>
{% endblock %} {% endblock %}

View file

@ -12,6 +12,7 @@
<form class="dropdown-item pt-0 pb-0" name="shelve" action="/shelve/" method="post"> <form class="dropdown-item pt-0 pb-0" name="shelve" action="/shelve/" method="post">
{% csrf_token %} {% csrf_token %}
<input type="hidden" name="book" value="{{ book.id }}"> <input type="hidden" name="book" value="{{ book.id }}">
<input type="hidden" name="change-shelf-from" value={{ current.identifier }}>
<input type="hidden" name="shelf" value="{{ shelf.identifier }}"> <input type="hidden" name="shelf" value="{{ shelf.identifier }}">
<button class="button is-fullwidth is-small" type="submit">{{ shelf.name }}</button> <button class="button is-fullwidth is-small" type="submit">{{ shelf.name }}</button>
</form> </form>

View file

@ -3,6 +3,7 @@
{% with book.id|uuid as uuid %} {% with book.id|uuid as uuid %}
{% active_shelf book as active_shelf %} {% active_shelf book as active_shelf %}
{% latest_read_through book request.user as readthrough %}
<div class="field has-addons mb-0"> <div class="field has-addons mb-0">
{% if switch_mode and active_shelf.book != book %} {% if switch_mode and active_shelf.book != book %}
<div class="control"> <div class="control">
@ -20,7 +21,6 @@
{% include 'snippets/shelve_button/start_reading_modal.html' with book=active_shelf.book controls_text="start-reading" controls_uid=uuid %} {% include 'snippets/shelve_button/start_reading_modal.html' with book=active_shelf.book controls_text="start-reading" controls_uid=uuid %}
{% latest_read_through book request.user as readthrough %}
{% include 'snippets/shelve_button/finish_reading_modal.html' with book=active_shelf.book controls_text="finish-reading" controls_uid=uuid readthrough=readthrough %} {% include 'snippets/shelve_button/finish_reading_modal.html' with book=active_shelf.book controls_text="finish-reading" controls_uid=uuid readthrough=readthrough %}
{% include 'snippets/shelve_button/progress_update_modal.html' with book=shelf_book.book controls_text="progress-update" controls_uid=uuid readthrough=readthrough %} {% include 'snippets/shelve_button/progress_update_modal.html' with book=shelf_book.book controls_text="progress-update" controls_uid=uuid readthrough=readthrough %}

View file

@ -28,6 +28,8 @@
{% if dropdown %}</li>{% endif %} {% if dropdown %}</li>{% endif %}
{% endfor %} {% endfor %}
{% if dropdown %} {% if dropdown %}
{% if readthrough and active_shelf.shelf.identifier != 'read' %}
<li role="menuitem"> <li role="menuitem">
<div class="dropdown-item pt-0 pb-0"> <div class="dropdown-item pt-0 pb-0">
{% trans "Update progress" as button_text %} {% trans "Update progress" as button_text %}
@ -35,3 +37,18 @@
</div> </div>
</li> </li>
{% endif %} {% endif %}
{% if active_shelf.shelf %}
<li role="menuitem">
<div class="dropdown-item pt-0 pb-0">
<form name="shelve" action="/unshelve/" method="post">
{% csrf_token %}
<input type="hidden" name="book" value="{{ active_shelf.book.id }}">
<input type="hidden" name="shelf" value="{{ active_shelf.shelf.id }}">
<button class="button is-fullwidth is-small is-danger is-light" type="submit">{% trans "Unshelve" %}</button>
</form>
</div>
</li>
{% endif %}
{% endif %}

View file

@ -42,7 +42,7 @@ class ReadingViews(TestCase):
def test_start_reading(self, _): def test_start_reading(self, _):
""" begin a book """ """ begin a book """
shelf = self.local_user.shelf_set.get(identifier="reading") shelf = self.local_user.shelf_set.get(identifier=models.Shelf.READING)
self.assertFalse(shelf.books.exists()) self.assertFalse(shelf.books.exists())
self.assertFalse(models.Status.objects.exists()) self.assertFalse(models.Status.objects.exists())
@ -73,7 +73,7 @@ class ReadingViews(TestCase):
def test_start_reading_reshelf(self, _): def test_start_reading_reshelf(self, _):
""" begin a book """ """ begin a book """
to_read_shelf = self.local_user.shelf_set.get(identifier="to-read") to_read_shelf = self.local_user.shelf_set.get(identifier=models.Shelf.TO_READ)
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"): with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
models.ShelfBook.objects.create( models.ShelfBook.objects.create(
shelf=to_read_shelf, book=self.book, user=self.local_user shelf=to_read_shelf, book=self.book, user=self.local_user
@ -93,7 +93,7 @@ class ReadingViews(TestCase):
def test_finish_reading(self, _): def test_finish_reading(self, _):
""" begin a book """ """ begin a book """
shelf = self.local_user.shelf_set.get(identifier="read") shelf = self.local_user.shelf_set.get(identifier=models.Shelf.READ_FINISHED)
self.assertFalse(shelf.books.exists()) self.assertFalse(shelf.books.exists())
self.assertFalse(models.Status.objects.exists()) self.assertFalse(models.Status.objects.exists())
readthrough = models.ReadThrough.objects.create( readthrough = models.ReadThrough.objects.create(

View file

@ -250,6 +250,11 @@ class Editions(View):
""" list of editions of a book """ """ list of editions of a book """
work = get_object_or_404(models.Work, id=book_id) work = get_object_or_404(models.Work, id=book_id)
try:
page = int(request.GET.get("page", 1))
except ValueError:
page = 1
if is_api_request(request): if is_api_request(request):
return ActivitypubResponse(work.to_edition_list(**request.GET)) return ActivitypubResponse(work.to_edition_list(**request.GET))
filters = {} filters = {}
@ -262,8 +267,9 @@ class Editions(View):
editions = work.editions.order_by("-edition_rank").all() editions = work.editions.order_by("-edition_rank").all()
languages = set(sum([e.languages for e in editions], [])) languages = set(sum([e.languages for e in editions], []))
paginated = Paginator(editions.filter(**filters).all(), PAGE_LENGTH)
data = { data = {
"editions": editions.filter(**filters).all(), "editions": paginated.page(page),
"work": work, "work": work,
"languages": languages, "languages": languages,
"formats": set( "formats": set(

View file

@ -19,7 +19,9 @@ from .shelf import handle_unshelve
def start_reading(request, book_id): def start_reading(request, book_id):
""" begin reading a book """ """ begin reading a book """
book = get_edition(book_id) book = get_edition(book_id)
shelf = models.Shelf.objects.filter(identifier="reading", user=request.user).first() reading_shelf = models.Shelf.objects.filter(
identifier=models.Shelf.READING, user=request.user
).first()
# create a readthrough # create a readthrough
readthrough = update_readthrough(request, book=book) readthrough = update_readthrough(request, book=book)
@ -29,20 +31,27 @@ def start_reading(request, book_id):
# create a progress update if we have a page # create a progress update if we have a page
readthrough.create_update() readthrough.create_update()
# shelve the book current_status_shelfbook = (
if request.POST.get("reshelve", True): models.ShelfBook.objects.select_related("shelf")
try: .filter(
current_shelf = models.Shelf.objects.get(user=request.user, edition=book) shelf__identifier__in=models.Shelf.READ_STATUS_IDENTIFIERS,
handle_unshelve(request.user, book, current_shelf) user=request.user,
except models.Shelf.DoesNotExist: book=book,
# this just means it isn't currently on the user's shelves )
pass .first()
models.ShelfBook.objects.create(book=book, shelf=shelf, user=request.user) )
if current_status_shelfbook is not None:
if current_status_shelfbook.shelf.identifier != models.Shelf.READING:
handle_unshelve(book, current_status_shelfbook.shelf)
else: # It already was on the shelf
return redirect(request.headers.get("Referer", "/"))
models.ShelfBook.objects.create(book=book, shelf=reading_shelf, user=request.user)
# post about it (if you want) # post about it (if you want)
if request.POST.get("post-status"): if request.POST.get("post-status"):
privacy = request.POST.get("privacy") privacy = request.POST.get("privacy")
handle_reading_status(request.user, shelf, book, privacy) handle_reading_status(request.user, reading_shelf, book, privacy)
return redirect(request.headers.get("Referer", "/")) return redirect(request.headers.get("Referer", "/"))
@ -52,27 +61,38 @@ def start_reading(request, book_id):
def finish_reading(request, book_id): def finish_reading(request, book_id):
""" a user completed a book, yay """ """ a user completed a book, yay """
book = get_edition(book_id) book = get_edition(book_id)
shelf = models.Shelf.objects.filter(identifier="read", user=request.user).first() finished_read_shelf = models.Shelf.objects.filter(
identifier=models.Shelf.READ_FINISHED, user=request.user
).first()
# update or create a readthrough # update or create a readthrough
readthrough = update_readthrough(request, book=book) readthrough = update_readthrough(request, book=book)
if readthrough: if readthrough:
readthrough.save() readthrough.save()
# shelve the book current_status_shelfbook = (
if request.POST.get("reshelve", True): models.ShelfBook.objects.select_related("shelf")
try: .filter(
current_shelf = models.Shelf.objects.get(user=request.user, edition=book) shelf__identifier__in=models.Shelf.READ_STATUS_IDENTIFIERS,
handle_unshelve(request.user, book, current_shelf) user=request.user,
except models.Shelf.DoesNotExist: book=book,
# this just means it isn't currently on the user's shelves )
pass .first()
models.ShelfBook.objects.create(book=book, shelf=shelf, user=request.user) )
if current_status_shelfbook is not None:
if current_status_shelfbook.shelf.identifier != models.Shelf.READ_FINISHED:
handle_unshelve(book, current_status_shelfbook.shelf)
else: # It already was on the shelf
return redirect(request.headers.get("Referer", "/"))
models.ShelfBook.objects.create(
book=book, shelf=finished_read_shelf, user=request.user
)
# post about it (if you want) # post about it (if you want)
if request.POST.get("post-status"): if request.POST.get("post-status"):
privacy = request.POST.get("privacy") privacy = request.POST.get("privacy")
handle_reading_status(request.user, shelf, book, privacy) handle_reading_status(request.user, finished_read_shelf, book, privacy)
return redirect(request.headers.get("Referer", "/")) return redirect(request.headers.get("Referer", "/"))

View file

@ -1,4 +1,5 @@
""" shelf views""" """ shelf views"""
from django.db import IntegrityError
from django.contrib.auth.decorators import login_required from django.contrib.auth.decorators import login_required
from django.core.paginator import Paginator from django.core.paginator import Paginator
from django.http import HttpResponseBadRequest, HttpResponseNotFound from django.http import HttpResponseBadRequest, HttpResponseNotFound
@ -126,7 +127,7 @@ def delete_shelf(request, shelf_id):
@login_required @login_required
@require_POST @require_POST
def shelve(request): def shelve(request):
""" put a on a user's shelf """ """ put a book on a user's shelf """
book = get_edition(request.POST.get("book")) book = get_edition(request.POST.get("book"))
desired_shelf = models.Shelf.objects.filter( desired_shelf = models.Shelf.objects.filter(
@ -135,21 +136,50 @@ def shelve(request):
if not desired_shelf: if not desired_shelf:
return HttpResponseNotFound() return HttpResponseNotFound()
if request.POST.get("reshelve", True): change_from_current_identifier = request.POST.get("change-shelf-from")
if change_from_current_identifier is not None:
current_shelf = models.Shelf.objects.get(
user=request.user, identifier=change_from_current_identifier
)
handle_unshelve(book, current_shelf)
# A book can be on multiple shelves, but only on one read status shelf at a time
if desired_shelf.identifier in models.Shelf.READ_STATUS_IDENTIFIERS:
current_read_status_shelfbook = (
models.ShelfBook.objects.select_related("shelf")
.filter(
shelf__identifier__in=models.Shelf.READ_STATUS_IDENTIFIERS,
user=request.user,
book=book,
)
.first()
)
if current_read_status_shelfbook is not None:
if (
current_read_status_shelfbook.shelf.identifier
!= desired_shelf.identifier
):
handle_unshelve(book, current_read_status_shelfbook.shelf)
else: # It is already on the shelf
return redirect(request.headers.get("Referer", "/"))
models.ShelfBook.objects.create(
book=book, shelf=desired_shelf, user=request.user
)
if desired_shelf.identifier == models.Shelf.TO_READ and request.POST.get(
"post-status"
):
privacy = request.POST.get("privacy") or desired_shelf.privacy
handle_reading_status(request.user, desired_shelf, book, privacy=privacy)
else:
try: try:
current_shelf = models.Shelf.objects.get(user=request.user, edition=book) models.ShelfBook.objects.create(
handle_unshelve(request.user, book, current_shelf) book=book, shelf=desired_shelf, user=request.user
except models.Shelf.DoesNotExist: )
# this just means it isn't currently on the user's shelves # The book is already on this shelf. Might be good to alert, or reject the action?
except IntegrityError:
pass pass
models.ShelfBook.objects.create(book=book, shelf=desired_shelf, user=request.user) return redirect(request.headers.get("Referer", "/"))
# post about "want to read" shelves
if desired_shelf.identifier == "to-read" and request.POST.get("post-status"):
privacy = request.POST.get("privacy") or desired_shelf.privacy
handle_reading_status(request.user, desired_shelf, book, privacy=privacy)
return redirect("/")
@login_required @login_required
@ -159,12 +189,12 @@ def unshelve(request):
book = models.Edition.objects.get(id=request.POST["book"]) book = models.Edition.objects.get(id=request.POST["book"])
current_shelf = models.Shelf.objects.get(id=request.POST["shelf"]) current_shelf = models.Shelf.objects.get(id=request.POST["shelf"])
handle_unshelve(request.user, book, current_shelf) handle_unshelve(book, current_shelf)
return redirect(request.headers.get("Referer", "/")) return redirect(request.headers.get("Referer", "/"))
# pylint: disable=unused-argument # pylint: disable=unused-argument
def handle_unshelve(user, book, shelf): def handle_unshelve(book, shelf):
""" unshelve a book """ """ unshelve a book """
row = models.ShelfBook.objects.get(book=book, shelf=shelf) row = models.ShelfBook.objects.get(book=book, shelf=shelf)
row.delete() row.delete()