PonyConf/proposals/views.py
2016-09-18 20:06:17 +02:00

282 lines
11 KiB
Python

from functools import reduce
from django.contrib import messages
from django.contrib.auth.decorators import login_required
from django.contrib.auth.mixins import LoginRequiredMixin
from django.contrib.auth.models import User
from django.contrib.sites.shortcuts import get_current_site
from django.core.exceptions import PermissionDenied
from django.core.urlresolvers import reverse
from django.db.models import Q
from django.shortcuts import get_object_or_404, redirect, render
from django.views.generic import CreateView, DetailView, ListView, UpdateView
from django.utils.translation import ugettext as _
from django.views.decorators.http import require_http_methods
from django.http import HttpResponse
from accounts.models import Participation
from accounts.mixins import OrgaRequiredMixin, StaffRequiredMixin
from accounts.decorators import orga_required, staff_required
from conversations.models import ConversationWithParticipant, ConversationAboutTalk, Message
from .forms import TalkForm, TopicCreateForm, TopicUpdateForm, ConferenceForm, TalkFilterForm, STATUS_VALUES, SpeakerFilterForm
from .models import Talk, Topic, Vote, Conference
from .signals import talk_added, talk_edited
from .utils import allowed_talks, markdown_to_html
@login_required
@require_http_methods(["POST"])
def markdown_preview(request):
content = request.POST.get('data', '')
return HttpResponse(markdown_to_html(content))
def home(request):
return render(request, 'proposals/home.html')
@orga_required
def conference(request):
conference = Conference.objects.get(site=get_current_site(request))
form = ConferenceForm(request.POST or None, instance=conference)
if request.method == 'POST' and form.is_valid():
form.save()
messages.success(request, 'Conference updated!')
return redirect(reverse('conference'))
return render(request, 'proposals/conference.html', {
'conference': conference,
'form': form,
})
@login_required
def participate(request):
talks = Talk.objects.filter(site=get_current_site(request))
my_talks = talks.filter(speakers=request.user)
proposed_talks = talks.exclude(speakers=request.user).filter(proposer=request.user)
return render(request, 'proposals/participate.html', {
'my_talks': my_talks,
'proposed_talks': proposed_talks,
})
@staff_required
def talk_list(request):
show_filters = False
talks = Talk.objects.filter(site=get_current_site(request))
filter_form = TalkFilterForm(request.GET or None, site=get_current_site(request))
# Filtering
if filter_form.is_valid():
data = filter_form.cleaned_data
if len(data['kind']):
show_filters = True
talks = talks.filter(reduce(lambda x, y: x | y, [Q(event__pk=pk) for pk in data['kind']]))
if len(data['status']):
show_filters = True
talks = talks.filter(reduce(lambda x, y: x | y, [Q(accepted=dict(STATUS_VALUES)[status]) for status in data['status']]))
if len(data['topic']):
show_filters = True
talks = talks.filter(reduce(lambda x, y: x | y, [Q(topics__slug=topic) for topic in data['topic']]))
# Sorting
if request.GET.get('order') == 'desc':
reverse = True
else:
reverse = False
SORT_MAPPING = {
'title': 'title',
'kind': 'event',
'status': 'accepted',
}
sort = request.GET.get('sort')
if sort in SORT_MAPPING.keys():
if reverse:
talks = talks.order_by('-' + SORT_MAPPING[sort])
else:
talks = talks.order_by(SORT_MAPPING[sort])
# Sorting URLs
sort_urls = dict()
sort_glyphicons = dict()
for c in SORT_MAPPING.keys():
url = request.GET.copy()
url['sort'] = c
if c == sort:
if reverse:
del url['order']
glyphicon = 'sort-by-attributes-alt'
else:
url['order'] = 'desc'
glyphicon = 'sort-by-attributes'
else:
glyphicon = 'sort'
sort_urls[c] = url.urlencode()
sort_glyphicons[c] = glyphicon
return render(request, 'proposals/talk_list.html', {
'show_filters': show_filters,
'talk_list': talks,
'filter_form': filter_form,
'sort_urls': sort_urls,
'sort_glyphicons': sort_glyphicons,
})
@login_required
def talk_edit(request, talk=None):
if talk:
talk = get_object_or_404(Talk, slug=talk, site=get_current_site(request))
if not talk.is_editable_by(request.user):
raise PermissionDenied()
form = TalkForm(request.POST or None, instance=talk, site=get_current_site(request))
if talk:
form.fields['title'].disabled = True
form.fields['topics'].disabled = True
else:
form.fields['speakers'].initial = [request.user]
if request.method == 'POST' and form.is_valid():
if hasattr(talk, 'id'):
talk = form.save()
talk_edited.send(talk.__class__, instance=talk, author=request.user)
messages.success(request, _('Talk modified successfully!'))
else:
form.instance.site = get_current_site(request)
form.instance.proposer = request.user
talk = form.save()
talk_added.send(talk.__class__, instance=talk, author=request.user)
messages.success(request, _('Talk proposed successfully!'))
return redirect(talk.get_absolute_url())
return render(request, 'proposals/talk_edit.html', {
'form': form,
})
class TalkDetail(LoginRequiredMixin, DetailView):
def get_queryset(self):
return Talk.objects.filter(site=get_current_site(self.request)).all()
def get_context_data(self, **ctx):
if self.object.is_moderable_by(self.request.user):
vote = Vote.objects.filter(talk=self.object, user=self.request.user).first()
ctx.update(edit_perm=True, moderate_perm=True, vote=vote,
form_url=reverse('talk-conversation', kwargs={'talk': self.object.slug}))
else:
ctx['edit_perm'] = self.object.is_editable_by(self.request.user)
return super().get_context_data(**ctx)
class TopicMixin(object):
def get_queryset(self):
return Topic.objects.filter(site=get_current_site(self.request)).all()
class TopicFormMixin(object):
def get_form_kwargs(self):
kwargs = super(TopicFormMixin, self).get_form_kwargs()
if self.get_form_class() == TopicCreateForm:
kwargs.update({'site_id': get_current_site(self.request).id})
return kwargs
class TopicList(LoginRequiredMixin, TopicMixin, ListView):
pass
class TopicCreate(OrgaRequiredMixin, TopicMixin, TopicFormMixin, CreateView):
model = Topic
form_class = TopicCreateForm
def form_valid(self, form):
form.instance.site = get_current_site(self.request)
return super().form_valid(form)
class TopicUpdate(OrgaRequiredMixin, TopicMixin, TopicFormMixin, UpdateView):
def get_form_class(self):
return TopicCreateForm if self.request.user.is_superuser else TopicUpdateForm
@login_required
def vote(request, talk, score):
talk = get_object_or_404(Talk, site=get_current_site(request), slug=talk)
if not talk.is_moderable_by(request.user):
raise PermissionDenied()
vote, created = Vote.objects.get_or_create(talk=talk, user=request.user)
vote.vote = int(score)
vote.save()
messages.success(request, _('Vote successfully created') if created else _('Vote successfully updated'))
return redirect(talk.get_absolute_url())
@login_required
def talk_decide(request, talk, accepted):
site = get_current_site(request)
talk = get_object_or_404(Talk, site=site, slug=talk)
if not talk.is_moderable_by(request.user):
raise PermissionDenied()
if request.method == 'POST':
# Does we need to send a notification to the proposer?
m = request.POST.get('message', '').strip()
if m:
participation = Participation.objects.get(site=site, user=talk.proposer)
conversation = ConversationWithParticipant.objects.get(participation=participation)
Message.objects.create(conversation=conversation, author=request.user, content=m)
# Save the decision in the talk's conversation
conversation = ConversationAboutTalk.objects.get(talk=talk)
if accepted:
note = "The talk has been accepted."
else:
note = "The talk has been declined."
Message.objects.create(conversation=conversation, author=request.user, content=note)
talk.accepted = accepted
talk.save()
messages.success(request, _('Decision taken in account'))
return redirect('show-talk', slug=talk.slug)
return render(request, 'proposals/talk_decide.html', {
'talk': talk,
'accept': accepted,
})
@staff_required
def speaker_list(request):
show_filters = False
site = get_current_site(request)
filter_form = SpeakerFilterForm(request.GET or None, site=site)
talks = Talk.objects.filter(site=site)
# Filtering
if filter_form.is_valid():
data = filter_form.cleaned_data
if len(data['topic']):
show_filters = True
talks = talks.filter(reduce(lambda x, y: x | y, [Q(topics__slug=topic) for topic in data['topic']]))
speakers = Participation.objects.filter(user__talk__in=talks).all().distinct()
if filter_form.is_valid():
data = filter_form.cleaned_data
if len(data['transport']):
show_filters = True
speakers = speakers.filter(need_transport=True).filter(reduce(lambda x, y: x | y, [Q(transport__pk=pk) for pk in data['transport']]))
if len(data['hosting']):
show_filters = True
queries = []
if 'hotel' in data['hosting']:
queries += [ Q(need_hosting=True, homestay=False) ]
if 'homestay' in data['hosting']:
queries += [ Q(need_hosting=True, homestay=True) ]
speakers = speakers.filter(reduce(lambda x, y: x | y, queries))
if data['sound'] != None:
show_filters = True
speakers = speakers.filter(sound=data['sound'])
return render(request, 'proposals/speaker_list.html', {
'speaker_list': speakers,
'filter_form': filter_form,
'show_filters': show_filters,
})
@login_required
def user_details(request, username):
user = get_object_or_404(User, username=username)
participation = get_object_or_404(Participation, user=user, site=get_current_site(request))
return render(request, 'proposals/user_details.html', {
'profile': user.profile,
'participation': participation,
'talk_list': allowed_talks(Talk.objects.filter(site=get_current_site(request), speakers=user), request),
})