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

455 lines
18 KiB
Python
Raw Normal View History

2017-07-14 15:32:14 +02:00
import typing
from contextlib import suppress
2017-05-14 20:21:33 +02:00
from django.contrib import messages
from django.core.exceptions import FieldDoesNotExist, ObjectDoesNotExist
2017-07-14 15:32:14 +02:00
from django.db import models
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 _
2016-09-23 15:23:02 +02:00
2017-07-26 13:52:23 +02:00
from c3nav.editor.forms import (GraphEdgeSettingsForm, GraphEditorActionForm, GraphEditorSettingsForm,
GraphNodeSettingsForm)
2017-06-20 14:02:30 +02:00
from c3nav.editor.views.base import sidebar_view
2017-05-14 20:21:33 +02:00
2017-07-14 15:32:14 +02:00
def child_model(request, model: typing.Union[str, models.Model], kwargs=None, parent=None):
model = request.changeset.wrap_model(model)
related_name = model._meta.default_related_name
2017-07-14 00:08:34 +02:00
if parent is not None:
qs = getattr(parent, related_name)
if hasattr(model, 'q_for_request'):
qs = qs.filter(model.q_for_request(request))
count = qs.count()
else:
count = None
return {
'title': model._meta.verbose_name_plural,
'url': reverse('editor.'+related_name+'.list', kwargs=kwargs),
2017-07-14 00:08:34 +02:00
'count': count,
}
2017-05-14 20:21:33 +02:00
@sidebar_view
def main_index(request):
Level = request.changeset.wrap_model('Level')
2017-05-14 20:21:33 +02:00
return render(request, 'editor/index.html', {
'levels': Level.objects.filter(Level.q_for_request(request), on_top_of__isnull=True),
2017-07-08 15:47:19 +02:00
'can_edit': request.changeset.can_edit(request),
'child_models': [
child_model(request, 'LocationGroupCategory'),
child_model(request, 'LocationGroup'),
child_model(request, 'AccessRestriction'),
child_model(request, 'Source'),
],
2017-05-14 20:21:33 +02:00
})
@sidebar_view
2017-06-11 14:43:14 +02:00
def level_detail(request, pk):
Level = request.changeset.wrap_model('Level')
qs = Level.objects.filter(Level.q_for_request(request))
level = get_object_or_404(qs.select_related('on_top_of').prefetch_related('levels_on_top'), pk=pk)
2017-05-16 12:34:45 +02:00
2017-06-11 14:43:14 +02:00
return render(request, 'editor/level.html', {
'levels': Level.objects.filter(Level.q_for_request(request), on_top_of__isnull=True),
2017-06-11 14:43:14 +02:00
'level': level,
'level_url': 'editor.levels.detail',
'level_as_pk': True,
'can_edit': request.changeset.can_edit(request),
'child_models': [child_model(request, model_name, kwargs={'level': pk}, parent=level)
2017-06-08 15:19:12 +02:00
for model_name in ('Building', 'Space', 'Door')],
'levels_on_top': level.levels_on_top.filter(Level.q_for_request(request)).all(),
2017-06-11 14:43:14 +02:00
'geometry_url': '/api/editor/geometries/?level='+str(level.primary_level_pk),
})
@sidebar_view
2017-06-11 14:43:14 +02:00
def space_detail(request, level, pk):
Space = request.changeset.wrap_model('Space')
qs = Space.objects.filter(Space.q_for_request(request))
space = get_object_or_404(qs.select_related('level'), level__pk=level, pk=pk)
return render(request, 'editor/space.html', {
2017-06-11 14:43:14 +02:00
'level': space.level,
'space': space,
'can_edit': request.changeset.can_edit(request),
'child_models': [child_model(request, model_name, kwargs={'space': pk}, parent=space)
2017-07-08 16:29:12 +02:00
for model_name in ('Hole', 'Area', 'Stair', 'Obstacle', 'LineObstacle', 'Column', 'POI')],
'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
2017-06-11 14:43:14 +02:00
def edit(request, pk=None, model=None, level=None, space=None, on_top_of=None, explicit_edit=False):
model = request.changeset.wrap_model(model)
related_name = model._meta.default_related_name
2016-10-13 15:55:15 +02:00
Level = request.changeset.wrap_model('Level')
Space = request.changeset.wrap_model('Space')
can_edit = request.changeset.can_edit(request)
obj = None
if pk is not None:
2016-11-27 23:51:44 +01:00
# Edit existing map item
kwargs = {'pk': pk}
2017-06-11 15:37:25 +02:00
qs = model.objects.all()
if hasattr(model, 'q_for_request'):
qs = qs.filter(model.q_for_request(request))
2017-06-11 14:43:14 +02:00
if level is not None:
kwargs.update({'level__pk': level})
2017-06-11 15:37:25 +02:00
qs = qs.select_related('level')
elif space is not None:
kwargs.update({'space__pk': space})
2017-06-11 15:37:25 +02:00
qs = qs.select_related('space')
obj = get_object_or_404(qs, **kwargs)
2017-06-11 14:43:14 +02:00
elif level is not None:
level = get_object_or_404(Level.objects.filter(Level.q_for_request(request)), pk=level)
2017-05-26 17:52:29 +02:00
elif space is not None:
space = get_object_or_404(Space.objects.filter(Space.q_for_request(request)), pk=space)
2017-06-11 01:19:37 +02:00
elif on_top_of is not None:
on_top_of = get_object_or_404(Level.objects.filter(Level.q_for_request(request), on_top_of__isnull=True),
pk=on_top_of)
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,
'can_edit': can_edit,
'new': new,
'title': obj.title if obj else None,
}
with suppress(FieldDoesNotExist):
2017-05-26 21:37:39 +02:00
ctx.update({
'geomtype': model._meta.get_field('geometry').geomtype,
})
2017-06-11 14:43:14 +02:00
if model == Level:
ctx.update({
2017-06-11 14:43:14 +02:00
'level': obj,
'back_url': reverse('editor.index') if new else reverse('editor.levels.detail', kwargs={'pk': pk}),
2017-06-18 01:08:04 +02:00
'nozoom': True,
})
if not new:
ctx.update({
2017-06-11 14:43:14 +02:00
'geometry_url': '/api/editor/geometries/?level='+str(obj.primary_level_pk),
2017-06-11 01:19:37 +02:00
'on_top_of': obj.on_top_of,
})
elif on_top_of:
ctx.update({
2017-06-11 14:43:14 +02:00
'geometry_url': '/api/editor/geometries/?level=' + str(on_top_of.pk),
2017-06-11 01:19:37 +02:00
'on_top_of': on_top_of,
2017-06-11 14:43:14 +02:00
'back_url': reverse('editor.levels.detail', kwargs={'pk': on_top_of.pk}),
})
elif model == Space and not new:
2017-06-11 14:43:14 +02:00
level = obj.level
ctx.update({
2017-06-11 14:43:14 +02:00
'level': obj.level,
'back_url': reverse('editor.spaces.detail', kwargs={'level': obj.level.pk, 'pk': pk}),
'geometry_url': '/api/editor/geometries/?space='+pk,
2017-06-18 01:08:04 +02:00
'nozoom': True,
})
2017-05-26 19:58:04 +02:00
elif model == Space and new:
ctx.update({
2017-06-11 14:43:14 +02:00
'level': level,
'back_url': reverse('editor.spaces.list', kwargs={'level': level.pk}),
'geometry_url': '/api/editor/geometries/?level='+str(level.primary_level_pk),
2017-06-18 01:08:04 +02:00
'nozoom': True,
2017-05-26 19:58:04 +02:00
})
2017-06-11 14:43:14 +02:00
elif hasattr(model, 'level'):
2017-06-13 03:31:10 +02:00
if not new:
2017-06-11 14:43:14 +02:00
level = obj.level
ctx.update({
2017-06-11 14:43:14 +02:00
'level': level,
'back_url': reverse('editor.'+related_name+'.list', kwargs={'level': level.pk}),
'geometry_url': '/api/editor/geometries/?level='+str(level.primary_level_pk),
})
2017-05-26 17:52:29 +02:00
elif hasattr(model, 'space'):
2017-06-13 03:31:10 +02:00
if not new:
2017-05-26 17:52:29 +02:00
space = obj.space
ctx.update({
2017-06-11 14:43:14 +02:00
'level': space.level,
'back_url': reverse('editor.'+related_name+'.list', kwargs={'space': space.pk}),
2017-05-26 17:52:29 +02:00
'geometry_url': '/api/editor/geometries/?space='+str(space.pk),
})
2017-05-16 17:45:56 +02:00
else:
kwargs = {}
2017-06-11 14:43:14 +02:00
if level is not None:
kwargs.update({'level': level})
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':
2017-06-13 03:31:10 +02:00
if not new and request.POST.get('delete') == '1':
2016-11-27 23:51:44 +01:00
# Delete this mapitem!
try:
if not request.changeset.get_changed_object(obj).can_delete():
raise PermissionError
except (ObjectDoesNotExist, PermissionError):
messages.error(request, _('You can not delete this object because other objects still depend on it.'))
return redirect(request.path)
2016-09-26 13:32:05 +02:00
if request.POST.get('delete_confirm') == '1':
2017-07-07 15:32:41 +02:00
with request.changeset.lock_to_edit(request) as changeset:
if changeset.can_edit(request):
obj.delete()
2017-07-07 15:32:41 +02:00
else:
messages.error(request, _('You can not edit changes on this changeset.'))
return redirect(request.path)
messages.success(request, _('Object was successfully deleted.'))
2017-06-11 14:43:14 +02:00
if model == Level:
2017-06-11 01:19:37 +02:00
if obj.on_top_of_id is not None:
2017-06-11 14:43:14 +02:00
return redirect(reverse('editor.levels.detail', kwargs={'pk': obj.on_top_of_id}))
2017-05-26 19:58:04 +02:00
return redirect(reverse('editor.index'))
elif model == Space:
2017-06-11 14:43:14 +02:00
return redirect(reverse('editor.spaces.list', kwargs={'level': obj.level.pk}))
2017-05-26 19:58:04 +02:00
return redirect(ctx['back_url'])
ctx['obj_title'] = obj.title
return render(request, 'editor/delete.html', ctx)
2017-06-13 03:31:10 +02:00
form = model.EditorForm(instance=model() if new else 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
2017-06-11 14:43:14 +02:00
if level is not None:
obj.level = level
2017-05-26 19:58:04 +02:00
if space is not None:
obj.space = space
2017-06-11 01:19:37 +02:00
if on_top_of is not None:
obj.on_top_of = on_top_of
2017-07-07 15:32:41 +02:00
with request.changeset.lock_to_edit(request) as changeset:
if changeset.can_edit(request):
obj.save()
2017-06-18 22:46:50 +02:00
if form.redirect_slugs is not None:
for slug in form.add_redirect_slugs:
obj.redirects.create(slug=slug)
2017-06-18 22:46:50 +02:00
for slug in form.remove_redirect_slugs:
obj.redirects.filter(slug=slug).delete()
2016-09-26 13:32:05 +02:00
form.save_m2m()
2017-07-07 15:32:41 +02:00
messages.success(request, _('Object was successfully saved.'))
return redirect(ctx['back_url'])
else:
messages.error(request, _('You can not edit changes on this changeset.'))
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-06-11 14:43:14 +02:00
def list_objects(request, model=None, level=None, space=None, explicit_edit=False):
2017-05-16 17:45:56 +02:00
if not request.resolver_match.url_name.endswith('.list'):
raise ValueError('url_name does not end with .list')
model = request.changeset.wrap_model(model)
Level = request.changeset.wrap_model('Level')
Space = request.changeset.wrap_model('Space')
can_edit = request.changeset.can_edit(request)
2017-05-16 17:45:56 +02:00
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,
'can_edit': can_edit,
2017-05-16 17:45:56 +02:00
}
2017-05-19 15:23:00 +02:00
queryset = model.objects.all().order_by('id')
if hasattr(model, 'q_for_request'):
queryset = queryset.filter(model.q_for_request(request))
2017-05-19 16:34:02 +02:00
reverse_kwargs = {}
2017-05-19 15:23:00 +02:00
2017-06-11 14:43:14 +02:00
if level is not None:
reverse_kwargs['level'] = level
2017-07-15 16:31:20 +02:00
level = get_object_or_404(Level.objects.filter(Level.q_for_request(request)), pk=level)
queryset = queryset.filter(level=level).defer('geometry')
2017-05-16 17:45:56 +02:00
ctx.update({
2017-06-11 14:43:14 +02:00
'back_url': reverse('editor.levels.detail', kwargs={'pk': level.pk}),
'back_title': _('back to level'),
'levels': Level.objects.filter(Level.q_for_request(request), on_top_of__isnull=True),
2017-06-11 14:43:14 +02:00
'level': level,
'level_url': request.resolver_match.url_name,
'geometry_url': '/api/editor/geometries/?level='+str(level.primary_level_pk),
2017-05-19 16:34:02 +02:00
})
elif space is not None:
reverse_kwargs['space'] = space
sub_qs = Space.objects.filter(Space.q_for_request(request)).select_related('level').defer('geometry')
space = get_object_or_404(sub_qs, pk=space)
queryset = queryset.filter(space=space).defer('geometry')
2017-05-19 16:34:02 +02:00
ctx.update({
2017-06-11 14:43:14 +02:00
'level': space.level,
'back_url': reverse('editor.spaces.detail', kwargs={'level': space.level.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)
2017-07-15 16:31:20 +02:00
@sidebar_view
def graph_edit(request, level=None, space=None):
Level = request.changeset.wrap_model('Level')
Space = request.changeset.wrap_model('Space')
2017-07-19 18:02:40 +02:00
GraphNode = request.changeset.wrap_model('GraphNode')
GraphEdge = request.changeset.wrap_model('GraphEdge')
2017-07-15 16:31:20 +02:00
can_edit = request.changeset.can_edit(request)
ctx = {
'path': request.path,
'can_edit': can_edit,
}
graph_editing_settings = {field.name: field.initial for field in GraphEditorSettingsForm()}
graph_editing_settings.update(request.session.get('graph_editing_settings', {}))
graph_editing = 'edit-nodes'
2017-07-26 13:20:55 +02:00
allow_clicked_position = False
2017-07-15 16:31:20 +02:00
if level is not None:
level = get_object_or_404(Level.objects.filter(Level.q_for_request(request)), pk=level)
ctx.update({
'back_url': reverse('editor.levels.detail', kwargs={'pk': level.pk}),
'back_title': _('back to level'),
'levels': Level.objects.filter(Level.q_for_request(request), on_top_of__isnull=True),
'level': level,
'level_url': request.resolver_match.url_name,
'geometry_url': '/api/editor/geometries/?level='+str(level.primary_level_pk),
})
elif space is not None:
queryset = Space.objects.filter(Space.q_for_request(request)).select_related('level').defer('geometry')
space = get_object_or_404(queryset, pk=space)
ctx.update({
'space': space,
'level': space.level,
'back_url': reverse('editor.spaces.detail', kwargs={'level': space.level.pk, 'pk': space.pk}),
'back_title': _('back to space'),
'geometry_url': '/api/editor/geometries/?space='+str(space.pk),
})
2017-07-26 12:17:44 +02:00
if graph_editing_settings['click_anywhere'] != 'noop':
graph_editing = 'edit-create-nodes'
2017-07-26 13:20:55 +02:00
allow_clicked_position = True
if request.method == 'POST':
2017-07-26 13:52:23 +02:00
node_settings_form = GraphNodeSettingsForm(instance=GraphNode(), data=request.POST)
edge_settings_form = GraphEdgeSettingsForm(instance=GraphEdge(), request=request, data=request.POST)
2017-07-26 13:20:55 +02:00
graph_action_form = GraphEditorActionForm(request=request, allow_clicked_position=allow_clicked_position,
data=request.POST)
if node_settings_form.is_valid() and edge_settings_form.is_valid() and graph_action_form.is_valid():
2017-07-26 14:39:15 +02:00
set_active_node = False
2017-07-26 13:52:23 +02:00
active_node = graph_action_form.cleaned_data['active_node']
clicked_node = graph_action_form.cleaned_data['clicked_node']
clicked_position = graph_action_form.cleaned_data.get('clicked_position')
if clicked_node is not None and clicked_position is None:
2017-07-26 14:39:15 +02:00
node_click_setting = graph_editing_settings['node_click']
if node_click_setting == 'connect_or_toggle':
set_active_node = True
active_node = clicked_node
else:
raise NotImplementedError
2017-07-26 13:52:23 +02:00
elif clicked_node is None and clicked_position is not None:
click_anywhere_setting = graph_editing_settings['click_anywhere']
if click_anywhere_setting != 'create_node_if_none_active' or active_node is None:
node = node_settings_form.instance
node.space = space
node.geometry = clicked_position
if space.geometry.contains(clicked_position):
with request.changeset.lock_to_edit(request) as changeset:
if changeset.can_edit(request):
node.save()
else:
messages.error(request, _('You can not edit changes on this changeset.'))
messages.success(request, _('New graph node created!'))
2017-07-26 14:39:15 +02:00
if set_active_node:
ctx.update({
'set_active_node': set_active_node,
'active_node': active_node,
})
2017-07-26 13:52:23 +02:00
ctx.update({
'nozoom': True,
})
2017-07-26 13:20:55 +02:00
else:
node_settings_form = GraphNodeSettingsForm()
edge_settings_form = GraphEdgeSettingsForm(request=request)
2017-07-26 13:52:23 +02:00
graph_action_form = GraphEditorActionForm(request=request, allow_clicked_position=allow_clicked_position)
2017-07-15 16:31:20 +02:00
2017-07-19 18:02:40 +02:00
ctx.update({
2017-07-26 13:20:55 +02:00
'node_settings_form': node_settings_form,
'edge_settings_form': edge_settings_form,
'graph_action_form': graph_action_form,
'graph_editing': graph_editing,
2017-07-19 18:02:40 +02:00
})
2017-07-15 16:31:20 +02:00
return render(request, 'editor/graph.html', ctx)
@sidebar_view
def graph_editing_settings(request):
ctx = {
'closemodal': False,
}
if request.method == 'POST':
form = GraphEditorSettingsForm(data=request.POST)
if form.is_valid():
messages.success(request, _('Graph Editing Settings were successfully saved.'))
2017-07-25 10:35:45 +02:00
request.session['graph_editing_settings'] = form.cleaned_data
if request.POST.get('can_close_modal') == '1':
ctx['closemodal'] = True
else:
2017-07-25 10:35:45 +02:00
form = GraphEditorSettingsForm(data=request.session.get('graph_editing_settings', {}))
ctx.update({
'form': form,
})
return render(request, 'editor/graph_editing_settings.html', ctx)