mirror of
https://github.com/bookwyrm-social/bookwyrm.git
synced 2025-01-20 22:18:07 +00:00
Merge branch 'main' into production
This commit is contained in:
commit
211bf318c4
11 changed files with 133 additions and 43 deletions
4
.gitignore
vendored
4
.gitignore
vendored
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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(
|
||||||
|
|
|
@ -33,4 +33,8 @@
|
||||||
</div>
|
</div>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
{% include 'snippets/pagination.html' with page=editions path=request.path %}
|
||||||
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
|
@ -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>
|
||||||
|
|
|
@ -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 %}
|
||||||
|
|
|
@ -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 %}
|
||||||
|
|
|
@ -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(
|
||||||
|
|
|
@ -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(
|
||||||
|
|
|
@ -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", "/"))
|
||||||
|
|
||||||
|
|
|
@ -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()
|
||||||
|
|
Loading…
Reference in a new issue