mirror of
https://github.com/ae-utbm/sith.git
synced 2024-11-22 22:23:23 +00:00
Lot of small improvement in the forum
This commit is contained in:
parent
3b16704227
commit
c66b9b0512
@ -6,6 +6,7 @@ from django.core.exceptions import ValidationError
|
|||||||
from django.db import IntegrityError, transaction
|
from django.db import IntegrityError, transaction
|
||||||
from django.core.urlresolvers import reverse
|
from django.core.urlresolvers import reverse
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
|
from django.utils.functional import cached_property
|
||||||
|
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
import pytz
|
import pytz
|
||||||
@ -16,6 +17,10 @@ from club.models import Club
|
|||||||
class Forum(models.Model):
|
class Forum(models.Model):
|
||||||
"""
|
"""
|
||||||
The Forum class, made as a tree to allow nice tidy organization
|
The Forum class, made as a tree to allow nice tidy organization
|
||||||
|
|
||||||
|
owner_club allows club members to moderate there own topics
|
||||||
|
edit_groups allows to put any group as a forum admin
|
||||||
|
view_groups allows some groups to view a forum
|
||||||
"""
|
"""
|
||||||
name = models.CharField(_('name'), max_length=64)
|
name = models.CharField(_('name'), max_length=64)
|
||||||
description = models.CharField(_('description'), max_length=256, default="")
|
description = models.CharField(_('description'), max_length=256, default="")
|
||||||
@ -77,6 +82,10 @@ class Forum(models.Model):
|
|||||||
def get_absolute_url(self):
|
def get_absolute_url(self):
|
||||||
return reverse('forum:view_forum', kwargs={'forum_id': self.id})
|
return reverse('forum:view_forum', kwargs={'forum_id': self.id})
|
||||||
|
|
||||||
|
@cached_property
|
||||||
|
def parent_list(self):
|
||||||
|
return self.get_parent_list()
|
||||||
|
|
||||||
def get_parent_list(self):
|
def get_parent_list(self):
|
||||||
l = []
|
l = []
|
||||||
p = self.parent
|
p = self.parent
|
||||||
@ -85,22 +94,27 @@ class Forum(models.Model):
|
|||||||
p = p.parent
|
p = p.parent
|
||||||
return l
|
return l
|
||||||
|
|
||||||
|
@cached_property
|
||||||
|
def topic_number(self):
|
||||||
|
return self.get_topic_number()
|
||||||
|
|
||||||
def get_topic_number(self):
|
def get_topic_number(self):
|
||||||
number = self.topics.all().count()
|
number = self.topics.all().count()
|
||||||
for c in self.children.all():
|
for c in self.children.all():
|
||||||
number += c.get_topic_number()
|
number += c.topic_number
|
||||||
return number
|
return number
|
||||||
|
|
||||||
|
@cached_property
|
||||||
|
def last_message(self):
|
||||||
|
return self.get_last_message()
|
||||||
|
|
||||||
def get_last_message(self):
|
def get_last_message(self):
|
||||||
last_msg = None
|
last_msg = None
|
||||||
for m in ForumMessage.objects.order_by('-id'):
|
for m in ForumMessage.objects.select_related('topic__forum', 'author').order_by('-id'):
|
||||||
forum = m.topic.forum
|
forum = m.topic.forum
|
||||||
if self in (forum.get_parent_list() + [forum]):
|
if self in (forum.parent_list + [forum]):
|
||||||
return m
|
return m
|
||||||
last_msg = m
|
last_msg = m
|
||||||
try:
|
|
||||||
pass
|
|
||||||
except: pass
|
|
||||||
return last_msg
|
return last_msg
|
||||||
|
|
||||||
class ForumTopic(models.Model):
|
class ForumTopic(models.Model):
|
||||||
@ -145,26 +159,34 @@ class ForumMessage(models.Model):
|
|||||||
ordering = ['id']
|
ordering = ['id']
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return "%s" % (self.title)
|
return "%s - %s" % (self.id, self.title)
|
||||||
|
|
||||||
def is_owned_by(self, user):
|
def is_owned_by(self, user): # Anyone can create a topic: it's better to
|
||||||
return self.topic.is_owned_by(user) or user.id == self.author.id
|
# check the rights at the forum level, since it's more controlled
|
||||||
|
return self.topic.forum.is_owned_by(user) or user.id == self.author.id
|
||||||
|
|
||||||
def can_be_edited_by(self, user):
|
def can_be_edited_by(self, user):
|
||||||
return user.is_owner(self.topic)
|
return user.can_edit(self.topic.forum)
|
||||||
|
|
||||||
def can_be_viewed_by(self, user):
|
def can_be_viewed_by(self, user):
|
||||||
return user.can_view(self.topic)
|
return user.can_view(self.topic)
|
||||||
|
|
||||||
def can_be_moderated_by(self, user):
|
def can_be_moderated_by(self, user):
|
||||||
return user.is_in_group(settings.SITH_GROUP_FORUM_ADMIN_ID)
|
return self.topic.forum.is_owned_by(user)
|
||||||
|
|
||||||
def get_absolute_url(self):
|
def get_absolute_url(self):
|
||||||
return self.topic.get_absolute_url() + "#first_unread"
|
return self.topic.get_absolute_url() + "#msg_" + str(self.id)
|
||||||
|
|
||||||
def mark_as_read(self, user):
|
def mark_as_read(self, user):
|
||||||
self.readers.add(user)
|
self.readers.add(user)
|
||||||
|
|
||||||
|
def is_read(self, user):
|
||||||
|
return (self.date < user.forum_infos.last_read_date) or (user in self.readers.all())
|
||||||
|
|
||||||
|
@cached_property
|
||||||
|
def deleted(self):
|
||||||
|
return self.is_deleted()
|
||||||
|
|
||||||
def is_deleted(self):
|
def is_deleted(self):
|
||||||
meta = self.metas.exclude(action="EDIT").order_by('-date').first()
|
meta = self.metas.exclude(action="EDIT").order_by('-date').first()
|
||||||
if meta:
|
if meta:
|
||||||
@ -184,7 +206,12 @@ class ForumMessageMeta(models.Model):
|
|||||||
action = models.CharField(_("action"), choices=MESSAGE_META_ACTIONS, max_length=16)
|
action = models.CharField(_("action"), choices=MESSAGE_META_ACTIONS, max_length=16)
|
||||||
|
|
||||||
class ForumUserInfo(models.Model):
|
class ForumUserInfo(models.Model):
|
||||||
user = models.OneToOneField(User, related_name="_forum_infos")
|
"""
|
||||||
|
This currently stores only the last date a user clicked "Mark all as read".
|
||||||
|
However, this can be extended with lot of user preferences dedicated to a
|
||||||
|
user, such as the favourite topics, the signature, and so on...
|
||||||
|
"""
|
||||||
|
user = models.OneToOneField(User, related_name="_forum_infos") # TODO: see to move that to the User class in order to reduce the number of db queries
|
||||||
last_read_date = models.DateTimeField(_('last read date'), default=datetime(year=settings.SITH_SCHOOL_START_YEAR,
|
last_read_date = models.DateTimeField(_('last read date'), default=datetime(year=settings.SITH_SCHOOL_START_YEAR,
|
||||||
month=1, day=1, tzinfo=pytz.UTC))
|
month=1, day=1, tzinfo=pytz.UTC))
|
||||||
|
|
||||||
|
@ -23,13 +23,12 @@
|
|||||||
{% if not forum.is_category %}
|
{% if not forum.is_category %}
|
||||||
<div class="ib w_small">
|
<div class="ib w_small">
|
||||||
<div class="ib w_medium">
|
<div class="ib w_medium">
|
||||||
{{ forum.get_topic_number() }}
|
{{ forum.topic_number }}
|
||||||
</div>
|
</div>
|
||||||
<div class="ib w_medium">
|
<div class="ib w_medium">
|
||||||
{% set last_msg = forum.get_last_message() %}
|
{% if forum.last_message %}
|
||||||
{% if last_msg %}
|
{{ forum.last_message.author }} <br/>
|
||||||
{{ last_msg.author }} <br/>
|
{{ forum.last_message.date|date(DATETIME_FORMAT) }} {{ forum.last_message.date|time(DATETIME_FORMAT) }}
|
||||||
{{ last_msg.date|date(DATETIME_FORMAT) }} {{ last_msg.date|time(DATETIME_FORMAT) }}
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -64,7 +63,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="ib w_medium" style="text-align: center;">
|
<div class="ib w_medium" style="text-align: center;">
|
||||||
{% set last_msg = topic.messages.order_by('id').last() %}
|
{% set last_msg = topic.messages.order_by('id').select_related('author').last() %}
|
||||||
{{ user_profile_link(last_msg.author) }} <br/>
|
{{ user_profile_link(last_msg.author) }} <br/>
|
||||||
{{ last_msg.date|date(DATETIME_FORMAT) }} {{ last_msg.date|time(DATETIME_FORMAT) }}
|
{{ last_msg.date|date(DATETIME_FORMAT) }} {{ last_msg.date|time(DATETIME_FORMAT) }}
|
||||||
</div>
|
</div>
|
||||||
|
@ -37,16 +37,13 @@
|
|||||||
<p>{{ topic.description }}</p>
|
<p>{{ topic.description }}</p>
|
||||||
<p><a href="{{ url('forum:new_message', topic_id=topic.id) }}">Reply</a></p>
|
<p><a href="{{ url('forum:new_message', topic_id=topic.id) }}">Reply</a></p>
|
||||||
|
|
||||||
{% set vars = {'unread': False} %} {# ugly hack to counter Jinja scopes #}
|
{% for m in topic.messages.select_related('author__profile_pict').all() %}
|
||||||
{% for m in topic.messages.all() %}
|
{% if m.id >= first_unread_message_id %}
|
||||||
{% if m.id == first_unread_message_id and vars.update({'unread': True}) %} {# idem #}
|
<div id="msg_{{ m.id }}" class="message unread">
|
||||||
<div class="message unread" id="first_unread">
|
|
||||||
{% elif vars.unread %}
|
|
||||||
<div class="message unread">
|
|
||||||
{% else %}
|
{% else %}
|
||||||
<div class="message">
|
<div id="msg_{{ m.id }}" class="message">
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<div class="msg_author">
|
<div class="msg_author">
|
||||||
{% if m.author.profile_pict %}
|
{% if m.author.profile_pict %}
|
||||||
<img src="{{ m.author.profile_pict.get_download_url() }}" alt="{% trans %}Profile{% endtrans %}" id="picture" />
|
<img src="{{ m.author.profile_pict.get_download_url() }}" alt="{% trans %}Profile{% endtrans %}" id="picture" />
|
||||||
{% else %}
|
{% else %}
|
||||||
@ -55,7 +52,7 @@
|
|||||||
<br/>
|
<br/>
|
||||||
<strong>{{ user_profile_link(m.author) }}</strong>
|
<strong>{{ user_profile_link(m.author) }}</strong>
|
||||||
</div>
|
</div>
|
||||||
<div style="display: inline-block; width: 80%; vertical-align: top;">
|
<div {% if m.id == first_unread_message_id %}id="first_unread"{% endif %} style="display: inline-block; width: 80%; vertical-align: top;">
|
||||||
<div style="display: inline-block; width: 74%;">
|
<div style="display: inline-block; width: 74%;">
|
||||||
{% if m.title %}
|
{% if m.title %}
|
||||||
<h5>{{ m.title }}</h5>
|
<h5>{{ m.title }}</h5>
|
||||||
@ -68,7 +65,7 @@
|
|||||||
<span> <a href="{{ url('forum:edit_message', message_id=m.id) }}">{% trans %}Edit{% endtrans %}</a></span>
|
<span> <a href="{{ url('forum:edit_message', message_id=m.id) }}">{% trans %}Edit{% endtrans %}</a></span>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if m.can_be_moderated_by(user) %}
|
{% if m.can_be_moderated_by(user) %}
|
||||||
{% if m.is_deleted() %}
|
{% if m.deleted %}
|
||||||
<span> <a href="{{ url('forum:undelete_message', message_id=m.id) }}">{% trans %}Undelete{% endtrans %}</a></span>
|
<span> <a href="{{ url('forum:undelete_message', message_id=m.id) }}">{% trans %}Undelete{% endtrans %}</a></span>
|
||||||
{% else %}
|
{% else %}
|
||||||
<span> <a href="{{ url('forum:delete_message', message_id=m.id) }}">{% trans %}Delete{% endtrans %}</a></span>
|
<span> <a href="{{ url('forum:delete_message', message_id=m.id) }}">{% trans %}Delete{% endtrans %}</a></span>
|
||||||
@ -81,17 +78,20 @@
|
|||||||
<div>
|
<div>
|
||||||
{{ m.message|markdown }}
|
{{ m.message|markdown }}
|
||||||
</div>
|
</div>
|
||||||
|
{% if m.can_be_moderated_by(user) %}
|
||||||
<ul>
|
<ul>
|
||||||
{% for meta in m.metas.all() %}
|
{% for meta in m.metas.select_related('user').all() %}
|
||||||
<li>{{ meta.get_action_display() }} {{ meta.user.get_display_name() }}
|
<li>{{ meta.get_action_display() }} {{ meta.user.get_display_name() }}
|
||||||
{% trans %} at {% endtrans %}{{ meta.date|time(DATETIME_FORMAT) }}
|
{% trans %} at {% endtrans %}{{ meta.date|time(DATETIME_FORMAT) }}
|
||||||
{% trans %} the {% endtrans %}{{ meta.date|date(DATETIME_FORMAT)}}</li>
|
{% trans %} the {% endtrans %}{{ meta.date|date(DATETIME_FORMAT)}}</li>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</ul>
|
</ul>
|
||||||
|
{% endif %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{{ m.mark_as_read(user) or "" }}
|
{{ m.mark_as_read(user) or "" }}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
<p><a href="{{ url('forum:new_message', topic_id=topic.id) }}">Reply</a></p>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
|
|
||||||
|
@ -10,6 +10,8 @@ from django import forms
|
|||||||
from django.db import models
|
from django.db import models
|
||||||
from django.core.exceptions import PermissionDenied
|
from django.core.exceptions import PermissionDenied
|
||||||
|
|
||||||
|
from math import inf
|
||||||
|
|
||||||
from core.views import CanViewMixin, CanEditMixin, CanEditPropMixin, CanCreateMixin, TabedViewMixin
|
from core.views import CanViewMixin, CanEditMixin, CanEditPropMixin, CanCreateMixin, TabedViewMixin
|
||||||
from forum.models import Forum, ForumMessage, ForumTopic, ForumMessageMeta
|
from forum.models import Forum, ForumMessage, ForumTopic, ForumMessageMeta
|
||||||
|
|
||||||
@ -36,9 +38,7 @@ class ForumLastUnread(ListView):
|
|||||||
def get_queryset(self):
|
def get_queryset(self):
|
||||||
l = ForumMessage.objects.exclude(readers=self.request.user).filter(
|
l = ForumMessage.objects.exclude(readers=self.request.user).filter(
|
||||||
date__gt=self.request.user.forum_infos.last_read_date).values_list('topic') # TODO try to do better
|
date__gt=self.request.user.forum_infos.last_read_date).values_list('topic') # TODO try to do better
|
||||||
return self.model.objects.filter(id__in=l).annotate(models.Max('messages__date')).order_by('-messages__date__max')
|
return self.model.objects.filter(id__in=l).annotate(models.Max('messages__date')).order_by('-messages__date__max').select_related('author')
|
||||||
# return self.model.objects.exclude(messages__readers=self.request.user).distinct().annotate(
|
|
||||||
# models.Max('messages__date')).order_by('-messages__date__max')
|
|
||||||
|
|
||||||
class ForumCreateView(CanCreateMixin, CreateView):
|
class ForumCreateView(CanCreateMixin, CreateView):
|
||||||
model = Forum
|
model = Forum
|
||||||
@ -118,14 +118,15 @@ class ForumTopicDetailView(CanViewMixin, DetailView):
|
|||||||
pk_url_kwarg = "topic_id"
|
pk_url_kwarg = "topic_id"
|
||||||
template_name = "forum/topic.jinja"
|
template_name = "forum/topic.jinja"
|
||||||
context_object_name = "topic"
|
context_object_name = "topic"
|
||||||
|
queryset = ForumTopic.objects.select_related('forum__parent')
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
def get_context_data(self, **kwargs):
|
||||||
kwargs = super(ForumTopicDetailView, self).get_context_data(**kwargs)
|
kwargs = super(ForumTopicDetailView, self).get_context_data(**kwargs)
|
||||||
msg = self.object.messages.exclude(readers=self.request.user).order_by('id').first()
|
msg = self.object.messages.exclude(readers=self.request.user).filter(date__gte=self.request.user.forum_infos.last_read_date).order_by('id').first()
|
||||||
try:
|
try:
|
||||||
kwargs['first_unread_message_id'] = msg.id
|
kwargs['first_unread_message_id'] = msg.id
|
||||||
except:
|
except:
|
||||||
kwargs['first_unread_message_id'] = -1
|
kwargs['first_unread_message_id'] = inf
|
||||||
return kwargs
|
return kwargs
|
||||||
|
|
||||||
class ForumMessageEditView(CanEditMixin, UpdateView):
|
class ForumMessageEditView(CanEditMixin, UpdateView):
|
||||||
@ -144,6 +145,7 @@ class ForumMessageDeleteView(SingleObjectMixin, RedirectView):
|
|||||||
permanent = False
|
permanent = False
|
||||||
|
|
||||||
def get_redirect_url(self, *args, **kwargs):
|
def get_redirect_url(self, *args, **kwargs):
|
||||||
|
self.object = self.get_object()
|
||||||
if self.object.can_be_moderated_by(self.request.user):
|
if self.object.can_be_moderated_by(self.request.user):
|
||||||
ForumMessageMeta(message=self.object, user=self.request.user, action="DELETE").save()
|
ForumMessageMeta(message=self.object, user=self.request.user, action="DELETE").save()
|
||||||
return self.object.get_absolute_url()
|
return self.object.get_absolute_url()
|
||||||
@ -154,6 +156,7 @@ class ForumMessageUndeleteView(SingleObjectMixin, RedirectView):
|
|||||||
permanent = False
|
permanent = False
|
||||||
|
|
||||||
def get_redirect_url(self, *args, **kwargs):
|
def get_redirect_url(self, *args, **kwargs):
|
||||||
|
self.object = self.get_object()
|
||||||
if self.object.can_be_moderated_by(self.request.user):
|
if self.object.can_be_moderated_by(self.request.user):
|
||||||
ForumMessageMeta(message=self.object, user=self.request.user, action="UNDELETE").save()
|
ForumMessageMeta(message=self.object, user=self.request.user, action="UNDELETE").save()
|
||||||
return self.object.get_absolute_url()
|
return self.object.get_absolute_url()
|
||||||
@ -172,10 +175,14 @@ class ForumMessageCreateView(CanCreateMixin, CreateView):
|
|||||||
def get_initial(self):
|
def get_initial(self):
|
||||||
init = super(ForumMessageCreateView, self).get_initial()
|
init = super(ForumMessageCreateView, self).get_initial()
|
||||||
try:
|
try:
|
||||||
init['message'] = "\n".join([
|
message = ForumMessage.objects.select_related('author').filter(id=self.request.GET['quote_id']).first()
|
||||||
"> " + line for line in ForumMessage.objects.filter(id=self.request.GET['quote_id']).first().message.split('\n')
|
init['message'] = "> ##### %s\n" % (_("%(author)s said") % {'author': message.author.get_short_name()})
|
||||||
|
init['message'] += "\n".join([
|
||||||
|
"> " + line for line in message.message.split('\n')
|
||||||
])
|
])
|
||||||
except: pass
|
init['message'] += "\n\n"
|
||||||
|
except Exception as e:
|
||||||
|
print(repr(e))
|
||||||
return init
|
return init
|
||||||
|
|
||||||
def form_valid(self, form):
|
def form_valid(self, form):
|
||||||
@ -183,3 +190,4 @@ class ForumMessageCreateView(CanCreateMixin, CreateView):
|
|||||||
form.instance.author = self.request.user
|
form.instance.author = self.request.user
|
||||||
return super(ForumMessageCreateView, self).form_valid(form)
|
return super(ForumMessageCreateView, self).form_valid(form)
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user