Leo's dev blog

Shopify Hydrogen: build a fully functional price range filter component

Published on
/8 mins read/---

A price range filter component is a common feature in e-commerce websites. It allows users to filter products by price range, and often be used in collection pages.

Hydrogen price range filter react component

In this article, I will show you how to build a fully functional price range filter component for a Shopify Hydrogen storefront.

Querying products and parsing price filters

In your collection route loader function, you will need to query the products and parse the price filters from the query parameters.

Get all filters from query parameters

All filters are stored in the request query parameters, and should have the same filter prefix to differentiate them from other params.

($locale).collection.$collectionHandle.tsx
import type { LoaderFunctionArgs } from "@shopify/remix-oxygen"
import type { ProductFilter } from "@shopify/hydrogen/storefront-api-types"
 
const FILTER_URL_PREFIX = "filter."
 
export async function loader({ request }: LoaderFunctionArgs) {
  let { searchParams } = new URL(request.url)
  let filters = [...searchParams.entries()].reduce((ft, [k, v]) => {
    if (k.startsWith(FILTER_URL_PREFIX)) {
      ft.push({
        [k.substring(FILTER_URL_PREFIX.length)]: JSON.parse(v),
      })
    }
    return filters
  }, [] as ProductFilter[])
 
  // ...
}

Query products with parsed filters

After parsing the filters, we will pass them to the collection query to get the products.

($locale).collection.$collectionHandle.tsx
import type { LoaderFunctionArgs } from "@shopify/remix-oxygen"
import type { CollectionQuery } from "storefront-api.generated"
 
export async function loader({ params, context }: LoaderFunctionArgs) {
  let { collection, collections } = await context.storefront
    .query<CollectionQuery>(COLLECTION_QUERY, {
      variables: {
        filters,
        handle: params.collectionHandle,
        // more variables including pagination, sort, language, country
      },
    })
    .catch(() => {
      return { collection: null, collections: [] }
    })
}

The COLLECTION_QUERY should also include the lowest and highest price products to be used for the price range filter.

($locale).collection.$collectionHandle.tsx
const COLLECTION_QUERY = `#graphql
  query collection(
    $handle: String!
    $filters: [ProductFilter!]
    // ... more variables
  ) {
    collection(handle: $handle) {
      id
      handle
      title
      products(
        filters: $filters,
        // ... more variables
      ) {
        filters {
          id
          label
          type
          values {
            id
            label
            count
            input
          }
        }
        // ... more fields
      }
      highestPriceProduct: products(first: 1, sortKey: PRICE, reverse: true) {
        nodes {
          priceRange {
            minVariantPrice {
              amount
              currencyCode
            }
            maxVariantPrice {
              amount
              currencyCode
            }
          }
          // ... more fields
        }
      }
      lowestPriceProduct: products(first: 1, sortKey: PRICE) {
        nodes {
          priceRange {
            minVariantPrice {
              amount
              currencyCode
            }
            maxVariantPrice {
              amount
              currencyCode
            }
          }
          // ... more fields
        }
      }
    }
    // ... more fields
  }
` as const

Extracting applied filters and prettifying the price filter label

Some filters are not having any values, so we need process the params to extract the applied filters and prettify the price filter label.

($locale).collection.$collectionHandle.tsx
import type { I18nBase } from '@shopify/hydrogen'
import type { ProductFilter, CurrencyCode } from "@shopify/hydrogen/storefront-api-types"
 
let allFilterValues = collection.products.filters.flatMap(
  (filter) => filter.values,
)
 
let appliedFilters = filters
  .map((filter) => {
    let foundValue = allFilterValues.find((value) => {
      let valueInput = JSON.parse(value.input as string) as ProductFilter
      // special case for price, the user can enter something freeform (still a number, though)
      // that may not make sense for the locale/currency.
      // Basically just check if the price filter is applied at all.
      if (valueInput.price && filter.price) {
        return true
      }
      return (
        // This comparison should be okay as long as we're not manipulating the input we
        // get from the API before using it as a URL param.
        JSON.stringify(valueInput) === JSON.stringify(filter)
      )
    })
    if (!foundValue) {
      console.error("Could not find filter value for filter", filter)
      return null
    }
 
    if (foundValue.id === "filter.v.price") {
      // Special case for price, we want to show the min and max values as the label.
      let input = JSON.parse(foundValue.input as string) as ProductFilter
      let min = parseAsCurrency(input.price?.min ?? 0, locale)
      let max = input.price?.max
        ? parseAsCurrency(input.price.max, locale)
        : ""
      let label = min && max ? `${min} - ${max}` : "Price"
      return { filter, label, }
    }
    return { filter, label: foundValue.label }
  })
  .filter((filter): filter is NonNullable<typeof filter> => filter !== null)
 
