Select Product Variants in Storefront

In this guide, you'll learn how to select a product variant to be added to the cart in the storefront.

Overview#

Before a customer can add a product to the cart, they have to select from the product's available options.

Then, since a variant is a combination of the product options' values (for example, size S and color Blue), you must find the product variant that matches the chosen option values.

Finally, when the customer adds the product to the cart, you must add the selected variant to the cart.

In this guide, you'll learn how to show the product options and find the selected variant in a React-based storefront.

NoteRefer to the Manage Cart's Items guide to learn how to add a product variant to the cart.

Example: React-Based Storefront#

Here's an example of a React component that shows the product options and allows the customer to select a variant:

TipLearn how to install and configure the JS SDK in the JS SDK documentation.
Code
1"use client" // include with Next.js 13+2
3import { useEffect, useMemo, useState } from "react"4import { HttpTypes } from "@medusajs/types"5import { sdk } from "@/lib/sdk"6
7type Props = {8  id: string9}10
11export default function Product({ id }: Props) {12  const [loading, setLoading] = useState(true)13  const [product, setProduct] = useState<14    HttpTypes.StoreProduct | undefined15  >()16  const [selectedOptions, setSelectedOptions] = useState<Record<string, string>>({})17
18  useEffect(() => {19    if (!loading) {20      return 21    }22
23    sdk.store.product.retrieve(id)24    .then(({ product: dataProduct }) => {25      setProduct(dataProduct)26      setLoading(false)27    })28  }, [loading])29
30  const selectedVariant = useMemo(() => {31    if (32      !product?.variants ||33      !product.options || 34      Object.keys(selectedOptions).length !== product.options?.length35    ) {36      return37    }38
39    return product.variants.find((variant) => variant.options?.every(40      (optionValue) => optionValue.id === selectedOptions[optionValue.option_id!]41    ))42  }, [selectedOptions, product])43
44  return (45    <div>46      {loading && <span>Loading...</span>}47      {product && (48        <>49          <h1>{product.title}</h1>50          {(product.options?.length || 0) > 0 && (51            <ul>52              {product.options!.map((option) => (53                <li key={option.id}>54                  {option.title}55                  {option.values?.map((optionValue) => (56                    <button 57                      key={optionValue.id}58                      onClick={() => {59                        setSelectedOptions((prev) => {60                          return {61                            ...prev,62                            [option.id!]: optionValue.value!,63                          }64                        })65                      }}66                    >67                      {optionValue.value}68                    </button>69                  ))}70                </li>71              ))}72            </ul>73          )}74          {selectedVariant && (75            <>76              <span>Selected Variant: {selectedVariant.id}</span>77              {/* TODO: Show add to cart button */}78            </>79          )}80          {product.images?.map((image) => (81            <img src={image.url} key={image.id} />82          ))}83        </>84      )}85    </div>86  )87}

In this example, you:

  • Retrieve the product details from Medusa.
  • Show the product's options and allow the customer to select an option value.
  • Store the selected options in the selectedOptions state variable. It's an object whose keys are options' ID, and values are the selected value of that option.
  • Compute the selected variable whenever the selected option is changed. When the customer chooses a value for all options, you find a product variant that has the same chosen option-value combinations.
  • Show the ID of the selected variant when it's found.

In your storefront, you should show the add-to-cart button instead of the variant ID. Refer to the Manage Cart's Items guide to learn how to add a product variant to the cart.

Was this page helpful?
Ask Anything
FAQ
What is Medusa?
How can I create a module?
How can I create a data model?
How do I create a workflow?
How can I extend a data model in the Product Module?
Recipes
How do I build a marketplace with Medusa?
How do I build digital products with Medusa?
How do I build subscription-based purchases with Medusa?
What other recipes are available in the Medusa documentation?
Chat is cleared on refresh
Line break