bgneal@81
|
1 """
|
bgneal@81
|
2 Views for the forums application.
|
bgneal@81
|
3 """
|
bgneal@83
|
4 from django.contrib.auth.decorators import login_required
|
bgneal@82
|
5 from django.http import Http404
|
bgneal@98
|
6 from django.http import HttpResponse
|
bgneal@89
|
7 from django.http import HttpResponseBadRequest
|
bgneal@90
|
8 from django.http import HttpResponseForbidden
|
bgneal@83
|
9 from django.http import HttpResponseRedirect
|
bgneal@83
|
10 from django.core.urlresolvers import reverse
|
bgneal@91
|
11 from django.core.paginator import InvalidPage
|
bgneal@82
|
12 from django.shortcuts import get_object_or_404
|
bgneal@81
|
13 from django.shortcuts import render_to_response
|
bgneal@97
|
14 from django.template.loader import render_to_string
|
bgneal@81
|
15 from django.template import RequestContext
|
bgneal@89
|
16 from django.views.decorators.http import require_POST
|
bgneal@108
|
17 from django.utils.text import wrap
|
bgneal@81
|
18
|
bgneal@90
|
19 from core.paginator import DiggPaginator
|
bgneal@98
|
20 from core.functions import email_admins
|
bgneal@81
|
21 from forums.models import Forum
|
bgneal@83
|
22 from forums.models import Topic
|
bgneal@91
|
23 from forums.models import Post
|
bgneal@98
|
24 from forums.models import FlaggedPost
|
bgneal@106
|
25 from forums.forms import NewTopicForm, NewPostForm, PostForm
|
bgneal@81
|
26
|
bgneal@90
|
27 #######################################################################
|
bgneal@90
|
28
|
bgneal@93
|
29 TOPICS_PER_PAGE = 50
|
bgneal@90
|
30 POSTS_PER_PAGE = 2
|
bgneal@90
|
31
|
bgneal@93
|
32 def create_topic_paginator(topics):
|
bgneal@93
|
33 return DiggPaginator(topics, TOPICS_PER_PAGE, body=5, tail=2, margin=3, padding=2)
|
bgneal@93
|
34
|
bgneal@93
|
35 def create_post_paginator(posts):
|
bgneal@93
|
36 return DiggPaginator(posts, POSTS_PER_PAGE, body=5, tail=2, margin=3, padding=2)
|
bgneal@90
|
37
|
bgneal@90
|
38 #######################################################################
|
bgneal@81
|
39
|
bgneal@81
|
40 def index(request):
|
bgneal@82
|
41 """
|
bgneal@82
|
42 This view displays all the forums available, ordered in each category.
|
bgneal@82
|
43 """
|
bgneal@100
|
44 forums = Forum.objects.forums_for_user(request.user)
|
bgneal@81
|
45 cats = {}
|
bgneal@81
|
46 for forum in forums:
|
bgneal@81
|
47 cat = cats.setdefault(forum.category.id, {
|
bgneal@81
|
48 'cat': forum.category,
|
bgneal@81
|
49 'forums': [],
|
bgneal@81
|
50 })
|
bgneal@81
|
51 cat['forums'].append(forum)
|
bgneal@81
|
52
|
bgneal@81
|
53 cmpdef = lambda a, b: cmp(a['cat'].position, b['cat'].position)
|
bgneal@81
|
54 cats = sorted(cats.values(), cmpdef)
|
bgneal@81
|
55
|
bgneal@81
|
56 return render_to_response('forums/index.html', {
|
bgneal@81
|
57 'cats': cats,
|
bgneal@81
|
58 },
|
bgneal@81
|
59 context_instance=RequestContext(request))
|
bgneal@81
|
60
|
bgneal@82
|
61
|
bgneal@81
|
62 def forum_index(request, slug):
|
bgneal@82
|
63 """
|
bgneal@82
|
64 Displays all the topics in a forum.
|
bgneal@82
|
65 """
|
bgneal@101
|
66 forum = get_object_or_404(Forum.objects.select_related(), slug=slug)
|
bgneal@100
|
67
|
bgneal@100
|
68 if not forum.category.can_access(request.user):
|
bgneal@100
|
69 return HttpResponseForbidden()
|
bgneal@100
|
70
|
bgneal@107
|
71 topics = forum.topics.select_related('user', 'last_post', 'last_post__user')
|
bgneal@93
|
72 paginator = create_topic_paginator(topics)
|
bgneal@93
|
73 page_num = int(request.GET.get('page', 1))
|
bgneal@93
|
74 try:
|
bgneal@93
|
75 page = paginator.page(page_num)
|
bgneal@93
|
76 except InvalidPage:
|
bgneal@93
|
77 raise Http404
|
bgneal@97
|
78
|
bgneal@97
|
79 # we do this for the template since it is rendered twice
|
bgneal@97
|
80 page_nav = render_to_string('forums/pagination.html', {'page': page})
|
bgneal@82
|
81
|
bgneal@82
|
82 return render_to_response('forums/forum_index.html', {
|
bgneal@82
|
83 'forum': forum,
|
bgneal@93
|
84 'page': page,
|
bgneal@97
|
85 'page_nav': page_nav,
|
bgneal@82
|
86 },
|
bgneal@82
|
87 context_instance=RequestContext(request))
|
bgneal@82
|
88
|
bgneal@82
|
89
|
bgneal@82
|
90 def topic_index(request, id):
|
bgneal@82
|
91 """
|
bgneal@82
|
92 Displays all the posts in a topic.
|
bgneal@82
|
93 """
|
bgneal@101
|
94 topic = get_object_or_404(Topic.objects.select_related(), pk=id)
|
bgneal@100
|
95
|
bgneal@100
|
96 if not topic.forum.category.can_access(request.user):
|
bgneal@100
|
97 return HttpResponseForbidden()
|
bgneal@100
|
98
|
bgneal@86
|
99 topic.view_count += 1
|
bgneal@86
|
100 topic.save()
|
bgneal@86
|
101
|
bgneal@86
|
102 posts = topic.posts.select_related()
|
bgneal@93
|
103 paginator = create_post_paginator(posts)
|
bgneal@93
|
104 page_num = int(request.GET.get('page', 1))
|
bgneal@90
|
105 try:
|
bgneal@90
|
106 page = paginator.page(page_num)
|
bgneal@90
|
107 except InvalidPage:
|
bgneal@90
|
108 raise Http404
|
bgneal@90
|
109
|
bgneal@90
|
110 last_page = page_num == paginator.num_pages
|
bgneal@86
|
111
|
bgneal@97
|
112 # we do this for the template since it is rendered twice
|
bgneal@97
|
113 page_nav = render_to_string('forums/pagination.html', {'page': page})
|
bgneal@97
|
114
|
bgneal@104
|
115 can_moderate = request.user.is_authenticated() and (
|
bgneal@104
|
116 request.user.is_superuser or \
|
bgneal@104
|
117 request.user in topic.forum.moderators.all())
|
bgneal@104
|
118
|
bgneal@104
|
119 can_reply = request.user.is_authenticated() and (
|
bgneal@104
|
120 not topic.locked or can_moderate)
|
bgneal@104
|
121
|
bgneal@86
|
122 return render_to_response('forums/topic.html', {
|
bgneal@86
|
123 'forum': topic.forum,
|
bgneal@86
|
124 'topic': topic,
|
bgneal@90
|
125 'page': page,
|
bgneal@97
|
126 'page_nav': page_nav,
|
bgneal@87
|
127 'last_page': last_page,
|
bgneal@104
|
128 'can_moderate': can_moderate,
|
bgneal@104
|
129 'can_reply': can_reply,
|
bgneal@106
|
130 'form': NewPostForm(initial={'topic_id': topic.id}),
|
bgneal@86
|
131 },
|
bgneal@86
|
132 context_instance=RequestContext(request))
|
bgneal@83
|
133
|
bgneal@83
|
134
|
bgneal@83
|
135 @login_required
|
bgneal@83
|
136 def new_topic(request, slug):
|
bgneal@83
|
137 """
|
bgneal@83
|
138 This view handles the creation of new topics.
|
bgneal@83
|
139 """
|
bgneal@101
|
140 forum = get_object_or_404(Forum.objects.select_related(), slug=slug)
|
bgneal@100
|
141
|
bgneal@100
|
142 if not forum.category.can_access(request.user):
|
bgneal@100
|
143 return HttpResponseForbidden()
|
bgneal@100
|
144
|
bgneal@83
|
145 if request.method == 'POST':
|
bgneal@102
|
146 form = NewTopicForm(request.user, forum, request.POST)
|
bgneal@83
|
147 if form.is_valid():
|
bgneal@102
|
148 topic = form.save(request.META.get("REMOTE_ADDR"))
|
bgneal@108
|
149 _bump_post_count(request.user)
|
bgneal@83
|
150 return HttpResponseRedirect(reverse('forums-new_topic_thanks',
|
bgneal@83
|
151 kwargs={'tid': topic.pk}))
|
bgneal@83
|
152 else:
|
bgneal@102
|
153 form = NewTopicForm(request.user, forum)
|
bgneal@83
|
154
|
bgneal@83
|
155 return render_to_response('forums/new_topic.html', {
|
bgneal@83
|
156 'forum': forum,
|
bgneal@83
|
157 'form': form,
|
bgneal@83
|
158 },
|
bgneal@83
|
159 context_instance=RequestContext(request))
|
bgneal@83
|
160
|
bgneal@83
|
161
|
bgneal@83
|
162 @login_required
|
bgneal@83
|
163 def new_topic_thanks(request, tid):
|
bgneal@83
|
164 """
|
bgneal@83
|
165 This view displays the success page for a newly created topic.
|
bgneal@83
|
166 """
|
bgneal@101
|
167 topic = get_object_or_404(Topic.objects.select_related(), pk=tid)
|
bgneal@83
|
168 return render_to_response('forums/new_topic_thanks.html', {
|
bgneal@83
|
169 'forum': topic.forum,
|
bgneal@83
|
170 'topic': topic,
|
bgneal@83
|
171 },
|
bgneal@83
|
172 context_instance=RequestContext(request))
|
bgneal@89
|
173
|
bgneal@89
|
174
|
bgneal@89
|
175 @require_POST
|
bgneal@89
|
176 def quick_reply_ajax(request):
|
bgneal@89
|
177 """
|
bgneal@89
|
178 This function handles the quick reply to a thread function. This
|
bgneal@89
|
179 function is meant to be the target of an AJAX post, and returns
|
bgneal@89
|
180 the HTML for the new post, which the client-side script appends
|
bgneal@89
|
181 to the document.
|
bgneal@89
|
182 """
|
bgneal@90
|
183 if not request.user.is_authenticated():
|
bgneal@108
|
184 return HttpResponseForbidden('Please login or register to post.')
|
bgneal@90
|
185
|
bgneal@106
|
186 form = NewPostForm(request.POST)
|
bgneal@89
|
187 if form.is_valid():
|
bgneal@108
|
188 if not _can_post_in_topic(form.topic, request.user):
|
bgneal@108
|
189 return HttpResponseForbidden("You don't have permission to post in this topic.")
|
bgneal@100
|
190
|
bgneal@108
|
191 post = form.save(request.user, request.META.get("REMOTE_ADDR", ""))
|
bgneal@108
|
192 _bump_post_count(request.user)
|
bgneal@89
|
193 return render_to_response('forums/display_post.html', {
|
bgneal@89
|
194 'post': post,
|
bgneal@89
|
195 },
|
bgneal@89
|
196 context_instance=RequestContext(request))
|
bgneal@89
|
197
|
bgneal@108
|
198 return HttpResponseBadRequest("Invalid post.");
|
bgneal@89
|
199
|
bgneal@91
|
200
|
bgneal@91
|
201 def goto_post(request, post_id):
|
bgneal@91
|
202 """
|
bgneal@91
|
203 This function calculates what page a given post is on, then redirects
|
bgneal@91
|
204 to that URL. This function is the target of get_absolute_url() for
|
bgneal@91
|
205 Post objects.
|
bgneal@91
|
206 """
|
bgneal@101
|
207 post = get_object_or_404(Post.objects.select_related(), pk=post_id)
|
bgneal@91
|
208 count = post.topic.posts.filter(creation_date__lt=post.creation_date).count()
|
bgneal@91
|
209 page = count / POSTS_PER_PAGE + 1
|
bgneal@91
|
210 url = reverse('forums-topic_index', kwargs={'id': post.topic.id}) + \
|
bgneal@91
|
211 '?page=%s#p%s' % (page, post.id)
|
bgneal@91
|
212 return HttpResponseRedirect(url)
|
bgneal@91
|
213
|
bgneal@98
|
214
|
bgneal@98
|
215 @require_POST
|
bgneal@98
|
216 def flag_post(request):
|
bgneal@98
|
217 """
|
bgneal@98
|
218 This function handles the flagging of posts by users. This function should
|
bgneal@98
|
219 be the target of an AJAX post.
|
bgneal@98
|
220 """
|
bgneal@98
|
221 if not request.user.is_authenticated():
|
bgneal@99
|
222 return HttpResponseForbidden('Please login or register to flag a post.')
|
bgneal@98
|
223
|
bgneal@98
|
224 id = request.POST.get('id')
|
bgneal@98
|
225 if id is None:
|
bgneal@98
|
226 return HttpResponseBadRequest('No post id')
|
bgneal@98
|
227
|
bgneal@98
|
228 try:
|
bgneal@98
|
229 post = Post.objects.get(pk=id)
|
bgneal@98
|
230 except Post.DoesNotExist:
|
bgneal@98
|
231 return HttpResponseBadRequest('No post with id %s' % id)
|
bgneal@98
|
232
|
bgneal@98
|
233 flag = FlaggedPost(user=request.user, post=post)
|
bgneal@98
|
234 flag.save()
|
bgneal@98
|
235 email_admins('A Post Has Been Flagged', """Hello,
|
bgneal@98
|
236
|
bgneal@98
|
237 A user has flagged a forum post for review.
|
bgneal@98
|
238 """)
|
bgneal@98
|
239 return HttpResponse('The post was flagged. A moderator will review the post shortly. ' \
|
bgneal@98
|
240 'Thanks for helping to improve the discussions on this site.')
|
bgneal@106
|
241
|
bgneal@106
|
242
|
bgneal@106
|
243 @login_required
|
bgneal@106
|
244 def edit_post(request, id):
|
bgneal@106
|
245 """
|
bgneal@106
|
246 This view function allows authorized users to edit posts.
|
bgneal@106
|
247 The superuser, forum moderators, and original author can edit posts.
|
bgneal@106
|
248 """
|
bgneal@106
|
249 post = get_object_or_404(Post.objects.select_related(), pk=id)
|
bgneal@108
|
250
|
bgneal@108
|
251 can_moderate = request.user.is_superuser or \
|
bgneal@108
|
252 request.user in post.topic.forum.moderators.all()
|
bgneal@108
|
253
|
bgneal@108
|
254 can_edit = can_moderate or request.user == post.user
|
bgneal@106
|
255
|
bgneal@106
|
256 if not can_edit:
|
bgneal@106
|
257 return HttpResponseForbidden("You don't have permission to edit that post.")
|
bgneal@106
|
258
|
bgneal@106
|
259 if request.method == "POST":
|
bgneal@106
|
260 form = PostForm(request.POST, instance=post)
|
bgneal@106
|
261 if form.is_valid():
|
bgneal@106
|
262 form.save()
|
bgneal@106
|
263 return HttpResponseRedirect(post.get_absolute_url())
|
bgneal@106
|
264 else:
|
bgneal@106
|
265 form = PostForm(instance=post)
|
bgneal@106
|
266
|
bgneal@106
|
267 return render_to_response('forums/edit_post.html', {
|
bgneal@106
|
268 'forum': post.topic.forum,
|
bgneal@106
|
269 'topic': post.topic,
|
bgneal@106
|
270 'post': post,
|
bgneal@106
|
271 'form': form,
|
bgneal@108
|
272 'can_moderate': can_moderate,
|
bgneal@106
|
273 },
|
bgneal@106
|
274 context_instance=RequestContext(request))
|
bgneal@107
|
275
|
bgneal@107
|
276
|
bgneal@107
|
277 @require_POST
|
bgneal@107
|
278 def delete_post(request):
|
bgneal@107
|
279 """
|
bgneal@107
|
280 This view function allows superusers and forum moderators to delete posts.
|
bgneal@107
|
281 This function is the target of AJAX calls from the client.
|
bgneal@107
|
282 """
|
bgneal@107
|
283 if not request.user.is_authenticated():
|
bgneal@107
|
284 return HttpResponseForbidden('Please login to delete a post.')
|
bgneal@107
|
285
|
bgneal@107
|
286 id = request.POST.get('id')
|
bgneal@107
|
287 if id is None:
|
bgneal@107
|
288 return HttpResponseBadRequest('No post id')
|
bgneal@107
|
289
|
bgneal@107
|
290 post = get_object_or_404(Post.objects.select_related(), pk=id)
|
bgneal@107
|
291
|
bgneal@107
|
292 can_delete = request.user.is_superuser or \
|
bgneal@107
|
293 request.user in post.topic.forum.moderators.all()
|
bgneal@107
|
294
|
bgneal@107
|
295 if not can_delete:
|
bgneal@107
|
296 return HttpResponseForbidden("You don't have permission to delete that post.")
|
bgneal@107
|
297
|
bgneal@107
|
298 if post.topic.post_count == 1 and post == post.topic.last_post:
|
bgneal@107
|
299 _delete_topic(post.topic)
|
bgneal@107
|
300 else:
|
bgneal@107
|
301 _delete_post(post)
|
bgneal@107
|
302
|
bgneal@107
|
303 return HttpResponse("The post has been deleted.")
|
bgneal@107
|
304
|
bgneal@107
|
305
|
bgneal@107
|
306 def _delete_post(post):
|
bgneal@107
|
307 """
|
bgneal@107
|
308 Internal function to delete a single post object.
|
bgneal@107
|
309 Decrements the post author's post count.
|
bgneal@107
|
310 Adjusts the parent topic and forum's last_post as needed.
|
bgneal@107
|
311 """
|
bgneal@107
|
312 # Adjust post creator's post count
|
bgneal@107
|
313 profile = post.user.get_profile()
|
bgneal@107
|
314 if profile.forum_post_count > 0:
|
bgneal@107
|
315 profile.forum_post_count -= 1
|
bgneal@107
|
316 profile.save()
|
bgneal@107
|
317
|
bgneal@107
|
318 # If this post is the last_post in a topic, we need to update
|
bgneal@107
|
319 # both the topic and parent forum's last post fields. If we don't
|
bgneal@107
|
320 # the cascading delete will delete them also!
|
bgneal@107
|
321
|
bgneal@107
|
322 topic = post.topic
|
bgneal@107
|
323 if topic.last_post == post:
|
bgneal@107
|
324 topic.last_post_pre_delete()
|
bgneal@107
|
325 topic.save()
|
bgneal@107
|
326
|
bgneal@107
|
327 forum = topic.forum
|
bgneal@107
|
328 if forum.last_post == post:
|
bgneal@107
|
329 forum.last_post_pre_delete()
|
bgneal@107
|
330 forum.save()
|
bgneal@107
|
331
|
bgneal@107
|
332 # Should be safe to delete the post now:
|
bgneal@107
|
333 post.delete()
|
bgneal@107
|
334
|
bgneal@107
|
335
|
bgneal@107
|
336 def _delete_topic(topic):
|
bgneal@107
|
337 """
|
bgneal@107
|
338 Internal function to delete an entire topic.
|
bgneal@107
|
339 Deletes the topic and all posts contained within.
|
bgneal@107
|
340 Adjusts the parent forum's last_post as needed.
|
bgneal@107
|
341 Note that we don't bother adjusting all the users'
|
bgneal@107
|
342 post counts as that doesn't seem to be worth the effort.
|
bgneal@107
|
343 """
|
bgneal@107
|
344 if topic.forum.last_post.topic == topic:
|
bgneal@107
|
345 topic.forum.last_post_pre_delete()
|
bgneal@107
|
346 topic.forum.save()
|
bgneal@107
|
347
|
bgneal@107
|
348 # It should be safe to just delete the topic now. This will
|
bgneal@107
|
349 # automatically delete all posts in the topic.
|
bgneal@107
|
350 topic.delete()
|
bgneal@108
|
351
|
bgneal@108
|
352
|
bgneal@108
|
353 @login_required
|
bgneal@108
|
354 def new_post(request, topic_id):
|
bgneal@108
|
355 """
|
bgneal@108
|
356 This function is the view for creating a normal, non-quick reply
|
bgneal@108
|
357 to a topic.
|
bgneal@108
|
358 """
|
bgneal@108
|
359 topic = get_object_or_404(Topic.objects.select_related(), pk=topic_id)
|
bgneal@108
|
360 can_post = _can_post_in_topic(topic, request.user)
|
bgneal@108
|
361
|
bgneal@108
|
362 if can_post:
|
bgneal@108
|
363 if request.method == 'POST':
|
bgneal@108
|
364 form = PostForm(request.POST)
|
bgneal@108
|
365 if form.is_valid():
|
bgneal@108
|
366 post = form.save(commit=False)
|
bgneal@108
|
367 post.topic = topic
|
bgneal@108
|
368 post.user = request.user
|
bgneal@108
|
369 post.user_ip = request.META.get("REMOTE_ADDR", "")
|
bgneal@108
|
370 post.save()
|
bgneal@108
|
371 _bump_post_count(request.user)
|
bgneal@108
|
372 return HttpResponseRedirect(post.get_absolute_url())
|
bgneal@108
|
373 else:
|
bgneal@108
|
374 quote_id = request.GET.get('quote')
|
bgneal@108
|
375 if quote_id:
|
bgneal@108
|
376 quote_post = get_object_or_404(Post.objects.select_related(),
|
bgneal@108
|
377 pk=quote_id)
|
bgneal@108
|
378 form = PostForm(initial={'body': _quote_message(quote_post.user.username,
|
bgneal@108
|
379 quote_post.body)})
|
bgneal@108
|
380 else:
|
bgneal@108
|
381 form = PostForm()
|
bgneal@108
|
382 else:
|
bgneal@108
|
383 form = None
|
bgneal@108
|
384
|
bgneal@108
|
385 return render_to_response('forums/new_post.html', {
|
bgneal@108
|
386 'forum': topic.forum,
|
bgneal@108
|
387 'topic': topic,
|
bgneal@108
|
388 'form': form,
|
bgneal@108
|
389 'can_post': can_post,
|
bgneal@108
|
390 },
|
bgneal@108
|
391 context_instance=RequestContext(request))
|
bgneal@108
|
392
|
bgneal@108
|
393
|
bgneal@108
|
394 def _can_post_in_topic(topic, user):
|
bgneal@108
|
395 """
|
bgneal@108
|
396 This function returns true if the given user can post in the given topic
|
bgneal@108
|
397 and false otherwise.
|
bgneal@108
|
398 """
|
bgneal@108
|
399 return (not topic.locked and topic.forum.category.can_access(user)) or \
|
bgneal@108
|
400 (user.is_superuser or user in topic.forum.moderators.all())
|
bgneal@108
|
401
|
bgneal@108
|
402
|
bgneal@108
|
403 def _bump_post_count(user):
|
bgneal@108
|
404 """
|
bgneal@108
|
405 Increments the forum_post_count for the given user.
|
bgneal@108
|
406 """
|
bgneal@108
|
407 profile = user.get_profile()
|
bgneal@108
|
408 profile.forum_post_count += 1
|
bgneal@108
|
409 profile.save()
|
bgneal@108
|
410
|
bgneal@108
|
411
|
bgneal@108
|
412 def _quote_message(who, message):
|
bgneal@108
|
413 """
|
bgneal@108
|
414 Builds a message reply by quoting the existing message in a
|
bgneal@108
|
415 typical email-like fashion. The quoting is compatible with Markdown.
|
bgneal@108
|
416 """
|
bgneal@108
|
417 header = '*%s wrote:*\n\n' % (who, )
|
bgneal@108
|
418 lines = wrap(message, 55).split('\n')
|
bgneal@108
|
419 for i, line in enumerate(lines):
|
bgneal@108
|
420 lines[i] = '> ' + line
|
bgneal@108
|
421 return header + '\n'.join(lines)
|