post.py 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. from django.contrib import messages
  2. from django.core.exceptions import PermissionDenied
  3. from django.db.transaction import atomic
  4. from django.http import JsonResponse
  5. from django.shortcuts import redirect
  6. from django.utils.translation import ugettext as _
  7. from misago.acl import add_acl
  8. from misago.core.errorpages import not_allowed
  9. from misago.threads import permissions, moderation, goto
  10. from misago.threads.forms.report import ReportPostForm
  11. from misago.threads.views.generic.base import ViewBase
  12. __all__ = [
  13. 'QuotePostView',
  14. 'ApprovePostView',
  15. 'UnhidePostView',
  16. 'HidePostView',
  17. 'DeletePostView',
  18. 'ReportPostView'
  19. ]
  20. class PostView(ViewBase):
  21. is_atomic = True
  22. require_post = True
  23. def dispatch(self, request, *args, **kwargs):
  24. if self.require_post and request.method != "POST":
  25. return not_allowed(request)
  26. post = None
  27. response = None
  28. if self.is_atomic:
  29. with atomic():
  30. post = self.get_post(request, True, **kwargs)
  31. response = self.real_dispatch(request, post)
  32. else:
  33. post = self.get_post(request, **kwargs)
  34. response = self.real_dispatch(request, post)
  35. if response:
  36. return response
  37. else:
  38. return self.redirect_to_post(request.user, post)
  39. def real_dispatch(self, request, post):
  40. raise NotImplementedError(
  41. "post views have to override real_dispatch method")
  42. def redirect_to_post(self, user, post):
  43. posts_qs = self.exclude_invisible_posts(post.thread.post_set,
  44. user,
  45. post.forum,
  46. post.thread)
  47. return redirect(goto.post(post.thread, posts_qs, post))
  48. class QuotePostView(PostView):
  49. is_atomic = False
  50. require_post = False
  51. def real_dispatch(self, request, post):
  52. quote_tpl = u'[quote="%s, post:%s, topic:%s"]\n%s\n[/quote]'
  53. formats = (post.poster_name, post.pk, post.thread_id, post.original)
  54. return JsonResponse({
  55. 'quote': quote_tpl % formats
  56. })
  57. class ApprovePostView(PostView):
  58. def real_dispatch(self, request, post):
  59. if not post.acl['can_approve']:
  60. raise PermissionDenied(_("You can't approve this post."))
  61. if post.id == post.thread.first_post_id:
  62. moderation.approve_thread(request.user, post.thread)
  63. messages.success(request, _("Thread has been approved."))
  64. else:
  65. moderation.approve_post(request.user, post)
  66. messages.success(request, _("Post has been approved."))
  67. post.thread.synchronize()
  68. post.thread.save()
  69. post.forum.synchronize()
  70. post.forum.save()
  71. class UnhidePostView(PostView):
  72. is_atomic = False
  73. def real_dispatch(self, request, post):
  74. permissions.allow_unhide_post(request.user, post)
  75. moderation.unhide_post(request.user, post)
  76. messages.success(request, _("Post has been made visible."))
  77. class HidePostView(PostView):
  78. is_atomic = False
  79. def real_dispatch(self, request, post):
  80. permissions.allow_hide_post(request.user, post)
  81. moderation.hide_post(request.user, post)
  82. messages.success(request, _("Post has been hidden."))
  83. class DeletePostView(PostView):
  84. def real_dispatch(self, request, post):
  85. post_id = post.id
  86. permissions.allow_delete_post(request.user, post)
  87. moderation.delete_post(request.user, post)
  88. post.thread.synchronize()
  89. post.thread.save()
  90. post.forum.synchronize()
  91. post.forum.save()
  92. posts_qs = self.exclude_invisible_posts(post.thread.post_set,
  93. request.user,
  94. post.forum,
  95. post.thread)
  96. posts_qs = posts_qs.select_related('thread', 'forum')
  97. if post_id < post.thread.last_post_id:
  98. target_post = posts_qs.order_by('id').filter(id__gt=post_id)
  99. else:
  100. target_post = posts_qs.order_by('-id').filter(id__lt=post_id)
  101. target_post = target_post[:1][0]
  102. target_post.thread.forum = target_post.forum
  103. add_acl(request.user, target_post.forum)
  104. add_acl(request.user, target_post.thread)
  105. add_acl(request.user, target_post)
  106. messages.success(request, _("Post has been deleted."))
  107. return self.redirect_to_post(request.user, target_post)
  108. class ReportPostView(PostView):
  109. is_atomic = False
  110. require_post = False
  111. template = 'misago/thread/report_modal.html'
  112. def dispatch(self, request, *args, **kwargs):
  113. if not request.is_ajax():
  114. return not_allowed(request)
  115. return super(ReportPostView, self).dispatch(request, *args, **kwargs)
  116. def real_dispatch(self, request, post):
  117. if not post.acl['can_report']:
  118. raise PermissionDenied(_("You can't report posts."))
  119. form = ReportPostForm()
  120. if request.method == 'POST':
  121. form = ReportPostForm(request.POST)
  122. if form.is_valid():
  123. message = _("%(user)s's post has been "
  124. "reported to moderators.")
  125. message = message % {'user': post.poster_name}
  126. return JsonResponse({'message': message})
  127. else:
  128. field_errors = form.errors.get('report_message')
  129. if field_errors:
  130. field_error = field_errors[0]
  131. else:
  132. field_error = _("Error reporting post.")
  133. return JsonResponse({'is_error': True, 'message': field_error})
  134. return self.render(request, {'form': form})