forms.py 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. from mptt.forms import TreeNodeChoiceField, TreeNodeMultipleChoiceField
  2. from django import forms
  3. from django.db import models
  4. from django.utils.html import conditional_escape, mark_safe
  5. from django.utils.translation import ugettext_lazy as _
  6. from misago.core.forms import YesNoSwitch
  7. from misago.core.validators import validate_sluggable
  8. from misago.threads.threadtypes import trees_map
  9. from . import THREADS_ROOT_NAME
  10. from .models import Category, CategoryRole
  11. class AdminCategoryFieldMixin(object):
  12. def __init__(self, *args, **kwargs):
  13. self.base_level = kwargs.pop('base_level', 1)
  14. kwargs['level_indicator'] = kwargs.get('level_indicator', '- - ')
  15. threads_tree_id = trees_map.get_tree_id_for_root(THREADS_ROOT_NAME)
  16. queryset = Category.objects.filter(tree_id=threads_tree_id)
  17. if not kwargs.pop('include_root', False):
  18. queryset = queryset.exclude(special_role="root_category")
  19. kwargs.setdefault('queryset', queryset)
  20. super(AdminCategoryFieldMixin, self).__init__(*args, **kwargs)
  21. def _get_level_indicator(self, obj):
  22. level = getattr(obj, obj._mptt_meta.level_attr) - self.base_level
  23. if level > 0:
  24. return mark_safe(conditional_escape(self.level_indicator) * level)
  25. else:
  26. return ''
  27. class AdminCategoryChoiceField(AdminCategoryFieldMixin, TreeNodeChoiceField):
  28. pass
  29. class AdminCategoryMultipleChoiceField(AdminCategoryFieldMixin, TreeNodeMultipleChoiceField):
  30. pass
  31. class CategoryFormBase(forms.ModelForm):
  32. name = forms.CharField(label=_("Name"), validators=[validate_sluggable()])
  33. description = forms.CharField(
  34. label=_("Description"),
  35. max_length=2048,
  36. required=False,
  37. widget=forms.Textarea(attrs={'rows': 3}),
  38. help_text=_("Optional description explaining category intented purpose."),
  39. )
  40. css_class = forms.CharField(
  41. label=_("CSS class"),
  42. required=False,
  43. help_text=_(
  44. "Optional CSS class used to customize this category appearance from templates."
  45. ),
  46. )
  47. is_closed = YesNoSwitch(
  48. label=_("Closed category"),
  49. required=False,
  50. help_text=_("Only members with valid permissions can post in closed categories."),
  51. )
  52. css_class = forms.CharField(
  53. label=_("CSS class"),
  54. required=False,
  55. help_text=_(
  56. "Optional CSS class used to customize this category appearance from templates."
  57. ),
  58. )
  59. prune_started_after = forms.IntegerField(
  60. label=_("Thread age"),
  61. min_value=0,
  62. help_text=_(
  63. "Prune thread if number of days since its creation is greater than specified. "
  64. "Enter 0 to disable this pruning criteria."
  65. ),
  66. )
  67. prune_replied_after = forms.IntegerField(
  68. label=_("Last reply"),
  69. min_value=0,
  70. help_text=_(
  71. "Prune thread if number of days since last reply is greater than specified. "
  72. "Enter 0 to disable this pruning criteria."
  73. ),
  74. )
  75. class Meta:
  76. model = Category
  77. fields = [
  78. 'name',
  79. 'description',
  80. 'css_class',
  81. 'is_closed',
  82. 'prune_started_after',
  83. 'prune_replied_after',
  84. 'archive_pruned_in',
  85. ]
  86. def clean_copy_permissions(self):
  87. data = self.cleaned_data['copy_permissions']
  88. if data and data.pk == self.instance.pk:
  89. message = _("Permissions cannot be copied from category into itself.")
  90. raise forms.ValidationError(message)
  91. return data
  92. def clean_archive_pruned_in(self):
  93. data = self.cleaned_data['archive_pruned_in']
  94. if data and data.pk == self.instance.pk:
  95. message = _("Category cannot act as archive for itself.")
  96. raise forms.ValidationError(message)
  97. return data
  98. def clean(self):
  99. data = super(CategoryFormBase, self).clean()
  100. self.instance.set_name(data.get('name'))
  101. return data
  102. def CategoryFormFactory(instance):
  103. parent_queryset = Category.objects.all_categories(True).order_by('lft')
  104. if instance.pk:
  105. not_siblings = models.Q(lft__lt=instance.lft)
  106. not_siblings = not_siblings | models.Q(rght__gt=instance.rght)
  107. parent_queryset = parent_queryset.filter(not_siblings)
  108. return type(
  109. 'CategoryFormFinal', (CategoryFormBase, ), {
  110. 'new_parent': AdminCategoryChoiceField(
  111. label=_("Parent category"),
  112. queryset=parent_queryset,
  113. initial=instance.parent,
  114. empty_label=None,
  115. ),
  116. 'copy_permissions': AdminCategoryChoiceField(
  117. label=_("Copy permissions"),
  118. help_text=_(
  119. "You can replace this category permissions with "
  120. "permissions copied from category selected here."
  121. ),
  122. queryset=Category.objects.all_categories(),
  123. empty_label=_("Don't copy permissions"),
  124. required=False,
  125. ),
  126. 'archive_pruned_in': AdminCategoryChoiceField(
  127. label=_("Archive"),
  128. help_text=_(
  129. "Instead of being deleted, pruned threads can be "
  130. "moved to designated category."
  131. ),
  132. queryset=Category.objects.all_categories(),
  133. empty_label=_("Don't archive pruned threads"),
  134. required=False,
  135. ),
  136. }
  137. )
  138. class DeleteCategoryFormBase(forms.ModelForm):
  139. class Meta:
  140. model = Category
  141. fields = []
  142. def clean(self):
  143. data = super(DeleteCategoryFormBase, self).clean()
  144. if data.get('move_threads_to'):
  145. if data['move_threads_to'].pk == self.instance.pk:
  146. message = _("You are trying to move this category threads to itself.")
  147. raise forms.ValidationError(message)
  148. moving_to_child = self.instance.has_child(data['move_threads_to'])
  149. if moving_to_child and not data.get('move_children_to'):
  150. message = _(
  151. "You are trying to move this category threads to a "
  152. "child category that will be deleted together with "
  153. "this category."
  154. )
  155. raise forms.ValidationError(message)
  156. return data
  157. def DeleteFormFactory(instance):
  158. content_queryset = Category.objects.all_categories().order_by('lft')
  159. fields = {
  160. 'move_threads_to': AdminCategoryChoiceField(
  161. label=_("Move category threads to"),
  162. queryset=content_queryset,
  163. initial=instance.parent,
  164. empty_label=_('Delete with category'),
  165. required=False,
  166. )
  167. }
  168. not_siblings = models.Q(lft__lt=instance.lft)
  169. not_siblings = not_siblings | models.Q(rght__gt=instance.rght)
  170. children_queryset = Category.objects.all_categories(True)
  171. children_queryset = children_queryset.filter(not_siblings).order_by('lft')
  172. if children_queryset.exists():
  173. fields['move_children_to'] = AdminCategoryChoiceField(
  174. label=_("Move child categories to"),
  175. queryset=children_queryset,
  176. empty_label=_('Delete with category'),
  177. required=False,
  178. )
  179. return type('DeleteCategoryFormFinal', (DeleteCategoryFormBase, ), fields)
  180. class CategoryRoleForm(forms.ModelForm):
  181. name = forms.CharField(label=_("Role name"))
  182. class Meta:
  183. model = CategoryRole
  184. fields = ['name']
  185. def RoleCategoryACLFormFactory(category, category_roles, selected_role):
  186. attrs = {
  187. 'category': category,
  188. 'role': forms.ModelChoiceField(
  189. label=_("Role"),
  190. required=False,
  191. queryset=category_roles,
  192. initial=selected_role,
  193. empty_label=_("No access"),
  194. )
  195. }
  196. return type('RoleCategoryACLForm', (forms.Form, ), attrs)
  197. def CategoryRolesACLFormFactory(role, category_roles, selected_role):
  198. attrs = {
  199. 'role': role,
  200. 'category_role': forms.ModelChoiceField(
  201. label=_("Role"),
  202. required=False,
  203. queryset=category_roles,
  204. initial=selected_role,
  205. empty_label=_("No access"),
  206. )
  207. }
  208. return type('CategoryRolesACLForm', (forms.Form, ), attrs)