adminFront/src/kitUI/Cart/calc.ts

253 lines
9.2 KiB
TypeScript
Raw Normal View History

2023-03-06 13:25:33 +00:00
import { CartItem, AnyDiscount, CartTotal, CartItemTotal, PrivilegeDiscount, CartPurchasesAmountDiscount, PurchasesAmountDiscount, ServiceToPriceMap, ServiceDiscount, UserDiscount } from "@root/model/cart";
2023-03-08 10:52:16 +00:00
import { ServiceType, SERVICE_LIST, Tariff } from "../../model/tariff";
import { User } from "../../model/user";
2023-02-25 09:23:28 +00:00
export function calcCartData(
user: User,
2023-03-06 13:25:33 +00:00
cartItems: CartItem[],
discounts: AnyDiscount[],
2023-03-14 08:28:42 +00:00
isNonCommercial: boolean = false,
2023-02-28 13:15:08 +00:00
coupon?: string,
2023-03-09 08:51:51 +00:00
): CartTotal | Error | null {
let isIncompatibleTariffs = false;
const defaultTariffTypePresent: { [Key in ServiceType]: boolean } = {
dwarfener: false,
squiz: false,
templategen: false,
};
cartItems.forEach(cartItem => {
if (cartItem.tariff.customPricePerUnit === undefined) return defaultTariffTypePresent[cartItem.tariff.privilege.serviceKey] = true;
if (
defaultTariffTypePresent[cartItem.tariff.privilege.serviceKey] &&
cartItem.tariff.customPricePerUnit !== undefined
) isIncompatibleTariffs = true;
});
if (isIncompatibleTariffs) return new Error("Если взят готовый тариф, то кастомный на этот сервис сделать уже нельзя");
if (!cartItems.length) return null;
2023-03-06 13:25:33 +00:00
const cartTotal: CartTotal = {
2023-02-25 09:23:28 +00:00
items: [],
totalPrice: 0,
priceByService: {
2023-03-08 10:52:16 +00:00
templategen: 0,
squiz: 0,
dwarfener: 0,
2023-02-25 09:23:28 +00:00
},
2023-03-08 10:52:16 +00:00
discountsByService: {
templategen: null,
squiz: null,
dwarfener: null,
},
2023-03-09 09:01:40 +00:00
envolvedCartDiscounts: [],
2023-03-14 12:47:05 +00:00
couponState: coupon ? "not found" : null,
2023-02-25 09:23:28 +00:00
};
2023-02-28 13:15:08 +00:00
// layer 0
2023-02-25 09:23:28 +00:00
for (const discount of discounts) {
2023-03-14 08:28:42 +00:00
if (discount.conditionType !== "userType" || !isNonCommercial) continue;
2023-02-25 09:23:28 +00:00
cartItems.forEach(cartItem => {
cartTotal.items.push({
2023-03-09 09:01:40 +00:00
envolvedDiscounts: [],
2023-02-28 13:15:08 +00:00
tariff: cartItem.tariff,
2023-02-25 09:23:28 +00:00
totalPrice: cartItem.price,
});
2023-02-28 13:15:08 +00:00
2023-03-08 10:52:16 +00:00
cartTotal.priceByService[cartItem.tariff.privilege.serviceKey] += cartItem.price;
2023-02-25 09:23:28 +00:00
cartTotal.totalPrice += cartItem.price;
});
2023-02-28 13:15:08 +00:00
cartTotal.totalPrice *= discount.target.factor;
2023-03-09 09:01:40 +00:00
cartTotal.envolvedCartDiscounts.push(discount);
2023-02-28 13:15:08 +00:00
2023-02-25 09:23:28 +00:00
return cartTotal;
}
2023-02-28 13:15:08 +00:00
const couponDiscount = coupon ? findUserDiscount(discounts, user, coupon) : null;
2023-02-25 09:23:28 +00:00
2023-02-28 13:15:08 +00:00
// layer 1
2023-02-25 09:23:28 +00:00
for (const cartItem of cartItems) {
2023-03-06 13:25:33 +00:00
const cartItemTotal: CartItemTotal = {
2023-02-28 13:15:08 +00:00
tariff: cartItem.tariff,
2023-03-09 09:01:40 +00:00
envolvedDiscounts: [],
2023-02-25 09:23:28 +00:00
totalPrice: cartItem.price,
};
2023-02-28 13:15:08 +00:00
const tariff = cartItem.tariff;
const privilegesAffectedByCoupon: string[] = [];
couponDiscount?.target.products.forEach(product => {
2023-03-08 10:52:16 +00:00
if (product.privilegeId !== tariff.privilege.privilegeId) return;
if (tariff.customPricePerUnit !== undefined && !couponDiscount.overwhelm) return;
cartItemTotal.totalPrice *= product.factor;
2023-03-09 09:01:40 +00:00
cartItemTotal.envolvedDiscounts.push(couponDiscount);
2023-03-14 12:47:05 +00:00
cartTotal.couponState = "applied";
2023-03-08 10:52:16 +00:00
privilegesAffectedByCoupon.push(product.privilegeId);
2023-02-28 13:15:08 +00:00
});
2023-02-25 09:23:28 +00:00
2023-02-28 13:15:08 +00:00
const privilegeDiscount = findMaxApplicablePrivilegeDiscount(discounts, tariff);
privilegeDiscount?.target.products.forEach(product => {
2023-03-08 10:52:16 +00:00
if (product.privilegeId !== tariff.privilege.privilegeId) return;
if (tariff.customPricePerUnit !== undefined) return;
if (privilegesAffectedByCoupon.includes(privilegeDiscount.condition.privilege.id)) return;
cartItemTotal.totalPrice *= product.factor;
2023-03-09 09:01:40 +00:00
cartItemTotal.envolvedDiscounts.push(privilegeDiscount);
2023-02-28 13:15:08 +00:00
});
2023-02-25 09:23:28 +00:00
cartTotal.items.push(cartItemTotal);
2023-03-08 10:52:16 +00:00
cartTotal.priceByService[tariff.privilege.serviceKey] += cartItemTotal.totalPrice;
2023-02-25 09:23:28 +00:00
}
2023-02-25 12:29:20 +00:00
// layer 2
2023-03-06 13:25:33 +00:00
SERVICE_LIST.map(service => service.serviceKey).forEach(service => {
2023-02-28 13:15:08 +00:00
const serviceDiscount = findMaxServiceDiscount(service, discounts, cartTotal.priceByService);
2023-02-27 13:33:15 +00:00
if (serviceDiscount) {
2023-03-08 10:52:16 +00:00
cartTotal.priceByService[service] *= serviceDiscount.target.factor;
cartTotal.discountsByService[service] = serviceDiscount;
2023-02-27 13:33:15 +00:00
}
2023-02-25 12:29:20 +00:00
2023-03-08 10:52:16 +00:00
cartTotal.totalPrice += cartTotal.priceByService[service];
2023-02-25 12:29:20 +00:00
});
// layer 3
const cartPurchasesAmountDiscount = findMaxCartPurchasesAmountDiscount(discounts, cartTotal);
2023-02-27 13:33:15 +00:00
if (cartPurchasesAmountDiscount) {
cartTotal.totalPrice *= cartPurchasesAmountDiscount.factor;
2023-03-09 09:01:40 +00:00
cartTotal.envolvedCartDiscounts.push(cartPurchasesAmountDiscount);
2023-02-27 13:33:15 +00:00
}
2023-02-25 12:29:20 +00:00
// layer 4
2023-03-14 08:28:42 +00:00
const totalPurchasesAmountDiscount = findMaxTotalPurchasesAmountDiscount(discounts, user.PurchasesAmount);
2023-02-27 13:33:15 +00:00
if (totalPurchasesAmountDiscount) {
cartTotal.totalPrice *= totalPurchasesAmountDiscount.factor;
2023-03-09 09:01:40 +00:00
cartTotal.envolvedCartDiscounts.push(totalPurchasesAmountDiscount);
2023-02-27 13:33:15 +00:00
}
2023-02-25 12:29:20 +00:00
2023-02-25 09:23:28 +00:00
return cartTotal;
}
2023-03-06 13:25:33 +00:00
function findMaxApplicablePrivilegeDiscount(discounts: AnyDiscount[], tariff: Tariff): PrivilegeDiscount | null {
const applicableDiscounts = discounts.filter((discount): discount is PrivilegeDiscount => {
2023-02-27 13:33:15 +00:00
return (
discount.conditionType === "privilege" &&
tariff.privilege.privilegeId === discount.condition.privilege.id &&
tariff.amount >= discount.condition.privilege.value
);
2023-02-25 09:23:28 +00:00
});
2023-02-25 12:29:20 +00:00
if (!applicableDiscounts.length) return null;
2023-02-25 09:23:28 +00:00
2023-02-25 12:29:20 +00:00
const maxValueDiscount = applicableDiscounts.reduce(
2023-02-25 09:23:28 +00:00
(prev, current) => current.condition.privilege.value > prev.condition.privilege.value ? current : prev
);
return maxValueDiscount;
2023-02-25 12:29:20 +00:00
}
2023-03-06 13:25:33 +00:00
function findMaxCartPurchasesAmountDiscount(discounts: AnyDiscount[], cartTotal: CartTotal): CartPurchasesAmountDiscount | null {
const applicableDiscounts = discounts.filter((discount): discount is CartPurchasesAmountDiscount => {
2023-02-27 13:33:15 +00:00
return discount.conditionType === "cartPurchasesAmount" && cartTotal.totalPrice >= discount.condition.cartPurchasesAmount;
2023-02-25 12:29:20 +00:00
});
if (!applicableDiscounts.length) return null;
const maxValueDiscount = applicableDiscounts.reduce(
(prev, current) => current.condition.cartPurchasesAmount > prev.condition.cartPurchasesAmount ? current : prev
);
return maxValueDiscount;
}
2023-03-14 08:28:42 +00:00
function findMaxTotalPurchasesAmountDiscount(discounts: AnyDiscount[], purchasesAmount: number): PurchasesAmountDiscount | null {
2023-03-06 13:25:33 +00:00
const applicableDiscounts = discounts.filter((discount): discount is PurchasesAmountDiscount => {
2023-03-14 08:28:42 +00:00
return discount.conditionType === "purchasesAmount" && purchasesAmount >= discount.condition.purchasesAmount;
2023-02-25 12:29:20 +00:00
});
if (!applicableDiscounts.length) return null;
const maxValueDiscount = applicableDiscounts.reduce(
(prev, current) => current.condition.purchasesAmount > prev.condition.purchasesAmount ? current : prev
);
return maxValueDiscount;
}
function findMaxServiceDiscount(
2023-03-06 13:25:33 +00:00
service: ServiceType,
discounts: AnyDiscount[],
priceByService: ServiceToPriceMap,
): ServiceDiscount | null {
const discountsForTariffService = discounts.filter((discount): discount is ServiceDiscount => {
2023-02-25 12:29:20 +00:00
return (
discount.conditionType === "service" &&
discount.condition.service.id === service &&
2023-03-08 10:52:16 +00:00
priceByService[service] >= discount.condition.service.value
2023-02-25 12:29:20 +00:00
);
});
if (!discountsForTariffService.length) return null;
const maxValueDiscount = discountsForTariffService.reduce((prev, current) => {
return current.condition.service.value > prev.condition.service.value ? current : prev;
});
return maxValueDiscount;
}
2023-02-25 09:23:28 +00:00
2023-03-06 13:25:33 +00:00
function findUserDiscount(discounts: AnyDiscount[], user: User, coupon: string,): UserDiscount | null {
const userDiscount = discounts.find((discount): discount is UserDiscount => {
2023-02-28 13:15:08 +00:00
return (
discount.conditionType === "user" &&
discount.condition.user === user.ID &&
discount.condition.coupon === coupon
);
2023-02-25 09:23:28 +00:00
});
2023-02-28 13:15:08 +00:00
return userDiscount ?? null;
2023-03-06 13:25:33 +00:00
}
export function createCartItem(tariff: Tariff): CartItem {
const pricePerUnit = tariff.customPricePerUnit ?? tariff.privilege.pricePerUnit;
const price = pricePerUnit * tariff.amount;
return { tariff, price, id: "someId" };
}
2023-03-07 14:39:46 +00:00
export function findDiscountFactor(discount: AnyDiscount, privilegeId?: string) {
2023-03-06 13:25:33 +00:00
switch (discount.conditionType) {
2023-03-06 17:30:35 +00:00
case "cartPurchasesAmount":
return discount.factor;
case "purchasesAmount":
2023-03-06 13:25:33 +00:00
return discount.factor;
case "privilege": {
const product = discount.target.products.find(product => product.privilegeId === privilegeId);
if (!product) throw new Error("Discount target product not found");
return product.factor;
}
case "user": {
const product = discount.target.products.find(product => product.privilegeId === privilegeId);
if (!product) throw new Error("Discount target product not found");
return product.factor;
}
case "service":
return discount.target.factor;
case "userType":
return discount.target.factor;
}
}
export function formatDiscountFactor(factor: number): string {
return `${((1 - factor) * 100).toFixed(1)}%`;
}