2017-06-18 16:52:50 +02:00
|
|
|
import json
|
2017-05-31 02:38:59 +02:00
|
|
|
from contextlib import suppress
|
2017-05-14 20:21:33 +02:00
|
|
|
from functools import wraps
|
|
|
|
|
2017-06-18 16:52:50 +02:00
|
|
|
from django.conf import settings
|
2017-06-18 20:43:20 +02:00
|
|
|
from django.contrib.auth import login, logout
|
|
|
|
from django.contrib.auth.forms import AuthenticationForm
|
2017-05-26 21:37:53 +02:00
|
|
|
from django.core.exceptions import FieldDoesNotExist, PermissionDenied
|
2017-05-16 16:56:50 +02:00
|
|
|
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-06-18 17:36:08 +02:00
|
|
|
from django.utils.formats import date_format
|
2017-05-16 17:45:56 +02:00
|
|
|
from django.utils.translation import ugettext_lazy as _
|
2017-05-15 00:11:09 +02:00
|
|
|
from django.views.decorators.cache import never_cache
|
2016-09-23 15:23:02 +02:00
|
|
|
|
2017-06-12 17:22:38 +02:00
|
|
|
from c3nav.editor.models import ChangeSet
|
2017-06-18 16:52:50 +02:00
|
|
|
from c3nav.editor.wrappers import is_created_pk
|
2017-05-09 09:36:08 +02:00
|
|
|
from c3nav.mapdata.models.base import EDITOR_FORM_MODELS
|
2017-06-18 16:52:50 +02:00
|
|
|
from c3nav.mapdata.models.locations import LocationRedirect, LocationSlug
|
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):
|
2017-06-12 17:22:38 +02:00
|
|
|
request.changeset = ChangeSet.get_for_request(request)
|
|
|
|
|
2017-05-16 16:22:33 +02:00
|
|
|
response = func(request, *args, **kwargs)
|
2017-05-19 15:22:13 +02:00
|
|
|
if request.is_ajax() or 'ajax' in request.GET:
|
2017-05-16 16:56:50 +02:00
|
|
|
if isinstance(response, HttpResponseRedirect):
|
|
|
|
return render(request, 'editor/redirect.html', {'target': response['location']})
|
2017-06-18 20:43:20 +02:00
|
|
|
response.write(render(request, 'editor/fragment_nav.html', {}).content)
|
2017-05-16 16:22:33 +02:00
|
|
|
return response
|
2017-06-18 05:17:41 +02:00
|
|
|
if isinstance(response, HttpResponseRedirect):
|
|
|
|
return response
|
2017-05-16 16:22:33 +02:00
|
|
|
return render(request, 'editor/map.html', {'content': response.content})
|
2017-05-15 00:11:09 +02:00
|
|
|
return never_cache(with_ajax_check)
|
2017-05-14 20:21:33 +02:00
|
|
|
|
|
|
|
|
2017-06-12 22:56:39 +02:00
|
|
|
def child_model(model, kwargs=None, parent=None):
|
2017-05-21 14:43:16 +02:00
|
|
|
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
|
2017-05-16 14:10:50 +02:00
|
|
|
def main_index(request):
|
2017-06-12 22:56:39 +02:00
|
|
|
Level = request.changeset.wrap('Level')
|
2017-05-14 20:21:33 +02:00
|
|
|
return render(request, 'editor/index.html', {
|
2017-06-11 14:43:14 +02:00
|
|
|
'levels': Level.objects.filter(on_top_of__isnull=True),
|
2017-05-21 14:43:16 +02:00
|
|
|
'child_models': [
|
2017-06-12 22:56:39 +02:00
|
|
|
child_model(request.changeset.wrap('LocationGroup')),
|
|
|
|
child_model(request.changeset.wrap('Source')),
|
2017-05-21 14:43:16 +02:00
|
|
|
],
|
2017-05-14 20:21:33 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
@sidebar_view
|
2017-06-11 14:43:14 +02:00
|
|
|
def level_detail(request, pk):
|
2017-06-12 22:56:39 +02:00
|
|
|
Level = request.changeset.wrap('Level')
|
2017-06-11 15:37:25 +02:00
|
|
|
level = get_object_or_404(Level.objects.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(on_top_of__isnull=True),
|
|
|
|
'level': level,
|
|
|
|
'level_url': 'editor.levels.detail',
|
|
|
|
'level_as_pk': True,
|
2017-05-19 16:08:46 +02:00
|
|
|
|
2017-06-12 22:56:39 +02:00
|
|
|
'child_models': [child_model(request.changeset.wrap(model_name), kwargs={'level': pk}, parent=level)
|
2017-06-08 15:19:12 +02:00
|
|
|
for model_name in ('Building', 'Space', 'Door')],
|
2017-06-11 14:43:14 +02:00
|
|
|
'levels_on_top': level.levels_on_top.all(),
|
|
|
|
'geometry_url': '/api/editor/geometries/?level='+str(level.primary_level_pk),
|
2017-05-19 16:08:46 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
@sidebar_view
|
2017-06-11 14:43:14 +02:00
|
|
|
def space_detail(request, level, pk):
|
2017-06-12 22:56:39 +02:00
|
|
|
Space = request.changeset.wrap('Space')
|
2017-06-15 01:19:25 +02:00
|
|
|
space = get_object_or_404(Space.objects.select_related('level'), level__pk=level, pk=pk)
|
2017-05-19 16:08:46 +02:00
|
|
|
|
2017-05-19 16:34:27 +02:00
|
|
|
return render(request, 'editor/space.html', {
|
2017-06-11 14:43:14 +02:00
|
|
|
'level': space.level,
|
2017-05-19 16:34:27 +02:00
|
|
|
'space': space,
|
2017-05-19 16:08:46 +02:00
|
|
|
|
2017-06-12 22:56:39 +02:00
|
|
|
'child_models': [child_model(request.changeset.wrap(model_name), kwargs={'space': pk}, parent=space)
|
2017-06-09 15:22:30 +02:00
|
|
|
for model_name in ('Hole', 'Area', 'Stair', 'Obstacle', 'LineObstacle', 'Column', 'Point')],
|
2017-05-21 23:39:26 +02:00
|
|
|
'geometry_url': '/api/editor/geometries/?space='+pk,
|
2017-05-16 12:34:45 +02:00
|
|
|
})
|
2017-05-14 20:21:33 +02:00
|
|
|
|
|
|
|
|
2017-05-16 14:10:50 +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):
|
2017-06-12 22:56:39 +02:00
|
|
|
model = request.changeset.wrap(EDITOR_FORM_MODELS[model])
|
2017-06-09 14:45:43 +02:00
|
|
|
related_name = model._meta.default_related_name
|
2016-10-13 15:55:15 +02:00
|
|
|
|
2017-06-12 22:56:39 +02:00
|
|
|
Level = request.changeset.wrap('Level')
|
|
|
|
Space = request.changeset.wrap('Space')
|
|
|
|
|
2017-05-16 14:10:50 +02:00
|
|
|
obj = None
|
|
|
|
if pk is not None:
|
2016-11-27 23:51:44 +01:00
|
|
|
# Edit existing map item
|
2017-05-19 16:08:46 +02:00
|
|
|
kwargs = {'pk': pk}
|
2017-06-11 15:37:25 +02:00
|
|
|
qs = model.objects.all()
|
2017-06-11 14:43:14 +02:00
|
|
|
if level is not None:
|
2017-06-15 01:39:33 +02:00
|
|
|
kwargs.update({'level__pk': level})
|
2017-06-11 15:37:25 +02:00
|
|
|
qs = qs.select_related('level')
|
2017-05-19 16:08:46 +02:00
|
|
|
elif space is not None:
|
2017-06-15 01:39:33 +02:00
|
|
|
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-05-16 14:10:50 +02:00
|
|
|
if False: # todo can access
|
2016-10-04 13:28:09 +02:00
|
|
|
raise PermissionDenied
|
2017-06-11 14:43:14 +02:00
|
|
|
elif level is not None:
|
|
|
|
level = get_object_or_404(Level, pk=level)
|
2017-05-26 17:52:29 +02:00
|
|
|
elif space is not None:
|
|
|
|
space = get_object_or_404(Space, pk=space)
|
2017-06-11 01:19:37 +02:00
|
|
|
elif on_top_of is not None:
|
2017-06-11 14:43:14 +02:00
|
|
|
on_top_of = get_object_or_404(Level.objects.filter(on_top_of__isnull=True), pk=on_top_of)
|
2016-09-26 13:32:05 +02:00
|
|
|
|
2017-05-16 14:10:50 +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(),
|
2017-05-16 14:10:50 +02:00
|
|
|
'model_title': model._meta.verbose_name,
|
|
|
|
'new': new,
|
|
|
|
'title': obj.title if obj else None,
|
|
|
|
}
|
|
|
|
|
2017-05-31 02:38:59 +02:00
|
|
|
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:
|
2017-05-16 14:10:50 +02:00
|
|
|
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,
|
2017-05-16 14:10:50 +02:00
|
|
|
})
|
2017-05-29 16:54:14 +02:00
|
|
|
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}),
|
2017-05-29 16:54:14 +02:00
|
|
|
})
|
2017-05-19 16:34:27 +02:00
|
|
|
elif model == Space and not new:
|
2017-06-11 14:43:14 +02:00
|
|
|
level = obj.level
|
2017-05-19 16:34:27 +02:00
|
|
|
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}),
|
2017-05-21 23:39:26 +02:00
|
|
|
'geometry_url': '/api/editor/geometries/?space='+pk,
|
2017-06-18 01:08:04 +02:00
|
|
|
'nozoom': True,
|
2017-05-19 16:34:27 +02:00
|
|
|
})
|
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
|
2017-05-16 14:10:50 +02:00
|
|
|
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-16 14:10:50 +02:00
|
|
|
})
|
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
|
2017-05-16 14:10:50 +02:00
|
|
|
ctx.update({
|
2017-06-11 14:43:14 +02:00
|
|
|
'level': space.level,
|
2017-06-09 14:45:43 +02:00
|
|
|
'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 14:10:50 +02:00
|
|
|
})
|
2017-05-16 17:45:56 +02:00
|
|
|
else:
|
2017-05-19 16:34:27 +02:00
|
|
|
kwargs = {}
|
2017-06-11 14:43:14 +02:00
|
|
|
if level is not None:
|
|
|
|
kwargs.update({'level': level})
|
2017-05-19 16:34:27 +02:00
|
|
|
elif space is not None:
|
|
|
|
kwargs.update({'space': space})
|
|
|
|
|
2017-05-16 17:45:56 +02:00
|
|
|
ctx.update({
|
2017-05-19 16:34:27 +02:00
|
|
|
'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!
|
2016-09-26 13:32:05 +02:00
|
|
|
if request.POST.get('delete_confirm') == '1':
|
2017-05-16 16:08:19 +02:00
|
|
|
obj.delete()
|
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'])
|
2017-06-11 13:34:53 +02:00
|
|
|
ctx['obj_title'] = obj.title
|
2017-05-16 14:10:50 +02:00
|
|
|
return render(request, 'editor/delete.html', ctx)
|
2016-09-26 13:36:09 +02:00
|
|
|
|
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():
|
2017-05-16 14:10:50 +02:00
|
|
|
# Update/create objects
|
|
|
|
obj = form.save(commit=False)
|
2016-10-13 15:55:15 +02:00
|
|
|
|
|
|
|
if form.titles is not None:
|
2017-05-16 14:10:50 +02:00
|
|
|
obj.titles = {}
|
2016-10-13 15:55:15 +02:00
|
|
|
for language, title in form.titles.items():
|
|
|
|
if title:
|
2017-05-16 14:10:50 +02:00
|
|
|
obj.titles[language] = title
|
2016-09-26 16:44:17 +02:00
|
|
|
|
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-05-16 14:10:50 +02:00
|
|
|
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)
|
|
|
|
|
|
|
|
for slug in form.remove_redirect_slugs:
|
|
|
|
obj.redirects.filter(slug=slug).delete()
|
|
|
|
|
2017-06-13 14:21:01 +02:00
|
|
|
form.save_m2m()
|
2017-06-13 03:31:10 +02:00
|
|
|
# request.changeset.changes.all().delete()
|
2016-09-26 13:32:05 +02:00
|
|
|
|
2017-05-16 16:56:50 +02:00
|
|
|
return redirect(ctx['back_url'])
|
2016-09-26 13:32:05 +02:00
|
|
|
else:
|
2017-05-16 14:10:50 +02:00
|
|
|
form = model.EditorForm(instance=obj, request=request)
|
2016-09-26 13:32:05 +02:00
|
|
|
|
2017-05-16 14:10:50 +02:00
|
|
|
ctx.update({
|
2016-09-26 13:32:05 +02:00
|
|
|
'form': form,
|
|
|
|
})
|
2017-05-16 14:10:50 +02:00
|
|
|
|
|
|
|
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')
|
|
|
|
|
2017-06-12 22:56:39 +02:00
|
|
|
model = request.changeset.wrap(EDITOR_FORM_MODELS[model])
|
|
|
|
|
|
|
|
Level = request.changeset.wrap('Level')
|
|
|
|
Space = request.changeset.wrap('Space')
|
|
|
|
|
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,
|
2017-05-19 16:34:27 +02:00
|
|
|
'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-06-11 14:43:14 +02:00
|
|
|
if level is not None:
|
|
|
|
reverse_kwargs['level'] = level
|
|
|
|
level = get_object_or_404(Level, pk=level)
|
|
|
|
queryset = queryset.filter(level=level)
|
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(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),
|
2017-05-19 16:34:02 +02:00
|
|
|
})
|
|
|
|
elif space is not None:
|
|
|
|
reverse_kwargs['space'] = space
|
2017-06-11 15:37:25 +02:00
|
|
|
space = get_object_or_404(Space.objects.select_related('level'), pk=space)
|
2017-05-19 16:34:02 +02:00
|
|
|
queryset = queryset.filter(space=space)
|
|
|
|
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'),
|
2017-05-21 23:39:26 +02:00
|
|
|
'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'),
|
|
|
|
})
|
|
|
|
|
2017-05-19 16:34:27 +02:00
|
|
|
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-06-13 15:31:54 +02:00
|
|
|
|
|
|
|
|
|
|
|
@sidebar_view
|
|
|
|
def changeset_detail(request, pk):
|
2017-06-18 18:36:09 +02:00
|
|
|
if str(pk) != str(request.changeset.pk):
|
|
|
|
changeset = get_object_or_404(ChangeSet.qs_for_request(request), pk=pk)
|
|
|
|
else:
|
|
|
|
changeset = request.changeset
|
2017-06-13 15:31:54 +02:00
|
|
|
|
2017-06-18 16:52:50 +02:00
|
|
|
# collect pks of relevant objects
|
|
|
|
object_pks = {}
|
|
|
|
for change in changeset.changes.all():
|
|
|
|
object_pks.setdefault(change.model_class, set()).add(change.obj_pk)
|
2017-06-18 17:20:57 +02:00
|
|
|
model = None
|
|
|
|
if change.action == 'update':
|
|
|
|
if change.model_class == LocationRedirect:
|
|
|
|
if change.field_name == 'target':
|
|
|
|
object_pks.setdefault(LocationSlug, set()).add(json.loads(change.field_value))
|
|
|
|
continue
|
|
|
|
elif not change.field_name.startswith('title_'):
|
|
|
|
field = change.model_class._meta.get_field(change.field_name)
|
|
|
|
model = getattr(field, 'related_model', None)
|
|
|
|
if change.action in ('m2m_add', 'm2m_remove'):
|
|
|
|
model = change.model_class._meta.get_field(change.field_name).related_model
|
|
|
|
if model is not None:
|
|
|
|
object_pks.setdefault(model, set()).add(json.loads(change.field_value))
|
2017-06-18 16:52:50 +02:00
|
|
|
|
|
|
|
# retrieve relevant objects
|
|
|
|
objects = {}
|
|
|
|
for model, pks in object_pks.items():
|
|
|
|
created_pks = set(pk for pk in pks if is_created_pk(pk))
|
|
|
|
existing_pks = pks-created_pks
|
|
|
|
model_objects = {}
|
|
|
|
if existing_pks:
|
|
|
|
for obj in model.objects.filter(pk__in=existing_pks):
|
|
|
|
if model == LocationSlug:
|
|
|
|
obj = obj.get_child()
|
|
|
|
model_objects[obj.pk] = obj
|
|
|
|
if created_pks:
|
|
|
|
for pk in created_pks:
|
2017-06-18 21:10:52 +02:00
|
|
|
model_objects[pk] = changeset.get_created_object(model, pk, allow_deleted=True)._obj
|
2017-06-18 17:20:57 +02:00
|
|
|
model_objects[pk].titles = {}
|
2017-06-18 16:52:50 +02:00
|
|
|
objects[model] = model_objects
|
|
|
|
|
|
|
|
grouped_changes = []
|
|
|
|
changes = []
|
|
|
|
last_obj = None
|
|
|
|
for change in changeset.changes.all():
|
|
|
|
pk = change.obj_pk
|
|
|
|
obj = objects[change.model_class][pk]
|
|
|
|
if change.model_class == LocationRedirect:
|
|
|
|
if change.action not in ('create', 'delete'):
|
|
|
|
continue
|
|
|
|
change.action = 'm2m_add' if change.action == 'create' else 'm2m_remove'
|
|
|
|
change.field_name = 'redirects'
|
|
|
|
change.field_value = obj.slug
|
|
|
|
pk = obj.target_id
|
|
|
|
obj = objects[LocationSlug][pk]
|
|
|
|
|
|
|
|
if obj != last_obj:
|
|
|
|
changes = []
|
2017-06-18 22:55:50 +02:00
|
|
|
obj_desc = _('%(model)s #%(id)s') % {'model': obj.__class__._meta.verbose_name, 'id': pk}
|
|
|
|
if is_created_pk(pk):
|
|
|
|
obj_desc = _('%s (created)') % obj_desc
|
2017-06-18 23:11:48 +02:00
|
|
|
obj_still_exists = int(pk[1:]) in changeset.created_objects[obj.__class__]
|
|
|
|
else:
|
|
|
|
obj_still_exists = pk not in changeset.deleted_existing[obj.__class__]
|
|
|
|
|
|
|
|
edit_url = None
|
|
|
|
if obj_still_exists and changeset == request.changeset:
|
|
|
|
reverse_kwargs = {'pk': obj.pk}
|
|
|
|
if hasattr(obj, 'level'):
|
|
|
|
reverse_kwargs['level'] = obj.level_id
|
|
|
|
elif hasattr(obj, 'space'):
|
|
|
|
reverse_kwargs['space'] = obj.space_id
|
|
|
|
edit_url = reverse('editor.'+obj.__class__._meta.default_related_name+'.edit', kwargs=reverse_kwargs)
|
|
|
|
|
2017-06-18 16:52:50 +02:00
|
|
|
grouped_changes.append({
|
|
|
|
'model': obj.__class__,
|
2017-06-18 23:11:48 +02:00
|
|
|
'model_title': obj.__class__._meta.verbose_name,
|
2017-06-18 22:55:50 +02:00
|
|
|
'obj': obj_desc,
|
2017-06-18 17:20:57 +02:00
|
|
|
'obj_title': obj.title if obj.titles else None,
|
2017-06-18 16:52:50 +02:00
|
|
|
'changes': changes,
|
2017-06-18 23:11:48 +02:00
|
|
|
'edit_url': edit_url,
|
2017-06-18 16:52:50 +02:00
|
|
|
})
|
|
|
|
last_obj = obj
|
|
|
|
|
|
|
|
change_data = {
|
2017-06-18 17:36:08 +02:00
|
|
|
'pk': change.pk,
|
|
|
|
'author': change.author,
|
|
|
|
'created': _('created at %(datetime)s') % {'datetime': date_format(change.created, 'DATETIME_FORMAT')},
|
2017-06-18 16:52:50 +02:00
|
|
|
}
|
|
|
|
changes.append(change_data)
|
|
|
|
if change.action == 'create':
|
|
|
|
change_data.update({
|
|
|
|
'icon': 'plus',
|
|
|
|
'class': 'success',
|
|
|
|
'title': _('created'),
|
|
|
|
})
|
|
|
|
elif change.action == 'delete':
|
|
|
|
change_data.update({
|
|
|
|
'icon': 'minus',
|
|
|
|
'class': 'danger',
|
|
|
|
'title': _('deleted')
|
|
|
|
})
|
|
|
|
elif change.action == 'update':
|
|
|
|
change_data.update({
|
|
|
|
'icon': 'option-vertical',
|
|
|
|
'class': 'muted',
|
|
|
|
})
|
|
|
|
if change.field_name == 'geometry':
|
|
|
|
change_data.update({
|
|
|
|
'icon': 'map-marker',
|
|
|
|
'class': 'info',
|
|
|
|
'title': _('edited geometry'),
|
|
|
|
})
|
|
|
|
else:
|
|
|
|
if change.field_name.startswith('title_'):
|
|
|
|
lang = change.field_name[6:]
|
|
|
|
field_title = _('Title (%(lang)s)') % {'lang': dict(settings.LANGUAGES).get(lang, lang)}
|
|
|
|
field_value = str(json.loads(change.field_value))
|
2017-06-18 17:20:57 +02:00
|
|
|
if field_value:
|
|
|
|
obj.titles[lang] = field_value
|
|
|
|
else:
|
|
|
|
obj.titles.pop(lang, None)
|
2017-06-18 16:52:50 +02:00
|
|
|
else:
|
|
|
|
field = obj.__class__._meta.get_field(change.field_name)
|
|
|
|
field_title = field.verbose_name
|
|
|
|
field_value = field.to_python(json.loads(change.field_value))
|
2017-06-18 17:20:57 +02:00
|
|
|
model = getattr(field, 'related_model', None)
|
|
|
|
if model is not None:
|
|
|
|
field_value = objects[model][field_value].title
|
2017-06-18 16:52:50 +02:00
|
|
|
if not field_value:
|
|
|
|
change_data.update({
|
|
|
|
'title': _('unset %(field_title)s') % {'field_title': field_title},
|
|
|
|
})
|
|
|
|
else:
|
|
|
|
change_data.update({
|
|
|
|
'title': field_title,
|
|
|
|
'value': field_value,
|
|
|
|
})
|
|
|
|
elif change.action in ('m2m_add', 'm2m_remove'):
|
|
|
|
change_data.update({
|
|
|
|
'icon': 'chevron-right' if change.action == 'm2m_add' else 'chevron-left',
|
|
|
|
'class': 'info',
|
|
|
|
})
|
|
|
|
if change.field_name == 'redirects':
|
|
|
|
change_data.update({
|
|
|
|
'title': _('Redirecting slugs'),
|
|
|
|
'value': change.field_value,
|
|
|
|
})
|
|
|
|
else:
|
|
|
|
field = obj.__class__._meta.get_field(change.field_name)
|
|
|
|
change_data.update({
|
2017-06-18 17:20:57 +02:00
|
|
|
'title': field.verbose_name,
|
2017-06-18 17:52:26 +02:00
|
|
|
'value': objects[field.related_model][json.loads(change.field_value)].title,
|
2017-06-18 16:52:50 +02:00
|
|
|
})
|
|
|
|
else:
|
|
|
|
change_data.update({
|
|
|
|
'title': '???',
|
|
|
|
})
|
|
|
|
|
2017-06-18 18:32:16 +02:00
|
|
|
if changeset.author:
|
2017-06-18 20:45:31 +02:00
|
|
|
desc = _('created at %(datetime)s by') % {'datetime': date_format(changeset.created, 'DATETIME_FORMAT')}
|
2017-06-18 18:32:16 +02:00
|
|
|
else:
|
2017-06-18 20:45:31 +02:00
|
|
|
desc = _('created at %(datetime)s') % {'datetime': date_format(changeset.created, 'DATETIME_FORMAT')}
|
2017-06-18 18:32:16 +02:00
|
|
|
|
2017-06-13 15:31:54 +02:00
|
|
|
ctx = {
|
|
|
|
'pk': pk,
|
|
|
|
'changeset': changeset,
|
2017-06-18 18:32:16 +02:00
|
|
|
'desc': desc,
|
2017-06-18 16:52:50 +02:00
|
|
|
'grouped_changes': grouped_changes,
|
2017-06-13 15:31:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if request.method == 'POST':
|
|
|
|
if request.POST.get('delete') == '1':
|
|
|
|
if request.POST.get('delete_confirm') == '1':
|
|
|
|
changeset.delete()
|
|
|
|
return redirect(reverse('editor.index'))
|
|
|
|
|
|
|
|
ctx.update({
|
|
|
|
'model_title': ChangeSet._meta.verbose_name,
|
|
|
|
'obj_title': changeset.title,
|
|
|
|
})
|
|
|
|
return render(request, 'editor/delete.html', ctx)
|
|
|
|
|
|
|
|
return render(request, 'editor/changeset.html', ctx)
|
2017-06-18 20:43:20 +02:00
|
|
|
|
|
|
|
|
|
|
|
@sidebar_view
|
|
|
|
def login_view(request):
|
2017-06-18 20:53:05 +02:00
|
|
|
redirect_path = request.GET['r'] if request.GET.get('r', '').startswith('/editor/') else reverse('editor.index')
|
|
|
|
if request.user.is_authenticated:
|
|
|
|
return redirect(redirect_path)
|
|
|
|
|
2017-06-18 20:43:20 +02:00
|
|
|
if request.method == 'POST':
|
|
|
|
form = AuthenticationForm(request, data=request.POST)
|
|
|
|
if form.is_valid():
|
|
|
|
login(request, form.user_cache)
|
2017-06-18 20:53:05 +02:00
|
|
|
|
2017-06-18 22:14:03 +02:00
|
|
|
if request.changeset.pk is not None:
|
|
|
|
if request.session.session_key is None:
|
|
|
|
request.session.save()
|
|
|
|
request.changeset.author = form.user_cache
|
|
|
|
request.changeset.session_key = request.session.session_key
|
|
|
|
request.changeset.save()
|
2017-06-18 20:53:05 +02:00
|
|
|
return redirect(redirect_path)
|
2017-06-18 20:43:20 +02:00
|
|
|
else:
|
|
|
|
form = AuthenticationForm(request)
|
|
|
|
|
|
|
|
return render(request, 'editor/login.html', {'form': form})
|
|
|
|
|
|
|
|
|
|
|
|
@sidebar_view
|
|
|
|
def logout_view(request):
|
2017-06-18 20:53:05 +02:00
|
|
|
redirect_path = request.GET['r'] if request.GET.get('r', '').startswith('/editor/') else reverse('editor.login')
|
2017-06-18 20:43:20 +02:00
|
|
|
logout(request)
|
2017-06-18 20:53:05 +02:00
|
|
|
return redirect(redirect_path)
|