Source code for shuup.admin.modules.products.forms.package_forms
# -*- 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 six
from django import forms
from django.contrib import messages
from django.db.transaction import atomic
from django.utils.translation import ugettext_lazy as _
from shuup.admin.forms.widgets import PackageProductChoiceWidget
from shuup.admin.modules.products.utils import clear_existing_package
from shuup.core.excs import ImpossibleProductModeException, Problem
from shuup.core.models import Product, Shop, ShopProduct
from .parent_forms import ProductChildBaseFormSet
[docs]class PackageChildForm(forms.Form):
def __init__(self, **kwargs):
initial = kwargs.get("initial", {})
self.product = initial.get("child")
super(PackageChildForm, self).__init__(**kwargs)
[docs] def get_shop_products(self, user):
if not user or not self.product:
return ShopProduct.objects.none()
else:
shop_products = []
shop_queryset = Shop.objects.all()
if not user.is_superuser:
shop_queryset = shop_queryset.filter(staff_members=user)
for shop in shop_queryset:
try:
shop_product = self.product.get_shop_instance(shop)
shop_products.append(shop_product)
except ShopProduct.DoesNotExist:
continue
return shop_products
[docs] def get_stock_statuses(self, user=None):
shop_products = self.get_shop_products(user)
return get_stock_statuses(self.product, shop_products)
[docs] def get_orderability_errors(self, user=None):
shop_products = self.get_shop_products(user)
return get_orderability_errors(self.product, shop_products)
child = forms.ModelChoiceField(
queryset=Product.objects.all(),
widget=PackageProductChoiceWidget(),
label="",
required=True,
)
quantity = forms.DecimalField(
min_value=0,
label="",
widget=forms.NumberInput(attrs={"class": "form-control"}),
required=True,
)
[docs]def get_stock_statuses(product, shop_products):
stocks = {}
if not product:
return stocks
sales_unit = product.sales_unit
sales_decimals = sales_unit.decimals if sales_unit else 0
sales_unit_symbol = sales_unit.symbol if sales_unit else ""
for shop_product in shop_products:
for supplier in shop_product.suppliers.enabled(shop=shop_product.shop):
if supplier in stocks.keys():
continue
stock_status = supplier.get_stock_status(product_id=product.id)
stocks[supplier] = (supplier, stock_status, sales_decimals, sales_unit_symbol)
return stocks
[docs]def get_orderability_errors(product, shop_products):
orderability_errors = []
if not product:
return orderability_errors
for shop_product in shop_products:
orderability_errors.extend(
[
"%s: %s" % (shop_product.shop.name, msg.message)
for msg in shop_product.get_orderability_errors(
supplier=None, quantity=shop_product.minimum_purchase_quantity, customer=None
)
]
)
for supplier in shop_product.suppliers.enabled(shop=shop_product.shop):
orderability_errors.extend(
[
"%s: %s" % (supplier.name, msg.message)
for msg in supplier.get_orderability_errors(
shop_product=shop_product, quantity=shop_product.minimum_purchase_quantity, customer=None
)
]
)
return orderability_errors
[docs]class PackageChildFormSet(ProductChildBaseFormSet):
deletion_label = ""
def __init__(self, **kwargs):
self.parent_product = kwargs.pop("parent_product")
kwargs["initial"] = [
{
"child": product,
"quantity": quantity,
}
for (product, quantity) in six.iteritems(self.parent_product.get_package_child_to_quantity_map())
]
super(PackageChildFormSet, self).__init__(**kwargs)
[docs] def save(self):
parent_product = self.parent_product
current_products = set(parent_product.get_package_child_to_quantity_map())
selected_products, removed_products, selected_quantities = self.get_selected_and_removed()
with atomic():
try:
clear_existing_package(parent_product)
parent_product.make_package(package_def=selected_quantities)
except ImpossibleProductModeException as ipme:
six.raise_from(
Problem(
_("Unable to make package %(product)s: %(error)s.") % {"product": parent_product, "error": ipme}
),
ipme,
)
products_to_add = selected_products - current_products
products_to_remove = current_products & removed_products
message_parts = []
if products_to_add:
message_parts.append(_("New products: %d.") % len(products_to_add))
if products_to_remove:
message_parts.append(_("Removed products: %d.") % len(products_to_remove))
if message_parts and self.request:
messages.success(self.request, ", ".join(message_parts))
[docs] def get_selected_and_removed(self):
deleted_forms = self.deleted_forms
removed_products = set()
selected_products = set()
selected_product_quantities = {}
for child_form in self.forms:
child_product = child_form.cleaned_data.get("child")
if not child_product:
continue
if child_form in deleted_forms:
removed_products.add(child_product)
elif child_product != self.parent_product:
selected_products.add(child_product)
elif self.request and child_product == self.parent_product:
messages.error(self.request, _("Couldn't add product %s to its own package.") % str(child_product))
quantity = child_form.cleaned_data.get("quantity")
selected_product_quantities[child_product] = quantity
selected_quantities = {
product: quantity
for product, quantity in selected_product_quantities.items()
if (product not in removed_products and product != self.parent_product)
}
return (selected_products, removed_products, selected_quantities)