Source code for shuup.admin.modules.contacts.views.detail

# -*- coding: utf-8 -*-
# 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

import warnings
from django.conf import settings
from django.contrib import messages
from django.contrib.auth import get_user_model
from django.http.response import HttpResponseRedirect
from django.utils.translation import ugettext_lazy as _
from django.views.generic import DetailView

from shuup.admin.modules.contacts.utils import check_contact_permission, request_limited
from shuup.admin.modules.users.views.detail import get_admin_url, get_front_url
from shuup.admin.shop_provider import get_shop
from shuup.admin.toolbar import DropdownActionButton, DropdownDivider, PostActionButton, Toolbar, URLActionButton
from shuup.apps.provides import get_provide_objects
from shuup.core.models import CompanyContact, Contact
from shuup.front.apps.registration.signals import company_contact_activated
from shuup.utils.deprecation import RemovedFromShuupWarning
from shuup.utils.django_compat import force_text, reverse
from shuup.utils.excs import Problem


[docs]class ContactDetailToolbar(Toolbar): def __init__(self, contact, request): self.contact = contact self.request = request self.user = getattr(self.contact, "user", None) super(ContactDetailToolbar, self).__init__() self.build()
[docs] def build_renew_password_button(self): disable_reason = None if "shuup.front.apps.auth" not in settings.INSTALLED_APPS: disable_reason = _("The Shuup frontend is not enabled.") elif not self.user: disable_reason = _("Contact has no associated user.") elif not getattr(self.user, "email", None): disable_reason = _("User has no associated email.") return PostActionButton( post_url=reverse("shuup_admin:contact.reset_password", kwargs={"pk": self.contact.pk}), name="pk", value=self.contact.pk, text=_("Reset password"), tooltip=_("Send a password renewal email."), confirm=_("Are you sure you wish to send a password recovery email to %s?") % self.contact.email, icon="fa fa-undo", disable_reason=disable_reason, extra_css_class="dropdown-item", )
[docs] def build_new_user_button(self): if self.user or isinstance(self.contact, CompanyContact): return return URLActionButton( url=reverse("shuup_admin:user.new") + "?contact_id=%s" % self.contact.pk, text=_("New User"), tooltip=_("Create a user for the contact."), icon="fa fa-user-plus", extra_css_class="dropdown-item", required_permissions=("user.new",), )
[docs] def build_new_order_button(self): return URLActionButton( url=reverse("shuup_admin:order.new") + "?contact_id=%s" % self.contact.pk, text=_("New Order"), tooltip=_("Create an order for the contact."), icon="fa fa-shopping-cart", extra_css_class="dropdown-item", required_permissions=("order.new",), )
[docs] def build_deactivate_button(self): return PostActionButton( post_url=self.request.path, name="set_is_active", value="0" if self.contact.is_active else "1", icon="fa fa-times-circle", text=_("Deactivate Contact") if self.contact.is_active else _("Activate Contact"), extra_css_class="dropdown-item btn-danger", )
[docs] def build_user_button(self): menu_items = [ self.build_new_order_button(), DropdownDivider(), self.build_new_user_button(), self.build_renew_password_button(), DropdownDivider(), self.build_deactivate_button(), ] self.append(DropdownActionButton(menu_items, icon="fa fa-user", text=_("Options"), extra_css_class="btn-info"))
[docs] def build_provides_buttons(self): action_menu_items = [] for button in get_provide_objects("admin_contact_toolbar_action_item"): if button.visible_for_object(self.contact): action_menu_items.append(button(object=self.contact)) if action_menu_items: self.append( DropdownActionButton( action_menu_items, icon="fa fa-star", text=_("Actions"), extra_css_class="btn-inverse", ) ) for button in get_provide_objects("admin_contact_toolbar_button"): warnings.warn( "`admin_contact_toolbar_button` provider is deprecated, " "use `admin_contact_toolbar_action_item` instead.", RemovedFromShuupWarning, ) self.append(button(self.contact))
[docs] def build_login_as_button(self): user = self.contact.user if hasattr(self.contact, "user") else None current_user = self.request.user if isinstance(user, get_user_model()): has_privileges = bool( getattr(current_user, "is_superuser", False) or getattr(current_user, "is_staff", False) ) can_impersonate = bool(has_privileges and user.is_active and not user.is_superuser) if can_impersonate and get_front_url() and not user.is_staff: self.append( PostActionButton( post_url=reverse("shuup_admin:user.login-as", kwargs={"pk": user.pk}), text=_("Login as User"), extra_css_class="btn-inverse", ) ) elif can_impersonate and get_admin_url() and user.is_staff: self.append( PostActionButton( post_url=reverse("shuup_admin:user.login-as-staff", kwargs={"pk": user.pk}), text=_("Login as Staff User"), extra_css_class="btn-inverse", ) )
[docs] def build(self): self.append( URLActionButton( url=reverse("shuup_admin:contact.edit", kwargs={"pk": self.contact.pk}), icon="fa fa-pencil", text=_("Edit"), extra_css_class="btn-primary", ) ) self.build_renew_password_button() self.build_new_user_button() self.build_user_button() self.build_new_order_button() self.build_provides_buttons() self.build_login_as_button()
[docs]class ContactDetailView(DetailView): model = Contact template_name = "shuup/admin/contacts/detail.jinja" context_object_name = "contact"
[docs] def get_object(self, *args, **kwargs): contact = super(ContactDetailView, self).get_object(*args, **kwargs) check_contact_permission(self.request, contact) return contact
[docs] def get_queryset(self): qs = super(ContactDetailView, self).get_queryset() if request_limited(self.request): qs = qs.filter(shops=get_shop(self.request)) if not self.request.user.is_superuser: # non superusers can't see superusers contacts qs = qs.exclude(PersonContact___user__is_superuser=True) return qs
[docs] def get_context_data(self, **kwargs): context = super(ContactDetailView, self).get_context_data(**kwargs) context["toolbar"] = ContactDetailToolbar(contact=self.object, request=self.request) context["title"] = "%s: %s" % (self.object._meta.verbose_name.title(), force_text(self.object)) context["contact_sections"] = [] contact_sections_provides = sorted(get_provide_objects("admin_contact_section"), key=lambda x: x.order) for admin_contact_section in contact_sections_provides: # Check whether the ContactSection should be visible for the current object if admin_contact_section.visible_for_object(self.object, self.request): context["contact_sections"].append(admin_contact_section) # add additional context data where the key is the contact_section identifier section_context = admin_contact_section.get_context_data(self.object, self.request) context[admin_contact_section.identifier] = section_context return context
def _handle_set_is_active(self): old_state = self.object.is_active state = bool(int(self.request.POST["set_is_active"])) if not state and hasattr(self.object, "user"): if getattr(self.object.user, "is_superuser", False) and not getattr( self.request.user, "is_superuser", False ): raise Problem(_("You can not deactivate a superuser. Remove superuser permission first.")) if self.object.user == self.request.user: raise Problem(_("You can not deactivate yourself. Use another account to deactivate the current one.")) self.object.is_active = state self.object.save(update_fields=("is_active",)) messages.success( self.request, _("%(contact)s is now %(state)s.") % {"contact": self.object, "state": _("active") if state else _("inactive")}, ) if self.object.is_active and self.object.is_active != old_state and isinstance(self.object, CompanyContact): company_contact_activated.send(sender=type(self.object), instance=self.object, request=self.request) return HttpResponseRedirect(self.request.path)
[docs] def post(self, request, *args, **kwargs): self.object = self.get_object() if "set_is_active" in request.POST: return self._handle_set_is_active() return super(ContactDetailView, self).post(request, *args, **kwargs)