forum.py 17 KB

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