team-3/src/c3nav/editor/views.py

277 lines
9.6 KiB
Python
Raw Normal View History

2017-05-14 20:21:33 +02:00
from functools import wraps
from django.apps import apps
2016-09-23 17:02:17 +02:00
from django.conf import settings
2017-05-26 21:37:53 +02:00
from django.core.exceptions import FieldDoesNotExist, PermissionDenied
from django.http import HttpResponseRedirect
from django.shortcuts import get_object_or_404, redirect, render
2017-05-14 20:21:33 +02:00
from django.urls import reverse
2017-05-16 17:45:56 +02:00
from django.utils.translation import ugettext_lazy as _
from django.views.decorators.cache import never_cache
2016-09-23 15:23:02 +02:00
from c3nav.mapdata.models import Section, Space
2017-05-09 09:36:08 +02:00
from c3nav.mapdata.models.base import EDITOR_FORM_MODELS
2016-11-27 23:51:44 +01:00
2017-05-14 20:21:33 +02:00
def sidebar_view(func):
@wraps(func)
def with_ajax_check(request, *args, **kwargs):
response = func(request, *args, **kwargs)
if request.is_ajax() or 'ajax' in request.GET:
if isinstance(response, HttpResponseRedirect):
return render(request, 'editor/redirect.html', {'target': response['location']})
return response
return render(request, 'editor/map.html', {'content': response.content})
return never_cache(with_ajax_check)
2017-05-14 20:21:33 +02:00
def child_model(model_name, kwargs=None, parent=None):
model = apps.get_model('mapdata', model_name)
related_name = model._meta.default_related_name
return {
'title': model._meta.verbose_name_plural,
'url': reverse('editor.'+related_name+'.list', kwargs=kwargs),
'count': None if parent is None else getattr(parent, related_name).count(),
}
2017-05-14 20:21:33 +02:00
@sidebar_view
def main_index(request):
2017-05-14 20:21:33 +02:00
return render(request, 'editor/index.html', {
'sections': Section.objects.all(),
'child_models': [
child_model('LocationGroup'),
2017-05-26 21:37:53 +02:00
child_model('Source'),
],
2017-05-14 20:21:33 +02:00
})
@sidebar_view
def section_detail(request, pk):
section = get_object_or_404(Section, pk=pk)
2017-05-16 12:34:45 +02:00
return render(request, 'editor/section.html', {
'sections': Section.objects.all(),
'section': section,
'section_url': 'editor.sections.detail',
'section_as_pk': True,
'child_models': [child_model(model_name, kwargs={'section': pk}, parent=section)
for model_name in ('Building', 'Space', 'Door', 'Hole')],
'geometry_url': '/api/editor/geometries/?section='+pk,
})
@sidebar_view
def space_detail(request, section, pk):
space = get_object_or_404(Space, section__id=section, pk=pk)
return render(request, 'editor/space.html', {
'section': space.section,
'space': space,
'child_models': [child_model(model_name, kwargs={'space': pk}, parent=space)
for model_name in ('Area', 'Stair', 'Obstacle', 'LineObstacle', 'Point')],
'geometry_url': '/api/editor/geometries/?space='+pk,
2017-05-16 12:34:45 +02:00
})
2017-05-14 20:21:33 +02:00
@sidebar_view
def edit(request, pk=None, model=None, section=None, space=None, explicit_edit=False):
model = EDITOR_FORM_MODELS[model]
2016-10-13 15:55:15 +02:00
obj = None
if pk is not None:
2016-11-27 23:51:44 +01:00
# Edit existing map item
kwargs = {'pk': pk}
if section is not None:
kwargs.update({'section__id': section})
elif space is not None:
kwargs.update({'space__id': space})
obj = get_object_or_404(model, **kwargs)
if False: # todo can access
raise PermissionDenied
2017-05-26 17:52:29 +02:00
elif section is not None:
section = get_object_or_404(Section, pk=section)
elif space is not None:
space = get_object_or_404(Space, pk=space)
2016-09-26 13:32:05 +02:00
new = obj is None
# noinspection PyProtectedMember
ctx = {
'path': request.path,
'pk': pk,
2017-05-26 17:06:52 +02:00
'model_name': model.__name__.lower(),
'model_title': model._meta.verbose_name,
'new': new,
'title': obj.title if obj else None,
}
2017-05-26 21:37:39 +02:00
try:
ctx.update({
'geomtype': model._meta.get_field('geometry').geomtype,
})
except FieldDoesNotExist:
pass
2017-05-16 15:51:56 +02:00
if model == Section:
ctx.update({
'section': obj,
2017-05-21 14:50:45 +02:00
'back_url': reverse('editor.index') if new else reverse('editor.sections.detail', kwargs={'pk': pk}),
'geometry_url': '/api/editor/geometries/?section='+pk,
})
elif model == Space and not new:
ctx.update({
'section': obj.section,
'back_url': reverse('editor.spaces.detail', kwargs={'section': obj.section.pk, 'pk': pk}),
'geometry_url': '/api/editor/geometries/?space='+pk,
})
2017-05-26 19:58:04 +02:00
elif model == Space and new:
ctx.update({
'section': section,
'back_url': reverse('editor.spaces.list', kwargs={'section': section.pk}),
'geometry_url': '/api/editor/geometries/?section='+str(section.pk),
})
2017-05-26 17:52:29 +02:00
elif hasattr(model, 'section'):
if obj:
section = obj.section
ctx.update({
2017-05-26 17:52:29 +02:00
'section': section,
'back_url': reverse('editor.sections.detail', kwargs={'pk': section.pk}),
'geometry_url': '/api/editor/geometries/?section='+str(section.pk),
})
2017-05-26 17:52:29 +02:00
elif hasattr(model, 'space'):
if obj:
space = obj.space
ctx.update({
2017-05-26 17:52:29 +02:00
'section': space.section,
'back_url': reverse('editor.spaces.detail', kwargs={'section': obj.space.section.pk, 'pk': space.pk}),
'geometry_url': '/api/editor/geometries/?space='+str(space.pk),
})
2017-05-16 17:45:56 +02:00
else:
kwargs = {}
if section is not None:
kwargs.update({'section': section})
elif space is not None:
kwargs.update({'space': space})
2017-05-16 17:45:56 +02:00
ctx.update({
'back_url': reverse('.'.join(request.resolver_match.url_name.split('.')[:-1]+['list']), kwargs=kwargs),
2017-05-16 17:45:56 +02:00
})
2016-09-26 13:32:05 +02:00
if request.method == 'POST':
if obj is not None and request.POST.get('delete') == '1':
2016-11-27 23:51:44 +01:00
# Delete this mapitem!
2016-09-26 13:32:05 +02:00
if request.POST.get('delete_confirm') == '1':
if not settings.DIRECT_EDITING:
# todo: suggest changes
raise NotImplementedError
2017-05-16 16:08:19 +02:00
obj.delete()
if model == Section:
2017-05-26 19:58:04 +02:00
return redirect(reverse('editor.index'))
elif model == Space:
return redirect(reverse('editor.spaces.list', kwargs={'section': obj.section.pk}))
return redirect(ctx['back_url'])
return render(request, 'editor/delete.html', ctx)
form = model.EditorForm(instance=obj, data=request.POST, request=request)
2016-09-26 13:32:05 +02:00
if form.is_valid():
# Update/create objects
obj = form.save(commit=False)
2016-10-13 15:55:15 +02:00
if form.titles is not None:
obj.titles = {}
2016-10-13 15:55:15 +02:00
for language, title in form.titles.items():
if title:
obj.titles[language] = title
if form.redirect_slugs is not None:
for slug in form.add_redirect_slugs:
obj.redirects.create(slug=slug)
for slug in form.remove_redirect_slugs:
obj.redirects.filter(slug=slug).delete()
2016-09-26 13:32:05 +02:00
if not settings.DIRECT_EDITING:
2017-04-28 11:03:08 +02:00
# todo: suggest changes
raise NotImplementedError
2016-09-26 13:32:05 +02:00
2017-05-26 19:58:04 +02:00
if section is not None:
obj.section = section
if space is not None:
obj.space = space
obj.save()
2016-12-01 12:25:02 +01:00
form.save_m2m()
2016-09-26 13:32:05 +02:00
return redirect(ctx['back_url'])
2016-09-26 13:32:05 +02:00
else:
form = model.EditorForm(instance=obj, request=request)
2016-09-26 13:32:05 +02:00
ctx.update({
2016-09-26 13:32:05 +02:00
'form': form,
})
return render(request, 'editor/edit.html', ctx)
2017-05-16 17:45:56 +02:00
@sidebar_view
2017-05-19 16:34:02 +02:00
def list_objects(request, model=None, section=None, space=None, explicit_edit=False):
2017-05-16 17:45:56 +02:00
model = EDITOR_FORM_MODELS[model]
if not request.resolver_match.url_name.endswith('.list'):
raise ValueError('url_name does not end with .list')
# noinspection PyProtectedMember
ctx = {
'path': request.path,
'model_name': model.__name__.lower(),
'model_title': model._meta.verbose_name,
'model_title_plural': model._meta.verbose_name_plural,
'explicit_edit': explicit_edit,
2017-05-16 17:45:56 +02:00
}
2017-05-19 15:23:00 +02:00
queryset = model.objects.all().order_by('id')
2017-05-19 16:34:02 +02:00
reverse_kwargs = {}
2017-05-19 15:23:00 +02:00
2017-05-19 16:34:02 +02:00
if section is not None:
reverse_kwargs['section'] = section
2017-05-19 15:23:00 +02:00
section = get_object_or_404(Section, pk=section)
queryset = queryset.filter(section=section)
2017-05-16 17:45:56 +02:00
ctx.update({
'back_url': reverse('editor.sections.detail', kwargs={'pk': section.pk}),
2017-05-19 16:34:02 +02:00
'back_title': _('back to section'),
'sections': Section.objects.all(),
'section': section,
'section_url': request.resolver_match.url_name,
'geometry_url': '/api/editor/geometries/?section='+str(section.pk),
2017-05-19 16:34:02 +02:00
})
elif space is not None:
reverse_kwargs['space'] = space
2017-05-19 16:50:09 +02:00
space = get_object_or_404(Space, pk=space)
2017-05-19 16:34:02 +02:00
queryset = queryset.filter(space=space)
ctx.update({
2017-05-19 16:50:09 +02:00
'section': space.section,
'back_url': reverse('editor.spaces.detail', kwargs={'section': space.section.pk, 'pk': space.pk}),
2017-05-16 17:45:56 +02:00
'back_title': _('back to space'),
'geometry_url': '/api/editor/geometries/?space='+str(space.pk),
2017-05-16 17:45:56 +02:00
})
else:
ctx.update({
'back_url': reverse('editor.index'),
'back_title': _('back to overview'),
})
edit_url_name = request.resolver_match.url_name[:-4]+('detail' if explicit_edit else 'edit')
2017-05-19 16:34:02 +02:00
for obj in queryset:
reverse_kwargs['pk'] = obj.pk
obj.edit_url = reverse(edit_url_name, kwargs=reverse_kwargs)
reverse_kwargs.pop('pk', None)
2017-05-19 15:23:00 +02:00
ctx.update({
2017-05-19 16:34:02 +02:00
'create_url': reverse(request.resolver_match.url_name[:-4] + 'create', kwargs=reverse_kwargs),
2017-05-19 15:23:00 +02:00
'objects': queryset,
})
2017-05-16 17:45:56 +02:00
return render(request, 'editor/list.html', ctx)