From b9b6393c6c550a38d5c3b28ae403aa3b75e84b1d Mon Sep 17 00:00:00 2001
From: Chung Zhi Xuan <97169613+spaceman03@users.noreply.github.com>
Date: Wed, 28 Aug 2024 22:25:34 +0800
Subject: [PATCH] Add merch promotions page (#149)
* Add merch promotions page
* Add button to create promotion
* Import Promotion class from types
* Fix linting issue
---------
Co-authored-by: limivann <71662324+limivann@users.noreply.github.com>
---
.../src/admin/components/AfterNavLinks.tsx | 5 +
apps/cms/src/admin/graphics/Logos.tsx | 2 +-
apps/cms/src/admin/styles.scss | 2 +-
.../cms/src/admin/utils/RenderCellFactory.tsx | 20 +++
apps/cms/src/admin/views/MerchProducts.tsx | 3 +-
apps/cms/src/admin/views/MerchPromotion.scss | 3 +
apps/cms/src/admin/views/MerchPromotion.tsx | 133 ++++++++++++++++++
apps/cms/src/admin/views/MerchSales.tsx | 19 +--
apps/cms/src/admin/views/ViewTemplate.tsx | 2 +-
apps/cms/src/apis/promotions.api.ts | 35 +++++
apps/cms/src/payload.config.ts | 19 ++-
apps/cms/src/types.ts | 5 +-
.../merch/components/checkout/Skeleton.tsx | 2 +
packages/eslint-config-custom/index.js | 1 +
packages/merch-helpers/src/lib/price.ts | 3 +-
packages/types/src/lib/merch.ts | 6 +-
16 files changed, 234 insertions(+), 26 deletions(-)
create mode 100644 apps/cms/src/admin/views/MerchPromotion.scss
create mode 100644 apps/cms/src/admin/views/MerchPromotion.tsx
create mode 100644 apps/cms/src/apis/promotions.api.ts
diff --git a/apps/cms/src/admin/components/AfterNavLinks.tsx b/apps/cms/src/admin/components/AfterNavLinks.tsx
index 772f6076..ee7edcfc 100644
--- a/apps/cms/src/admin/components/AfterNavLinks.tsx
+++ b/apps/cms/src/admin/components/AfterNavLinks.tsx
@@ -19,6 +19,11 @@ const merchRoutes = [
label: "Products",
href: "/admin/merch/products",
},
+ {
+ id: "merch_promotions",
+ label: "Promotions",
+ href: "/admin/merch/promotions",
+ },
];
const MerchLinks: React.FC = () => {
diff --git a/apps/cms/src/admin/graphics/Logos.tsx b/apps/cms/src/admin/graphics/Logos.tsx
index 29e59d7c..590187b8 100644
--- a/apps/cms/src/admin/graphics/Logos.tsx
+++ b/apps/cms/src/admin/graphics/Logos.tsx
@@ -14,4 +14,4 @@ const SCSEImage = ({ classname }: SCSEImageProps) => {
};
export const SCSEIcon = () => ;
-export const SCSELogo = () => ;
+export const SCSELogo = () => ;
\ No newline at end of file
diff --git a/apps/cms/src/admin/styles.scss b/apps/cms/src/admin/styles.scss
index f7307f65..f4418f4e 100644
--- a/apps/cms/src/admin/styles.scss
+++ b/apps/cms/src/admin/styles.scss
@@ -1,2 +1,2 @@
header {
-}
+}
\ No newline at end of file
diff --git a/apps/cms/src/admin/utils/RenderCellFactory.tsx b/apps/cms/src/admin/utils/RenderCellFactory.tsx
index 9000c172..5790390e 100644
--- a/apps/cms/src/admin/utils/RenderCellFactory.tsx
+++ b/apps/cms/src/admin/utils/RenderCellFactory.tsx
@@ -1,5 +1,6 @@
import React from "react";
import payload from "payload";
+import { Promotion } from "types";
export class RenderCellFactory {
static get(element: unknown, key: string) {
@@ -103,6 +104,25 @@ export class RenderCellFactory {
};
}
+ if (key === "discounts" && typeof element[key] === "object") {
+ // Render discounts
+ const DiscountsComponent: React.FC<{ data: Promotion["discounts"] }> = ({ data }) => (
+
+
+
+
+ Promo Type: {data.promoType}
+ Promo Value: {data.promoValue}
+ Applies To: {data.appliesTo?.join(", ") || "All"}
+ Minimum Quantity: {data.minimumQty ?? "None"}
+
+
+
+
+ );
+ return (row: any, data: Promotion["discounts"]) => ;
+ }
+
if (typeof element[key] == "object") {
const DateComponent: React.FC<{ children?: React.ReactNode }> = ({
children,
diff --git a/apps/cms/src/admin/views/MerchProducts.tsx b/apps/cms/src/admin/views/MerchProducts.tsx
index fbfce61e..f9dfea9a 100644
--- a/apps/cms/src/admin/views/MerchProducts.tsx
+++ b/apps/cms/src/admin/views/MerchProducts.tsx
@@ -37,7 +37,8 @@ const MerchProducts: AdminView = ({ user, canAccessAdmin }) => {
const sampleProduct = data[0];
const keys = Object.keys(sampleProduct);
for (const key of keys) {
- const renderCell: React.FC<{ children?: React.ReactNode }> = RenderCellFactory.get(sampleProduct, key);
+ const renderCell: React.FC<{ children?: React.ReactNode }> =
+ RenderCellFactory.get(sampleProduct, key);
const col: Column = {
accessor: key,
components: {
diff --git a/apps/cms/src/admin/views/MerchPromotion.scss b/apps/cms/src/admin/views/MerchPromotion.scss
new file mode 100644
index 00000000..ccd39949
--- /dev/null
+++ b/apps/cms/src/admin/views/MerchPromotion.scss
@@ -0,0 +1,3 @@
+table {
+ width: 100%;
+}
\ No newline at end of file
diff --git a/apps/cms/src/admin/views/MerchPromotion.tsx b/apps/cms/src/admin/views/MerchPromotion.tsx
new file mode 100644
index 00000000..8c5ef3f6
--- /dev/null
+++ b/apps/cms/src/admin/views/MerchPromotion.tsx
@@ -0,0 +1,133 @@
+import React, { useEffect, useState } from "react";
+import { Button } from "payload/components/elements";
+import { AdminView } from "payload/config";
+import ViewTemplate from "./ViewTemplate";
+import { Column } from "payload/dist/admin/components/elements/Table/types";
+import { RenderCellFactory } from "../utils/RenderCellFactory";
+import SortedColumn from "../utils/SortedColumn";
+import { Table } from "payload/dist/admin/components/elements/Table";
+import { Promotion } from "types";
+import PromotionsApi from "../../apis/promotions.api";
+import './MerchPromotion.scss';
+
+const MerchPromotion: AdminView = ({ user, canAccessAdmin }) => {
+ // Get data from API
+ const [data, setData] = useState(null);
+ useEffect(() => {
+ PromotionsApi.getPromotions()
+ .then((res: Promotion[]) => setData(res))
+ .catch((error) => console.log(error));
+ }, []);
+
+ // Output human-readable table headers based on the attribute names from the API
+ function prettifyKey(str: string): string {
+ let res = "";
+ for (const i of str.split("_")) {
+ res += i.charAt(0).toUpperCase() + i.slice(1) + " ";
+ }
+ return res;
+ }
+
+ // Do not load table until we receive the data
+ if (data == null) {
+ return Loading...
;
+ }
+
+ const tableCols = new Array();
+ if (data && data.length > 0) {
+ const sampleProduct = data[0];
+ const keys = Object.keys(sampleProduct);
+ for (const key of keys) {
+ const renderCell: React.FC<{ children?: React.ReactNode }> =
+ RenderCellFactory.get(sampleProduct, key);
+ const col: Column = {
+ accessor: key,
+ components: {
+ Heading: (
+
+ ),
+ renderCell: renderCell,
+ },
+ label: "",
+ name: "",
+ active: true,
+ };
+ tableCols.push(col);
+ }
+ }
+
+ const editColumn: Column = {
+ accessor: "edit",
+ components: {
+ Heading: Edit
,
+ renderCell: ({ children }) => (
+
+ ),
+ },
+ label: "Edit",
+ name: "edit",
+ active: true,
+ };
+
+ tableCols.push(editColumn);
+
+ const deleteColumn: Column = {
+ accessor: "delete",
+ components: {
+ Heading: Delete
,
+ renderCell: ({ children }) => (
+
+ ),
+ },
+ label: "Delete",
+ name: "delete",
+ active: true,
+ };
+
+ tableCols.push(deleteColumn);
+
+ const handleEdit = (promotionID: string) => {
+ console.log(`Dummy. Promotion ID: ${promotionID}`);
+ };
+
+ const handleDelete = (promotionID: string) => {
+ console.log(`Dummy. Promotion ID: ${promotionID}`);
+ };
+
+ const handleCreatePromotion = () => {
+ console.log("Creating a new promotion...");
+ };
+
+ console.log(tableCols);
+
+ return (
+
+
+
+
+
+
+
+
+
+
+
+
+ );
+};
+
+export default MerchPromotion;
\ No newline at end of file
diff --git a/apps/cms/src/admin/views/MerchSales.tsx b/apps/cms/src/admin/views/MerchSales.tsx
index 81d2d3f2..db3679a8 100644
--- a/apps/cms/src/admin/views/MerchSales.tsx
+++ b/apps/cms/src/admin/views/MerchSales.tsx
@@ -10,14 +10,15 @@ import { RenderCellFactory } from "../utils/RenderCellFactory";
import SortedColumn from "../utils/SortedColumn";
import { Table } from "payload/dist/admin/components/elements/Table";
+
const MerchSales: AdminView = ({ user, canAccessAdmin }) => {
- // Get data from API
- const [data, setData] = useState(null);
- useEffect(() => {
- OrdersApi.getOrders()
- .then((res: IOrder[]) => setData(res))
- .catch((error) => console.log(error));
- }, []);
+ // Get data from API
+ const [data, setData] = useState(null);
+ useEffect(() => {
+ OrdersApi.getOrders()
+ .then((res: IOrder[]) => setData(res))
+ .catch((error) => console.log(error));
+ }, []);
// Output human-readable table headers based on the attribute names from the API
function prettifyKey(str: string): string {
@@ -30,7 +31,7 @@ const MerchSales: AdminView = ({ user, canAccessAdmin }) => {
// Do not load table until we receive the data
if (data == null) {
- return Loading...
;
+ return Loading...
;
}
const tableCols = new Array();
@@ -92,7 +93,7 @@ const MerchSales: AdminView = ({ user, canAccessAdmin }) => {
const handleEdit = (orderId: string) => {
console.log(`Dummy. Order ID: ${orderId}`);
- };
+ }
const handleDelete = (orderId: string) => {
console.log(`Dummy. Order ID: ${orderId}`);
diff --git a/apps/cms/src/admin/views/ViewTemplate.tsx b/apps/cms/src/admin/views/ViewTemplate.tsx
index f2e6adfe..aefb58c4 100644
--- a/apps/cms/src/admin/views/ViewTemplate.tsx
+++ b/apps/cms/src/admin/views/ViewTemplate.tsx
@@ -38,7 +38,7 @@ const ViewTemplate = ({
label: title,
},
]);
- }, [setStepNav]);
+ }, [setStepNav, title]);
// If an unauthorized user tries to navigate straight to this page,
// Boot 'em out
diff --git a/apps/cms/src/apis/promotions.api.ts b/apps/cms/src/apis/promotions.api.ts
new file mode 100644
index 00000000..cc23fd83
--- /dev/null
+++ b/apps/cms/src/apis/promotions.api.ts
@@ -0,0 +1,35 @@
+import { Promotion } from "types";
+import { PromoType } from "types";
+// todo turn into real api
+class PromotionsApi {
+ async getPromotions(): Promise {
+ const res: Promotion[] = [
+ {
+ promoCode: "MARCHSALES",
+ maxRedemptions: 10,
+ redemptionsRemaining: 10,
+ discounts: {
+ promoType: PromoType.FIXED_VALUE,
+ promoValue: 5,
+ appliesTo: ["1"],
+ minimumQty: 1,
+ },
+ },
+ {
+ promoCode: "TSHIRTPROMO",
+ maxRedemptions: 10,
+ redemptionsRemaining: 10,
+ discounts: {
+ promoType: PromoType.FIXED_VALUE,
+ promoValue: 5,
+ appliesTo: ["1"],
+ minimumQty: 1,
+ },
+ },
+ ];
+
+ return Promise.resolve(res);
+ }
+}
+
+export default new PromotionsApi();
diff --git a/apps/cms/src/payload.config.ts b/apps/cms/src/payload.config.ts
index 63a5afce..85033c0b 100644
--- a/apps/cms/src/payload.config.ts
+++ b/apps/cms/src/payload.config.ts
@@ -14,6 +14,7 @@ import AfterNavLinks from "./admin/components/AfterNavLinks";
import MerchSales from "./admin/views/MerchSales";
import MerchOverview from "./admin/views/MerchOverview";
import MerchProducts from "./admin/views/MerchProducts";
+import MerchPromotion from "./admin/views/MerchPromotion";
import { SCSEIcon, SCSELogo } from "./admin/graphics/Logos";
import BeforeNavLinks from "./admin/components/BeforeNavLinks";
import Order from "./collections/Orders";
@@ -51,6 +52,10 @@ export default buildConfig({
path: "/merch/products",
Component: MerchProducts,
},
+ {
+ path: "/merch/promotions",
+ Component: MerchPromotion,
+ },
],
beforeNavLinks: BeforeNavLinks,
afterNavLinks: AfterNavLinks,
@@ -90,13 +95,13 @@ export default buildConfig({
},
plugins: isUsingCloudStore()
? [
- cloudStorage({
- collections: {
- media: {
- adapter: adapter,
+ cloudStorage({
+ collections: {
+ media: {
+ adapter: adapter,
+ },
},
- },
- }),
- ]
+ }),
+ ]
: [],
});
diff --git a/apps/cms/src/types.ts b/apps/cms/src/types.ts
index ece41843..7236123c 100644
--- a/apps/cms/src/types.ts
+++ b/apps/cms/src/types.ts
@@ -1,6 +1,7 @@
/* tslint:disable */
+/* eslint-disable */
/**
- * This file was automatically generated by Payload CMS.
+ * This file was automatically generated by Payload.
* DO NOT MODIFY IT BY HAND. Instead, modify your source Payload config,
* and re-run `payload generate:types` to regenerate this file.
*/
@@ -89,7 +90,7 @@ export interface User {
name?: string;
updatedAt: string;
createdAt: string;
- email?: string;
+ email: string;
resetPasswordToken?: string;
resetPasswordExpiration?: string;
salt?: string;
diff --git a/apps/web/features/merch/components/checkout/Skeleton.tsx b/apps/web/features/merch/components/checkout/Skeleton.tsx
index 06c3e9d6..0aa6b5c3 100644
--- a/apps/web/features/merch/components/checkout/Skeleton.tsx
+++ b/apps/web/features/merch/components/checkout/Skeleton.tsx
@@ -14,6 +14,7 @@ const ItemSkeleton = () => (
);
+
const CheckoutSkeleton: React.FC = () => {
return (
{
};
export default CheckoutSkeleton;
+
diff --git a/packages/eslint-config-custom/index.js b/packages/eslint-config-custom/index.js
index c92d6e71..813bb43c 100644
--- a/packages/eslint-config-custom/index.js
+++ b/packages/eslint-config-custom/index.js
@@ -27,6 +27,7 @@ module.exports = {
"@next/next/no-html-link-for-pages": "off",
// react
"react/jsx-key": "off",
+ "react/display-name": "off",
// cypress
"cypress/no-assigning-return-values": "error",
"cypress/no-unnecessary-waiting": "error",
diff --git a/packages/merch-helpers/src/lib/price.ts b/packages/merch-helpers/src/lib/price.ts
index 02e61f58..86651a52 100644
--- a/packages/merch-helpers/src/lib/price.ts
+++ b/packages/merch-helpers/src/lib/price.ts
@@ -66,7 +66,8 @@ export const calculatePricing = (
discountedPrice: itemPrice,
};
}
- for (const discount of promotion.discounts) {
+ const discountsArray = promotion.discounts as unknown as Promotion["discounts"][];
+ for (const discount of discountsArray) {
if (discount.appliesTo && !discount.appliesTo.includes(item.id)) {
continue;
}
diff --git a/packages/types/src/lib/merch.ts b/packages/types/src/lib/merch.ts
index ac5ddfc8..ac03f82d 100644
--- a/packages/types/src/lib/merch.ts
+++ b/packages/types/src/lib/merch.ts
@@ -72,12 +72,12 @@ export interface Promotion {
promoCode: string;
maxRedemptions: number;
redemptionsRemaining: number;
- discounts: Array<{
+ discounts: {
promoType: PromoType;
promoValue: number; // percent off or fixed value off based on promoType property
- appliesTo?: Array; // array of product ids
+ appliesTo?: string[]; // array of product ids
minimumQty?: number; // minimum quantity of items in the order to apply the discount
- }>;
+ };
}
export type PricedCart = {