mirror of
https://github.com/bookwyrm-social/bookwyrm.git
synced 2025-01-05 14:58:43 +00:00
Merge branch 'main' into production
This commit is contained in:
commit
a868e6c6fd
16 changed files with 85 additions and 39 deletions
|
@ -75,7 +75,7 @@ class ActivityObject:
|
||||||
for field in fields(self):
|
for field in fields(self):
|
||||||
try:
|
try:
|
||||||
value = kwargs[field.name]
|
value = kwargs[field.name]
|
||||||
if value in (None, MISSING):
|
if value in (None, MISSING, {}):
|
||||||
raise KeyError()
|
raise KeyError()
|
||||||
try:
|
try:
|
||||||
is_subclass = issubclass(field.type, ActivityObject)
|
is_subclass = issubclass(field.type, ActivityObject)
|
||||||
|
|
|
@ -24,7 +24,7 @@ class Person(ActivityObject):
|
||||||
outbox: str
|
outbox: str
|
||||||
followers: str
|
followers: str
|
||||||
publicKey: PublicKey
|
publicKey: PublicKey
|
||||||
endpoints: Dict
|
endpoints: Dict = None
|
||||||
name: str = None
|
name: str = None
|
||||||
summary: str = None
|
summary: str = None
|
||||||
icon: Image = field(default_factory=lambda: {})
|
icon: Image = field(default_factory=lambda: {})
|
||||||
|
|
|
@ -115,7 +115,14 @@ class StatusForm(CustomForm):
|
||||||
class EditUserForm(CustomForm):
|
class EditUserForm(CustomForm):
|
||||||
class Meta:
|
class Meta:
|
||||||
model = models.User
|
model = models.User
|
||||||
fields = ["avatar", "name", "email", "summary", "manually_approves_followers"]
|
fields = [
|
||||||
|
"avatar",
|
||||||
|
"name",
|
||||||
|
"email",
|
||||||
|
"summary",
|
||||||
|
"manually_approves_followers",
|
||||||
|
"show_goal",
|
||||||
|
]
|
||||||
help_texts = {f: None for f in fields}
|
help_texts = {f: None for f in fields}
|
||||||
|
|
||||||
|
|
||||||
|
|
18
bookwyrm/migrations/0052_user_show_goal.py
Normal file
18
bookwyrm/migrations/0052_user_show_goal.py
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
# Generated by Django 3.0.7 on 2021-03-18 15:51
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("bookwyrm", "0051_auto_20210316_1950"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="user",
|
||||||
|
name="show_goal",
|
||||||
|
field=models.BooleanField(default=True),
|
||||||
|
),
|
||||||
|
]
|
|
@ -73,7 +73,7 @@ class ActivitypubFieldMixin:
|
||||||
raise
|
raise
|
||||||
value = getattr(data, "actor")
|
value = getattr(data, "actor")
|
||||||
formatted = self.field_from_activity(value)
|
formatted = self.field_from_activity(value)
|
||||||
if formatted is None or formatted is MISSING:
|
if formatted is None or formatted is MISSING or formatted == {}:
|
||||||
return
|
return
|
||||||
setattr(instance, self.name, formatted)
|
setattr(instance, self.name, formatted)
|
||||||
|
|
||||||
|
@ -101,7 +101,7 @@ class ActivitypubFieldMixin:
|
||||||
|
|
||||||
def field_from_activity(self, value):
|
def field_from_activity(self, value):
|
||||||
""" formatter to convert activitypub into a model value """
|
""" formatter to convert activitypub into a model value """
|
||||||
if hasattr(self, "activitypub_wrapper"):
|
if value and hasattr(self, "activitypub_wrapper"):
|
||||||
value = value.get(self.activitypub_wrapper)
|
value = value.get(self.activitypub_wrapper)
|
||||||
return value
|
return value
|
||||||
|
|
||||||
|
|
|
@ -102,6 +102,7 @@ class User(OrderedCollectionPageMixin, AbstractUser):
|
||||||
updated_date = models.DateTimeField(auto_now=True)
|
updated_date = models.DateTimeField(auto_now=True)
|
||||||
last_active_date = models.DateTimeField(auto_now=True)
|
last_active_date = models.DateTimeField(auto_now=True)
|
||||||
manually_approves_followers = fields.BooleanField(default=False)
|
manually_approves_followers = fields.BooleanField(default=False)
|
||||||
|
show_goal = models.BooleanField(default=True)
|
||||||
|
|
||||||
name_field = "username"
|
name_field = "username"
|
||||||
|
|
||||||
|
|
27
bookwyrm/static/js/localstorage.js
Normal file
27
bookwyrm/static/js/localstorage.js
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
// set javascript listeners
|
||||||
|
window.onload = function() {
|
||||||
|
// display based on localstorage vars
|
||||||
|
document.querySelectorAll('[data-hide]')
|
||||||
|
.forEach(t => setDisplay(t));
|
||||||
|
|
||||||
|
// update localstorage
|
||||||
|
Array.from(document.getElementsByClassName('set-display'))
|
||||||
|
.forEach(t => t.onclick = updateDisplay);
|
||||||
|
};
|
||||||
|
|
||||||
|
function updateDisplay(e) {
|
||||||
|
// used in set reading goal
|
||||||
|
var key = e.target.getAttribute('data-id');
|
||||||
|
var value = e.target.getAttribute('data-value');
|
||||||
|
window.localStorage.setItem(key, value);
|
||||||
|
|
||||||
|
document.querySelectorAll('[data-hide="' + key + '"]')
|
||||||
|
.forEach(t => setDisplay(t));
|
||||||
|
}
|
||||||
|
|
||||||
|
function setDisplay(el) {
|
||||||
|
// used in set reading goal
|
||||||
|
var key = el.getAttribute('data-hide');
|
||||||
|
var value = window.localStorage.getItem(key);
|
||||||
|
addRemoveClass(el, 'hidden', value);
|
||||||
|
}
|
|
@ -23,14 +23,6 @@ window.onload = function() {
|
||||||
Array.from(document.getElementsByClassName('pulldown-menu'))
|
Array.from(document.getElementsByClassName('pulldown-menu'))
|
||||||
.forEach(t => t.onclick = toggleMenu);
|
.forEach(t => t.onclick = toggleMenu);
|
||||||
|
|
||||||
// display based on localstorage vars
|
|
||||||
document.querySelectorAll('[data-hide]')
|
|
||||||
.forEach(t => setDisplay(t));
|
|
||||||
|
|
||||||
// update localstorage
|
|
||||||
Array.from(document.getElementsByClassName('set-display'))
|
|
||||||
.forEach(t => t.onclick = updateDisplay);
|
|
||||||
|
|
||||||
// hidden submit button in a form
|
// hidden submit button in a form
|
||||||
document.querySelectorAll('.hidden-form input')
|
document.querySelectorAll('.hidden-form input')
|
||||||
.forEach(t => t.onchange = revealForm);
|
.forEach(t => t.onchange = revealForm);
|
||||||
|
@ -78,24 +70,6 @@ function revealForm(e) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
function updateDisplay(e) {
|
|
||||||
// used in set reading goal
|
|
||||||
var key = e.target.getAttribute('data-id');
|
|
||||||
var value = e.target.getAttribute('data-value');
|
|
||||||
window.localStorage.setItem(key, value);
|
|
||||||
|
|
||||||
document.querySelectorAll('[data-hide="' + key + '"]')
|
|
||||||
.forEach(t => setDisplay(t));
|
|
||||||
}
|
|
||||||
|
|
||||||
function setDisplay(el) {
|
|
||||||
// used in set reading goal
|
|
||||||
var key = el.getAttribute('data-hide');
|
|
||||||
var value = window.localStorage.getItem(key);
|
|
||||||
addRemoveClass(el, 'hidden', value);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
function toggleAction(e) {
|
function toggleAction(e) {
|
||||||
var el = e.currentTarget;
|
var el = e.currentTarget;
|
||||||
var pressed = el.getAttribute('aria-pressed') == 'false';
|
var pressed = el.getAttribute('aria-pressed') == 'false';
|
||||||
|
|
|
@ -19,9 +19,9 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{# announcements and system messages #}
|
{# announcements and system messages #}
|
||||||
{% if not goal and tab == 'home' %}
|
{% if request.user.show_goal and not goal and tab == 'home' %}
|
||||||
{% now 'Y' as year %}
|
{% now 'Y' as year %}
|
||||||
<section class="block hidden" aria-title="{% trans 'Announcements' %}" data-hide="hide-{{ year }}-reading-goal">
|
<section class="block">
|
||||||
{% include 'snippets/goal_card.html' with year=year %}
|
{% include 'snippets/goal_card.html' with year=year %}
|
||||||
<hr>
|
<hr>
|
||||||
</section>
|
</section>
|
||||||
|
|
|
@ -41,6 +41,12 @@
|
||||||
<p class="help is-danger">{{ error | escape }}</p>
|
<p class="help is-danger">{{ error | escape }}</p>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</div>
|
</div>
|
||||||
|
<div class="block">
|
||||||
|
<label class="checkbox label" for="id_show_goal">
|
||||||
|
{% trans "Show set reading goal prompt in feed:" %}
|
||||||
|
{{ form.show_goal }}
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
<div class="block">
|
<div class="block">
|
||||||
<label class="checkbox label" for="id_manually_approves_followers">
|
<label class="checkbox label" for="id_manually_approves_followers">
|
||||||
{% trans "Manually approve followers:" %}
|
{% trans "Manually approve followers:" %}
|
||||||
|
|
|
@ -17,8 +17,9 @@
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block card-footer %}
|
{% block card-footer %}
|
||||||
<div class="card-footer-item is-flex-direction-column">
|
<form class="card-footer-item is-flex-direction-column" method="post" action="{% url 'hide-goal' %}">
|
||||||
<button class="button is-danger is-light is-block set-display" data-id="hide-{{ year }}-reading-goal" data-value="true">{% trans "Dismiss message" %}</button>
|
{% csrf_token %}
|
||||||
|
<button type="submit" class="button is-danger is-light is-block set-display" >{% trans "Dismiss message" %}</button>
|
||||||
<p class="help">{% blocktrans with path=request.user.local_path %}You can set or change your reading goal any time from your <a href="{{ path }}">profile page</a>{% endblocktrans %}</p>
|
<p class="help">{% blocktrans with path=request.user.local_path %}You can set or change your reading goal any time from your <a href="{{ path }}">profile page</a>{% endblocktrans %}</p>
|
||||||
</div>
|
</form>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
|
@ -35,7 +35,7 @@
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
{% if status.content and status.status_type != 'GeneratedNote' and status.status_type != 'Announce' %}
|
{% if status.content and status.status_type != 'GeneratedNote' and status.status_type != 'Announce' %}
|
||||||
{% include 'snippets/trimmed_text.html' with full=status.content|safe %}
|
{% include 'snippets/trimmed_text.html' with full=status.content|safe no_trim=status.content_warning %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if status.attachments.exists %}
|
{% if status.attachments.exists %}
|
||||||
<div class="block">
|
<div class="block">
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
{% with full|to_markdown|safe as full %}
|
{% with full|to_markdown|safe as full %}
|
||||||
|
|
||||||
{% with full|to_markdown|safe|truncatewords_html:60 as trimmed %}
|
{% with full|to_markdown|safe|truncatewords_html:60 as trimmed %}
|
||||||
{% if trimmed != full %}
|
{% if not no_trim and trimmed != full %}
|
||||||
<div id="hide-full-{{ uuid }}">
|
<div id="hide-full-{{ uuid }}">
|
||||||
<div class="content" id="trimmed-{{ uuid }}">
|
<div class="content" id="trimmed-{{ uuid }}">
|
||||||
<div dir="auto">{{ trimmed }}</div>
|
<div dir="auto">{{ trimmed }}</div>
|
||||||
|
|
|
@ -109,11 +109,13 @@ urlpatterns = [
|
||||||
),
|
),
|
||||||
re_path(r"%s/shelves/?$" % user_path, views.user_shelves_page, name="user-shelves"),
|
re_path(r"%s/shelves/?$" % user_path, views.user_shelves_page, name="user-shelves"),
|
||||||
re_path(r"%s/lists/?$" % user_path, views.UserLists.as_view(), name="user-lists"),
|
re_path(r"%s/lists/?$" % user_path, views.UserLists.as_view(), name="user-lists"),
|
||||||
|
# goals
|
||||||
re_path(
|
re_path(
|
||||||
r"%s/goal/(?P<year>\d{4})/?$" % user_path,
|
r"%s/goal/(?P<year>\d{4})/?$" % user_path,
|
||||||
views.Goal.as_view(),
|
views.Goal.as_view(),
|
||||||
name="user-goal",
|
name="user-goal",
|
||||||
),
|
),
|
||||||
|
re_path(r"^hide-goal/?$", views.hide_goal, name="hide-goal"),
|
||||||
# lists
|
# lists
|
||||||
re_path(r"^list/?$", views.Lists.as_view(), name="lists"),
|
re_path(r"^list/?$", views.Lists.as_view(), name="lists"),
|
||||||
re_path(r"^list/(?P<list_id>\d+)(.json)?/?$", views.List.as_view(), name="list"),
|
re_path(r"^list/(?P<list_id>\d+)(.json)?/?$", views.List.as_view(), name="list"),
|
||||||
|
|
|
@ -9,7 +9,7 @@ from .federation import Federation
|
||||||
from .feed import DirectMessage, Feed, Replies, Status
|
from .feed import DirectMessage, Feed, Replies, Status
|
||||||
from .follow import follow, unfollow
|
from .follow import follow, unfollow
|
||||||
from .follow import accept_follow_request, delete_follow_request
|
from .follow import accept_follow_request, delete_follow_request
|
||||||
from .goal import Goal
|
from .goal import Goal, hide_goal
|
||||||
from .import_data import Import, ImportStatus
|
from .import_data import Import, ImportStatus
|
||||||
from .inbox import Inbox
|
from .inbox import Inbox
|
||||||
from .interaction import Favorite, Unfavorite, Boost, Unboost
|
from .interaction import Favorite, Unfavorite, Boost, Unboost
|
||||||
|
|
|
@ -6,6 +6,7 @@ from django.template.loader import get_template
|
||||||
from django.template.response import TemplateResponse
|
from django.template.response import TemplateResponse
|
||||||
from django.utils.decorators import method_decorator
|
from django.utils.decorators import method_decorator
|
||||||
from django.views import View
|
from django.views import View
|
||||||
|
from django.views.decorators.http import require_POST
|
||||||
|
|
||||||
from bookwyrm import forms, models
|
from bookwyrm import forms, models
|
||||||
from bookwyrm.status import create_generated_note
|
from bookwyrm.status import create_generated_note
|
||||||
|
@ -65,3 +66,12 @@ class Goal(View):
|
||||||
)
|
)
|
||||||
|
|
||||||
return redirect(request.headers.get("Referer", "/"))
|
return redirect(request.headers.get("Referer", "/"))
|
||||||
|
|
||||||
|
|
||||||
|
@require_POST
|
||||||
|
@login_required
|
||||||
|
def hide_goal(request):
|
||||||
|
""" don't keep bugging people to set a goal """
|
||||||
|
request.user.show_goal = False
|
||||||
|
request.user.save(broadcast=False)
|
||||||
|
return redirect(request.headers.get("Referer", "/"))
|
||||||
|
|
Loading…
Reference in a new issue