Source code for shuup.simple_cms.admin_module.views

# This file is part of Shuup.
#
# Copyright (c) 2012-2021, Shuup Commerce Inc. All rights reserved.
#
# This source code is licensed under the OSL-3.0 license found in the
# LICENSE file in the root directory of this source tree.
from __future__ import unicode_literals

from django.conf import settings
from django.contrib import messages
from django.core.exceptions import ValidationError
from django.forms import ChoiceField, DateTimeField
from django.http import HttpResponseRedirect
from django.utils.translation import ugettext_lazy as _
from django_jinja.views.generic import DetailView

from shuup.admin.form_part import FormPart, FormPartsViewMixin, SaveFormPartsMixin, TemplatedFormDef
from shuup.admin.forms.widgets import TextEditorWidget
from shuup.admin.shop_provider import get_shop
from shuup.admin.supplier_provider import get_supplier
from shuup.admin.toolbar import get_default_edit_toolbar
from shuup.admin.utils.picotable import Column, TextFilter
from shuup.admin.utils.views import CreateOrUpdateView, PicotableListView
from shuup.apps.provides import get_provide_objects
from shuup.simple_cms.models import Page
from shuup.utils.django_compat import reverse, reverse_lazy
from shuup.utils.i18n import get_language_name
from shuup.utils.multilanguage_model_form import MultiLanguageModelForm


