threads.py 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  1. from django.contrib import messages
  2. from django.core.urlresolvers import reverse
  3. from django.shortcuts import redirect
  4. from django.utils.translation import ugettext_lazy, ugettext as _
  5. from misago.core.shortcuts import paginate
  6. from misago.readtracker import threadstracker
  7. from misago.threads.models import Label
  8. from misago.threads.moderation import ModerationError
  9. from misago.threads.views.generic.base import ViewBase
  10. __all__ = ['Actions', 'Sorting', 'Threads', 'ThreadsView']
  11. class Actions(object):
  12. select_threads_message = ugettext_lazy(
  13. "You have to select at least one thread")
  14. def __init__(self, **kwargs):
  15. if kwargs.get('user').is_authenticated():
  16. self.available_actions = self.get_available_actions(kwargs)
  17. else:
  18. self.available_actions = []
  19. self.selected_ids = []
  20. def get_available_actions(self, kwargs):
  21. raise NotImplementedError("get_available_actions has to return list "
  22. "of dicts with allowed actions")
  23. def resolve_action(self, request):
  24. action_name = request.POST.get('action')
  25. if ':' in action_name:
  26. action_bits = action_name.split(':')
  27. action_name = action_bits[0]
  28. action_arg = action_bits[1]
  29. else:
  30. action_arg = None
  31. for action in self.available_actions:
  32. if action['action'] == action_name:
  33. action_callable = 'action_%s' % action_name
  34. return getattr(self, action_callable), action_arg
  35. else:
  36. raise ModerationError(_("Requested action is invalid."))
  37. def clean_selection(self, data):
  38. filtered_data = []
  39. for pk in data[:50]: # a tiny fail-safe to avoid too big workloads
  40. try:
  41. filtered_data.append(int(pk))
  42. except ValueError:
  43. pass
  44. if not filtered_data:
  45. raise ModerationError(self.select_threads_message)
  46. return filtered_data
  47. def handle_post(self, request, queryset):
  48. try:
  49. action, action_arg = self.resolve_action(request)
  50. self.selected_ids = self.clean_selection(
  51. request.POST.getlist('thread', []))
  52. filtered_queryset = queryset.filter(pk__in=self.selected_ids)
  53. if filtered_queryset.exists():
  54. if action_arg:
  55. response = action(request, filtered_queryset, action_arg)
  56. else:
  57. response = action(request, filtered_queryset)
  58. if response:
  59. return response
  60. else:
  61. # prepare default response: page reload
  62. return redirect(request.path)
  63. else:
  64. raise ModerationError(self.select_threads_message)
  65. except ModerationError as e:
  66. messages.error(request, e.message)
  67. return False
  68. def get_list(self):
  69. return self.available_actions
  70. def get_selected_ids(self):
  71. return self.selected_ids
  72. class Sorting(object):
  73. sortings = (
  74. {
  75. 'method': 'recently-replied',
  76. 'name': ugettext_lazy("Recently replied"),
  77. 'order_by': '-last_post_on',
  78. },
  79. {
  80. 'method': 'last-replied',
  81. 'name': ugettext_lazy("Last replied"),
  82. 'order_by': 'last_post_on',
  83. },
  84. {
  85. 'method': 'most-replied',
  86. 'name': ugettext_lazy("Most replied"),
  87. 'order_by': '-replies',
  88. },
  89. {
  90. 'method': 'least-replied',
  91. 'name': ugettext_lazy("Least replied"),
  92. 'order_by': 'replies',
  93. },
  94. {
  95. 'method': 'newest',
  96. 'name': ugettext_lazy("Newest"),
  97. 'order_by': '-id',
  98. },
  99. {
  100. 'method': 'oldest',
  101. 'name': ugettext_lazy("Oldest"),
  102. 'order_by': 'id',
  103. },
  104. )
  105. def __init__(self, link_name, link_params):
  106. self.link_name = link_name
  107. self.link_params = link_params.copy()
  108. self.default_method = self.sortings[0]['method']
  109. def clean_kwargs(self, kwargs):
  110. sorting = kwargs.get('sort')
  111. if sorting:
  112. if sorting == self.default_method:
  113. kwargs.pop('sort')
  114. return kwargs
  115. available_sortings = [method['method'] for method in self.sortings]
  116. if sorting not in available_sortings:
  117. kwargs.pop('sort')
  118. return kwargs
  119. else:
  120. sorting = self.default_method
  121. self.set_sorting(sorting)
  122. return kwargs
  123. def set_sorting(self, method):
  124. for sorting in self.sortings:
  125. if sorting['method'] == method:
  126. self.sorting = sorting
  127. break
  128. @property
  129. def name(self):
  130. return self.sorting['name']
  131. def choices(self):
  132. choices = []
  133. for sorting in self.sortings:
  134. if sorting['method'] != self.sorting['method']:
  135. if sorting['method'] == self.default_method:
  136. self.link_params.pop('sort', None)
  137. else:
  138. self.link_params['sort'] = sorting['method']
  139. url = reverse(self.link_name, kwargs=self.link_params)
  140. choices.append({
  141. 'name': sorting['name'],
  142. 'url': url,
  143. })
  144. return choices
  145. def sort(self, threads):
  146. threads.sort(self.sorting['order_by'])
  147. class Threads(object):
  148. def __init__(self, user):
  149. self.user = user
  150. def sort(self, sort_by):
  151. self.queryset = self.queryset.order_by(sort_by)
  152. def label_threads(self, threads, labels=None):
  153. if labels:
  154. labels_dict = dict([(label.pk, label) for label in labels])
  155. else:
  156. labels_dict = Label.objects.get_cached_labels_dict()
  157. for thread in threads:
  158. thread.label = labels_dict.get(thread.label_id)
  159. def make_threads_read_aware(self, threads):
  160. threadstracker.make_read_aware(self.user, threads)
  161. class ThreadsView(ViewBase):
  162. def get_threads(self, request, kwargs):
  163. queryset = self.get_threads_queryset(request, forum)
  164. queryset = threads_qs.order_by('-last_post_id')
  165. page = paginate(threads_qs, kwargs.get('page', 0), 30, 10)
  166. threads = [thread for thread in page.object_list]
  167. return page, threads
  168. def get_threads_queryset(self, request):
  169. return forum.thread_set.all().order_by('-last_post_id')
  170. def clean_kwargs(self, request, kwargs):
  171. cleaned_kwargs = kwargs.copy()
  172. if request.user.is_anonymous():
  173. """we don't allow sort/filter for guests"""
  174. cleaned_kwargs.pop('sort', None)
  175. cleaned_kwargs.pop('show', None)
  176. return cleaned_kwargs