A deep dive into creating a production-ready e-commerce solution using Next.js 15, TypeScript, and modern web development practices
Website: https://shopfake.joetohdjojo.com
Github Repository: https://github.com/joe-tohdjojo/shop-fake
Welcome to a comprehensive exploration of how I built SHPFK (Shop Fake), a modern e-commerce platform that prioritizes performance, accessibility, and developer experience. In this post, I'll walk you through the technical decisions, architecture patterns, and implementation details that make this project stand out.
Our foundation is built on cutting-edge technologies that provide a robust development experience:
One of the most critical aspects of any e-commerce platform is user authentication. Here's how we've implemented a secure and seamless authentication system:
We faced several interesting challenges with cookie management, particularly when working with DummyJSON's API:
Domain
and SameSite
)Domain
attributesSameSite
for securityhttpOnly
and secure
flagsThe homepage showcases how we leverage Next.js 15's Server Components for optimal performance:
Server Component that fetches and renders featured products. This is done with Suspense
boundary for better UX and the data is cached using Next.js's cache: 'force-cache'
feature.
Server Component that fetches and renders a preview of all categories. This is done with Suspense
boundary for better UX and the data is not only cached using Next.js's cache: 'force-cache'
feature, but also deduplicated to ensure only one fetch call happens.
I've implemented several patterns to optimize data fetching and state management:
One of our most valuable abstractions is the WithServerData
component:
<Suspense fallback={<div className="w-full space-y-6">Loading filter panel...</div>}>
<WithServerData fetchFunction={fetchProductCategories}>
{({ data, error }) => {
if (error || !data) return null;
return <FilterPanel.Desktop categories={data} />;
}}
</WithServerData>
</Suspense>
This pattern allows us to:
The cart system demonstrates advanced state management techniques:
export function AddToCartButton({
productId,
productTitle,
stock,
}: {
productId: number;
productTitle: string;
stock: number;
}) {
const router = useRouter();
const { toast } = useToast();
const queryClient = useQueryClient();
const { data: user } = useUser();
const { data: cart } = useCart();
const { mutate } = useMutation({
mutationKey: ['addToCart', productId],
mutationFn: () => {
if (!cart) {
return addCart({
products: [{ id: productId, quantity: 1 }],
userId: user?.id,
});
}
let productAdded = false;
const products = cart.products.map(
(product: Product & { quantity: number }) => {
if (product.id === productId) productAdded = true;
return {
id: product.id,
quantity: productAdded ? product.quantity + 1 : product.quantity,
};
},
);
if (!productAdded) products.push({ id: productId, quantity: 1 });
return addCart({
userId: user?.id,
products,
});
},
onSuccess: (data) => {
if (!user || !data) return;
queryClient.setQueryData(['cart'], (oldData: Cart) => {
const newData = { ...data };
newData.products = [
...((oldData || {}).products || []),
data.products[data.products.length - 1],
];
return newData;
});
toast({
title: `${productTitle} has been added to your cart`,
});
},
});
const handleClick = () => {
if (!user)
return router.push(`${ROUTES.LOGIN.path}?redirect=${window.location}`);
mutate();
};
return (
<Button
className="w-full"
disabled={stock === 0}
onClick={handleClick}
>
{stock === 0 ? 'Out of Stock' : 'Add to Cart'}
</Button>
);
}
The shop page showcases several advanced patterns:
Throughout this project, several key principles emerged:
While the current implementation serves its purpose well, here are some areas for future enhancement:
Building SHPFK has been an exercise in applying modern web development practices to create a performant, maintainable e-commerce platform. The combination of Next.js 15's powerful features with carefully chosen patterns and abstractions has resulted in a codebase that's both powerful and pleasant to work with.
The source code is available on GitHub, and I welcome any feedback or contributions from the community.
Remember: while this implementation uses specific tools like Next.js and React Query, the patterns and principles discussed here can be applied to any modern web application.