// The helper function to parse the price as currency.
function parseAsCurrency(
  value: number,
  locale: I18nBase & {
    label: string
    currency: CurrencyCode
    pathPrefix: string
}) {
  return new Intl.NumberFormat(`${locale.language}-${locale.country}`, {
    style: "currency",
    currency: locale.currency,
  }).format(value)
}

That's all the data needed for the price range filter component. Send them to the client by returning in the end of the loader function.

($locale).collection.$collectionHandle.tsx
import type { LoaderFunctionArgs } from "@shopify/remix-oxygen"
 
export async function loader({ params, context }: LoaderFunctionArgs) {
  // ... loader function logics
  return json({
    collection,
    appliedFilters,
    // ... more data
  })
}

Building the price range filter component

Since Hydrogen is built on top of Remix, this component will simply be a React component.

I will use Radix UI for the slider component, and Tailwind CSS for the styling.

price-range-filter.tsx
import * as Slider from "@radix-ui/react-slider"
import { useLocation, useNavigate, useSearchParams } from "@remix-run/react"
import clsx from "clsx"
import { useRef, useState } from "react"
import type { CollectionQuery } from "storefront-api.generated"
 
const FILTER_URL_PREFIX = "filter."
 
export function PriceRangeFilter({
  collection,
}: {
  collection: CollectionQuery["collection"]
}) {
  let [params] = useSearchParams()
  let location = useLocation()
  let navigate = useNavigate()
  let thumbRef = useRef<"from" | "to" | null>(null)
 
  let { minVariantPrice, maxVariantPrice } = getPricesRange(collection)
  let { min, max } = getPricesFromFilter(params)
 
  let [minPrice, setMinPrice] = useState(min)
  let [maxPrice, setMaxPrice] = useState(max)
 
  function handleFilter() {
    let paramsClone = new URLSearchParams(params)
    if (minPrice === undefined && maxPrice === undefined) {
      // Remove the price filter from the URL if both min and max are undefined.
      paramsClone.delete(`${FILTER_URL_PREFIX}price`)
    } else {
      // Safely update the price filter.
      let price = {
        ...(minPrice === undefined ? {} : { min: minPrice }),
        ...(maxPrice === undefined ? {} : { max: maxPrice }),
      }
      paramsClone = filterInputToParams({ price }, paramsClone)
    }
    if (params.toString() !== paramsClone.toString()) {
      // Navigate to the new URL with the updated filters.
      navigate(`${location.pathname}?${paramsClone.toString()}`, {
        preventScrollReset: true,
      })
    }
  }
 
  return (
    <Slider.Root
      min={minVariantPrice}
      max={maxVariantPrice}
      step={1}
      minStepsBetweenThumbs={1}
      value={[minPrice || minVariantPrice, maxPrice || maxVariantPrice]}
      onValueChange={([newMin, newMax]) => {
        if (thumbRef.current) {
          // Prevent the min thumb from being dragged to the right of the max thumb.
          // And prevent the max thumb from being dragged to the left of the min thumb.
          if (thumbRef.current === "from") {
            if (maxPrice === undefined || newMin < maxPrice) {
              setMinPrice(newMin)
            }
          } else {
            if (minPrice === undefined || newMax > minPrice) {
              setMaxPrice(newMax)
            }
          }
        } else {
          setMinPrice(newMin)
          setMaxPrice(newMax)
        }
      }}
      onValueCommit={handleFilter}
      className="relative flex h-4 w-full items-center"
    >
      <Slider.Track className="relative h-1 grow rounded-full bg-gray-200">
        <Slider.Range className="absolute h-full rounded-full bg-gray-800" />
      </Slider.Track>
      {["from", "to"].map((s: "from" | "to") => (
        <Slider.Thumb
          key={s}
          onPointerUp={() => (thumbRef.current = null)}
          onPointerDown={() => (thumbRef.current = s)}
          className={clsx(
            "block h-4 w-4 bg-gray-800 cursor-grab rounded-full shadow-md",
            "focus-visible:outline-none",
          )}
        />
      ))}
    </Slider.Root>
  )
}

