list.py 19 KB

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