Documentation
Feedback
Guides
Storefront Development

Storefront Development
Storefront development

SKUMatrix

Displays a table of product variations with quantity selectors for bulk purchases.

The SKUMatrix component displays a table of product variations with quantity selectors, enabling customers to add multiple SKUs to the cart at once. This component is ideal for B2B scenarios or bulk purchasing.
The final component is a compound of the following:
  • SKUMatrix: Wraps the SKU Matrix structure that provides context and state management.
  • SKUMatrixTrigger: Renders a button that opens the SKU Matrix sidebar.
  • SKUMatrixSidebar: Renders a SlideOver containing a table with product variations, prices, availability, and quantity selectors.
Example
Code

Usage

Import the component

Import the compound from @faststore/ui.

_10
import {
_10
SKUMatrix,
_10
SKUMatrixTrigger,
_10
SKUMatrixSidebar,
_10
useSKUMatrix,
_10
} from "@faststore/ui";

Import styles

Add the sidebar styling tokens and layout rules to your theme stylesheet:

_10
@import "@faststore/ui/src/components/organisms/SKUMatrix/styles.scss";

Follow the Importing FastStore UI component styles instructions. Additionally, the quantity selector, badges, and table components rely on their respective partials.

Examples

Loading State

While fetching product variants, display skeleton loaders to improve perceived performance by setting loading={true} on SKUMatrixSidebar.
Example
Code

Full Width Sidebar

Display the matrix in a full-width slide-over for complex product catalogs with many columns by setting size="full".
Example
Code

Left Side Direction

Open the sidebar from the left side of the screen by setting direction="leftSide".
Example
Code

Design tokens

Local tokenDefault value/Global token linked
--fs-sku-matrix-sidebar-bkg-color
var(--fs-color-body-bkg)
--fs-sku-matrix-sidebar-title-sizevar(--fs-text-size-6)
--fs-sku-matrix-sidebar-title-text-weightvar(--fs-text-weight-semibold)
--fs-sku-matrix-sidebar-table-cell-font-sizevar(--fs-text-size-tiny)
--fs-sku-matrix-sidebar-table-cell-text-weightvar(--fs-text-weight-medium)
--fs-sku-matrix-sidebar-table-cell-image-widthvar(--fs-spacing-7)
--fs-sku-matrix-sidebar-table-cell-image-border-radiusvar(--fs-border-radius)
--fs-sku-matrix-slide-over-partial-gapcalc(2 * var(--fs-grid-padding))
--fs-sku-matrix-slide-over-partial-width-mobilecalc(100vw - var(--fs-sku-matrix-slide-over-partial-gap))

Data attributes

Use these selectors to customize styles or attach test logic:
  • data-fs-sku-matrix
  • data-testid="fs-sku-matrix"
  • data-fs-sku-matrix-sidebar
  • data-fs-sku-matrix-sidebar-title
  • data-fs-sku-matrix-sidebar-cell-image
  • data-fs-sku-matrix-sidebar-table-price
  • data-fs-sku-matrix-sidebar-table-cell-quantity-selector
  • data-fs-sku-matrix-sidebar-table-action
  • data-fs-sku-matrix-sidebar-footer
  • SlideOver attributes propagated to the sidebar: data-fs-slide-over, data-fs-slide-over-direction, data-fs-slide-over-size, data-fs-slide-over-state

Props

NameTypeDescriptionDefault
testIdstringID to find this component in testing tools (e.g.: cypress, testing library, and jest).fs-sku-matrix
NameTypeDescriptionDefault
testIdstringID to find this component in testing tools (e.g.: cypress, testing library, and jest).
variant"primary" | "secondary" | "tertiary"Specifies the component color variant.secondary
size"small" | "regular"Specifies the size variant.
inversefalse | trueDefines the use of inverted colors.
disabledfalse | trueSpecifies that this button should be disabled.
iconstring | number | false | true | {} | ReactElement<any, string | JSXElementConstructor<any>> | Iterable<ReactNode> | ReactPortalA React component that will be rendered as an icon.
loadingfalse | trueBoolean that represents a loading state.
loadingLabelstringSpecifies a label for loading state.
iconPosition"left" | "right"Specifies where the icon should be positioned
NameTypeDescriptionDefault
titlestringTitle for the SKUMatrixSidebar component.
columns*VariationProductColumnRepresents the variations products to building the table.
buyProps*{ 'data-testid': string; 'data-sku': string; 'data-seller': string; onClick(e: React.MouseEvent<HTMLButtonElement>): void; }Properties related to the 'add to cart' button
formatterPriceFormatterFormatter function that transforms the raw price value and render the result.
loadingfalse | trueCheck if some result is still loading before render the result.
ImageComponentFunctionComponent<{ src: string; alt: string; width?: number; height?: number; }>Function that returns a React component that will be used to render images.({ src, alt, ...otherProps }) => <img src={src} alt={alt} {...otherProps} />
aria-labelledbystringIdentifies the element (or elements) that labels the current element. @see aria-labelledby https://www.w3.org/TR/wai-aria-1.1/#aria-labelledby
children*string | number | false | true | {} | ReactElement<any, string | JSXElementConstructor<any>> | Iterable<ReactNode> | ReactPortalChildren or function as a children.
testIdstringID to find this component in testing tools (e.g.: cypress, testing library, and jest).
onEntered() => voidCallback function when the modal is opened.
onDismiss() => voidThis function is called whenever the user clicks outside. the modal content
overlayPropsPropsProps forwarded to the `Overlay` component.
disableEscapeKeyDownfalse | trueDisable being closed using the Escape key.
direction"leftSide" | "rightSide"Represents the side that the SlideOver comes from.rightSide
size"full" | "partial"Represents the size of the SlideOver.partial

Best practices

✅ Dos

  • Use concise variation labels (example: "Black - 42") to ensure the table remains scannable on partial-width slide-overs.
  • Keep price and availability in sync with your store OMS; display a toast message when quantity adjustments are coerced.
  • Localize both column headers and toast messages for multinational catalogs.

❌ Don'ts

  • Don't expose the matrix when a product is single-SKU; it adds unnecessary friction.
  • Don't rely on color alone for stock communication; pair badge colors with descriptive text.
  • Don't allow quantities beyond available stock; the component exposes validation hooks for this reason.
  • Don't hardcode dimensions; merchants should be able to adjust columns per category via Headless CMS.

Contributors
2
Photo of the contributor
Photo of the contributor
Was this helpful?
Yes
No
Suggest Edits (GitHub)
Contributors
2
Photo of the contributor
Photo of the contributor
Was this helpful?
Suggest edits (GitHub)
On this page