From d7dbd7a04fc8cb87cf223fb5a17af8d59c6431ea Mon Sep 17 00:00:00 2001
From: Chancellor Clark
Date: Tue, 10 Dec 2024 12:25:49 -0500
Subject: [PATCH] feat(core): use toasts for address deletions (#1728)
---
.changeset/quiet-sloths-beg.md | 5 +
.../addresses/_actions/delete-address.ts | 34 +++----
.../addresses/_components/address-book.tsx | 96 ++++++++-----------
.../[slug]/_components/edit-address-form.tsx | 4 +-
.../(default)/account/addresses/page-data.ts | 5 +-
.../(default)/account/addresses/page.tsx | 4 +-
core/client/tags.ts | 1 +
core/tests/ui/e2e/account.spec.ts | 14 +--
core/tests/ui/e2e/addresses.spec.ts | 4 +-
9 files changed, 79 insertions(+), 88 deletions(-)
create mode 100644 .changeset/quiet-sloths-beg.md
diff --git a/.changeset/quiet-sloths-beg.md b/.changeset/quiet-sloths-beg.md
new file mode 100644
index 0000000000..abc03295cb
--- /dev/null
+++ b/.changeset/quiet-sloths-beg.md
@@ -0,0 +1,5 @@
+---
+"@bigcommerce/catalyst-core": minor
+---
+
+Convert the messages that were displayed when deleting an address over to using the toast functionality.
diff --git a/core/app/[locale]/(default)/account/addresses/_actions/delete-address.ts b/core/app/[locale]/(default)/account/addresses/_actions/delete-address.ts
index ec5f3cf241..d3b2941f99 100644
--- a/core/app/[locale]/(default)/account/addresses/_actions/delete-address.ts
+++ b/core/app/[locale]/(default)/account/addresses/_actions/delete-address.ts
@@ -1,13 +1,12 @@
'use server';
-import { revalidatePath } from 'next/cache';
+import { revalidateTag } from 'next/cache';
import { getTranslations } from 'next-intl/server';
import { getSessionCustomerAccessToken } from '~/auth';
import { client } from '~/client';
import { graphql } from '~/client/graphql';
-
-import { State } from '../../settings/change-password/_actions/change-password';
+import { TAGS } from '~/client/tags';
const DeleteCustomerAddressMutation = graphql(`
mutation DeleteCustomerAddressMutation(
@@ -19,11 +18,9 @@ const DeleteCustomerAddressMutation = graphql(`
errors {
__typename
... on CustomerAddressDeletionError {
- __typename
message
}
... on CustomerNotLoggedInError {
- __typename
message
}
}
@@ -32,7 +29,12 @@ const DeleteCustomerAddressMutation = graphql(`
}
`);
-export const deleteAddress = async (addressId: number): Promise => {
+interface DeleteAddressResponse {
+ status: 'success' | 'error';
+ message: string;
+}
+
+export const deleteAddress = async (addressId: number): Promise => {
const t = await getTranslations('Account.Addresses.Delete');
const customerAccessToken = await getSessionCustomerAccessToken();
@@ -50,24 +52,24 @@ export const deleteAddress = async (addressId: number): Promise => {
const result = response.data.customer.deleteCustomerAddress;
- revalidatePath('/account/addresses', 'page');
-
- if (result.errors.length === 0) {
- return { status: 'success', messages: [t('success')] };
+ if (result.errors.length > 0) {
+ result.errors.forEach((error) => {
+ // Throw the first error message, as we should only handle one error at a time
+ throw new Error(error.message);
+ });
}
- return {
- status: 'error',
- messages: result.errors.map((error) => error.message),
- };
+ revalidateTag(TAGS.customer);
+
+ return { status: 'success', message: t('success') };
} catch (error: unknown) {
if (error instanceof Error) {
return {
status: 'error',
- messages: [error.message],
+ message: error.message,
};
}
- return { status: 'error', messages: [t('error')] };
+ return { status: 'error', message: t('error') };
}
};
diff --git a/core/app/[locale]/(default)/account/addresses/_components/address-book.tsx b/core/app/[locale]/(default)/account/addresses/_components/address-book.tsx
index 22ff3ba679..2af1a60832 100644
--- a/core/app/[locale]/(default)/account/addresses/_components/address-book.tsx
+++ b/core/app/[locale]/(default)/account/addresses/_components/address-book.tsx
@@ -1,88 +1,51 @@
'use client';
+import { AlertCircle, Check } from 'lucide-react';
import { useTranslations } from 'next-intl';
-import { PropsWithChildren, useEffect } from 'react';
+import { PropsWithChildren } from 'react';
+import { toast } from 'react-hot-toast';
import { ExistingResultType } from '~/client/util';
import { Link } from '~/components/link';
import { Button } from '~/components/ui/button';
-import { Message } from '~/components/ui/message';
-import { useRouter } from '~/i18n/routing';
-import { useAccountStatusContext } from '../../_components/account-status-provider';
import { Modal } from '../../_components/modal';
-import { SubmitMessagesList } from '../../_components/submit-messages-list';
import { deleteAddress } from '../_actions/delete-address';
import { getCustomerAddresses } from '../page-data';
export type Addresses = ExistingResultType['addresses'];
-interface AddressChangeProps {
- addressId: number;
- isAddressRemovable: boolean;
-}
-
-const AddressChangeButtons = ({ addressId, isAddressRemovable }: AddressChangeProps) => {
- const { setAccountState } = useAccountStatusContext();
- const t = useTranslations('Account.Addresses');
-
- const handleDeleteAddress = async () => {
- const submit = await deleteAddress(addressId);
-
- if (submit.status === 'success') {
- setAccountState({
- status: 'success',
- messages: submit.messages || [''],
- });
- }
- };
-
- return (
-
-
-
-
-
-
- );
-};
-
interface AddressBookProps {
customerAddresses: Addresses;
- addressesCount: number;
+ totalAddresses: number;
}
export const AddressBook = ({
children,
- addressesCount,
+ totalAddresses,
customerAddresses,
}: PropsWithChildren) => {
const t = useTranslations('Account.Addresses');
- const { accountState } = useAccountStatusContext();
- const router = useRouter();
- useEffect(() => {
- if (customerAddresses.length === 0) {
- router.push(`/account/addresses/`);
+ const handleDeleteAddress = (addressId: number) => async () => {
+ const { status, message } = await deleteAddress(addressId);
+
+ if (status === 'error') {
+ toast.error(message, {
+ icon: ,
+ });
+
+ return;
}
- }, [customerAddresses, router]);
+
+ toast.success(message, {
+ icon: ,
+ });
+ };
return (
<>
- {(accountState.status === 'error' || accountState.status === 'success') && (
-
-
-
- )}
- {!addressesCount && {t('emptyAddresses')}
}
+ {totalAddresses === 0 && {t('emptyAddresses')}
}
{customerAddresses.map(
({
@@ -111,7 +74,24 @@ export const AddressBook = ({
{countryCode}
- 1} />
+
+
+
+
+
+
),
)}
diff --git a/core/app/[locale]/(default)/account/addresses/edit/[slug]/_components/edit-address-form.tsx b/core/app/[locale]/(default)/account/addresses/edit/[slug]/_components/edit-address-form.tsx
index e88a6e8a0b..42ec7557ab 100644
--- a/core/app/[locale]/(default)/account/addresses/edit/[slug]/_components/edit-address-form.tsx
+++ b/core/app/[locale]/(default)/account/addresses/edit/[slug]/_components/edit-address-form.tsx
@@ -232,11 +232,11 @@ export const EditAddressForm = ({
const submit = await deleteAddress(address.entityId);
if (submit.status === 'success') {
- setAccountState({ status: submit.status, messages: submit.messages || [''] });
+ setAccountState({ status: submit.status, messages: [submit.message] });
}
if (submit.status === 'error') {
- setAccountState({ status: submit.status, messages: submit.messages || [''] });
+ setAccountState({ status: submit.status, messages: [submit.message] });
}
router.push('/account/addresses');
diff --git a/core/app/[locale]/(default)/account/addresses/page-data.ts b/core/app/[locale]/(default)/account/addresses/page-data.ts
index 1a635e1fb2..e61e09a5dd 100644
--- a/core/app/[locale]/(default)/account/addresses/page-data.ts
+++ b/core/app/[locale]/(default)/account/addresses/page-data.ts
@@ -6,6 +6,7 @@ import { client } from '~/client';
import { FormFieldValuesFragment } from '~/client/fragments/form-fields-values';
import { PaginationFragment } from '~/client/fragments/pagination';
import { graphql } from '~/client/graphql';
+import { TAGS } from '~/client/tags';
const GetCustomerAddressesQuery = graphql(
`
@@ -59,7 +60,7 @@ export const getCustomerAddresses = cache(
document: GetCustomerAddressesQuery,
variables: { ...paginationArgs },
customerAccessToken,
- fetchOptions: { cache: 'no-store' },
+ fetchOptions: { cache: 'no-store', next: { tags: [TAGS.customer] } },
});
const addresses = response.data.customer?.addresses;
@@ -70,7 +71,7 @@ export const getCustomerAddresses = cache(
return {
pageInfo: addresses.pageInfo,
- addressesCount: addresses.collectionInfo?.totalItems ?? 0,
+ totalAddresses: addresses.collectionInfo?.totalItems ?? 0,
addresses: removeEdgesAndNodes({ edges: addresses.edges }),
};
},
diff --git a/core/app/[locale]/(default)/account/addresses/page.tsx b/core/app/[locale]/(default)/account/addresses/page.tsx
index a6384f368e..27e9b76fa2 100644
--- a/core/app/[locale]/(default)/account/addresses/page.tsx
+++ b/core/app/[locale]/(default)/account/addresses/page.tsx
@@ -36,13 +36,13 @@ export default async function Addresses({ searchParams }: Props) {
notFound();
}
- const { addresses, pageInfo, addressesCount } = data;
+ const { addresses, pageInfo, totalAddresses } = data;
const { hasNextPage, hasPreviousPage, startCursor, endCursor } = pageInfo;
return (
<>
-
+
// More info: https://github.com/amannn/next-intl/issues/1335
await expect(page).toHaveURL('/account/');
- await expect(page.getByRole('link', { name: 'Orders' })).toBeVisible();
- await expect(page.getByRole('link', { name: 'Addresses' })).toBeVisible();
- await expect(page.getByRole('link', { name: 'Account settings' })).toBeVisible();
+ const tabs = page.getByRole('navigation', { name: 'Account Tabs' });
- await page.getByRole('link', { name: 'Orders' }).click();
+ await expect(tabs.getByRole('link', { name: 'Orders' })).toBeVisible();
+ await expect(tabs.getByRole('link', { name: 'Addresses' })).toBeVisible();
+ await expect(tabs.getByRole('link', { name: 'Account settings' })).toBeVisible();
+
+ await tabs.getByRole('link', { name: 'Orders' }).click();
await expect(page).toHaveURL('/account/orders/');
await expect(page.getByRole('heading', { name: 'Orders' })).toBeVisible();
- await page.getByRole('link', { name: 'Addresses' }).click();
+ await tabs.getByRole('link', { name: 'Addresses' }).click();
await expect(page).toHaveURL('/account/addresses/');
await expect(page.getByRole('heading', { name: 'Addresses' })).toBeVisible();
- await page.getByRole('link', { name: 'Account settings' }).click();
+ await tabs.getByRole('link', { name: 'Account settings' }).click();
await expect(page).toHaveURL('/account/settings/');
await expect(page.getByRole('heading', { name: 'Account settings' })).toBeVisible();
diff --git a/core/tests/ui/e2e/addresses.spec.ts b/core/tests/ui/e2e/addresses.spec.ts
index 4463cd1d79..69f0e04e32 100644
--- a/core/tests/ui/e2e/addresses.spec.ts
+++ b/core/tests/ui/e2e/addresses.spec.ts
@@ -31,7 +31,7 @@ test('Add new address', async ({ page, account }) => {
await page.getByRole('button', { name: 'Add new address' }).click();
await page.waitForURL('/account/addresses/');
- await expect(page.getByText('Address added to your account.')).toBeVisible();
+ // await expect(page.getByText('Address added to your account.')).toBeVisible();
await customer.logout();
});
@@ -47,7 +47,7 @@ test('Edit address', async ({ page, account }) => {
await page.getByRole('button', { name: 'Edit address' }).click();
await page.waitForURL('/account/addresses/');
- await expect(page.getByText('The address has been successfully updated.')).toBeVisible();
+ // await expect(page.getByText('The address has been successfully updated.')).toBeVisible();
await customer.logout();
});