useProductOptions
The useProductOptions hook returns an object that enables you to keep track of the
selected variant and/or selling plan state, as well as callbacks for modifying the state. The useProductOptions hook must be a child of the ProductOptionsProvider component.
Example code
/**
 * Iterate through a list of variants and allow the customer to select a specific variant.
 */
import {useProductOptions} from '@shopify/hydrogen';
export function MyComponent() {
  const {variants, selectedVariant, setSelectedVariant} = useProductOptions();
  return (
    <>
      <label htmlFor="variants">Select a variant</label>
      <select
        id="variants"
        name="variants"
        value={selectedVariant?.id}
        onChange={(e) =>
          setSelectedVariant(
            variants.find((variant) => variant.id === e.target.value)
          )
        }
      >
        {variants.map((variant) => (
          <option key={variant.id} value={variant.id}>
            {variant.title}
          </option>
        ))}
      </select>
    </>
  );
}
/**
 * Support selling plans. You should display a selling plan selector to a user
 * when a product has selling plans enabled. You need to pass `sellingPlanGroups` to the hook.
 */
import {useProductOptions} from '@shopify/hydrogen';
export function MyComponent() {
  const {
    selectedSellingPlan,
    setSelectedSellingPlan,
    selectedSellingPlanAllocation,
    sellingPlanGroups,
  } = useProductOptions({
    variants: product.variants,
    sellingPlanGroups: product.sellingPlanGroups,
    initialVariantId: product.variants.edges[0].node.id,
  });
  return (
    <>
      {/* Code for your variant selector goes here */}
      {sellingPlanGroups.map((sellingPlanGroup) => (
        <div key={sellingPlanGroup.id}>
          <h2>{sellingPlanGroup.name}</h2>
          <ul>
            {sellingPlanGroup.sellingPlans.map((sellingPlan) => {
              return (
                <li key={sellingPlan.id}>
                  <button onClick={() => setSelectedSellingPlan(sellingPlan)}>
                    {sellingPlan.name}
                  </button>
                </li>
              );
            })}
          </ul>
        </div>
      ))}
    </>
  );
}
/**
 * Use product options.
 */
import {useProductOptions} from '@shopify/hydrogen';
export function MyComponent() {
  const {options, selectedVariant, selectedOptions, setSelectedOption} =
    useProductOptions({variants: product.variants});
  return (
    <>
      {options.map(({name, values}) => (
        <fieldset key={name}>
          <legend>{name}</legend>
          {values.map((value) => (
            <label htmlFor={`option[${name}][${value}]`}>
              <input
                type="radio"
                id={`option[${name}][${value}]`}
                name={`option[${name}]`}
                checked={selectedOptions[name] === value}
                onChange={() => setSelectedOption(name, value)}
              />
              {value}
            </label>
          ))}
        </fieldset>
      ))}
    </>
  );
}
Considerations
- To make sure you have all the data necessary for the useProductOptionshook, refer to the Storefront API's ProductVariant object.
- If your product requires a selling plan, then make sure to display that as required in your user interface. If it doesn't require a selling plan, then you might want to offer a "one-time purchase" option which triggers setSelectedSellingPlan(null).
- You can use selectedSellingPlanAllocationto display the price adjustments for the selected variant when a given selling plan is active.
- You can manually deselect a variant by calling setSelectedVariant(null).
Arguments
This hook takes a single object with the following keys:
| Key | Type | Description | 
|---|---|---|
| variants? | PartialDeep<ProductVariantConnection> | The product's VariantConnection. | 
| sellingPlanGroups? | PartialDeep<SellingPlanGroupConnection> | The product's SellingPlanGroups. | 
| initialVariantId? | ProductVariantType['id'] | The initially selected variant. | 
Return value
This hook returns a single object with the following keys:
| Key | Description | 
|---|---|
| variants | An array of the variant nodesfrom theVariantConnection. | 
| options | An array of the product's options and values. | 
| selectedVariant | The selected variant. | 
| setSelectedVariant | A callback to set the selected variant to the variant passed as an argument. | 
| selectedOptions | The current selected options. | 
| setSelectedOption | A callback to set the selected option. | 
| setSelectedOptions | A callback to set multiple selected options at once. | 
| isOptionInStock | A callback that returns a boolean indicating if the option is in stock. | 
| setSelectedSellingPlan | A callback to set the selected selling plan to the one passed as an argument. | 
| selectedSellingPlan | The selected selling plan. | 
| selectedSellingPlanAllocation | The selected selling plan allocation. | 
| sellingPlanGroups | The selling plan groups. | 
Variables
The Product object includes variables that you will need to provide values for when performing your query.
| Variable | Description | 
|---|---|
| $numProductVariantMetafields | The number of Metafieldobjects to query for in a variant'sMetafieldConnection. | 
| $numProductVariantSellingPlanAllocations | The number of SellingPlanAllocationsto query for in a variant'sSellingPlanAllocationConnection. | 
| $numProductSellingPlanGroups | The number of SellingPlanGroupsobjects to query for in aSellingPlanGroupConnection. | 
| $$numProductSellingPlans | The number of SellingPlanobjects to query for in aSellingPlanConnection. |