forum.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558
  1. from django.contrib import messages
  2. from django.core.urlresolvers import reverse
  3. from django.db.transaction import atomic
  4. from django.shortcuts import redirect
  5. from django.utils.translation import ugettext_lazy, ugettext as _, ungettext
  6. from misago.core.shortcuts import paginate
  7. from misago.forums.lists import get_forums_list, get_forum_path
  8. from misago.threads import moderation
  9. from misago.threads.models import ANNOUNCEMENT, Thread, Label
  10. from misago.threads.permissions import exclude_invisible_threads
  11. from misago.threads.views.generic.threads import (Actions, Sorting, Threads,
  12. ThreadsView)
  13. __all__ = ['ForumFiltering', 'ForumThreads', 'ForumView']
  14. class ForumActions(Actions):
  15. def get_available_actions(self, kwargs):
  16. self.forum = kwargs['forum']
  17. actions = []
  18. if self.forum.acl['can_change_threads_labels'] == 2:
  19. for label in self.forum.labels:
  20. actions.append({
  21. 'action': 'label:%s' % label.slug,
  22. 'icon': 'tag',
  23. 'name': _('Label as "%(label)s"') % {'label': label.name}
  24. })
  25. if self.forum.labels:
  26. actions.append({
  27. 'action': 'unlabel',
  28. 'icon': 'times-circle',
  29. 'name': _("Remove labels")
  30. })
  31. if self.forum.acl['can_change_threads_weight'] == 2:
  32. actions.append({
  33. 'action': 'announce',
  34. 'icon': 'star',
  35. 'name': _("Change to announcements")
  36. })
  37. if self.forum.acl['can_change_threads_weight']:
  38. actions.append({
  39. 'action': 'pin',
  40. 'icon': 'bookmark',
  41. 'name': _("Change to pinned")
  42. })
  43. actions.append({
  44. 'action': 'reset',
  45. 'icon': 'circle',
  46. 'name': _("Reset weight")
  47. })
  48. if self.forum.acl['can_review_moderated_content']:
  49. actions.append({
  50. 'action': 'approve',
  51. 'icon': 'check',
  52. 'name': _("Approve threads")
  53. })
  54. if self.forum.acl['can_move_threads']:
  55. actions.append({
  56. 'action': 'move',
  57. 'icon': 'arrow-right',
  58. 'name': _("Move threads")
  59. })
  60. if self.forum.acl['can_merge_threads']:
  61. actions.append({
  62. 'action': 'merge',
  63. 'icon': 'reply-all',
  64. 'name': _("Merge threads")
  65. })
  66. if self.forum.acl['can_close_threads']:
  67. actions.append({
  68. 'action': 'open',
  69. 'icon': 'unlock-alt',
  70. 'name': _("Open threads")
  71. })
  72. actions.append({
  73. 'action': 'close',
  74. 'icon': 'lock',
  75. 'name': _("Close threads")
  76. })
  77. if self.forum.acl['can_hide_threads']:
  78. actions.append({
  79. 'action': 'unhide',
  80. 'icon': 'eye',
  81. 'name': _("Unhide threads")
  82. })
  83. actions.append({
  84. 'action': 'hide',
  85. 'icon': 'eye-slash',
  86. 'name': _("Hide threads")
  87. })
  88. if self.forum.acl['can_hide_threads'] == 2:
  89. actions.append({
  90. 'action': 'delete',
  91. 'icon': 'times',
  92. 'name': _("Delete threads")
  93. })
  94. return actions
  95. def action_label(self, request, threads, label_slug):
  96. for label in self.forum.labels:
  97. if label.slug == label_slug:
  98. break
  99. else:
  100. raise ModerationError(_("Requested action is invalid."))
  101. changed_threads = 0
  102. for thread in threads:
  103. if moderation.label_thread(request.user, thread, label):
  104. changed_threads += 1
  105. if changed_threads:
  106. message = ungettext(
  107. '%(changed)d thread was labeled "%(label)s".',
  108. '%(changed)d threads were labeled "%(label)s".',
  109. changed_threads)
  110. messages.success(request, message % {
  111. 'changed': changed_threads,
  112. 'label': label.name
  113. })
  114. else:
  115. message = ("No threads were labeled.")
  116. messages.info(request, message)
  117. def action_unlabel(self, request, threads):
  118. changed_threads = 0
  119. for thread in threads:
  120. if moderation.unlabel_thread(request.user, thread):
  121. changed_threads += 1
  122. if changed_threads:
  123. message = ungettext(
  124. '%(changed)d thread label was remoded.',
  125. '%(changed)d threads labels were removed.',
  126. changed_threads)
  127. messages.success(request, message % {'changed': changed_threads})
  128. else:
  129. message = ("No threads were unlabeled.")
  130. messages.info(request, message)
  131. def action_announce(self, request, threads):
  132. changed_threads = 0
  133. for thread in threads:
  134. if moderation.announce_thread(request.user, thread):
  135. changed_threads += 1
  136. if changed_threads:
  137. message = ungettext(
  138. '%(changed)d thread was changed to announcement.',
  139. '%(changed)d threads were changed to announcements.',
  140. changed_threads)
  141. messages.success(request, message % {'changed': changed_threads})
  142. else:
  143. message = ("No threads were changed to announcements.")
  144. messages.info(request, message)
  145. def action_pin(self, request, threads):
  146. changed_threads = 0
  147. for thread in threads:
  148. if moderation.pin_thread(request.user, thread):
  149. changed_threads += 1
  150. if changed_threads:
  151. message = ungettext(
  152. '%(changed)d thread was pinned.',
  153. '%(changed)d threads were pinned.',
  154. changed_threads)
  155. messages.success(request, message % {'changed': changed_threads})
  156. else:
  157. message = ("No threads were pinned.")
  158. messages.info(request, message)
  159. def action_reset(self, request, threads):
  160. changed_threads = 0
  161. for thread in threads:
  162. if moderation.reset_thread(request.user, thread):
  163. changed_threads += 1
  164. if changed_threads:
  165. message = ungettext(
  166. '%(changed)d thread weight was reset.',
  167. '%(changed)d threads weight was reset.',
  168. changed_threads)
  169. messages.success(request, message % {'changed': changed_threads})
  170. else:
  171. message = ("No threads weight was reset.")
  172. messages.info(request, message)
  173. def action_close(self, request, threads):
  174. changed_threads = 0
  175. for thread in threads:
  176. if moderation.close_thread(request.user, thread):
  177. changed_threads += 1
  178. if changed_threads:
  179. message = ungettext(
  180. '%(changed)d thread was closed.',
  181. '%(changed)d threads were closed.',
  182. changed_threads)
  183. messages.success(request, message % {'changed': changed_threads})
  184. else:
  185. message = ("No threads were closed.")
  186. messages.info(request, message)
  187. def action_open(self, request, threads):
  188. changed_threads = 0
  189. for thread in threads:
  190. if moderation.open_thread(request.user, thread):
  191. changed_threads += 1
  192. if changed_threads:
  193. message = ungettext(
  194. '%(changed)d thread was opened.',
  195. '%(changed)d threads were opened.',
  196. changed_threads)
  197. messages.success(request, message % {'changed': changed_threads})
  198. else:
  199. message = ("No threads were opened.")
  200. messages.info(request, message)
  201. def action_unhide(self, request, threads):
  202. changed_threads = 0
  203. for thread in threads:
  204. if moderation.unhide_thread(request.user, thread):
  205. changed_threads += 1
  206. if changed_threads:
  207. with atomic():
  208. self.forum.synchronize()
  209. self.forum.save()
  210. message = ungettext(
  211. '%(changed)d thread was made visible.',
  212. '%(changed)d threads were made visible.',
  213. changed_threads)
  214. messages.success(request, message % {'changed': changed_threads})
  215. else:
  216. message = ("No threads were made visible.")
  217. messages.info(request, message)
  218. def action_hide(self, request, threads):
  219. changed_threads = 0
  220. for thread in threads:
  221. if moderation.hide_thread(request.user, thread):
  222. changed_threads += 1
  223. if changed_threads:
  224. with atomic():
  225. self.forum.synchronize()
  226. self.forum.save()
  227. message = ungettext(
  228. '%(changed)d thread was hidden.',
  229. '%(changed)d threads were hidden.',
  230. changed_threads)
  231. messages.success(request, message % {'changed': changed_threads})
  232. else:
  233. message = ("No threads were hidden.")
  234. messages.info(request, message)
  235. def action_delete(self, request, threads):
  236. changed_threads = 0
  237. for thread in threads:
  238. if moderation.delete_thread(request.user, thread):
  239. changed_threads += 1
  240. if changed_threads:
  241. with atomic():
  242. self.forum.synchronize()
  243. self.forum.save()
  244. message = ungettext(
  245. '%(changed)d thread was deleted.',
  246. '%(changed)d threads were deleted.',
  247. changed_threads)
  248. messages.success(request, message % {'changed': changed_threads})
  249. else:
  250. message = ("No threads were deleted.")
  251. messages.info(request, message)
  252. class ForumFiltering(object):
  253. def __init__(self, forum, link_name, link_params):
  254. self.forum = forum
  255. self.link_name = link_name
  256. self.link_params = link_params.copy()
  257. self.filters = self.get_available_filters()
  258. def get_available_filters(self):
  259. filters = []
  260. if self.forum.acl['can_see_all_threads']:
  261. filters.append({
  262. 'type': 'my-threads',
  263. 'name': _("My threads"),
  264. 'is_label': False,
  265. })
  266. if self.forum.acl['can_see_reports']:
  267. filters.append({
  268. 'type': 'reported',
  269. 'name': _("With reported posts"),
  270. 'is_label': False,
  271. })
  272. if self.forum.acl['can_review_moderated_content']:
  273. filters.extend(({
  274. 'type': 'moderated-threads',
  275. 'name': _("Moderated threads"),
  276. 'is_label': False,
  277. },
  278. {
  279. 'type': 'moderated-posts',
  280. 'name': _("With moderated posts"),
  281. 'is_label': False,
  282. }))
  283. for label in self.forum.labels:
  284. filters.append({
  285. 'type': label.slug,
  286. 'name': label.name,
  287. 'is_label': True,
  288. 'css_class': label.css_class,
  289. })
  290. return filters
  291. def clean_kwargs(self, kwargs):
  292. show = kwargs.get('show')
  293. if show:
  294. available_filters = [method['type'] for method in self.filters]
  295. if show in available_filters:
  296. self.show = show
  297. else:
  298. kwargs.pop('show')
  299. else:
  300. self.show = None
  301. return kwargs
  302. def filter(self, threads):
  303. threads.filter(self.show)
  304. def get_filtering_dics(self):
  305. try:
  306. return self._dicts
  307. except AttributeError:
  308. self._dicts = self.create_dicts()
  309. return self._dicts
  310. def create_dicts(self):
  311. dicts = []
  312. if self.forum.acl['can_see_all_threads']:
  313. default_name = _("All threads")
  314. else:
  315. default_name = _("Your threads")
  316. self.link_params.pop('show', None)
  317. dicts.append({
  318. 'type': None,
  319. 'url': reverse(self.link_name, kwargs=self.link_params),
  320. 'name': default_name,
  321. 'is_label': False,
  322. })
  323. for filtering in self.filters:
  324. self.link_params['show'] = filtering['type']
  325. filtering['url'] = reverse(self.link_name, kwargs=self.link_params)
  326. dicts.append(filtering)
  327. return dicts
  328. @property
  329. def is_active(self):
  330. return bool(self.show)
  331. @property
  332. def current(self):
  333. try:
  334. return self._current
  335. except AttributeError:
  336. for filtering in self.get_filtering_dics():
  337. if filtering['type'] == self.show:
  338. self._current = filtering
  339. return filtering
  340. def choices(self):
  341. if self.show:
  342. choices = []
  343. for filtering in self.get_filtering_dics():
  344. if filtering['type'] != self.show:
  345. choices.append(filtering)
  346. return choices
  347. else:
  348. return self.get_filtering_dics()[1:]
  349. class ForumThreads(Threads):
  350. def __init__(self, user, forum):
  351. self.user = user
  352. self.forum = forum
  353. self.filter_by = None
  354. self.sort_by = ('-weight', '-last_post_on')
  355. def filter(self, filter_by):
  356. self.filter_by = filter_by
  357. def sort(self, sort_by):
  358. if sort_by[0] == '-':
  359. weight = '-weight'
  360. else:
  361. weight = 'weight'
  362. self.sort_by = (weight, sort_by)
  363. def list(self, page=0):
  364. queryset = self.get_queryset()
  365. queryset = queryset.order_by(*self.sort_by)
  366. announcements_qs = queryset.filter(weight=ANNOUNCEMENT)
  367. threads_qs = queryset.filter(weight__lt=ANNOUNCEMENT)
  368. self._page = paginate(threads_qs, page, 20, 10)
  369. self._paginator = self._page.paginator
  370. threads = []
  371. for announcement in announcements_qs:
  372. threads.append(announcement)
  373. for thread in self._page.object_list:
  374. threads.append(thread)
  375. for thread in threads:
  376. thread.forum = self.forum
  377. self.label_threads(threads, self.forum.labels)
  378. self.make_threads_read_aware(threads)
  379. return threads
  380. def filter_threads(self, queryset):
  381. if self.filter_by == 'my-threads':
  382. return queryset.filter(starter_id=self.user.id)
  383. else:
  384. if self.forum.acl['can_see_own_threads']:
  385. if self.user.is_authenticated():
  386. queryset = queryset.filter(starter_id=self.user.id)
  387. else:
  388. queryset = queryset.filter(starter_id=0)
  389. if self.filter_by == 'reported':
  390. return queryset.filter(has_reported_posts=True)
  391. elif self.filter_by == 'moderated-threads':
  392. return queryset.filter(is_moderated=True)
  393. elif self.filter_by == 'moderated-posts':
  394. return queryset.filter(has_moderated_posts=True)
  395. else:
  396. for label in self.forum.labels:
  397. if label.slug == self.filter_by:
  398. return queryset.filter(label_id=label.pk)
  399. else:
  400. return queryset
  401. def get_queryset(self):
  402. queryset = exclude_invisible_threads(
  403. self.user, self.forum, self.forum.thread_set)
  404. return self.filter_threads(queryset)
  405. error_message = ("threads list has to be loaded via call to list() before "
  406. "pagination data will be available")
  407. @property
  408. def paginator(self):
  409. try:
  410. return self._paginator
  411. except AttributeError:
  412. raise AttributeError(self.error_message)
  413. @property
  414. def page(self):
  415. try:
  416. return self._page
  417. except AttributeError:
  418. raise AttributeError(self.error_message)
  419. class ForumView(ThreadsView):
  420. """
  421. Basic view for forum threads lists
  422. """
  423. template = 'misago/threads/forum.html'
  424. Threads = ForumThreads
  425. Sorting = Sorting
  426. Filtering = ForumFiltering
  427. Actions = ForumActions
  428. def dispatch(self, request, *args, **kwargs):
  429. forum = self.get_forum(request, **kwargs)
  430. forum.labels = Label.objects.get_forum_labels(forum)
  431. if forum.lft + 1 < forum.rght:
  432. forum.subforums = get_forums_list(request.user, forum)
  433. else:
  434. forum.subforums = []
  435. page_number = kwargs.pop('page', None)
  436. cleaned_kwargs = self.clean_kwargs(request, kwargs)
  437. sorting = self.Sorting(self.link_name, cleaned_kwargs)
  438. cleaned_kwargs = sorting.clean_kwargs(cleaned_kwargs)
  439. filtering = self.Filtering(forum, self.link_name, cleaned_kwargs)
  440. cleaned_kwargs = filtering.clean_kwargs(cleaned_kwargs)
  441. if cleaned_kwargs != kwargs:
  442. return redirect('misago:forum', **cleaned_kwargs)
  443. threads = self.Threads(request.user, forum)
  444. sorting.sort(threads)
  445. filtering.filter(threads)
  446. actions = self.Actions(user=request.user, forum=forum)
  447. if request.method == 'POST':
  448. # see if we can delegate anything to actions manager
  449. response = actions.handle_post(request, threads.get_queryset())
  450. if response:
  451. return response
  452. return self.render(request, {
  453. 'link_name': self.link_name,
  454. 'links_params': cleaned_kwargs,
  455. 'forum': forum,
  456. 'path': get_forum_path(forum),
  457. 'threads': threads.list(page_number),
  458. 'page': threads.page,
  459. 'paginator': threads.paginator,
  460. 'list_actions': actions.get_list(),
  461. 'selected_threads': actions.get_selected_ids(),
  462. 'sorting': sorting,
  463. 'filtering': filtering,
  464. })