[docs]class PageForm(MultiLanguageModelForm): available_from = DateTimeField(label=_("Available since"), required=False, localize=True) available_to = DateTimeField(label=_("Available until"), required=False, localize=True)
[docs] class Meta: model = Page fields = [ "title", "url", "content", "available_from", "available_to", "identifier", "visible_in_menu", "parent", "template_name", "list_children_on_page", "show_child_timestamps", "render_title", "available_permission_groups", ] widgets = {"content": TextEditorWidget(attrs={"data-height": 500, "data-noresize": "true"})}
def __init__(self, **kwargs): self.request = kwargs.pop("request") kwargs.setdefault("required_languages", ()) # No required languages here super(PageForm, self).__init__(**kwargs) self.fields["parent"].queryset = Page.objects.filter(shop=get_shop(self.request)) self.fields["template_name"] = ChoiceField( label=_("Template"), required=False, choices=[ (simple_cms_template.template_path, simple_cms_template.name) for simple_cms_template in get_provide_objects("simple_cms_template") ], )
[docs] def clean(self): """ If title or content has been given on any language we must enforce that the other fields are also required in that language. This is done the way it is because url is not required by default in model level. """ data = super(PageForm, self).clean() something_filled = False urls = [] for language in self.languages: field_names = self.trans_name_map[language] if not any(data.get(field_name) for field_name in field_names.values()): # Let's not complain about this language continue something_filled = True for field_name in field_names.values(): value = data.get(field_name) if value: # No need to bother complaining about this field if field_name.startswith("url__"): # url needs a second look though if not self.is_url_valid(language, field_name, value): self.add_error(field_name, ValidationError(_("URL already exists."), code="invalid_url")) if value in urls: self.add_error( field_name, ValidationError(_("URL must be unique."), code="invalid_unique_url") ) urls.append(value) continue self.add_error( field_name, _("%(label)s is required when any %(language)s field is filled.") % {"label": self.fields[field_name].label, "language": get_language_name(language)}, ) if not something_filled: title_field = "title__%s" % self.default_language self.add_error(title_field, _("Please fill at least one language fully.")) return data
[docs] def clean_parent(self): parent = self.cleaned_data["parent"] if self.instance and parent and self.instance.id == parent.id: self.add_error("parent", _("A page may not be made a child of itself.")) else: return parent
[docs] def save(self, commit=True): if not hasattr(self.instance, "shop") or not self.instance.shop: self.instance.shop = get_shop(self.request) return super(PageForm, self).save(commit)
[docs] def is_url_valid(self, language_code, field_name, url): """ Ensure URL given is unique. Check through the pages translation model objects to make sure that the url given doesn't already exist. Possible failure cases: * for new page: 1. URL already exists * or existing page: 1. URL (other than owned by existing page) exists 2. URL exists in other languages of existing page """ pages_ids = Page.objects.for_shop(get_shop(self.request)).exclude(deleted=True).values_list("id", flat=True) qs = self._get_translation_model().objects.filter(url=url, master_id__in=pages_ids) if not self.instance.pk and qs.exists(): return False other_qs = qs.exclude(master=self.instance) if other_qs.exists(): return False own_qs = qs.filter(master=self.instance).exclude(language_code=language_code) if own_qs.exists(): return False return True
def _save_translation(self, instance, translation): if not translation.url: # No url? Skip saving this. if translation.pk: # Oh, it's an old one? translation.delete() # Well, it's not anymore. return translation.save()
[docs]class PageBaseFormPart(FormPart): priority = 1 name = "base"
[docs] def get_form_defs(self): yield TemplatedFormDef( self.name, PageForm, template_name="shuup/simple_cms/admin/_edit_base_page_form.jinja", required=True, kwargs={"instance": self.object, "languages": settings.LANGUAGES, "request": self.request}, )
[docs] def form_valid(self, form): self.object = form[self.name].save() if not self.object.created_by: self.object.created_by = self.request.user self.object.modified_by = self.request.user self.object.save()
[docs]class PageEditView(SaveFormPartsMixin, FormPartsViewMixin, CreateOrUpdateView): model = Page template_name = "shuup/simple_cms/admin/edit.jinja" base_form_part_classes = [ PageBaseFormPart, ] context_object_name = "page" form_part_class_provide_key = "admin_page_form_part" add_form_errors_as_messages = True
[docs] def get_toolbar(self): save_form_id = self.get_save_form_id() return get_default_edit_toolbar(self, save_form_id, delete_url=self.get_delete_url())
[docs] def get_delete_url(self): url = None if self.object.pk: url = reverse_lazy("shuup_admin:simple_cms.page.delete", kwargs={"pk": self.object.pk}) return url
[docs] def get_queryset(self): return super(PageEditView, self).get_queryset().for_shop(get_shop(self.request)).not_deleted()
[docs] def form_valid(self, form): return self.save_form_parts(form)
[docs]class PageListView(PicotableListView): url_identifier = "simple_cms.page" model = Page default_columns = [ Column( "title", _("Title"), sort_field="translations__title", display="title", linked=True, filter_config=TextFilter(operator="startswith", filter_field="translations__title"), ), Column("available_from", _("Available since")), Column("available_to", _("Available until")), Column("created_by", _("Created by")), Column("created_on", _("Date created")), ]
[docs] def get_object_abstract(self, instance, item): return [ {"text": "%s" % (instance or _("Page")), "class": "header"}, {"title": _("Available since"), "text": item.get("available_from")}, {"title": _("Available until"), "text": item.get("available_to")} if instance.available_to else None, ]
[docs] def get_queryset(self): return super(PageListView, self).get_queryset().for_shop(get_shop(self.request)).not_deleted()
[docs]class PageDeleteView(DetailView): queryset = Page.objects.all()
[docs] def get_success_url(self, *args, **kwargs): return reverse("shuup_admin:simple_cms.page.list")
[docs] def get_queryset(self, *args, **kwargs): queryset = super(PageDeleteView, self).get_queryset() queryset = queryset.for_shop(get_shop(self.request)).not_deleted() supplier = get_supplier(self.request) if supplier: queryset = queryset.filter(supplier=supplier) return queryset
[docs] def post(self, request, *args, **kwargs): page = self.get_object() page.soft_delete(user=request.user) messages.success(request, _("%s has been marked deleted.") % page) return HttpResponseRedirect(self.get_success_url())