The utils needed for the price range filter component are:

price-range-filter.tsx
import type { ProductFilter } from "@shopify/hydrogen/storefront-api-types"
import type { CollectionQuery } from "storefront-api.generated"
 
function filterInputToParams(
  input: string | ProductFilter,
  params: URLSearchParams,
) {
  let filter =
    typeof input === "string" ? (JSON.parse(input) as ProductFilter) : input
 
  for (let [k, v] of Object.entries(filter)) {
    let key = `${FILTER_URL_PREFIX}${k}`
    let value = JSON.stringify(v)
    if (params.has(key, value)) {
      return params
    }
    if (k === "price") {
      // For price, we want to overwrite
      params.set(key, value)
    } else {
      params.append(key, value)
    }
  }
 
  return params
}
 
function getPricesRange(collection: CollectionQuery["collection"]) {
  let { highestPriceProduct, lowestPriceProduct } = collection
  let minVariantPrice =
    lowestPriceProduct.nodes[0]?.priceRange?.minVariantPrice
  let maxVariantPrice =
    highestPriceProduct.nodes[0]?.priceRange?.maxVariantPrice
  return {
    minVariantPrice: Number(minVariantPrice?.amount) || 0,
    maxVariantPrice: Number(maxVariantPrice?.amount) || 1000,
  }
}
 
function getPricesFromFilter(params: URLSearchParams) {
  let priceFilter = params.get(`${FILTER_URL_PREFIX}price`)
  let price = priceFilter
    ? (JSON.parse(priceFilter) as ProductFilter["price"])
    : undefined
  let min = Number.isNaN(Number(price?.min)) ? undefined : Number(price?.min)
  let max = Number.isNaN(Number(price?.max)) ? undefined : Number(price?.max)
  return { min, max }
}

You might include 2 inputs for the min and max price to make it more user-friendly. The 2 inputs values will be synced with the slider.

price-range-filter.tsx
export function PriceRangeFilter({
  collection,
}: {
  collection: CollectionQuery["collection"]
}) {
  return (
    <div className="space-y-4">
      <Slider.Root>
        // Slider component
      </Slider.Root>
      <div className="flex items-center gap-4">
        <div className="flex items-center gap-1 px-4 border border-line-subtle bg-gray-50 shrink">
          <VisuallyHidden.Root asChild>
            <label htmlFor="minPrice" aria-label="Min price">
              Min price
            </label>
          </VisuallyHidden.Root>
          <span>$</span>
          <input
            name="minPrice"
            type="number"
            value={minPrice ?? ""}
            min={minVariantPrice}
            placeholder={minVariantPrice.toString()}
            onChange={(e) => {
              let { value } = e.target;
              let newMinPrice = Number.isNaN(Number.parseFloat(value))
                ? undefined
                : Number.parseFloat(value);
              setMinPrice(newMinPrice);
            }}
            onBlur={handleFilter}
            className="text-right focus-visible:outline-none py-3 bg-transparent w-full"
          />
        </div>
        <span>To</span>
        <div className="flex items-center gap-1 px-4 border border-line-subtle bg-gray-50">
          <VisuallyHidden.Root asChild>
            <label htmlFor="maxPrice" aria-label="Max price">
              Max price
            </label>
          </VisuallyHidden.Root>
          <span>$</span>
          <input
            name="maxPrice"
            type="number"
            value={maxPrice ?? ""}
            max={maxVariantPrice}
            placeholder={maxVariantPrice.toString()}
            onChange={(e) => {
              let { value } = e.target;
              let newMaxPrice = Number.isNaN(Number.parseFloat(value))
                ? undefined
                : Number.parseFloat(value);
              setMaxPrice(newMaxPrice);
            }}
            onBlur={handleFilter}
            className="text-right focus-visible:outline-none py-3 bg-transparent w-full"
          />
        </div>
      </div>
    </div>
  )
}

That's how we build a fully functional price range filter component for collection pages in Shopify Hydrogen.

Check out the demo video below:

References

The component is fully open-sourced and available on GitHub.

Bonus

I'm building the first-ever Shopify Hydrogen theme customizer and CMS, including multiple Hydrogen Themes (all are free and open-sourced). If you are interested in building your own Shopify Hydrogen online store, checkout these links:

Happy filtering .