from django import forms
from django.utils.translation import get_language
from django.utils.translation import gettext_lazy as _
from enumfields import Enum
from shuup.core.catalog import ProductCatalog, ProductCatalogContext
from shuup.core.models import Product, ProductCrossSell, ProductCrossSellType, ProductMode, ShopProductVisibility
from shuup.front.template_helpers.general import (
get_best_selling_products,
get_newest_products,
get_products_for_categories,
get_random_products,
)
from shuup.front.template_helpers.product import map_relation_type
from shuup.xtheme import TemplatedPlugin
from shuup.xtheme.plugins.forms import GenericPluginForm, TranslatableField
from shuup.xtheme.plugins.widgets import XThemeSelect2ModelChoiceField, XThemeSelect2ModelMultipleChoiceField
[docs]
class HighlightType(Enum):
NEWEST = "newest"
BEST_SELLING = "best_selling"
RANDOM = "random"
class Labels:
NEWEST = _("Newest")
BEST_SELLING = _("Best Selling")
RANDOM = _("Random")
[docs]
class ProductHighlightPlugin(TemplatedPlugin):
identifier = "product_highlight"
name = _("Product Highlights")
template_name = "shuup/xtheme/plugins/highlight_plugin.jinja"
cacheable = True
fields = [
("title", TranslatableField(label=_("Title"), required=False, initial="")),
(
"type",
forms.ChoiceField(
label=_("Type"),
choices=HighlightType.choices(),
initial=HighlightType.NEWEST.value,
),
),
("count", forms.IntegerField(label=_("Count"), min_value=1, initial=4)),
(
"orderable_only",
forms.BooleanField(
label=_("Only show in-stock and orderable items"),
help_text=_(
"Warning: The final number of products can be lower than 'Count' "
"as it will filter out unorderable products from a set of 'Count' products."
),
initial=True,
required=False,
),
),
]
[docs]
def get_cache_key(self, context, **kwargs) -> str:
title = self.get_translated_value("title")
highlight_type = self.config.get("type", HighlightType.NEWEST.value)
count = self.config.get("count", 4)
orderable_only = self.config.get("orderable_only", True)
return str((get_language(), title, highlight_type, orderable_only, count))
[docs]
def get_context_data(self, context):
highlight_type = self.config.get("type", HighlightType.NEWEST.value)
count = self.config.get("count", 4)
orderable_only = self.config.get("orderable_only", True)
if highlight_type == HighlightType.NEWEST.value:
products = get_newest_products(context, count, orderable_only)
elif highlight_type == HighlightType.BEST_SELLING.value:
products = get_best_selling_products(
context,
count,
orderable_only=orderable_only,
)
elif highlight_type == HighlightType.RANDOM.value:
products = get_random_products(context, count, orderable_only)
else:
products = []
return {
"request": context["request"],
"title": self.get_translated_value("title"),
"products": products,
}
[docs]
class ProductCrossSellsPlugin(TemplatedPlugin):
identifier = "product_cross_sells"
name = _("Product Cross Sells")
template_name = "shuup/xtheme/plugins/cross_sells_plugin.jinja"
cacheable = True
required_context_variables = ["product"]
fields = [
("title", TranslatableField(label=_("Title"), required=False, initial="")),
("type", ProductCrossSell.type.field.formfield()),
("count", forms.IntegerField(label=_("Count"), min_value=1, initial=4)),
(
"use_variation_parents",
forms.BooleanField(
label=_("Show variation parents"),
help_text=_("Render variation parents instead of the children."),
initial=False,
required=False,
),
),
(
"orderable_only",
forms.BooleanField(
label=_("Only show in-stock and orderable items"),
initial=True,
required=False,
help_text=_(
"Warning: The final number of products can be lower than 'Count' "
"as it will filter out unorderable products from a set of 'Count' products."
),
),
),
]
[docs]
def __init__(self, config):
relation_type = config.get("type", None)
if relation_type:
# Map initial config string to enum type
try:
type = map_relation_type(relation_type)
except LookupError:
type = ProductCrossSellType.RELATED
config["type"] = type
super().__init__(config)
[docs]
def get_cache_key(self, context, **kwargs) -> str:
title = self.get_translated_value("title")
relation_type = self.config.get("type")
count = self.config.get("count", 4)
orderable_only = self.config.get("orderable_only", True)
use_variation_parents = self.config.get("use_variation_parents", False)
return str(
(
get_language(),
title,
relation_type,
orderable_only,
count,
use_variation_parents,
)
)
[docs]
def get_context_data(self, context):
count = self.config.get("count", 4)
product = context.get("product", None)
orderable_only = self.config.get("orderable_only", True)
relation_type = self.config.get("type")
try:
type = map_relation_type(relation_type)
except LookupError:
type = ProductCrossSellType.RELATED
return {
"request": context["request"],
"title": self.get_translated_value("title"),
"use_variation_parents": self.config.get("use_variation_parents", False),
"product": product,
"type": type,
"count": count,
"orderable_only": orderable_only,
}
[docs]
class ProductsFromCategoryPlugin(TemplatedPlugin):
identifier = "category_products"
name = _("Category Products Highlight")
template_name = "shuup/xtheme/plugins/highlight_plugin.jinja"
editor_form_class = ProductsFromCategoryForm
cacheable = True
fields = [
("title", TranslatableField(label=_("Title"), required=False, initial="")),
("count", forms.IntegerField(label=_("Count"), min_value=1, initial=4)),
(
"orderable_only",
forms.BooleanField(
label=_("Only show in-stock and orderable items"),
initial=True,
required=False,
help_text=_(
"Warning: The final number of products can be lower than 'Count' "
"as it will filter out unorderable products from a set of 'Count' products."
),
),
),
]
[docs]
def get_cache_key(self, context, **kwargs) -> str:
title = self.get_translated_value("title")
category_id = self.config.get("category")
count = self.config.get("count")
orderable_only = self.config.get("orderable_only", True)
return str((get_language(), title, category_id, orderable_only, count))
[docs]
def get_context_data(self, context):
products = []
category_id = self.config.get("category")
count = self.config.get("count")
orderable_only = self.config.get("orderable_only", True)
if category_id:
products = get_products_for_categories(
context, [category_id], n_products=count, orderable_only=orderable_only
)
return {
"request": context["request"],
"title": self.get_translated_value("title"),
"products": products,
}
[docs]
class ProductSelectionPlugin(TemplatedPlugin):
"""
A plugin that renders a selection of products
"""
identifier = "product_selection"
name = _("Product Selection")
template_name = "shuup/xtheme/plugins/product_selection_plugin.jinja"
editor_form_class = ProductSelectionConfigForm
cacheable = True
fields = [("title", TranslatableField(label=_("Title"), required=False, initial=""))]
[docs]
def get_cache_key(self, context, **kwargs) -> str:
title = self.get_translated_value("title")
products = self.config.get("products")
return str((get_language(), title, products))
[docs]
def get_context_data(self, context):
request = context["request"]
products = self.config.get("products")
products_qs = Product.objects.none()
if products:
catalog = ProductCatalog(
ProductCatalogContext(
shop=request.shop,
user=getattr(request, "user", None),
contact=getattr(request, "customer", None),
purchasable_only=True,
visibility=ShopProductVisibility.LISTED,
)
)
products_qs = catalog.get_products_queryset().filter(
pk__in=products, mode__in=ProductMode.get_parent_modes()
)
return {
"request": request,
"title": self.get_translated_value("title"),
"products": products_qs,
}