Source code for shuup.admin.views.select

# -*- 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

from django.apps import apps
from django.contrib.auth import get_user_model
from django.core.exceptions import FieldDoesNotExist
from django.db import models
from django.db.models import Q
from django.http import JsonResponse
from django.utils.translation import ugettext_lazy as _
from django.views.generic import TemplateView

from shuup.admin.supplier_provider import get_supplier
from shuup.core.models import (
    Carrier,
    Category,
    Contact,
    Product,
    ProductMode,
    Shop,
    ShopProduct,
    ShopProductVisibility,
    Supplier,
)
from shuup.utils.django_compat import force_text


def _field_exists(model, field):
    try:
        model._meta.get_field(field)
        return True
    except FieldDoesNotExist:
        return False


[docs]class MultiselectAjaxView(TemplateView): model = None search_fields = [] result_limit = 20
[docs] def init_search_fields(self, cls): """ Configure the fields to use for searching. If the `cls` object has a search_fields attribute, it will be used, otherwise, the class will be inspected and the attribute `name` or `translations__name` will mainly be used. Other fields will be used for already known `cls` instances. """ if hasattr(cls, "search_fields"): self.search_fields = cls.search_fields return self.search_fields = [] key = "%sname" % ("translations__" if hasattr(cls, "translations") else "") self.search_fields.append(key) if issubclass(cls, Carrier): self.search_fields.append("base_translations__name") self.search_fields.remove("name") if issubclass(cls, Contact): self.search_fields.append("email") if issubclass(cls, Product): self.search_fields.append("sku") self.search_fields.append("barcode") if issubclass(cls, ShopProduct): self.search_fields.append("product__translations__name") user_model = get_user_model() if issubclass(cls, user_model): if _field_exists(user_model, "username"): self.search_fields.append("username") if _field_exists(user_model, "email"): self.search_fields.append("email") if not _field_exists(user_model, "name"): self.search_fields.remove("name")
[docs] def get_data(self, request, *args, **kwargs): # noqa model_name = request.GET.get("model") if not model_name: return [] cls = apps.get_model(model_name) qs = cls.objects.all() shop = request.shop # if shop is informed, make sure user has access to it if request.GET.get("shop"): query_shop = Shop.objects.get_for_user(request.user).filter(pk=request.GET["shop"]).first() if query_shop: shop = query_shop search_mode = request.GET.get("searchMode") qs = self._filter_query(request, cls, qs, shop, search_mode) self.init_search_fields(cls) if not self.search_fields: return [{"id": None, "name": _("Couldn't get selections for %s.") % model_name}] if request.GET.get("search"): query = Q() keyword = request.GET.get("search", "").strip() for field in self.search_fields: query |= Q(**{"%s__icontains" % field: keyword}) if issubclass(cls, Contact) or issubclass(cls, get_user_model()): query &= Q(is_active=True) qs = qs.filter(query) if search_mode and issubclass(cls, Product): if search_mode == "main": qs = qs.filter( mode__in=[ ProductMode.SIMPLE_VARIATION_PARENT, ProductMode.VARIABLE_VARIATION_PARENT, ProductMode.NORMAL, ] ) elif search_mode == "parent_product": qs = qs.filter(mode__in=[ProductMode.SIMPLE_VARIATION_PARENT, ProductMode.VARIABLE_VARIATION_PARENT]) elif search_mode == "sellable_mode_only": qs = qs.exclude( Q(mode__in=[ProductMode.SIMPLE_VARIATION_PARENT, ProductMode.VARIABLE_VARIATION_PARENT]) | Q(deleted=True) | Q(shop_products__visibility=ShopProductVisibility.NOT_VISIBLE) ).filter(shop_products__purchasable=True) sales_units = request.GET.get("salesUnits") if sales_units and issubclass(cls, Product): qs = qs.filter(sales_unit__translations__symbol__in=sales_units.strip().split(",")) qs = qs.distinct() return sorted( [{"id": obj.id, "name": force_text(obj)} for obj in qs[: self.result_limit]], key=lambda x: x["name"] )
def _filter_query(self, request, cls, qs, shop, search_mode=None): # the supplier provider returned a valid supplier # make sure to filter the search by the current supplier supplier = get_supplier(request) if search_mode == "visible" and issubclass(cls, Category): qs = cls.objects.all_visible(self.request.customer, shop=self.request.shop) elif search_mode == "enabled" and issubclass(cls, Supplier): qs = cls.objects.enabled(shop=shop) elif hasattr(cls.objects, "all_except_deleted"): qs = cls.objects.all_except_deleted(shop=shop) elif hasattr(cls.objects, "get_for_user"): qs = cls.objects.get_for_user(self.request.user) if issubclass(cls, Product): qs = qs.filter(shop_products__shop=shop) if supplier: qs = qs.filter(shop_products__suppliers=supplier) related_fields = [models.OneToOneField, models.ForeignKey, models.ManyToManyField] # Get all relation fields and check whether this models has # relation to Shop mode, if so, filter by the current shop allowed_shop_fields = ["shop", "shops"] shop_related_fields = [ field for field in cls._meta.get_fields() if type(field) in related_fields and field.related_model == Shop and field.name in allowed_shop_fields ] for shop_field in shop_related_fields: qs = qs.filter(**{shop_field.name: shop}) if supplier: allowed_supplier_fields = ["supplier", "suppliers"] supplier_related_fields = [ field for field in cls._meta.get_fields() if ( type(field) in related_fields and field.related_model == Supplier and field.name in allowed_supplier_fields ) ] for supplier_field in supplier_related_fields: qs = qs.filter(**{supplier_field.name: supplier}) return qs
[docs] def get(self, request, *args, **kwargs): return JsonResponse({"results": self.get_data(request, *args, **kwargs)})