diff --git a/bookwyrm/incoming.py b/bookwyrm/incoming.py index 134c54845..a91193e41 100644 --- a/bookwyrm/incoming.py +++ b/bookwyrm/incoming.py @@ -1,6 +1,6 @@ ''' handles all of the activity coming in to the server ''' import json -from urllib.parse import urldefrag +from urllib.parse import urldefrag, unquote_plus import django.db.utils from django.http import HttpResponse @@ -217,29 +217,7 @@ def handle_create(activity): # we really oughtn't even be sending in this case return - # render the json into an activity object - serializer = activitypub.activity_objects[activity['object']['type']] - activity = serializer(**activity['object']) - - # ignore notes that aren't replies to known statuses - if activity.type == 'Note': - reply = models.Status.objects.filter( - remote_id=activity.inReplyTo - ).first() - if not reply: - return - - # look up books - book_urls = [] - if hasattr(activity, 'inReplyToBook'): - book_urls.append(activity.inReplyToBook) - if hasattr(activity, 'tag'): - book_urls += [t['href'] for t in activity.tag if t['type'] == 'Book'] - for remote_id in book_urls: - books_manager.get_or_create_book(remote_id) - - model = models.activity_models[activity.type] - status = activity.to_model(model) + status = status_builder.create_status(activity['object']) # create a notification if this is a reply if status.reply_parent and status.reply_parent.user.local: @@ -308,10 +286,11 @@ def handle_boost(activity): @app.task def handle_unboost(activity): ''' someone gave us a boost! ''' - boost = models.Boost.objects.filter(remote_id=activity['object']['id']).first() - if not boost: - return - status_builder.delete_status(boost) + boost = models.Boost.objects.filter( + remote_id=activity['object']['id'] + ).first() + if boost: + status_builder.delete_status(boost) @app.task @@ -319,8 +298,15 @@ def handle_tag(activity): ''' someone is tagging a book ''' user = get_or_create_remote_user(activity['actor']) if not user.local: - book = activity['target']['id'] - status_builder.create_tag(user, book, activity['object']['name']) + # ordered collection weirndess so we can't just to_model + book = books_manager.get_or_create_book(activity['object']['id']) + name = activity['object']['target'].split('/')[-1] + name = unquote_plus(name) + models.Tag.objects.get_or_create( + user=user, + book=book, + name=name + ) @app.task diff --git a/bookwyrm/models/tag.py b/bookwyrm/models/tag.py index 2c6d7a980..510ad13bc 100644 --- a/bookwyrm/models/tag.py +++ b/bookwyrm/models/tag.py @@ -35,8 +35,8 @@ class Tag(OrderedCollectionMixin, BookWyrmModel): return activitypub.Add( id='%s#add' % self.remote_id, actor=user.remote_id, - object=self.book.to_activity(), - target=self.to_activity(), + object=self.book.local_id, + target=self.remote_id, ).serialize() def to_remove_activity(self, user): diff --git a/bookwyrm/outgoing.py b/bookwyrm/outgoing.py index 0ea10ab91..1b21603e4 100644 --- a/bookwyrm/outgoing.py +++ b/bookwyrm/outgoing.py @@ -9,7 +9,7 @@ import requests from bookwyrm import activitypub from bookwyrm import models from bookwyrm.broadcast import broadcast -from bookwyrm.status import create_tag, create_notification +from bookwyrm.status import create_notification from bookwyrm.status import create_generated_note from bookwyrm.status import delete_status from bookwyrm.remote_user import get_or_create_remote_user @@ -257,9 +257,8 @@ def handle_status(user, form): broadcast(user, remote_activity, software='other') -def handle_tag(user, book, name): +def handle_tag(user, tag): ''' tag a book ''' - tag = create_tag(user, book, name) broadcast(user, tag.to_add_activity(user)) diff --git a/bookwyrm/remote_user.py b/bookwyrm/remote_user.py index 372f97a8c..b3408c712 100644 --- a/bookwyrm/remote_user.py +++ b/bookwyrm/remote_user.py @@ -7,6 +7,8 @@ from django.core.files.base import ContentFile from django.db import transaction from bookwyrm import activitypub, models +from bookwyrm import status as status_builder +from bookwyrm.tasks import app def get_or_create_remote_user(actor): @@ -29,7 +31,7 @@ def get_or_create_remote_user(actor): user.avatar.save(*avatar) if user.bookwyrm_user: - get_remote_reviews(user) + get_remote_reviews.delay(user.id) return user @@ -78,8 +80,10 @@ def get_avatar(data): return [image_name, image_content] -def get_remote_reviews(user): +@app.task +def get_remote_reviews(user_id): ''' ingest reviews by a new remote bookwyrm user ''' + user = models.User.objects.get(id=user_id) outbox_page = user.outbox + '?page=true' response = requests.get( outbox_page, @@ -87,9 +91,8 @@ def get_remote_reviews(user): ) data = response.json() # TODO: pagination? - for status in data['orderedItems']: - if status.get('bookwyrmType') == 'Review': - activitypub.Review(**status).to_model(models.Review) + for activity in data['orderedItems']: + status_builder.create_status(activity) def get_or_create_remote_server(domain): diff --git a/bookwyrm/static/css/fonts/icomoon.eot b/bookwyrm/static/css/fonts/icomoon.eot index e031e7d28..c787c9bd3 100644 Binary files a/bookwyrm/static/css/fonts/icomoon.eot and b/bookwyrm/static/css/fonts/icomoon.eot differ diff --git a/bookwyrm/static/css/fonts/icomoon.svg b/bookwyrm/static/css/fonts/icomoon.svg index 0323df053..e58847cdf 100644 --- a/bookwyrm/static/css/fonts/icomoon.svg +++ b/bookwyrm/static/css/fonts/icomoon.svg @@ -30,6 +30,7 @@ + diff --git a/bookwyrm/static/css/fonts/icomoon.ttf b/bookwyrm/static/css/fonts/icomoon.ttf index d112fb65c..8ba512daf 100644 Binary files a/bookwyrm/static/css/fonts/icomoon.ttf and b/bookwyrm/static/css/fonts/icomoon.ttf differ diff --git a/bookwyrm/static/css/fonts/icomoon.woff b/bookwyrm/static/css/fonts/icomoon.woff index bbac704f0..50261c681 100644 Binary files a/bookwyrm/static/css/fonts/icomoon.woff and b/bookwyrm/static/css/fonts/icomoon.woff differ diff --git a/bookwyrm/static/css/icons.css b/bookwyrm/static/css/icons.css index 1f34f78c8..09c22e1a2 100644 --- a/bookwyrm/static/css/icons.css +++ b/bookwyrm/static/css/icons.css @@ -1,10 +1,10 @@ @font-face { font-family: 'icomoon'; - src: url('fonts/icomoon.eot?ouoizu'); - src: url('fonts/icomoon.eot?ouoizu#iefix') format('embedded-opentype'), - url('fonts/icomoon.ttf?ouoizu') format('truetype'), - url('fonts/icomoon.woff?ouoizu') format('woff'), - url('fonts/icomoon.svg?ouoizu#icomoon') format('svg'); + src: url('fonts/icomoon.eot?aeb2zb'); + src: url('fonts/icomoon.eot?aeb2zb#iefix') format('embedded-opentype'), + url('fonts/icomoon.ttf?aeb2zb') format('truetype'), + url('fonts/icomoon.woff?aeb2zb') format('woff'), + url('fonts/icomoon.svg?aeb2zb#icomoon') format('svg'); font-weight: normal; font-style: normal; font-display: block; @@ -25,6 +25,9 @@ -moz-osx-font-smoothing: grayscale; } +.icon-check:before { + content: "\e917"; +} .icon-dots-three:before { content: "\e916"; } diff --git a/bookwyrm/status.py b/bookwyrm/status.py index c8c01c990..ed1befac6 100644 --- a/bookwyrm/status.py +++ b/bookwyrm/status.py @@ -1,8 +1,7 @@ ''' Handle user activity ''' from datetime import datetime -from django.db import IntegrityError -from bookwyrm import models +from bookwyrm import activitypub, books_manager, models from bookwyrm.books_manager import get_or_create_book from bookwyrm.sanitize_html import InputHtmlParser @@ -14,6 +13,37 @@ def delete_status(status): status.save() +def create_status(activity): + ''' unfortunately, it's not QUITE as simple as deserialiing it ''' + # render the json into an activity object + serializer = activitypub.activity_objects[activity['type']] + activity = serializer(**activity) + try: + model = models.activity_models[activity.type] + except KeyError: + # not a type of status we are prepared to deserialize + return None + + # ignore notes that aren't replies to known statuses + if activity.type == 'Note': + reply = models.Status.objects.filter( + remote_id=activity.inReplyTo + ).first() + if not reply: + return None + + # look up books + book_urls = [] + if hasattr(activity, 'inReplyToBook'): + book_urls.append(activity.inReplyToBook) + if hasattr(activity, 'tag'): + book_urls += [t['href'] for t in activity.tag if t['type'] == 'Book'] + for remote_id in book_urls: + books_manager.get_or_create_book(remote_id) + + return activity.to_model(model) + + def create_generated_note(user, content, mention_books=None, privacy='public'): ''' a note created by the app about user activity ''' # sanitize input html @@ -34,17 +64,6 @@ def create_generated_note(user, content, mention_books=None, privacy='public'): return status -def create_tag(user, possible_book, name): - ''' add a tag to a book ''' - book = get_or_create_book(possible_book) - - try: - tag = models.Tag.objects.create(name=name, book=book, user=user) - except IntegrityError: - return models.Tag.objects.get(name=name, book=book, user=user) - return tag - - def create_notification(user, notification_type, related_user=None, \ related_book=None, related_status=None, related_import=None): ''' let a user know when someone interacts with their content ''' diff --git a/bookwyrm/templates/book.html b/bookwyrm/templates/book.html index d271f9c11..af79f57ff 100644 --- a/bookwyrm/templates/book.html +++ b/bookwyrm/templates/book.html @@ -27,11 +27,18 @@ {% include 'snippets/shelve_button.html' %} {% if request.user.is_authenticated and not book.cover %} -
- {% csrf_token %} - {{ cover_form.as_p }} - -
+
+
+ {% csrf_token %} +
+ + +
+
+ +
+
+
{% endif %}
@@ -60,59 +67,85 @@ + + +
+ +
-
- -
- +
diff --git a/bookwyrm/templates/search_results.html b/bookwyrm/templates/search_results.html index cb997298a..add82a7d4 100644 --- a/bookwyrm/templates/search_results.html +++ b/bookwyrm/templates/search_results.html @@ -49,7 +49,7 @@ {% csrf_token %}
{% include 'snippets/search_result_text.html' with result=result link=False %}
- + {% endfor %} diff --git a/bookwyrm/templates/shelf.html b/bookwyrm/templates/shelf.html index 962795437..8e6cc9f88 100644 --- a/bookwyrm/templates/shelf.html +++ b/bookwyrm/templates/shelf.html @@ -1,7 +1,7 @@ {% extends 'layout.html' %} {% load fr_display %} {% block content %} -{% include 'user_header.html' with user=user %} +{% include 'snippets/user_header.html' with user=user %}
diff --git a/bookwyrm/templates/snippets/create_status.html b/bookwyrm/templates/snippets/create_status.html index 9055e2d4d..f8c5b3cf1 100644 --- a/bookwyrm/templates/snippets/create_status.html +++ b/bookwyrm/templates/snippets/create_status.html @@ -1,109 +1,31 @@ {% load humanize %} {% load fr_display %} -
-
-
- -
- -
- - -
- -
- - -
- -
- - -
-
+
+ +
+ +
+ + {% include 'snippets/create_status_form.html' with type='review' %} +
+ +
+ + {% include 'snippets/create_status_form.html' with type="comment" placeholder="Some thougts on '"|add:book.title|add:"'" %} +
+ +
+ + {% include 'snippets/create_status_form.html' with type="quote" placeholder="An excerpt from '"|add:book.title|add:"'" %}
diff --git a/bookwyrm/templates/snippets/create_status_form.html b/bookwyrm/templates/snippets/create_status_form.html new file mode 100644 index 000000000..fd88426ad --- /dev/null +++ b/bookwyrm/templates/snippets/create_status_form.html @@ -0,0 +1,40 @@ + + diff --git a/bookwyrm/templates/snippets/follow_button.html b/bookwyrm/templates/snippets/follow_button.html index 1f914a6ca..3fdcbab81 100644 --- a/bookwyrm/templates/snippets/follow_button.html +++ b/bookwyrm/templates/snippets/follow_button.html @@ -11,14 +11,14 @@ Follow request already sent. {% csrf_token %} {% if user.manually_approves_followers %} - + {% else %} - + {% endif %} {% endif %} diff --git a/bookwyrm/templates/snippets/follow_request_buttons.html b/bookwyrm/templates/snippets/follow_request_buttons.html index 165887e0d..b6296d3fd 100644 --- a/bookwyrm/templates/snippets/follow_request_buttons.html +++ b/bookwyrm/templates/snippets/follow_request_buttons.html @@ -3,11 +3,11 @@
{% csrf_token %} - +
{% csrf_token %} - +
{% endif %} diff --git a/bookwyrm/templates/snippets/shelf.html b/bookwyrm/templates/snippets/shelf.html index adee004ed..a9238a45f 100644 --- a/bookwyrm/templates/snippets/shelf.html +++ b/bookwyrm/templates/snippets/shelf.html @@ -71,7 +71,7 @@ {% csrf_token %} - + {% elif active_shelf.identifier == 'reading' %}
- +
@@ -99,7 +104,7 @@