123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365 |
- from urllib import urlencode
- from django.contrib import messages
- from django.core.paginator import Paginator, EmptyPage
- from django.db import transaction
- from django.shortcuts import redirect
- from django.utils.translation import ugettext_lazy as _
- from django.views.generic import View
- from misago.core.exceptions import ExplicitFirstPage
- from misago.admin import site
- from misago.admin.views import render
- class AdminBaseMixin(object):
- """
- Admin mixin abstraciton used for configuring admin CRUD views.
- Takes following attributes:
- Model = Model instance
- root_link = name of link leading to root action (eg. list of all items
- templates_dir = directory with templates
- message_404 = string used in "requested item not found" messages
- """
- Model = None
- root_link = None
- templates_dir = None
- message_404 = None
- def get_model(self):
- """
- Basic method for retrieving Model, used in cases such as User model.
- """
- return self.Model
- class AdminView(View):
- def final_template(self):
- return '%s/%s' % (self.templates_dir, self.template)
- def current_link(self, request):
- matched_url = request.resolver_match.url_name
- return '%s:%s' % (request.resolver_match.namespace, matched_url)
- def process_context(self, request, context):
- """
- Simple hook for extending and manipulating template context.
- """
- return context
- def render(self, request, context=None):
- context = context or {}
- context['root_link'] = self.root_link
- context['current_link'] = self.current_link(request)
- context = self.process_context(request, context)
- return render(request, self.final_template(), context)
- class ListView(AdminView):
- """
- Admin items list view
- Uses following attributes:
- template = template name used to render items list
- items_per_page = number of items displayed on single page
- (enter 0 or don't define for no pagination)
- ordering = tuple of tuples defining allowed orderings
- typles should follow this format: (name, order_by)
- """
- template = 'list.html'
- items_per_page = 0
- ordering = None
- extra_actions = None
- @classmethod
- def add_item_action(cls, name, icon, link, style=None):
- if not cls.extra_actions:
- cls.extra_actions = []
- cls.extra_actions.append({
- 'name': name,
- 'icon': icon,
- 'link': link,
- 'style': style,
- })
- def get_queryset(self):
- return self.get_model().objects.all()
- def paginate_items(self, context, page):
- try:
- page = int(page)
- if page == 1:
- raise ExplicitFirstPage()
- elif page == 0:
- page = 1
- except ValueError:
- page_no = 1
- context['paginator'] = Paginator(context['items'],
- self.items_per_page,
- allow_empty_first_page=True)
- context['page'] = context['paginator'].page(page)
- context['items'] = context['page'].object_list
- @property
- def filters_token(self):
- return '%s:filters' % self.root_link
- def search_form(self, request, context):
- pass
- def filter_items(self, context):
- pass
- @property
- def ordering_session_key(self):
- return 'misago_admin_%s_order_by' % self.root_link
- def get_ordering_from_get(self, request):
- sort = request.GET.get('sort')
- if request.GET.get('direction') == 'desc':
- new_ordering = '-%s' % sort
- elif request.GET.get('direction') == 'asc':
- new_ordering = sort
- else:
- new_ordering = '?nope'
- return self.clean_ordering(new_ordering)
- def get_ordering_from_session(self, request):
- new_ordering = request.session.get(self.ordering_session_key)
- return self.clean_ordering(new_ordering)
- def get_default_ordering(self):
- pass
- def clean_ordering(self, new_ordering):
- for order_by, name in self.ordering:
- if order_by == new_ordering:
- return order_by
- else:
- return None
- def get_ordering_methods(self, request):
- methods = {
- 'GET': self.get_ordering_from_get(request),
- 'session': self.get_ordering_from_session(request),
- 'default': self.get_default_ordering(),
- }
- if methods['GET'] and methods['GET'] != methods['session']:
- request.session[self.ordering_session_key] = methods['GET']
- return methods
- def get_ordering_method(self, methods):
- for method in ('GET', 'session', 'default'):
- if methods.get(method):
- return methods.get(method)
- def order_items(self, method, context):
- for order_by, name in self.ordering:
- order_as_dict = {
- 'type': 'desc' if order_by[0] == '-' else 'asc',
- 'order_by': order_by,
- 'name': name,
- }
- if order_by == method:
- context['order'] = order_as_dict
- context['items'] = context['items'].order_by(
- order_as_dict['order_by'])
- elif order_as_dict['name']:
- if order_as_dict['type'] == 'desc':
- order_as_dict['order_by'] = order_as_dict['order_by'][1:]
- context['order_by'].append(order_as_dict)
- def make_querystrings(self, request, context):
- values = {}
- filter_values = {}
- order_values = {}
- if context['active_filters']:
- filter_values = context['active_filters']
- values.update(filter_values)
- if context['order']:
- order_values = {
- 'sort': context['order']['order_by'],
- 'direction': context['order']['type'],
- }
- values.update(order_values)
- if values:
- context['querystring'] = '?%s' % urlencode(values)
- if order_values:
- context['querystring_order'] = '?%s' % urlencode(order_values)
- if filter_values:
- context['querystring_filter'] = '?%s' % urlencode(filter_values)
- def dispatch(self, request, *args, **kwargs):
- active_filters = request.session.get(self.filters_token, None)
- extra_actions_list = self.extra_actions or []
- set_querystring = False
- context = {
- 'items': self.get_queryset(),
- 'paginator': None,
- 'page': None,
- 'order_by': [],
- 'order': None,
- 'search_form': None,
- 'active_filters': active_filters,
- 'querystring': '',
- 'querystring_order': '',
- 'querystring_filter': '',
- 'extra_actions': extra_actions_list,
- 'extra_actions_len': len(extra_actions_list),
- }
- if self.ordering:
- ordering_methods = self.get_ordering_methods(request)
- current_method = self.get_ordering_method(ordering_methods)
- self.order_items(current_method, context)
- if len(self.ordering) > 1 and not ordering_methods.get('GET'):
- set_querystring = True
- self.search_form(request, context)
- if active_filters:
- self.filter_items(context)
- if self.items_per_page:
- self.paginate_items(context, kwargs.get('page', 0))
- self.make_querystrings(request, context)
- if set_querystring:
- return redirect('%s%s' % (request.path, context['querystring']))
- return self.render(request, context)
- class TargetedView(AdminView):
- def check_permissions(self, request, target):
- pass
- def get_target(self, kwargs):
- if len(kwargs) == 1:
- select_for_update = self.get_model().objects.select_for_update()
- return select_for_update.get(pk=kwargs[kwargs.keys()[0]])
- else:
- return self.get_model()()
- def get_target_or_none(self, request, kwargs):
- try:
- return self.get_target(kwargs)
- except self.get_model().DoesNotExist:
- return None
- def dispatch(self, request, *args, **kwargs):
- with transaction.atomic():
- target = self.get_target_or_none(request, kwargs)
- if not target:
- messages.error(request, self.message_404)
- return redirect(self.root_link)
- error = self.check_permissions(request, target)
- if error:
- messages.error(request, error)
- return redirect(self.root_link)
- return self.real_dispatch(request, target)
- def real_dispatch(self, request, target):
- pass
- class FormView(TargetedView):
- Form = None
- template = 'form.html'
- def create_form_type(self, request):
- return self.Form
- def initialize_form(self, FormType, request):
- if request.method == 'POST':
- return FormType(request.POST, request.FILES)
- else:
- return FormType()
- def handle_form(self, form, request):
- raise NotImplementedError(
- "You have to define your own handle_form method to handle "
- "form submissions.")
- def real_dispatch(self, request, target):
- FormType = self.create_form_type(request)
- form = self.initialize_form(FormType, request)
- if request.method == 'POST' and form.is_valid():
- response = self.handle_form(form, request)
- if response:
- return response
- elif 'stay' in request.POST:
- return redirect(request.path)
- else:
- return redirect(self.root_link)
- return self.render(request, {'form': form})
- class ModelFormView(FormView):
- message_submit = None
- def create_form_type(self, request, target):
- return self.Form
- def initialize_form(self, FormType, request, target):
- if request.method == 'POST':
- return FormType(request.POST, request.FILES, instance=target)
- else:
- return FormType(instance=target)
- def handle_form(self, form, request, target):
- form.instance.save()
- if self.message_submit:
- messages.success(request, self.message_submit % target.name)
- def real_dispatch(self, request, target):
- FormType = self.create_form_type(request, target)
- form = self.initialize_form(FormType, request, target)
- if request.method == 'POST' and form.is_valid():
- response = self.handle_form(form, request, target)
- if response:
- return response
- elif 'stay' in request.POST:
- return redirect(request.path)
- else:
- return redirect(self.root_link)
- return self.render(request, {'form': form, 'target': target})
- class ButtonView(TargetedView):
- def real_dispatch(self, request, target):
- if request.method == 'POST':
- new_response = self.button_action(request, target)
- if new_response:
- return new_response
- return redirect(self.root_link)
- def button_action(self, request, target):
- raise NotImplementedError("You have to define custom button_action.")
|