posting.py 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. from django.contrib.auth import get_user_model
  2. from django.core.exceptions import PermissionDenied
  3. from django.utils.translation import ugettext_lazy as _, ungettext
  4. from misago.conf import settings
  5. from misago.core import forms
  6. from misago.markup import common_flavour
  7. from misago.threads.permissions import allow_message_user
  8. from misago.threads.validators import validate_title
  9. class ReplyForm(forms.Form):
  10. is_main = True
  11. legend = _("Reply")
  12. template = "misago/posting/replyform.html"
  13. post = forms.CharField(label=_("Message body"), required=False)
  14. def __init__(self, post=None, request=None, *args, **kwargs):
  15. self.request = request
  16. self.post_instance = post
  17. self.parsing_result = {}
  18. super(ReplyForm, self).__init__(*args, **kwargs)
  19. def validate_post(self, post):
  20. if not self.post_instance.pk or self.post_instance.original != post:
  21. self._validate_post(post)
  22. self.parse_post(post)
  23. def _validate_post(self, post):
  24. post_len = len(post)
  25. if not post_len:
  26. raise forms.ValidationError(_("Enter message."))
  27. if post_len < settings.post_length_min:
  28. message = ungettext(
  29. "Posted message should be at least %(limit)s character long.",
  30. "Posted message should be at least %(limit)s characters long.",
  31. settings.post_length_min)
  32. message = message % {'limit': settings.post_length_min}
  33. raise forms.ValidationError(message)
  34. if settings.post_length_max and post_len > settings.post_length_max:
  35. message = ungettext(
  36. "Posted message can't be longer than %(limit)s character.",
  37. "Posted message can't be longer than %(limit)s characters.",
  38. settings.post_length_max)
  39. message = message % {'limit': settings.post_length_max}
  40. raise forms.ValidationError(message)
  41. def parse_post(self, post):
  42. self.parsing_result = common_flavour(
  43. self.request, self.post_instance.poster, post)
  44. self.post_instance.original = self.parsing_result['original_text']
  45. self.post_instance.parsed = self.parsing_result['parsed_text']
  46. def validate_data(self, data):
  47. self.validate_post(data.get('post', ''))
  48. def clean(self):
  49. data = super(ReplyForm, self).clean()
  50. self.validate_data(data)
  51. return data
  52. class ThreadForm(ReplyForm):
  53. legend = _("Thread ")
  54. title = forms.CharField(label=_("Thread title"), required=False)
  55. def __init__(self, thread=None, *args, **kwargs):
  56. self.thread_instance = thread
  57. super(ThreadForm, self).__init__(*args, **kwargs)
  58. def validate_data(self, data):
  59. errors = []
  60. if not data.get('title') and not data.get('post'):
  61. raise forms.ValidationError(_("Enter thread title and message."))
  62. try:
  63. validate_title(data.get('title', ''))
  64. except forms.ValidationError as e:
  65. errors.append(e)
  66. try:
  67. self.validate_post(data.get('post', ''))
  68. except forms.ValidationError as e:
  69. errors.append(e)
  70. if errors:
  71. raise forms.ValidationError(errors)
  72. class ThreadParticipantsForm(forms.Form):
  73. is_supporting = True
  74. location = 'reply_top'
  75. template = "misago/posting/threadparticipantsform.html"
  76. users = forms.CharField(label=_("Invite users to thread"), required=False)
  77. def __init__(self, *args, **kwargs):
  78. self.users_cache = []
  79. self.user = kwargs.pop('user', None)
  80. super(ThreadParticipantsForm, self).__init__(*args, **kwargs)
  81. def clean(self):
  82. cleaned_data = super(ThreadParticipantsForm, self).clean()
  83. if not cleaned_data.get('users'):
  84. raise forms.ValidationError(
  85. _("You have to specify message recipients."))
  86. clean_usernames = []
  87. for name in cleaned_data['users'].split(','):
  88. clean_name = name.strip().lower()
  89. if clean_name == self.user.slug:
  90. raise forms.ValidationError(
  91. _("You can't addres message to yourself."))
  92. if clean_name not in clean_usernames:
  93. clean_usernames.append(clean_name)
  94. max_participants = self.user.acl['max_private_thread_participants']
  95. if max_participants and len(clean_usernames) > max_participants:
  96. message = ungettext("You can't start private thread "
  97. "with more than than %(users)s user.",
  98. "You can't start private thread "
  99. "with more than than %(users)s users.",
  100. max_participants)
  101. message = message % {'users': max_participants}
  102. raise forms.ValidationError(message)
  103. users_qs = get_user_model().objects.filter(slug__in=clean_usernames)
  104. for user in users_qs:
  105. try:
  106. allow_message_user(self.user, user)
  107. except PermissionDenied as e:
  108. raise forms.ValidationError(unicode(e))
  109. self.users_cache.append(user)
  110. if len(self.users_cache) != len(clean_usernames):
  111. valid_usernames = [u.slug for u in self.users_cache]
  112. invalid_usernames = []
  113. for username in clean_usernames:
  114. if username not in valid_usernames:
  115. invalid_usernames.append(username)
  116. message = _("One or more message recipients could "
  117. "not be found: %(usernames)s")
  118. formats = {'usernames': ', '.join(invalid_usernames)}
  119. raise forms.ValidationError(message % formats)
  120. valid_usernames = [u.username for u in self.users_cache]
  121. cleaned_data['users'] = ','.join(valid_usernames)
  122. return cleaned_data
  123. class ThreadLabelFormBase(forms.Form):
  124. is_supporting = True
  125. location = 'after_title'
  126. template = "misago/posting/threadlabelform.html"
  127. def ThreadLabelForm(*args, **kwargs):
  128. labels = kwargs.pop('labels')
  129. choices = [(0, _("No label"))]
  130. choices.extend([(label.pk, label.name ) for label in labels])
  131. field = forms.TypedChoiceField(
  132. label=_("Thread label"),
  133. coerce=int,
  134. choices=choices)
  135. FormType = type("ThreadLabelFormFinal",
  136. (ThreadLabelFormBase,),
  137. {'label': field})
  138. return FormType(*args, **kwargs)
  139. class ThreadPinForm(forms.Form):
  140. is_supporting = True
  141. location = 'lefthand'
  142. template = "misago/posting/threadpinform.html"
  143. is_pinned = forms.YesNoSwitch(
  144. label=_("Pin thread"),
  145. yes_label=_("Pinned thread"),
  146. no_label=_("Unpinned thread"))
  147. class ThreadCloseForm(forms.Form):
  148. is_supporting = True
  149. location = 'lefthand'
  150. template = "misago/posting/threadcloseform.html"
  151. is_closed = forms.YesNoSwitch(
  152. label=_("Close thread"),
  153. yes_label=_("Closed thread"),
  154. no_label=_("Open thread"))