threads.py 6.9 KB

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