annotate gpp/forums/views.py @ 160:2eb3984ccb15

Implement #45, add a who's online feature for the forums. Created middleware that caches usernames and guest session ids in the cache. Added a tag that displays this info.
author Brian Neal <bgneal@gmail.com>
date Tue, 22 Dec 2009 02:08:05 +0000
parents 152d77265da6
children 445e1466a98d
rev   line source
bgneal@81 1 """
bgneal@81 2 Views for the forums application.
bgneal@81 3 """
bgneal@113 4 import datetime
bgneal@113 5
bgneal@83 6 from django.contrib.auth.decorators import login_required
bgneal@82 7 from django.http import Http404
bgneal@98 8 from django.http import HttpResponse
bgneal@89 9 from django.http import HttpResponseBadRequest
bgneal@90 10 from django.http import HttpResponseForbidden
bgneal@83 11 from django.http import HttpResponseRedirect
bgneal@83 12 from django.core.urlresolvers import reverse
bgneal@91 13 from django.core.paginator import InvalidPage
bgneal@82 14 from django.shortcuts import get_object_or_404
bgneal@81 15 from django.shortcuts import render_to_response
bgneal@97 16 from django.template.loader import render_to_string
bgneal@81 17 from django.template import RequestContext
bgneal@89 18 from django.views.decorators.http import require_POST
bgneal@108 19 from django.utils.text import wrap
bgneal@81 20
bgneal@90 21 from core.paginator import DiggPaginator
bgneal@98 22 from core.functions import email_admins
bgneal@113 23 from forums.models import Forum, Topic, Post, FlaggedPost, TopicLastVisit, \
bgneal@113 24 ForumLastVisit
bgneal@115 25 from forums.forms import NewTopicForm, NewPostForm, PostForm, MoveTopicForm, \
bgneal@115 26 SplitTopicForm
bgneal@114 27 from forums.unread import get_forum_unread_status, get_topic_unread_status, \
bgneal@114 28 get_post_unread_status
bgneal@81 29
bgneal@117 30 from bio.models import UserProfile
bgneal@90 31 #######################################################################
bgneal@90 32
bgneal@93 33 TOPICS_PER_PAGE = 50
bgneal@113 34 POSTS_PER_PAGE = 20
bgneal@90 35
bgneal@93 36 def create_topic_paginator(topics):
bgneal@93 37 return DiggPaginator(topics, TOPICS_PER_PAGE, body=5, tail=2, margin=3, padding=2)
bgneal@93 38
bgneal@93 39 def create_post_paginator(posts):
bgneal@93 40 return DiggPaginator(posts, POSTS_PER_PAGE, body=5, tail=2, margin=3, padding=2)
bgneal@90 41
bgneal@90 42 #######################################################################
bgneal@81 43
bgneal@81 44 def index(request):
bgneal@82 45 """
bgneal@82 46 This view displays all the forums available, ordered in each category.
bgneal@82 47 """
bgneal@100 48 forums = Forum.objects.forums_for_user(request.user)
bgneal@113 49 get_forum_unread_status(forums, request.user)
bgneal@81 50 cats = {}
bgneal@81 51 for forum in forums:
bgneal@81 52 cat = cats.setdefault(forum.category.id, {
bgneal@81 53 'cat': forum.category,
bgneal@81 54 'forums': [],
bgneal@81 55 })
bgneal@81 56 cat['forums'].append(forum)
bgneal@81 57
bgneal@81 58 cmpdef = lambda a, b: cmp(a['cat'].position, b['cat'].position)
bgneal@81 59 cats = sorted(cats.values(), cmpdef)
bgneal@81 60
bgneal@81 61 return render_to_response('forums/index.html', {
bgneal@81 62 'cats': cats,
bgneal@81 63 },
bgneal@81 64 context_instance=RequestContext(request))
bgneal@81 65
bgneal@82 66
bgneal@81 67 def forum_index(request, slug):
bgneal@82 68 """
bgneal@82 69 Displays all the topics in a forum.
bgneal@82 70 """
bgneal@101 71 forum = get_object_or_404(Forum.objects.select_related(), slug=slug)
bgneal@100 72
bgneal@100 73 if not forum.category.can_access(request.user):
bgneal@100 74 return HttpResponseForbidden()
bgneal@100 75
bgneal@107 76 topics = forum.topics.select_related('user', 'last_post', 'last_post__user')
bgneal@114 77 get_topic_unread_status(forum, topics, request.user)
bgneal@114 78
bgneal@93 79 paginator = create_topic_paginator(topics)
bgneal@93 80 page_num = int(request.GET.get('page', 1))
bgneal@93 81 try:
bgneal@93 82 page = paginator.page(page_num)
bgneal@93 83 except InvalidPage:
bgneal@93 84 raise Http404
bgneal@97 85
bgneal@97 86 # we do this for the template since it is rendered twice
bgneal@97 87 page_nav = render_to_string('forums/pagination.html', {'page': page})
bgneal@111 88
bgneal@111 89 can_moderate = _can_moderate(forum, request.user)
bgneal@82 90
bgneal@82 91 return render_to_response('forums/forum_index.html', {
bgneal@82 92 'forum': forum,
bgneal@93 93 'page': page,
bgneal@97 94 'page_nav': page_nav,
bgneal@111 95 'can_moderate': can_moderate,
bgneal@82 96 },
bgneal@82 97 context_instance=RequestContext(request))
bgneal@82 98
bgneal@82 99
bgneal@82 100 def topic_index(request, id):
bgneal@82 101 """
bgneal@82 102 Displays all the posts in a topic.
bgneal@82 103 """
bgneal@101 104 topic = get_object_or_404(Topic.objects.select_related(), pk=id)
bgneal@100 105
bgneal@100 106 if not topic.forum.category.can_access(request.user):
bgneal@100 107 return HttpResponseForbidden()
bgneal@100 108
bgneal@86 109 topic.view_count += 1
bgneal@86 110 topic.save()
bgneal@86 111
bgneal@86 112 posts = topic.posts.select_related()
bgneal@114 113
bgneal@93 114 paginator = create_post_paginator(posts)
bgneal@93 115 page_num = int(request.GET.get('page', 1))
bgneal@90 116 try:
bgneal@90 117 page = paginator.page(page_num)
bgneal@90 118 except InvalidPage:
bgneal@90 119 raise Http404
bgneal@117 120 get_post_unread_status(topic, page.object_list, request.user)
bgneal@117 121
bgneal@117 122 # Attach user profiles to each post to avoid using get_user_profile() in
bgneal@117 123 # the template.
bgneal@117 124 users = set(post.user.id for post in page.object_list)
bgneal@117 125
bgneal@117 126 profiles = UserProfile.objects.filter(user__id__in=users).select_related()
bgneal@117 127 user_profiles = dict((profile.user.id, profile) for profile in profiles)
bgneal@117 128
bgneal@117 129 for post in page.object_list:
bgneal@117 130 post.user_profile = user_profiles[post.user.id]
bgneal@90 131
bgneal@90 132 last_page = page_num == paginator.num_pages
bgneal@86 133
bgneal@113 134 if request.user.is_authenticated() and last_page:
bgneal@113 135 _update_last_visit(request.user, topic)
bgneal@113 136
bgneal@97 137 # we do this for the template since it is rendered twice
bgneal@97 138 page_nav = render_to_string('forums/pagination.html', {'page': page})
bgneal@97 139
bgneal@109 140 can_moderate = _can_moderate(topic.forum, request.user)
bgneal@104 141
bgneal@104 142 can_reply = request.user.is_authenticated() and (
bgneal@104 143 not topic.locked or can_moderate)
bgneal@104 144
bgneal@86 145 return render_to_response('forums/topic.html', {
bgneal@86 146 'forum': topic.forum,
bgneal@86 147 'topic': topic,
bgneal@90 148 'page': page,
bgneal@97 149 'page_nav': page_nav,
bgneal@87 150 'last_page': last_page,
bgneal@104 151 'can_moderate': can_moderate,
bgneal@104 152 'can_reply': can_reply,
bgneal@106 153 'form': NewPostForm(initial={'topic_id': topic.id}),
bgneal@86 154 },
bgneal@86 155 context_instance=RequestContext(request))
bgneal@83 156
bgneal@83 157
bgneal@83 158 @login_required
bgneal@83 159 def new_topic(request, slug):
bgneal@83 160 """
bgneal@83 161 This view handles the creation of new topics.
bgneal@83 162 """
bgneal@101 163 forum = get_object_or_404(Forum.objects.select_related(), slug=slug)
bgneal@100 164
bgneal@100 165 if not forum.category.can_access(request.user):
bgneal@100 166 return HttpResponseForbidden()
bgneal@100 167
bgneal@83 168 if request.method == 'POST':
bgneal@102 169 form = NewTopicForm(request.user, forum, request.POST)
bgneal@83 170 if form.is_valid():
bgneal@102 171 topic = form.save(request.META.get("REMOTE_ADDR"))
bgneal@108 172 _bump_post_count(request.user)
bgneal@83 173 return HttpResponseRedirect(reverse('forums-new_topic_thanks',
bgneal@83 174 kwargs={'tid': topic.pk}))
bgneal@83 175 else:
bgneal@102 176 form = NewTopicForm(request.user, forum)
bgneal@83 177
bgneal@83 178 return render_to_response('forums/new_topic.html', {
bgneal@83 179 'forum': forum,
bgneal@83 180 'form': form,
bgneal@83 181 },
bgneal@83 182 context_instance=RequestContext(request))
bgneal@83 183
bgneal@83 184
bgneal@83 185 @login_required
bgneal@83 186 def new_topic_thanks(request, tid):
bgneal@83 187 """
bgneal@83 188 This view displays the success page for a newly created topic.
bgneal@83 189 """
bgneal@101 190 topic = get_object_or_404(Topic.objects.select_related(), pk=tid)
bgneal@83 191 return render_to_response('forums/new_topic_thanks.html', {
bgneal@83 192 'forum': topic.forum,
bgneal@83 193 'topic': topic,
bgneal@83 194 },
bgneal@83 195 context_instance=RequestContext(request))
bgneal@89 196
bgneal@89 197
bgneal@89 198 @require_POST
bgneal@89 199 def quick_reply_ajax(request):
bgneal@89 200 """
bgneal@89 201 This function handles the quick reply to a thread function. This
bgneal@89 202 function is meant to be the target of an AJAX post, and returns
bgneal@89 203 the HTML for the new post, which the client-side script appends
bgneal@89 204 to the document.
bgneal@89 205 """
bgneal@90 206 if not request.user.is_authenticated():
bgneal@108 207 return HttpResponseForbidden('Please login or register to post.')
bgneal@90 208
bgneal@106 209 form = NewPostForm(request.POST)
bgneal@89 210 if form.is_valid():
bgneal@108 211 if not _can_post_in_topic(form.topic, request.user):
bgneal@108 212 return HttpResponseForbidden("You don't have permission to post in this topic.")
bgneal@100 213
bgneal@108 214 post = form.save(request.user, request.META.get("REMOTE_ADDR", ""))
bgneal@114 215 post.unread = True
bgneal@122 216 post.user_profile = request.user.get_profile()
bgneal@108 217 _bump_post_count(request.user)
bgneal@113 218 _update_last_visit(request.user, form.topic)
bgneal@89 219 return render_to_response('forums/display_post.html', {
bgneal@89 220 'post': post,
bgneal@113 221 'can_moderate': _can_moderate(form.topic.forum, request.user),
bgneal@120 222 'can_reply': True,
bgneal@89 223 },
bgneal@89 224 context_instance=RequestContext(request))
bgneal@89 225
bgneal@108 226 return HttpResponseBadRequest("Invalid post.");
bgneal@89 227
bgneal@91 228
bgneal@91 229 def goto_post(request, post_id):
bgneal@91 230 """
bgneal@91 231 This function calculates what page a given post is on, then redirects
bgneal@91 232 to that URL. This function is the target of get_absolute_url() for
bgneal@91 233 Post objects.
bgneal@91 234 """
bgneal@101 235 post = get_object_or_404(Post.objects.select_related(), pk=post_id)
bgneal@91 236 count = post.topic.posts.filter(creation_date__lt=post.creation_date).count()
bgneal@91 237 page = count / POSTS_PER_PAGE + 1
bgneal@91 238 url = reverse('forums-topic_index', kwargs={'id': post.topic.id}) + \
bgneal@91 239 '?page=%s#p%s' % (page, post.id)
bgneal@91 240 return HttpResponseRedirect(url)
bgneal@91 241
bgneal@98 242
bgneal@98 243 @require_POST
bgneal@98 244 def flag_post(request):
bgneal@98 245 """
bgneal@98 246 This function handles the flagging of posts by users. This function should
bgneal@98 247 be the target of an AJAX post.
bgneal@98 248 """
bgneal@98 249 if not request.user.is_authenticated():
bgneal@99 250 return HttpResponseForbidden('Please login or register to flag a post.')
bgneal@98 251
bgneal@98 252 id = request.POST.get('id')
bgneal@98 253 if id is None:
bgneal@98 254 return HttpResponseBadRequest('No post id')
bgneal@98 255
bgneal@98 256 try:
bgneal@98 257 post = Post.objects.get(pk=id)
bgneal@98 258 except Post.DoesNotExist:
bgneal@98 259 return HttpResponseBadRequest('No post with id %s' % id)
bgneal@98 260
bgneal@98 261 flag = FlaggedPost(user=request.user, post=post)
bgneal@98 262 flag.save()
bgneal@98 263 email_admins('A Post Has Been Flagged', """Hello,
bgneal@98 264
bgneal@98 265 A user has flagged a forum post for review.
bgneal@98 266 """)
bgneal@98 267 return HttpResponse('The post was flagged. A moderator will review the post shortly. ' \
bgneal@98 268 'Thanks for helping to improve the discussions on this site.')
bgneal@106 269
bgneal@106 270
bgneal@106 271 @login_required
bgneal@106 272 def edit_post(request, id):
bgneal@106 273 """
bgneal@106 274 This view function allows authorized users to edit posts.
bgneal@106 275 The superuser, forum moderators, and original author can edit posts.
bgneal@106 276 """
bgneal@106 277 post = get_object_or_404(Post.objects.select_related(), pk=id)
bgneal@108 278
bgneal@109 279 can_moderate = _can_moderate(post.topic.forum, request.user)
bgneal@108 280 can_edit = can_moderate or request.user == post.user
bgneal@106 281
bgneal@106 282 if not can_edit:
bgneal@106 283 return HttpResponseForbidden("You don't have permission to edit that post.")
bgneal@106 284
bgneal@106 285 if request.method == "POST":
bgneal@106 286 form = PostForm(request.POST, instance=post)
bgneal@106 287 if form.is_valid():
bgneal@115 288 post = form.save(commit=False)
bgneal@115 289 post.touch()
bgneal@115 290 post.save()
bgneal@106 291 return HttpResponseRedirect(post.get_absolute_url())
bgneal@106 292 else:
bgneal@106 293 form = PostForm(instance=post)
bgneal@106 294
bgneal@123 295 post.user_profile = request.user.get_profile()
bgneal@123 296
bgneal@106 297 return render_to_response('forums/edit_post.html', {
bgneal@106 298 'forum': post.topic.forum,
bgneal@106 299 'topic': post.topic,
bgneal@106 300 'post': post,
bgneal@106 301 'form': form,
bgneal@108 302 'can_moderate': can_moderate,
bgneal@106 303 },
bgneal@106 304 context_instance=RequestContext(request))
bgneal@107 305
bgneal@107 306
bgneal@107 307 @require_POST
bgneal@107 308 def delete_post(request):
bgneal@107 309 """
bgneal@107 310 This view function allows superusers and forum moderators to delete posts.
bgneal@107 311 This function is the target of AJAX calls from the client.
bgneal@107 312 """
bgneal@107 313 if not request.user.is_authenticated():
bgneal@107 314 return HttpResponseForbidden('Please login to delete a post.')
bgneal@107 315
bgneal@107 316 id = request.POST.get('id')
bgneal@107 317 if id is None:
bgneal@107 318 return HttpResponseBadRequest('No post id')
bgneal@107 319
bgneal@107 320 post = get_object_or_404(Post.objects.select_related(), pk=id)
bgneal@107 321
bgneal@107 322 can_delete = request.user.is_superuser or \
bgneal@107 323 request.user in post.topic.forum.moderators.all()
bgneal@107 324
bgneal@107 325 if not can_delete:
bgneal@107 326 return HttpResponseForbidden("You don't have permission to delete that post.")
bgneal@107 327
bgneal@147 328 delete_single_post(post)
bgneal@147 329 return HttpResponse("The post has been deleted.")
bgneal@147 330
bgneal@147 331
bgneal@147 332 def delete_single_post(post):
bgneal@147 333 """
bgneal@147 334 This function deletes a single post. It handles the case of where
bgneal@147 335 a post is the sole post in a topic by deleting the topic also. It
bgneal@147 336 adjusts any foreign keys in Topic or Forum objects that might be pointing
bgneal@147 337 to this post before deleting the post to avoid a cascading delete.
bgneal@147 338 """
bgneal@107 339 if post.topic.post_count == 1 and post == post.topic.last_post:
bgneal@107 340 _delete_topic(post.topic)
bgneal@107 341 else:
bgneal@107 342 _delete_post(post)
bgneal@107 343
bgneal@107 344
bgneal@107 345 def _delete_post(post):
bgneal@107 346 """
bgneal@107 347 Internal function to delete a single post object.
bgneal@107 348 Decrements the post author's post count.
bgneal@107 349 Adjusts the parent topic and forum's last_post as needed.
bgneal@107 350 """
bgneal@107 351 # Adjust post creator's post count
bgneal@107 352 profile = post.user.get_profile()
bgneal@107 353 if profile.forum_post_count > 0:
bgneal@107 354 profile.forum_post_count -= 1
bgneal@107 355 profile.save()
bgneal@107 356
bgneal@107 357 # If this post is the last_post in a topic, we need to update
bgneal@107 358 # both the topic and parent forum's last post fields. If we don't
bgneal@107 359 # the cascading delete will delete them also!
bgneal@107 360
bgneal@107 361 topic = post.topic
bgneal@107 362 if topic.last_post == post:
bgneal@107 363 topic.last_post_pre_delete()
bgneal@107 364 topic.save()
bgneal@107 365
bgneal@107 366 forum = topic.forum
bgneal@107 367 if forum.last_post == post:
bgneal@107 368 forum.last_post_pre_delete()
bgneal@107 369 forum.save()
bgneal@107 370
bgneal@107 371 # Should be safe to delete the post now:
bgneal@107 372 post.delete()
bgneal@107 373
bgneal@107 374
bgneal@107 375 def _delete_topic(topic):
bgneal@107 376 """
bgneal@107 377 Internal function to delete an entire topic.
bgneal@107 378 Deletes the topic and all posts contained within.
bgneal@107 379 Adjusts the parent forum's last_post as needed.
bgneal@107 380 Note that we don't bother adjusting all the users'
bgneal@107 381 post counts as that doesn't seem to be worth the effort.
bgneal@107 382 """
bgneal@147 383 if topic.forum.last_post and topic.forum.last_post.topic == topic:
bgneal@107 384 topic.forum.last_post_pre_delete()
bgneal@107 385 topic.forum.save()
bgneal@107 386
bgneal@107 387 # It should be safe to just delete the topic now. This will
bgneal@107 388 # automatically delete all posts in the topic.
bgneal@107 389 topic.delete()
bgneal@108 390
bgneal@108 391
bgneal@108 392 @login_required
bgneal@108 393 def new_post(request, topic_id):
bgneal@108 394 """
bgneal@108 395 This function is the view for creating a normal, non-quick reply
bgneal@108 396 to a topic.
bgneal@108 397 """
bgneal@108 398 topic = get_object_or_404(Topic.objects.select_related(), pk=topic_id)
bgneal@108 399 can_post = _can_post_in_topic(topic, request.user)
bgneal@108 400
bgneal@108 401 if can_post:
bgneal@108 402 if request.method == 'POST':
bgneal@108 403 form = PostForm(request.POST)
bgneal@108 404 if form.is_valid():
bgneal@108 405 post = form.save(commit=False)
bgneal@108 406 post.topic = topic
bgneal@108 407 post.user = request.user
bgneal@108 408 post.user_ip = request.META.get("REMOTE_ADDR", "")
bgneal@108 409 post.save()
bgneal@108 410 _bump_post_count(request.user)
bgneal@113 411 _update_last_visit(request.user, topic)
bgneal@108 412 return HttpResponseRedirect(post.get_absolute_url())
bgneal@108 413 else:
bgneal@108 414 quote_id = request.GET.get('quote')
bgneal@108 415 if quote_id:
bgneal@108 416 quote_post = get_object_or_404(Post.objects.select_related(),
bgneal@108 417 pk=quote_id)
bgneal@108 418 form = PostForm(initial={'body': _quote_message(quote_post.user.username,
bgneal@108 419 quote_post.body)})
bgneal@108 420 else:
bgneal@108 421 form = PostForm()
bgneal@108 422 else:
bgneal@108 423 form = None
bgneal@108 424
bgneal@108 425 return render_to_response('forums/new_post.html', {
bgneal@108 426 'forum': topic.forum,
bgneal@108 427 'topic': topic,
bgneal@108 428 'form': form,
bgneal@108 429 'can_post': can_post,
bgneal@108 430 },
bgneal@108 431 context_instance=RequestContext(request))
bgneal@108 432
bgneal@108 433
bgneal@109 434 @login_required
bgneal@109 435 def mod_topic_stick(request, id):
bgneal@109 436 """
bgneal@109 437 This view function is for moderators to toggle the sticky status of a topic.
bgneal@109 438 """
bgneal@109 439 topic = get_object_or_404(Topic.objects.select_related(), pk=id)
bgneal@109 440 if _can_moderate(topic.forum, request.user):
bgneal@109 441 topic.sticky = not topic.sticky
bgneal@109 442 topic.save()
bgneal@109 443 return HttpResponseRedirect(topic.get_absolute_url())
bgneal@109 444
bgneal@110 445 return HttpResponseForbidden()
bgneal@109 446
bgneal@109 447
bgneal@109 448 @login_required
bgneal@109 449 def mod_topic_lock(request, id):
bgneal@109 450 """
bgneal@109 451 This view function is for moderators to toggle the locked status of a topic.
bgneal@109 452 """
bgneal@109 453 topic = get_object_or_404(Topic.objects.select_related(), pk=id)
bgneal@109 454 if _can_moderate(topic.forum, request.user):
bgneal@109 455 topic.locked = not topic.locked
bgneal@109 456 topic.save()
bgneal@109 457 return HttpResponseRedirect(topic.get_absolute_url())
bgneal@109 458
bgneal@110 459 return HttpResponseForbidden()
bgneal@109 460
bgneal@109 461
bgneal@109 462 @login_required
bgneal@109 463 def mod_topic_delete(request, id):
bgneal@109 464 """
bgneal@109 465 This view function is for moderators to delete an entire topic.
bgneal@109 466 """
bgneal@109 467 topic = get_object_or_404(Topic.objects.select_related(), pk=id)
bgneal@109 468 if _can_moderate(topic.forum, request.user):
bgneal@109 469 forum_url = topic.forum.get_absolute_url()
bgneal@109 470 _delete_topic(topic)
bgneal@109 471 return HttpResponseRedirect(forum_url)
bgneal@109 472
bgneal@110 473 return HttpResponseForbidden()
bgneal@110 474
bgneal@110 475
bgneal@110 476 @login_required
bgneal@110 477 def mod_topic_move(request, id):
bgneal@110 478 """
bgneal@110 479 This view function is for moderators to move a topic to a different forum.
bgneal@110 480 """
bgneal@110 481 topic = get_object_or_404(Topic.objects.select_related(), pk=id)
bgneal@110 482 if not _can_moderate(topic.forum, request.user):
bgneal@110 483 return HttpResponseForbidden()
bgneal@110 484
bgneal@110 485 if request.method == 'POST':
bgneal@110 486 form = MoveTopicForm(request.user, request.POST)
bgneal@110 487 if form.is_valid():
bgneal@110 488 new_forum = form.cleaned_data['forums']
bgneal@110 489 old_forum = topic.forum
bgneal@111 490 _move_topic(topic, old_forum, new_forum)
bgneal@110 491 return HttpResponseRedirect(topic.get_absolute_url())
bgneal@110 492 else:
bgneal@110 493 form = MoveTopicForm(request.user)
bgneal@110 494
bgneal@110 495 return render_to_response('forums/move_topic.html', {
bgneal@110 496 'forum': topic.forum,
bgneal@110 497 'topic': topic,
bgneal@110 498 'form': form,
bgneal@110 499 },
bgneal@110 500 context_instance=RequestContext(request))
bgneal@109 501
bgneal@109 502
bgneal@111 503 @login_required
bgneal@111 504 def mod_forum(request, slug):
bgneal@111 505 """
bgneal@111 506 Displays a view to allow moderators to perform various operations
bgneal@111 507 on topics in a forum in bulk. We currently support mass locking/unlocking,
bgneal@111 508 stickying and unstickying, moving, and deleting topics.
bgneal@111 509 """
bgneal@111 510 forum = get_object_or_404(Forum.objects.select_related(), slug=slug)
bgneal@111 511 if not _can_moderate(forum, request.user):
bgneal@111 512 return HttpResponseForbidden()
bgneal@111 513
bgneal@111 514 topics = forum.topics.select_related('user', 'last_post', 'last_post__user')
bgneal@111 515 paginator = create_topic_paginator(topics)
bgneal@111 516 page_num = int(request.REQUEST.get('page', 1))
bgneal@111 517 try:
bgneal@111 518 page = paginator.page(page_num)
bgneal@111 519 except InvalidPage:
bgneal@111 520 raise Http404
bgneal@111 521
bgneal@111 522 # we do this for the template since it is rendered twice
bgneal@111 523 page_nav = render_to_string('forums/pagination.html', {'page': page})
bgneal@111 524 form = None
bgneal@111 525
bgneal@111 526 if request.method == 'POST':
bgneal@111 527 topic_ids = request.POST.getlist('topic_ids')
bgneal@111 528 url = reverse('forums-mod_forum', kwargs={'slug':forum.slug})
bgneal@111 529 url += '?page=%s' % page_num
bgneal@111 530
bgneal@111 531 if len(topic_ids):
bgneal@111 532 if request.POST.get('sticky'):
bgneal@111 533 _bulk_sticky(forum, topic_ids)
bgneal@111 534 return HttpResponseRedirect(url)
bgneal@111 535 elif request.POST.get('lock'):
bgneal@111 536 _bulk_lock(forum, topic_ids)
bgneal@111 537 return HttpResponseRedirect(url)
bgneal@111 538 elif request.POST.get('delete'):
bgneal@111 539 _bulk_delete(forum, topic_ids)
bgneal@111 540 return HttpResponseRedirect(url)
bgneal@111 541 elif request.POST.get('move'):
bgneal@111 542 form = MoveTopicForm(request.user, request.POST, hide_label=True)
bgneal@111 543 if form.is_valid():
bgneal@111 544 _bulk_move(topic_ids, forum, form.cleaned_data['forums'])
bgneal@111 545 return HttpResponseRedirect(url)
bgneal@111 546
bgneal@111 547 if form is None:
bgneal@111 548 form = MoveTopicForm(request.user, hide_label=True)
bgneal@111 549
bgneal@111 550 return render_to_response('forums/mod_forum.html', {
bgneal@111 551 'forum': forum,
bgneal@111 552 'page': page,
bgneal@111 553 'page_nav': page_nav,
bgneal@111 554 'form': form,
bgneal@111 555 },
bgneal@111 556 context_instance=RequestContext(request))
bgneal@111 557
bgneal@111 558
bgneal@113 559 @login_required
bgneal@113 560 @require_POST
bgneal@113 561 def forum_catchup(request, slug):
bgneal@113 562 """
bgneal@113 563 This view marks all the topics in the forum as being read.
bgneal@113 564 """
bgneal@113 565 forum = get_object_or_404(Forum.objects.select_related(), slug=slug)
bgneal@113 566
bgneal@113 567 if not forum.category.can_access(request.user):
bgneal@113 568 return HttpResponseForbidden()
bgneal@113 569
bgneal@113 570 forum.catchup(request.user)
bgneal@113 571 return HttpResponseRedirect(forum.get_absolute_url())
bgneal@113 572
bgneal@113 573
bgneal@115 574 @login_required
bgneal@115 575 def mod_topic_split(request, id):
bgneal@115 576 """
bgneal@115 577 This view function allows moderators to split posts off to a new topic.
bgneal@115 578 """
bgneal@115 579 topic = get_object_or_404(Topic.objects.select_related(), pk=id)
bgneal@115 580 if not _can_moderate(topic.forum, request.user):
bgneal@115 581 return HttpResponseRedirect(topic.get_absolute_url())
bgneal@115 582
bgneal@115 583 if request.method == "POST":
bgneal@115 584 form = SplitTopicForm(request.user, request.POST)
bgneal@115 585 if form.is_valid():
bgneal@115 586 if form.split_at:
bgneal@115 587 _split_topic_at(topic, form.post_ids[0],
bgneal@115 588 form.cleaned_data['forums'],
bgneal@115 589 form.cleaned_data['name'])
bgneal@115 590 else:
bgneal@115 591 _split_topic(topic, form.post_ids,
bgneal@115 592 form.cleaned_data['forums'],
bgneal@115 593 form.cleaned_data['name'])
bgneal@115 594
bgneal@115 595 return HttpResponseRedirect(topic.get_absolute_url())
bgneal@115 596 else:
bgneal@115 597 form = SplitTopicForm(request.user)
bgneal@115 598
bgneal@115 599 posts = topic.posts.select_related()
bgneal@115 600
bgneal@115 601 return render_to_response('forums/mod_split_topic.html', {
bgneal@115 602 'forum': topic.forum,
bgneal@115 603 'topic': topic,
bgneal@115 604 'posts': posts,
bgneal@115 605 'form': form,
bgneal@115 606 },
bgneal@115 607 context_instance=RequestContext(request))
bgneal@115 608
bgneal@115 609
bgneal@109 610 def _can_moderate(forum, user):
bgneal@109 611 """
bgneal@109 612 Determines if a user has permission to moderate a given forum.
bgneal@109 613 """
bgneal@109 614 return user.is_authenticated() and (
bgneal@109 615 user.is_superuser or user in forum.moderators.all())
bgneal@109 616
bgneal@109 617
bgneal@108 618 def _can_post_in_topic(topic, user):
bgneal@108 619 """
bgneal@108 620 This function returns true if the given user can post in the given topic
bgneal@108 621 and false otherwise.
bgneal@108 622 """
bgneal@108 623 return (not topic.locked and topic.forum.category.can_access(user)) or \
bgneal@108 624 (user.is_superuser or user in topic.forum.moderators.all())
bgneal@108 625
bgneal@108 626
bgneal@108 627 def _bump_post_count(user):
bgneal@108 628 """
bgneal@108 629 Increments the forum_post_count for the given user.
bgneal@108 630 """
bgneal@108 631 profile = user.get_profile()
bgneal@108 632 profile.forum_post_count += 1
bgneal@108 633 profile.save()
bgneal@108 634
bgneal@108 635
bgneal@108 636 def _quote_message(who, message):
bgneal@111 637 """
bgneal@111 638 Builds a message reply by quoting the existing message in a
bgneal@111 639 typical email-like fashion. The quoting is compatible with Markdown.
bgneal@111 640 """
bgneal@111 641 header = '*%s wrote:*\n\n' % (who, )
bgneal@111 642 lines = wrap(message, 55).split('\n')
bgneal@111 643 for i, line in enumerate(lines):
bgneal@111 644 lines[i] = '> ' + line
bgneal@111 645 return header + '\n'.join(lines)
bgneal@111 646
bgneal@111 647
bgneal@111 648 def _move_topic(topic, old_forum, new_forum):
bgneal@111 649 if new_forum != old_forum:
bgneal@111 650 topic.forum = new_forum
bgneal@111 651 topic.save()
bgneal@111 652 # Have to adjust foreign keys to last_post, denormalized counts, etc.:
bgneal@112 653 old_forum.sync()
bgneal@111 654 old_forum.save()
bgneal@112 655 new_forum.sync()
bgneal@111 656 new_forum.save()
bgneal@111 657
bgneal@111 658
bgneal@111 659 def _bulk_sticky(forum, topic_ids):
bgneal@111 660 """
bgneal@111 661 Performs a toggle on the sticky status for a given list of topic ids.
bgneal@111 662 """
bgneal@111 663 topics = Topic.objects.filter(pk__in=topic_ids)
bgneal@111 664 for topic in topics:
bgneal@111 665 if topic.forum == forum:
bgneal@111 666 topic.sticky = not topic.sticky
bgneal@111 667 topic.save()
bgneal@111 668
bgneal@111 669
bgneal@111 670 def _bulk_lock(forum, topic_ids):
bgneal@111 671 """
bgneal@111 672 Performs a toggle on the locked status for a given list of topic ids.
bgneal@111 673 """
bgneal@111 674 topics = Topic.objects.filter(pk__in=topic_ids)
bgneal@111 675 for topic in topics:
bgneal@111 676 if topic.forum == forum:
bgneal@111 677 topic.locked = not topic.locked
bgneal@111 678 topic.save()
bgneal@111 679
bgneal@111 680
bgneal@111 681 def _bulk_delete(forum, topic_ids):
bgneal@111 682 """
bgneal@111 683 Deletes the list of topics.
bgneal@111 684 """
bgneal@111 685 topics = Topic.objects.filter(pk__in=topic_ids).select_related()
bgneal@111 686 for topic in topics:
bgneal@111 687 if topic.forum == forum:
bgneal@111 688 _delete_topic(topic)
bgneal@111 689
bgneal@111 690
bgneal@111 691 def _bulk_move(topic_ids, old_forum, new_forum):
bgneal@111 692 """
bgneal@111 693 Moves the list of topics to a new forum.
bgneal@111 694 """
bgneal@111 695 topics = Topic.objects.filter(pk__in=topic_ids).select_related()
bgneal@111 696 for topic in topics:
bgneal@111 697 if topic.forum == old_forum:
bgneal@111 698 _move_topic(topic, old_forum, new_forum)
bgneal@111 699
bgneal@113 700
bgneal@113 701 def _update_last_visit(user, topic):
bgneal@113 702 """
bgneal@113 703 Does the bookkeeping for the last visit status for the user to the
bgneal@113 704 topic/forum.
bgneal@113 705 """
bgneal@113 706 now = datetime.datetime.now()
bgneal@113 707 try:
bgneal@113 708 flv = ForumLastVisit.objects.get(user=user, forum=topic.forum)
bgneal@113 709 except ForumLastVisit.DoesNotExist:
bgneal@113 710 flv = ForumLastVisit(user=user, forum=topic.forum)
bgneal@113 711 flv.begin_date = now
bgneal@113 712
bgneal@113 713 flv.end_date = now
bgneal@113 714 flv.save()
bgneal@113 715
bgneal@113 716 if topic.update_date > flv.begin_date:
bgneal@113 717 try:
bgneal@113 718 tlv = TopicLastVisit.objects.get(user=user, topic=topic)
bgneal@113 719 except TopicLastVisit.DoesNotExist:
bgneal@113 720 tlv = TopicLastVisit(user=user, topic=topic)
bgneal@113 721
bgneal@113 722 tlv.touch()
bgneal@113 723 tlv.save()
bgneal@113 724
bgneal@115 725
bgneal@115 726 def _split_topic_at(topic, post_id, new_forum, new_name):
bgneal@115 727 """
bgneal@115 728 This function splits the post given by post_id and all posts that come
bgneal@115 729 after it in the given topic to a new topic in a new forum.
bgneal@115 730 It is assumed the caller has been checked for moderator rights.
bgneal@115 731 """
bgneal@115 732 post = get_object_or_404(Post, id=post_id)
bgneal@115 733 if post.topic == topic:
bgneal@115 734 post_ids = Post.objects.filter(topic=topic,
bgneal@115 735 creation_date__gte=post.creation_date).values_list('id', flat=True)
bgneal@115 736 _split_topic(topic, post_ids, new_forum, new_name)
bgneal@115 737
bgneal@115 738
bgneal@115 739 def _split_topic(topic, post_ids, new_forum, new_name):
bgneal@115 740 """
bgneal@115 741 This function splits the posts given by the post_ids list in the
bgneal@115 742 given topic to a new topic in a new forum.
bgneal@115 743 It is assumed the caller has been checked for moderator rights.
bgneal@115 744 """
bgneal@115 745 posts = Post.objects.filter(topic=topic, id__in=post_ids)
bgneal@115 746 if len(posts) > 0:
bgneal@115 747 new_topic = Topic(forum=new_forum, name=new_name, user=posts[0].user)
bgneal@115 748 new_topic.save()
bgneal@115 749 for post in posts:
bgneal@115 750 post.topic = new_topic
bgneal@115 751 post.save()
bgneal@115 752
bgneal@115 753 topic.post_count_update()
bgneal@115 754 topic.save()
bgneal@115 755 new_topic.post_count_update()
bgneal@115 756 new_topic.save()
bgneal@115 757 topic.forum.sync()
bgneal@115 758 topic.forum.save()
bgneal@115 759 new_forum.sync()
bgneal@115 760 new_forum.save()