widgets.py 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476
  1. from django import forms
  2. from django.core.urlresolvers import reverse
  3. from django.shortcuts import redirect
  4. from django.template import RequestContext
  5. from django.utils.translation import ugettext_lazy as _
  6. from jinja2 import TemplateNotFound
  7. import math
  8. from misago.forms import Form
  9. from misago.forms.layouts import *
  10. from misago.messages import Message, BasicMessage
  11. from misago.search import SearchException
  12. """
  13. Class widgets
  14. """
  15. class BaseWidget(object):
  16. """
  17. Admin Widget abstract class, providing widgets with common or shared functionality
  18. """
  19. admin = None
  20. id = None
  21. fallback = None
  22. name = None
  23. help = None
  24. notfound_message = None
  25. def __new__(cls, request, **kwargs):
  26. obj = super(BaseWidget, cls).__new__(cls)
  27. if not obj.name:
  28. obj.name = obj.get_name()
  29. if not obj.help:
  30. obj.help = obj.get_help()
  31. return obj(request, **kwargs)
  32. def get_token(self, token):
  33. return '%s_%s_%s' % (self.id, token, str('%s.%s' % (self.admin.model.__module__, self.admin.model.__name__)))
  34. def get_url(self):
  35. return reverse(self.admin.get_action_attr(self.id, 'route'))
  36. def get_name(self):
  37. return self.admin.get_action_attr(self.id, 'name')
  38. def get_help(self):
  39. return self.admin.get_action_attr(self.id, 'help')
  40. def get_id(self):
  41. return 'admin_%s' % self.id
  42. def get_templates(self, template):
  43. return ('%s/%s/%s.html' % (str(self.admin.model.__module__).split('.')[1], str(self.admin.route).lower(), template),
  44. '%s/%s.html' % (str(self.admin.model.__module__).split('.')[1], template),
  45. 'admin/%s.html' % template)
  46. def get_fallback_url(self, request):
  47. return reverse(self.fallback)
  48. def get_target(self, request, model):
  49. pass
  50. def get_target_name(self, model):
  51. try:
  52. return model.__dict__[self.target_name]
  53. except AttributeError:
  54. return None
  55. def get_and_validate_target(self, request, target):
  56. try:
  57. model = self.admin.model.objects.select_related().get(pk=target)
  58. self.get_target(request, model)
  59. return model
  60. except self.admin.model.DoesNotExist:
  61. request.messages.set_flash(BasicMessage(self.notfound_message), 'error', self.admin.id)
  62. except ValueError as e:
  63. request.messages.set_flash(BasicMessage(e.args[0]), 'error', self.admin.id)
  64. return None
  65. class ListWidget(BaseWidget):
  66. """
  67. Items list widget
  68. """
  69. actions =[]
  70. columns = []
  71. sortables = {}
  72. default_sorting = None
  73. search_form = None
  74. is_filtering = False
  75. pagination = None
  76. template = 'list'
  77. hide_actions = False
  78. empty_message = _('There are no items to display')
  79. empty_search_message = _('Search has returned no items')
  80. nothing_checked_message = _('You have to select at least one item.')
  81. prompt_select = False
  82. def get_item_actions(self, request, item):
  83. """
  84. Provides request and item, should return list of tuples with item actions in following format:
  85. (id, name, help, icon, link)
  86. """
  87. return []
  88. def action(self, icon=None, name=None, url=None, post=False, prompt=None):
  89. """
  90. Function call to make hash with item actions
  91. """
  92. if prompt:
  93. self.prompt_select = True
  94. return {
  95. 'icon': icon,
  96. 'name': name,
  97. 'url': url,
  98. 'post': post,
  99. 'prompt': prompt,
  100. }
  101. def get_search_form(self, request):
  102. """
  103. Build a form object with items search
  104. """
  105. return self.search_form
  106. def set_filters(self, model, filters):
  107. """
  108. Set filters on model using filters from session
  109. """
  110. return None
  111. def get_table_form(self, request, page_items):
  112. """
  113. Build a form object with list of all items fields
  114. """
  115. return None
  116. def get_actions_form(self, page_items):
  117. """
  118. Build a form object with list of all items actions
  119. """
  120. if not self.actions:
  121. return None # Dont build form
  122. form_fields = {}
  123. list_choices = []
  124. for action in self.actions:
  125. list_choices.append((action[0], action[1]))
  126. form_fields['list_action'] = forms.ChoiceField(choices=list_choices)
  127. list_choices = []
  128. for item in page_items:
  129. list_choices.append((item.pk, None))
  130. form_fields['list_items'] = forms.MultipleChoiceField(choices=list_choices,widget=forms.CheckboxSelectMultiple)
  131. return type('AdminListForm', (Form,), form_fields)
  132. def get_sorting(self, request):
  133. """
  134. Return list sorting method.
  135. A list with three values:
  136. - Field we use to sort over
  137. - Sorting direction
  138. - order_by() argument
  139. """
  140. sorting_method = None
  141. if request.session.get(self.get_token('sort')) and request.session.get(self.get_token('sort'))[0] in self.sortables:
  142. sorting_method = request.session.get(self.get_token('sort'))
  143. if request.GET.get('sort') and request.GET.get('sort') in self.sortables:
  144. new_sorting = request.GET.get('sort')
  145. sorting_dir = int(request.GET.get('dir')) == 1
  146. sorting_method = [
  147. new_sorting,
  148. sorting_dir,
  149. new_sorting if sorting_dir else '-%s' % new_sorting
  150. ]
  151. request.session[self.get_token('sort')] = sorting_method
  152. if not sorting_method:
  153. new_sorting = self.sortables.keys()[0]
  154. if self.default_sorting in self.sortables:
  155. new_sorting = self.default_sorting
  156. sorting_method = [
  157. new_sorting,
  158. self.sortables[new_sorting] == True,
  159. new_sorting if self.sortables[new_sorting] else '-%s' % new_sorting
  160. ]
  161. return sorting_method
  162. def get_pagination_url(self, page):
  163. return reverse(self.admin.get_action_attr(self.id, 'route'), kwargs={'page': page})
  164. def get_pagination(self, request, total, page):
  165. """
  166. Return list pagination.
  167. A list with three values:
  168. - Offset for ORM slicing
  169. - Length of slice
  170. - no. of prev page (or -1 for first page)
  171. - no. of next page (or -1 for last page)
  172. - Current page
  173. - Pages total
  174. """
  175. if not self.pagination or total < 0:
  176. # Dont do anything if we are not paging
  177. return None
  178. # Set basic pagination, use either Session cache or new page value
  179. pagination = {'start': 0, 'stop': 0, 'prev': -1, 'next': -1}
  180. if request.session.get(self.get_token('pagination')):
  181. pagination['start'] = request.session.get(self.get_token('pagination'))
  182. page = int(page)
  183. if page > 0:
  184. pagination['start'] = (page - 1) * self.pagination
  185. # Set page and total stat
  186. pagination['page'] = int(pagination['start'] / self.pagination) + 1
  187. pagination['total'] = int(math.ceil(total / float(self.pagination)))
  188. # Fix too large offset
  189. if pagination['start'] > total:
  190. pagination['start'] = 0
  191. # Allow prev/next?
  192. if total > self.pagination:
  193. if pagination['page'] > 1:
  194. pagination['prev'] = pagination['page'] - 1
  195. if pagination['page'] < pagination['total']:
  196. pagination['next'] = pagination['page'] + 1
  197. # Set stop offset
  198. pagination['stop'] = pagination['start'] + self.pagination
  199. return pagination
  200. def __call__(self, request, page=0):
  201. """
  202. Use widget as view
  203. """
  204. # Get basic list attributes
  205. if request.session.get(self.get_token('filter')):
  206. self.is_filtering = True
  207. items_total = self.set_filters(self.admin.model.objects, request.session.get(self.get_token('filter'))).count()
  208. else:
  209. items_total = self.admin.model.objects.count()
  210. sorting_method = self.get_sorting(request)
  211. paginating_method = self.get_pagination(request, items_total, page)
  212. # List items
  213. items = self.admin.model.objects
  214. # Filter items?
  215. if request.session.get(self.get_token('filter')):
  216. items = self.set_filters(items, request.session.get(self.get_token('filter')))
  217. else:
  218. items = items.all()
  219. items = items.order_by(sorting_method[2])
  220. if self.pagination:
  221. items = items[paginating_method['start']:paginating_method['stop']]
  222. # Default message
  223. message = request.messages.get_message(self.admin.id)
  224. # See if we should make and handle search form
  225. search_form = None
  226. SearchForm = self.get_search_form(request)
  227. if SearchForm:
  228. if request.method == 'POST':
  229. # New search
  230. if request.POST.get('origin') == 'search':
  231. search_form = SearchForm(request.POST, request=request)
  232. if search_form.is_valid():
  233. search_criteria = {}
  234. for field, criteria in search_form.cleaned_data.items():
  235. if len(criteria) > 0:
  236. search_criteria[field] = criteria
  237. if not search_criteria:
  238. message = BasicMessage(_("No search criteria have been defined."))
  239. else:
  240. request.session[self.get_token('filter')] = search_criteria
  241. return redirect(self.get_url())
  242. else:
  243. message = BasicMessage(_("Search form contains errors."))
  244. message.type = 'error'
  245. else:
  246. search_form = SearchForm(request=request)
  247. # Kill search
  248. if request.POST.get('origin') == 'clear' and self.is_filtering and request.csrf.request_secure(request):
  249. request.session[self.get_token('filter')] = None
  250. request.messages.set_flash(BasicMessage(_("Search criteria have been cleared.")), 'info', self.admin.id)
  251. return redirect(self.get_url())
  252. else:
  253. if self.is_filtering:
  254. search_form = SearchForm(request=request, initial=request.session.get(self.get_token('filter')))
  255. else:
  256. search_form = SearchForm(request=request)
  257. # See if we sould make and handle tab form
  258. table_form = None
  259. TableForm = self.get_table_form(request, items)
  260. if TableForm:
  261. if request.method == 'POST' and request.POST.get('origin') == 'table':
  262. pass
  263. else:
  264. table_form = TableForm(request=request)
  265. # See if we should make and handle list form
  266. list_form = None
  267. ListForm = self.get_actions_form(items)
  268. if ListForm:
  269. if request.method == 'POST' and request.POST.get('origin') == 'list':
  270. list_form = ListForm(request.POST, request=request)
  271. if list_form.is_valid():
  272. try:
  273. form_action = getattr(self, 'action_' + list_form.cleaned_data['list_action'])
  274. message, redirect_url = form_action(request, items, list_form.cleaned_data['list_items'])
  275. if redirect:
  276. request.messages.set_flash(message, message.type, self.admin.id)
  277. return redirect(redirect_url)
  278. except AttributeError:
  279. message = BasicMessage(_("Action requested is incorrect."))
  280. message.type = 'error'
  281. else:
  282. if 'list_items' in list_form.errors:
  283. message = BasicMessage(self.nothing_checked_message)
  284. elif 'list_action' in list_form.errors:
  285. message = BasicMessage(_("Action requested is incorrect."))
  286. else:
  287. message = Message(request, list_form.non_field_errors()[0])
  288. message.type = 'error'
  289. else:
  290. list_form = ListForm(request=request)
  291. # Render list
  292. return request.theme.render_to_response(self.get_templates(self.template),
  293. {
  294. 'admin': self.admin,
  295. 'action': self,
  296. 'request': request,
  297. 'url': self.get_url(),
  298. 'message': message,
  299. 'sorting': self.sortables,
  300. 'sorting_method': sorting_method,
  301. 'pagination': paginating_method,
  302. 'list_form': FormLayout(list_form) if list_form else None,
  303. 'search_form': FormLayout(search_form) if search_form else None,
  304. 'table_form': FormFields(table_form) if table_form else None,
  305. 'items': items,
  306. 'items_total': items_total,
  307. },
  308. context_instance=RequestContext(request));
  309. class FormWidget(BaseWidget):
  310. """
  311. Form page widget
  312. """
  313. template = 'form'
  314. submit_button = _("Save Changes")
  315. form = None
  316. layout = None
  317. target_name = None
  318. submit_fallback = False
  319. def get_url(self, request, model):
  320. return reverse(self.admin.get_action_attr(self.id, 'route'))
  321. def get_form(self, request, model):
  322. return self.form
  323. def get_layout(self, request, form, model):
  324. if self.layout:
  325. return self.layout
  326. return form.layout
  327. def get_initial_data(self, request, model):
  328. return {}
  329. def submit_form(self, request, form, model):
  330. """
  331. Handle form submission, ALWAYS return tuple with model and message
  332. """
  333. pass
  334. def __call__(self, request, target=None, slug=None):
  335. # Default message
  336. message = request.messages.get_message(self.admin.id)
  337. # Fetch target?
  338. model = None
  339. if target:
  340. model = self.get_and_validate_target(request, target)
  341. if not model:
  342. return redirect(self.get_fallback_url(request))
  343. original_model = model
  344. # Get form type to instantiate
  345. FormType = self.get_form(request, target)
  346. #Submit form
  347. if request.method == 'POST':
  348. form = FormType(request.POST, request=request, initial=self.get_initial_data(request, model))
  349. if form.is_valid():
  350. model, message = self.submit_form(request, form, model)
  351. if message.type != 'error':
  352. request.messages.set_flash(message, message.type, self.admin.id)
  353. # Redirect back to right page
  354. try:
  355. if 'save_new' in request.POST and self.get_new_url:
  356. return redirect(self.get_new_url(request, model))
  357. except AttributeError:
  358. pass
  359. try:
  360. if 'save_edit' in request.POST and self.get_edit_url:
  361. return redirect(self.get_edit_url(request, model))
  362. except AttributeError:
  363. pass
  364. try:
  365. if self.get_submit_url:
  366. return redirect(self.get_submit_url(request, model))
  367. except AttributeError:
  368. pass
  369. return redirect(self.get_fallback_url(request))
  370. else:
  371. message = Message(request, form.non_field_errors()[0])
  372. message.type = 'error'
  373. else:
  374. form = FormType(request=request, initial=self.get_initial_data(request, model))
  375. # Render form
  376. return request.theme.render_to_response(self.get_templates(self.template),
  377. {
  378. 'admin': self.admin,
  379. 'action': self,
  380. 'request': request,
  381. 'url': self.get_url(request, model),
  382. 'fallback': self.get_fallback_url(request),
  383. 'message': message,
  384. 'target': self.get_target_name(original_model),
  385. 'target_model': original_model,
  386. 'form': FormLayout(form, self.get_layout(request, form, target)),
  387. },
  388. context_instance=RequestContext(request));
  389. class ButtonWidget(BaseWidget):
  390. """
  391. Button Action Widget
  392. This widget handles most basic and common type of admin action - button press:
  393. - User presses button on list (for example "delete this user!")
  394. - Widget checks if request is CSRF-valid and POST
  395. - Widget optionally chcecks if target has been provided and action is allowed at all
  396. - Widget does action and redirects us back to fallback url
  397. """
  398. def __call__(self, request, target=None, slug=None):
  399. # Fetch target?
  400. model = None
  401. if target:
  402. model = self.get_and_validate_target(request, target)
  403. if not model:
  404. return redirect(self.get_fallback_url(request))
  405. original_model = model
  406. # Crash if this is invalid request
  407. if not request.csrf.request_secure(request):
  408. request.messages.set_flash(BasicMessage(_("Action authorization is invalid.")), 'error', self.admin.id)
  409. return redirect(self.get_fallback_url(request))
  410. # Do something
  411. message, url = self.action(request, model)
  412. request.messages.set_flash(message, message.type, self.admin.id)
  413. if url:
  414. return redirect(url)
  415. return redirect(self.get_fallback_url(request))
  416. def action(self, request, target):
  417. """
  418. Action to be executed when button is pressed
  419. Define custom one in your Admin action.
  420. It should return response and message objects
  421. """
  422. pass