list.py 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375
  1. from django.core.urlresolvers import reverse
  2. from django.db.models import Q, F
  3. from django import forms
  4. from django.forms import ValidationError
  5. from django.shortcuts import redirect
  6. from django.template import RequestContext
  7. from django.utils import timezone
  8. from django.utils.translation import ugettext as _
  9. from misago.acl.utils import ACLError403, ACLError404
  10. from misago.forms import Form, FormLayout, FormFields
  11. from misago.forums.models import Forum
  12. from misago.messages import Message
  13. from misago.readstracker.trackers import ForumsTracker, ThreadsTracker
  14. from misago.threads.forms import MoveThreadsForm, MergeThreadsForm
  15. from misago.threads.models import Thread, Post
  16. from misago.threads.views.base import BaseView
  17. from misago.views import error403, error404
  18. from misago.utils import make_pagination, slugify
  19. class ThreadsView(BaseView):
  20. def fetch_forum(self, forum):
  21. self.forum = Forum.objects.get(pk=forum, type='forum')
  22. self.proxy = Forum.objects.parents_aware_forum(self.forum)
  23. self.request.acl.forums.allow_forum_view(self.forum)
  24. self.parents = Forum.objects.forum_parents(self.forum.pk)
  25. if self.forum.lft + 1 != self.forum.rght:
  26. self.forum.subforums = Forum.objects.treelist(self.request.acl.forums, self.forum, tracker=ForumsTracker(self.request.user))
  27. self.tracker = ThreadsTracker(self.request, self.forum)
  28. def fetch_threads(self, page):
  29. self.count = self.request.acl.threads.filter_threads(self.request, self.forum, Thread.objects.filter(forum=self.forum).filter(weight__lt=2)).count()
  30. self.pagination = make_pagination(page, self.count, self.request.settings.threads_per_page)
  31. self.threads = []
  32. queryset_anno = Thread.objects.filter(Q(forum=Forum.objects.token_to_pk('annoucements')) | (Q(forum=self.forum) & Q(weight=2)))
  33. queryset_threads = self.request.acl.threads.filter_threads(self.request, self.forum, Thread.objects.filter(forum=self.forum).filter(weight__lt=2)).order_by('-weight', '-last')
  34. if self.request.settings.avatars_on_threads_list:
  35. queryset_anno = queryset_anno.prefetch_related('start_poster', 'last_post')
  36. queryset_threads = queryset_threads.prefetch_related('start_poster', 'last_poster')
  37. for thread in queryset_anno:
  38. self.threads.append(thread)
  39. for thread in queryset_threads:
  40. self.threads.append(thread)
  41. if self.request.settings.threads_per_page < self.count:
  42. self.threads = self.threads[self.pagination['start']:self.pagination['stop']]
  43. for thread in self.threads:
  44. thread.is_read = self.tracker.is_read(thread)
  45. def get_thread_actions(self):
  46. acl = self.request.acl.threads.get_role(self.forum)
  47. actions = []
  48. try:
  49. if acl['can_approve']:
  50. actions.append(('accept', _('Accept threads')))
  51. if acl['can_pin_threads'] == 2:
  52. actions.append(('annouce', _('Change to annoucements')))
  53. if acl['can_pin_threads'] > 0:
  54. actions.append(('sticky', _('Change to sticky threads')))
  55. if acl['can_pin_threads'] > 0:
  56. actions.append(('normal', _('Change to standard thread')))
  57. if acl['can_move_threads_posts']:
  58. actions.append(('move', _('Move threads')))
  59. actions.append(('merge', _('Merge threads')))
  60. if acl['can_close_threads']:
  61. actions.append(('open', _('Open threads')))
  62. actions.append(('close', _('Close threads')))
  63. if acl['can_delete_threads']:
  64. actions.append(('undelete', _('Undelete threads')))
  65. actions.append(('soft', _('Soft delete threads')))
  66. if acl['can_delete_threads'] == 2:
  67. actions.append(('hard', _('Hard delete threads')))
  68. except KeyError:
  69. pass
  70. return actions
  71. def make_form(self):
  72. self.form = None
  73. list_choices = self.get_thread_actions();
  74. if (not self.request.user.is_authenticated()
  75. or not list_choices):
  76. return
  77. form_fields = {}
  78. form_fields['list_action'] = forms.ChoiceField(choices=list_choices)
  79. list_choices = []
  80. for item in self.threads:
  81. if item.forum_id == self.forum.pk:
  82. list_choices.append((item.pk, None))
  83. if not list_choices:
  84. return
  85. form_fields['list_items'] = forms.MultipleChoiceField(choices=list_choices, widget=forms.CheckboxSelectMultiple)
  86. self.form = type('ThreadsViewForm', (Form,), form_fields)
  87. def handle_form(self):
  88. if self.request.method == 'POST':
  89. self.form = self.form(self.request.POST, request=self.request)
  90. if self.form.is_valid():
  91. checked_items = []
  92. posts = []
  93. for thread in self.threads:
  94. if str(thread.pk) in self.form.cleaned_data['list_items'] and thread.forum_id == self.forum.pk:
  95. posts.append(thread.start_post_id)
  96. if thread.start_post_id != thread.last_post_id:
  97. posts.append(thread.last_post_id)
  98. checked_items.append(thread.pk)
  99. if checked_items:
  100. if posts:
  101. for post in Post.objects.filter(id__in=posts).prefetch_related('user'):
  102. for thread in self.threads:
  103. if thread.start_post_id == post.pk:
  104. thread.start_post = post
  105. if thread.last_post_id == post.pk:
  106. thread.last_post = post
  107. if thread.start_post_id == post.pk or thread.last_post_id == post.pk:
  108. break
  109. form_action = getattr(self, 'action_' + self.form.cleaned_data['list_action'])
  110. try:
  111. response = form_action(checked_items)
  112. if response:
  113. return response
  114. return redirect(self.request.path)
  115. except forms.ValidationError as e:
  116. self.message = Message(e.messages[0], 'error')
  117. else:
  118. self.message = Message(_("You have to select at least one thread."), 'error')
  119. else:
  120. if 'list_action' in self.form.errors:
  121. self.message = Message(_("Action requested is incorrect."), 'error')
  122. else:
  123. self.message = Message(form.non_field_errors()[0], 'error')
  124. else:
  125. self.form = self.form(request=self.request)
  126. def action_accept(self, ids):
  127. accepted = 0
  128. users = []
  129. for thread in self.threads:
  130. if thread.pk in ids and thread.moderated:
  131. accepted += 1
  132. # Sync thread and post
  133. thread.moderated = False
  134. thread.replies_moderated -= 1
  135. thread.save(force_update=True)
  136. thread.start_post.moderated = False
  137. thread.start_post.save(force_update=True)
  138. thread.last_post.set_checkpoint(self.request, 'accepted')
  139. # Sync user
  140. if thread.last_post.user:
  141. thread.start_post.user.threads += 1
  142. thread.start_post.user.posts += 1
  143. users.append(thread.start_post.user)
  144. if accepted:
  145. self.request.monitor['threads'] = int(self.request.monitor['threads']) + accepted
  146. self.request.monitor['posts'] = int(self.request.monitor['posts']) + accepted
  147. self.forum.threads_delta += 1
  148. self.forum.posts_delta += self.thread.replies + 1
  149. self.forum.sync()
  150. self.forum.save(force_update=True)
  151. for user in users:
  152. user.save(force_update=True)
  153. self.request.messages.set_flash(Message(_('Selected threads have been marked as reviewed and made visible to other members.')), 'success', 'threads')
  154. def action_annouce(self, ids):
  155. acl = self.request.acl.threads.get_role(self.forum)
  156. annouced = []
  157. for thread in self.threads:
  158. if thread.pk in ids and thread.weight < 2:
  159. annouced.append(thread.pk)
  160. if annouced:
  161. Thread.objects.filter(id__in=annouced).update(weight=2)
  162. self.request.messages.set_flash(Message(_('Selected threads have been turned into annoucements.')), 'success', 'threads')
  163. def action_sticky(self, ids):
  164. sticky = []
  165. for thread in self.threads:
  166. if thread.pk in ids and thread.weight != 1 and (acl['can_pin_threads'] == 2 or thread.weight < 2):
  167. sticky.append(thread.pk)
  168. if sticky:
  169. Thread.objects.filter(id__in=sticky).update(weight=1)
  170. self.request.messages.set_flash(Message(_('Selected threads have been sticked to the top of list.')), 'success', 'threads')
  171. def action_normal(self, ids):
  172. normalised = []
  173. for thread in self.threads:
  174. if thread.pk in ids and thread.weight > 0:
  175. normalised.append(thread.pk)
  176. if normalised:
  177. Thread.objects.filter(id__in=normalised).update(weight=0)
  178. self.request.messages.set_flash(Message(_('Selected threads weight has been removed.')), 'success', 'threads')
  179. def action_move(self, ids):
  180. threads = []
  181. for thread in self.threads:
  182. if thread.pk in ids:
  183. threads.append(thread)
  184. if self.request.POST.get('origin') == 'move_form':
  185. form = MoveThreadsForm(self.request.POST, request=self.request, forum=self.forum)
  186. if form.is_valid():
  187. new_forum = form.cleaned_data['new_forum']
  188. for thread in threads:
  189. thread.forum = new_forum
  190. thread.post_set.update(forum=new_forum)
  191. thread.change_set.update(forum=new_forum)
  192. thread.checkpoint_set.update(forum=new_forum)
  193. thread.save(force_update=True)
  194. new_forum.sync()
  195. new_forum.save(force_update=True)
  196. self.forum.sync()
  197. self.forum.save(force_update=True)
  198. self.request.messages.set_flash(Message(_('Selected threads have been moved to "%(forum)s".') % {'forum': new_forum.name}), 'success', 'threads')
  199. return None
  200. self.message = Message(form.non_field_errors()[0], 'error')
  201. else:
  202. form = MoveThreadsForm(request=self.request, forum=self.forum)
  203. return self.request.theme.render_to_response('threads/move.html',
  204. {
  205. 'message': self.message,
  206. 'forum': self.forum,
  207. 'parents': self.parents,
  208. 'threads': threads,
  209. 'form': FormLayout(form),
  210. },
  211. context_instance=RequestContext(self.request));
  212. def action_merge(self, ids):
  213. if len(ids) < 2:
  214. raise ValidationError(_("You have to pick two or more threads to merge."))
  215. threads = []
  216. for thread in self.threads:
  217. if thread.pk in ids:
  218. threads.append(thread)
  219. if self.request.POST.get('origin') == 'merge_form':
  220. form = MergeThreadsForm(self.request.POST, request=self.request, threads=threads)
  221. if form.is_valid():
  222. new_thread = Thread.objects.create(
  223. forum=self.forum,
  224. name=form.cleaned_data['thread_name'],
  225. slug=slugify(form.cleaned_data['thread_name']),
  226. start=timezone.now(),
  227. last=timezone.now()
  228. )
  229. last_merge = 0
  230. last_thread = None
  231. merged = []
  232. for i in range(0, len(threads)):
  233. thread = form.merge_order[i]
  234. merged.append(thread.pk)
  235. if last_thread and last_thread.last > thread.start:
  236. last_merge += thread.merges + 1
  237. thread.post_set.update(thread=new_thread, merge=F('merge') + last_merge)
  238. thread.change_set.update(thread=new_thread)
  239. thread.checkpoint_set.update(thread=new_thread)
  240. last_thread = thread
  241. Thread.objects.filter(id__in=merged).delete()
  242. new_thread.sync()
  243. new_thread.save(force_update=True)
  244. self.forum.sync()
  245. self.forum.save(force_update=True)
  246. self.request.messages.set_flash(Message(_('Selected threads have been merged into new one.')), 'success', 'threads')
  247. return None
  248. self.message = Message(form.non_field_errors()[0], 'error')
  249. else:
  250. form = MergeThreadsForm(request=self.request, threads=threads)
  251. return self.request.theme.render_to_response('threads/merge.html',
  252. {
  253. 'message': self.message,
  254. 'forum': self.forum,
  255. 'parents': self.parents,
  256. 'threads': threads,
  257. 'form': FormLayout(form),
  258. },
  259. context_instance=RequestContext(self.request));
  260. def action_open(self, ids):
  261. opened = []
  262. for thread in self.threads:
  263. if thread.pk in ids and thread.closed:
  264. opened.append(thread.pk)
  265. thread.last_post.set_checkpoint(self.request, 'opened')
  266. if opened:
  267. Thread.objects.filter(id__in=opened).update(closed=False)
  268. self.request.messages.set_flash(Message(_('Selected threads have been opened.')), 'success', 'threads')
  269. def action_close(self, ids):
  270. closed = []
  271. for thread in self.threads:
  272. if thread.pk in ids and not thread.closed:
  273. closed.append(thread.pk)
  274. thread.last_post.set_checkpoint(self.request, 'closed')
  275. if closed:
  276. Thread.objects.filter(id__in=closed).update(closed=True)
  277. self.request.messages.set_flash(Message(_('Selected threads have been closed.')), 'success', 'threads')
  278. def action_undelete(self, ids):
  279. undeleted = []
  280. posts = 0
  281. for thread in self.threads:
  282. if thread.pk in ids and thread.deleted:
  283. undeleted.append(thread.pk)
  284. posts += thread.replies + 1
  285. thread.start_post.deleted = False
  286. thread.start_post.save(force_update=True)
  287. thread.last_post.set_checkpoint(self.request, 'undeleted')
  288. if undeleted:
  289. self.request.monitor['threads'] = int(self.request.monitor['threads']) + len(undeleted)
  290. self.request.monitor['posts'] = int(self.request.monitor['posts']) + posts
  291. self.forum.sync()
  292. self.forum.save(force_update=True)
  293. Thread.objects.filter(id__in=undeleted).update(deleted=False)
  294. self.request.messages.set_flash(Message(_('Selected threads have been undeleted.')), 'success', 'threads')
  295. def action_soft(self, ids):
  296. deleted = []
  297. posts = 0
  298. for thread in self.threads:
  299. if thread.pk in ids and not thread.deleted:
  300. deleted.append(thread.pk)
  301. posts += thread.replies + 1
  302. thread.start_post.deleted = True
  303. thread.start_post.save(force_update=True)
  304. thread.last_post.set_checkpoint(self.request, 'deleted')
  305. if deleted:
  306. self.request.monitor['threads'] = int(self.request.monitor['threads']) - len(deleted)
  307. self.request.monitor['posts'] = int(self.request.monitor['posts']) - posts
  308. self.forum.sync()
  309. self.forum.save(force_update=True)
  310. Thread.objects.filter(id__in=deleted).update(deleted=True)
  311. self.request.messages.set_flash(Message(_('Selected threads have been softly deleted.')), 'success', 'threads')
  312. def action_hard(self, ids):
  313. deleted = []
  314. posts = 0
  315. for thread in self.threads:
  316. if thread.pk in ids:
  317. deleted.append(thread.pk)
  318. posts += thread.replies + 1
  319. thread.delete()
  320. if deleted:
  321. self.request.monitor['threads'] = int(self.request.monitor['threads']) - len(deleted)
  322. self.request.monitor['posts'] = int(self.request.monitor['posts']) - posts
  323. self.forum.sync()
  324. self.forum.save(force_update=True)
  325. self.request.messages.set_flash(Message(_('Selected threads have been deleted.')), 'success', 'threads')
  326. def __call__(self, request, slug=None, forum=None, page=0):
  327. self.request = request
  328. self.pagination = None
  329. self.parents = None
  330. self.message = request.messages.get_message('threads')
  331. try:
  332. self.fetch_forum(forum)
  333. self.fetch_threads(page)
  334. self.make_form()
  335. if self.form:
  336. response = self.handle_form()
  337. if response:
  338. return response
  339. except Forum.DoesNotExist:
  340. return error404(request)
  341. except ACLError403 as e:
  342. return error403(request, e.message)
  343. except ACLError404 as e:
  344. return error404(request, e.message)
  345. # Merge proxy into forum
  346. self.forum.closed = self.proxy.closed
  347. return request.theme.render_to_response('threads/list.html',
  348. {
  349. 'message': self.message,
  350. 'forum': self.forum,
  351. 'parents': self.parents,
  352. 'count': self.count,
  353. 'list_form': FormFields(self.form).fields if self.form else None,
  354. 'threads': self.threads,
  355. 'pagination': self.pagination,
  356. },
  357. context_instance=RequestContext(request));