> ## Documentation Index
> Fetch the complete documentation index at: https://anvil.servicetitan.com/llms.txt
> Use this file to discover all available pages before exploring further.

# Select Field – Code

> SelectField components provide searchable dropdown selection with support for async data loading and client-side filtering.

export const LiveCode = ({children, customHeight, clickToLoad, example, fullWidth, fullHeight, hideCodeInLiveCode, screenshot, screenshotOnly, showCode: showCodeProp}) => {
  const SCREENSHOTS_BASE = "https://servicetitan.github.io/anvil2-docs-live-code/screenshots";
  const STACKBLITZ_BASE = "https://stackblitz.com/github/servicetitan/anvil2-docs-live-code/tree/main/examples";
  const [showCodeBlock, setShowCodeBlock] = useState(showCodeProp ?? false);
  const [isLocalOverride, setIsLocalOverride] = useState(false);
  useEffect(() => {
    const examplePath = `/images/live-code-screenshots-tmp/${example}.png`;
    fetch(examplePath, {
      method: "HEAD"
    }).then(r => {
      if (r.ok) setIsLocalOverride(true);
    }).catch(() => {});
  }, [example]);
  const screenshotBase = isLocalOverride ? "/images/live-code-screenshots-tmp" : SCREENSHOTS_BASE;
  if (screenshotOnly) {
    return <Frame className="flex flex-col">
        <div className="flex dark:hidden" style={{
      justifyContent: "center",
      alignItems: "center",
      width: fullWidth ? "100%" : "50%",
      minHeight: fullHeight ? "284px" : undefined,
      background: "#FFFFFF"
    }}>
          <img srcset={`${screenshotBase}/${example}.png, ${screenshotBase}/${example}-2x.png 2x`} src={`${screenshotBase}/${example}.png`} alt={example} noZoom />
        </div>
        <div className="hidden dark:flex" style={{
      justifyContent: "center",
      alignItems: "center",
      width: fullWidth ? "100%" : "50%",
      minHeight: fullHeight ? "284px" : undefined,
      background: "#141414"
    }}>
          <img srcset={`${screenshotBase}/${example}-dark.png, ${screenshotBase}/${example}-dark-2x.png 2x`} src={`${screenshotBase}/${example}-dark.png`} alt={example} noZoom />
        </div>
      </Frame>;
  }
  if (screenshot) {
    return <Frame className="flex flex-col -mb-2">
        <div className="flex dark:hidden bg-white dark:bg-codeblock border border-gray-950/10 dark:border-white/10 dark:twoslash-dark rounded-2xl overflow-hidden" style={{
      justifyContent: "center",
      alignItems: "center",
      width: fullWidth ? "100%" : "50%",
      minHeight: fullHeight ? "284px" : undefined
    }}>
          <img srcset={`${screenshotBase}/${example}.png, ${screenshotBase}/${example}-2x.png 2x`} src={`${screenshotBase}/${example}.png`} alt={example} noZoom />
        </div>

        <div className="hidden dark:flex bg-white dark:bg-codeblock border border-gray-950/10 dark:border-white/10 dark:twoslash-dark rounded-2xl overflow-hidden" style={{
      background: "#141414",
      justifyContent: "center",
      alignItems: "center",
      width: fullWidth ? "100%" : "50%",
      minHeight: fullHeight ? "284px" : undefined
    }}>
          <img srcset={`${screenshotBase}/${example}-dark.png, ${screenshotBase}/${example}-dark-2x.png 2x`} src={`${screenshotBase}/${example}-dark.png`} alt={example} noZoom />
        </div>

        <div className="flex justify-end items-center text-xs py-2 px-1 gap-4">
          {!showCodeProp ? <button className="inline-flex justify-end items-center text-gray-700 dark:text-gray-50 hover:text-blue-500 dark:hover:text-blue-300 transition-colors group self-end gap-1 cursor-pointer" onClick={() => setShowCodeBlock(!showCodeBlock)} style={{
      appearance: "none"
    }}>
              <Icon icon="code" size="12px" className="group-hover:bg-blue-500 dark:group-hover:bg-blue-300" />
              <span>{showCodeBlock ? "Hide code" : "Show code"}</span>
            </button> : null}

          <a className="inline-flex justify-end items-center hover:text-blue-500 dark:hover:text-blue-300 transition-colors group self-end gap-1" href={`${STACKBLITZ_BASE}/${example}?file=src/App.tsx`} target="_blank" rel="noreferrer">
            <Icon icon="bolt" size="12px" className="group-hover:bg-blue-500 dark:group-hover:bg-blue-300" />
            <span>StackBlitz demo</span>
          </a>
        </div>

        <div className="grid transition-[grid-template-rows] duration-300 ease-in-out overflow-auto overflow-y-hidden overflow-x-auto" style={showCodeBlock ? {
      gridTemplateRows: "1fr"
    } : {
      gridTemplateRows: "0fr"
    }}>
          <div style={{
      minHeight: 0,
      overflowX: "auto",
      overflowY: "hidden",
      marginBlockStart: "-1.25rem",
      marginBlockEnd: "-1.5rem"
    }}>
            {children}
          </div>
        </div>
      </Frame>;
  } else {
    return <div style={{
      display: "flex",
      width: fullWidth ? "100%" : "50%",
      minHeight: customHeight ? customHeight : "316px",
      resize: "vertical",
      overflow: "auto"
    }}>
        <iframe title={example} style={{
      flex: 1,
      width: fullWidth ? "100%" : "50%",
      minHeight: customHeight ? customHeight : "316px"
    }} src={`${STACKBLITZ_BASE}/${example}?embed=1&hideNavigation=1&hideExplorer=1&terminalHeight=0&file=src/App.tsx${clickToLoad ? "&ctl=1" : ""}${hideCodeInLiveCode ? "&view=preview" : ""}`} allow="accelerometer; ambient-light-sensor; camera; encrypted-media; geolocation; gyroscope; hid; microphone; midi; payment; usb; vr; xr-spatial-tracking" sandbox="allow-forms allow-modals allow-popups allow-presentation allow-same-origin allow-scripts" />
      </div>;
  }
};

<Note>
  **Beta Feature**

  This feature is currently in beta, and needs to be imported from `@servicetitan/anvil2/beta`.

  While we hope to minimize breaking changes, they may occur due to feedback we receive or other improvements. These will always be documented in the changelog and communicated in Slack.

  Please reach out in the [#ask-designsystem](https://servicetitan.enterprise.slack.com/archives/CBSRGHTRS) channel with any questions or feedback!
</Note>

<Tabs>
  <Tab title="Implementation">
    ## Overview

    The select field family includes two components for different use cases:

    * **`SelectField`** — For async data loading with support for pagination (page-based, offset-based, or group-based lazy loading)
      * Includes automatic debouncing of the search input (configurable via the `debounceMs` prop)
      * Includes automatic caching of the search input (configurable via the `cache` prop)
    * **`SelectFieldSync`** — For client-side filtering of static option arrays

    Both components provide a searchable dropdown interface and adaptive display modes (popover or dialog).

    ## SelectFieldSync (Static Options)

    Use `SelectFieldSync` when you have a static list of options that can be filtered client-side.

    ```tsx theme={null}
    import { useState } from "react";
    import { SelectFieldSync } from "@servicetitan/anvil2/beta";

    const options = [
      { id: 1, label: "Option One" },
      { id: 2, label: "Option Two" },
      { id: 3, label: "Option Three" },
    ];

    const ExampleComponent = () => {
      const [selectedOption, setSelectedOption] = useState(null);

      return (
        <SelectFieldSync
          label="Select an option"
          placeholder="Search options..."
          options={options}
          value={selectedOption}
          onSelectedOptionChange={setSelectedOption}
        />
      );
    };
    ```

    ### Filtering and Sorting

    By default, `SelectFieldSync` uses [match-sorter](https://github.com/kentcdodds/match-sorter) to filter options by their `label` and `searchText` fields. Results are also ranked by match quality, so the best matches appear first. Before any search is performed, options appear in the order they are supplied.
    You can customize this behavior in two ways:

    #### Using match-sorter options

    Pass a [match-sorter options object](https://github.com/kentcdodds/match-sorter#options) to customize the default filtering and sorting behavior (e.g., change which keys are matched or adjust ranking):

    ```tsx theme={null}
    <SelectFieldSync
      options={options}
      filter={{ 
        keys: ["label", "searchText"], 
        // ...other match-sorter options
      }}
      // ...other props
    />
    ```

    #### Using a custom filter function

    Pass a function for full control over both filtering and sort order. The returned array determines the exact order options appear in the dropdown:

    ```tsx theme={null}
    <SelectFieldSync
      options={options}
      filter={(options, searchValue) => {
        return options.filter((option) =>
          option.label?.toLowerCase().includes(searchValue.toLowerCase())
        );
      }}
      // ...other props
    />
    ```

    ## SelectField (Async Loading)

    Use `SelectField` when options need to be fetched from an API or when dealing with large datasets that require server-side filtering.

    ### Basic Async Loading

    ```tsx theme={null}
    import { useState } from "react";
    import { SelectField } from "@servicetitan/anvil2/beta";

    const ExampleComponent = () => {
      const [selectedOption, setSelectedOption] = useState(null);

      return (
        <SelectField
          label="Search users"
          placeholder="Type to search..."
          loadOptions={async (searchValue) => {
            const response = await fetch(`/api/users?q=${searchValue}`);
            const users = await response.json();
            return users.map((user) => ({
              id: user.id,
              label: user.name,
            }));
          }}
          value={selectedOption}
          onSelectedOptionChange={setSelectedOption}
        />
      );
    };
    ```

    ### Lazy Loading Modes

    `SelectField` supports three lazy loading modes for paginated data:

    #### Page-based Pagination

    ```tsx theme={null}
    <SelectField
      lazy="page"
      lazyOptions={{ pageSize: 10 }}
      loadOptions={async (searchValue, pageNumber, pageSize) => {
        const response = await fetch(
          `/api/items?q=${searchValue}&page=${pageNumber}&size=${pageSize}`
        );
        const { data, totalCount } = await response.json();
        return {
          options: data.map((item) => ({ id: item.id, label: item.name })),
          hasMore: pageNumber * pageSize + data.length < totalCount,
        };
      }}
      // ...other props
    />
    ```

    #### Offset-based Pagination

    ```tsx theme={null}
    <SelectField
      lazy="offset"
      lazyOptions={{ limit: 15 }}
      loadOptions={async (searchValue, offset, limit) => {
        const response = await fetch(
          `/api/items?q=${searchValue}&offset=${offset}&limit=${limit}`
        );
        const { data, totalCount } = await response.json();
        return {
          options: data.map((item) => ({ id: item.id, label: item.name })),
          hasMore: offset + limit < totalCount,
        };
      }}
      // ...other props
    />
    ```

    #### Group-based Loading

    For loading grouped options incrementally:

    ```tsx theme={null}
    <SelectField
      lazy="group"
      groupToString={(groupValue) => String(groupValue)}
      loadOptions={async (searchValue, previousGroupKey) => {
        const response = await fetch(
          `/api/items?q=${searchValue}&afterGroup=${previousGroupKey || ""}`
        );
        const { data, hasMore } = await response.json();
        return {
          options: data.map((item) => ({
            id: item.id,
            label: item.name,
            group: item.category, // Required for grouped options
          })),
          hasMore,
        };
      }}
      // ...other props
    />
    ```

    ## Display Modes

    Control how the options menu is displayed using the `displayMenuAs` prop:

    ```tsx theme={null}
    // Automatically choose based on device (default)
    <SelectFieldSync displayMenuAs="auto" {...props} />

    // Always show as popover (not recommended for mobile)
    <SelectFieldSync displayMenuAs="popover" {...props} />

    // Always show as dialog
    <SelectFieldSync displayMenuAs="dialog" {...props} />
    ```

    ## Caching

    `SelectField` caches `loadOptions` results by default. Configure caching behavior:

    ```tsx theme={null}
    // Disable caching
    <SelectField cache={{ enabled: false }} {...props} />

    // Configure max cache size (default: 15)
    <SelectField cache={{ maxSize: 100 }} {...props} />
    ```

    ### Clearing the Cache

    Use a ref to imperatively clear the cache:

    ```tsx theme={null}
    import { useRef } from "react";
    import { SelectField } from "@servicetitan/anvil2/beta";

    const ExampleComponent = () => {
      const selectFieldRef = useRef(null);

      const handleRefresh = () => {
        selectFieldRef.current?.clearCache();
      };

      return (
        <>
          <SelectField ref={selectFieldRef} {...props} />
          <button onClick={handleRefresh}>Refresh Options</button>
        </>
      );
    };
    ```

    ### Invalidating Options

    Call `invalidate()` to clear the cache and reload options from the data source. Use this when the underlying data has changed and the component needs to reflect the update:

    ```tsx theme={null}
    import { useRef } from "react";
    import { SelectField } from "@servicetitan/anvil2/beta";

    const ExampleComponent = () => {
      const selectFieldRef = useRef(null);

      const handleDataSourceChange = () => {
        selectFieldRef.current?.invalidate();
      };

      return (
        <SelectField ref={selectFieldRef} {...props} />
      );
    };
    ```

    <Note>`SelectFieldSync` handles this automatically when its `options` prop changes.</Note>

    ## Initial Load Behavior

    Control when options are first loaded with the `initialLoad` prop:

    ```tsx theme={null}
    // Load immediately on mount (default)
    <SelectField initialLoad="immediate" {...props} />

    // Load when user opens the dropdown
    <SelectField initialLoad="open" {...props} />

    // Auto (currently equivalent to "immediate")
    <SelectField initialLoad="auto" {...props} />
    ```

    ## Field States

    ### Error State

    Display validation errors using the `error` prop:

    ```tsx theme={null}
    <SelectFieldSync
      label="Category"
      options={options}
      value={selectedOption}
      onSelectedOptionChange={setSelectedOption}
      error="Please select a category"
    />
    ```

    ### Hint and Description

    Provide additional context with `hint` and `description`:

    ```tsx theme={null}
    <SelectFieldSync
      label="Category"
      options={options}
      value={selectedOption}
      onSelectedOptionChange={setSelectedOption}
      hint="Select the most relevant category"
      description="This will be used for filtering"
    />
    ```

    ### Required Field

    Mark a field as required with the `required` prop:

    ```tsx theme={null}
    <SelectFieldSync
      label="Category"
      options={options}
      value={selectedOption}
      onSelectedOptionChange={setSelectedOption}
      required
    />
    ```

    ### Disabled and ReadOnly

    When `disabled` is set, users cannot interact with the field.

    ```tsx theme={null}
    // Disabled - cannot interact with the field
    <SelectFieldSync
      label="Category"
      options={options}
      value={selectedOption}
      onSelectedOptionChange={setSelectedOption}
      disabled
    />
    ```

    When `readOnly` is set, users can see what options exist but cannot change the current value.

    ```tsx theme={null}
    // ReadOnly - can see options but cannot change the current value
    <SelectFieldSync
      label="Category"
      options={options}
      value={selectedOption}
      onSelectedOptionChange={setSelectedOption}
      readOnly
    />
    ```

    ### Prefix and Suffix

    Add content before or after the input with `prefix` and `suffix`:

    ```tsx theme={null}
    <SelectFieldSync
      label="Price Range"
      options={priceRanges}
      value={selectedOption}
      onSelectedOptionChange={setSelectedOption}
      prefix="$"
      suffix="USD"
    />
    ```

    ### Markdown in labels

    The `label` prop supports inline markdown: bold (`**text**`), italic (`*text*`), bold and italic (`***text***`), highlight (`==text==`), and code (`` `text` ``).

    <LiveCode showCode example="selectfield-markdownlabel" screenshot fullWidth>
      ```tsx lines expandable theme={null}
      import { useState } from "react";
      import { Flex } from "@servicetitan/anvil2";
      import { SelectFieldSync, SelectFieldOption } from "@servicetitan/anvil2/beta";

      const options = [
        { id: 1, label: "Option A" },
        { id: 2, label: "Option B" },
        { id: 3, label: "Option C" },
      ];

      function App() {
        const [value, setValue] = useState<SelectFieldOption | null>(null);

        return (
          <Flex direction="column" gap="4" style={{ maxWidth: 400 }}>
            <SelectFieldSync
              label="**Bold** label"
              placeholder="Bold"
              options={options}
              value={value}
              onSelectedOptionChange={setValue}
            />
            <SelectFieldSync
              label="*Italic* label"
              placeholder="Italic"
              options={options}
              value={value}
              onSelectedOptionChange={setValue}
            />
            <SelectFieldSync
              label="***Bold and italic*** label"
              placeholder="Bold and italic"
              options={options}
              value={value}
              onSelectedOptionChange={setValue}
            />
            <SelectFieldSync
              label="==Highlight== label"
              placeholder="Highlight"
              options={options}
              value={value}
              onSelectedOptionChange={setValue}
            />
            <SelectFieldSync
              label="`Code` label"
              placeholder="Code"
              options={options}
              value={value}
              onSelectedOptionChange={setValue}
            />
          </Flex>
        );
      }

      export default App;
      ```
    </LiveCode>

    ### Hide the label

    Use `hideLabel` to visually hide the label. The `label` string is converted to an `aria-label` on the input so it remains accessible to screen readers — any inline markdown is stripped to plain text.

    <LiveCode showCode example="selectfield-hidelabel" screenshot fullWidth>
      ```tsx lines expandable theme={null}
      import { useState } from "react";
      import { Flex } from "@servicetitan/anvil2";
      import { SelectFieldSync, SelectFieldOption } from "@servicetitan/anvil2/beta";

      const options = [
        { id: 1, label: "Option A" },
        { id: 2, label: "Option B" },
        { id: 3, label: "Option C" },
      ];

      function App() {
        const [value, setValue] = useState<SelectFieldOption | null>(null);

        return (
          <Flex direction="row" gap="2" style={{ maxWidth: 500 }}>
            <SelectFieldSync
              label="First option"
              hideLabel
              placeholder="First option"
              options={options}
              value={value}
              onSelectedOptionChange={setValue}
            />
            <SelectFieldSync
              label="Second option"
              hideLabel
              placeholder="Second option"
              options={options}
              value={value}
              onSelectedOptionChange={setValue}
            />
          </Flex>
        );
      }

      export default App;
      ```
    </LiveCode>

    ## Disabled Options

    Individual options can be disabled by setting `disabled: true` on the option:

    ```tsx theme={null}
    const options = [
      { id: 1, label: "Available Option" },
      { id: 2, label: "Unavailable Option", disabled: true },
      { id: 3, label: "Another Available Option" },
    ];

    <SelectFieldSync
      label="Select an option"
      options={options}
      value={selectedOption}
      onSelectedOptionChange={setSelectedOption}
    />
    ```

    ## Pinned Options

    Pin frequently used or suggested options above the list using the `pinned` prop. Each pinned section requires a `label` and an `options` value, which can be a static array or a dynamic loader function.

    ### Static Pinned Options

    Pass an object with `label` and a static `options` array:

    ```tsx theme={null}
    <SelectField
      label="Books"
      pinned={{
        label: "Favorites",
        options: [
          { id: "fav-1", label: "The Martian" },
          { id: "fav-2", label: "Dune" },
        ],
      }}
      loadOptions={fetchBooks}
      value={selectedOption}
      onSelectedOptionChange={setSelectedOption}
    />
    ```

    ### Dynamic Pinned Options

    Pass a function as `options` to compute pinned options based on the current search value. This is useful for AI-powered suggestions or context-aware recommendations:

    ```tsx theme={null}
    <SelectField
      label="Books"
      pinned={{
        label: "AI Suggestions",
        options: async (searchValue) => {
          const suggestions = await fetchAISuggestions(searchValue);
          return suggestions.map((s) => ({ id: s.id, label: s.title }));
        },
      }}
      loadOptions={fetchBooks}
      value={selectedOption}
      onSelectedOptionChange={setSelectedOption}
    />
    ```

    By default, the loader re-runs whenever the search value changes. Set `searchReactive: false` to call the loader once and reuse the result across all search values:

    ```tsx theme={null}
    <SelectField
      label="Books"
      pinned={{
        label: "Your Favorites",
        options: async () => {
          return await fetchFavorites();
        },
        searchReactive: false,
      }}
      loadOptions={fetchBooks}
      value={selectedOption}
      onSelectedOptionChange={setSelectedOption}
    />
    ```

    ### Multiple Pinned Sections

    Pass an array of pinned section objects:

    ```tsx theme={null}
    <SelectField
      label="Books"
      pinned={[
        {
          label: "AI Suggestions",
          options: async (searchValue) => {
            const suggestions = await fetchAISuggestions(searchValue);
            return suggestions.map((s) => ({ id: s.id, label: s.title }));
          },
        },
        {
          label: "Your Favorites",
          options: [
            { id: "fav-1", label: "Dune" },
            { id: "fav-2", label: "Foundation" },
          ],
        },
      ]}
      loadOptions={fetchBooks}
      value={selectedOption}
      onSelectedOptionChange={setSelectedOption}
    />
    ```

    ## Grouping Options

    Options can be organized into visual groups by adding a `group` property to each option. Groups appear as labeled sections in the dropdown.

    ### Basic Grouping

    Add a `group` property to options and provide a `groupToString` function to display group labels:

    ```tsx theme={null}
    const options = [
      { id: 1, label: "Apple", group: "fruits" },
      { id: 2, label: "Banana", group: "fruits" },
      { id: 3, label: "Carrot", group: "vegetables" },
      { id: 4, label: "Broccoli", group: "vegetables" },
    ];

    <SelectFieldSync
      label="Food"
      options={options}
      groupToString={(group) =>
        group === "fruits" ? "Fruits" : "Vegetables"
      }
      value={selectedOption}
      onSelectedOptionChange={setSelectedOption}
    />
    ```

    ### Group Sorting

    Use `groupSorter` to control the order of groups. By default, groups appear in the order they are first encountered in the options array. With `groupSorter`, you can apply custom sorting (e.g., alphabetical, numerical priority).

    This prop is available on `SelectFieldSync` and non-lazy `SelectField`:

    <Warning>
      Avoid using `groupSorter` with `lazy="group"`. When groups load incrementally from the server and get re-sorted, the menu content shifts unexpectedly, creating a disorienting user experience. Instead, have your server return groups in the desired order.
    </Warning>

    ```tsx theme={null}
    const options = [
      { id: 1, label: "Item A", group: 3 },
      { id: 2, label: "Item B", group: 1 },
      { id: 3, label: "Item C", group: 2 },
    ];

    // With SelectFieldSync
    <SelectFieldSync
      label="Items"
      options={options}
      groupToString={(group) => `Priority ${group}`}
      groupSorter={(a, b) => Number(a) - Number(b)}
      value={selectedOption}
      onSelectedOptionChange={setSelectedOption}
    />

    // With async SelectField (non-lazy)
    <SelectField
      label="Items"
      loadOptions={async () => options}
      groupToString={(group) => `Priority ${group}`}
      groupSorter={(a, b) => Number(a) - Number(b)}
      value={selectedOption}
      onSelectedOptionChange={setSelectedOption}
    />
    ```

    ### Mixed Grouped and Ungrouped Options

    Options without a `group` property appear after all grouped sections:

    ```tsx theme={null}
    const options = [
      { id: 1, label: "Apple", group: "fruits" },
      { id: 2, label: "Banana", group: "fruits" },
      { id: 3, label: "Other Item" }, // No group - appears last
      { id: 4, label: "Carrot", group: "vegetables" },
    ];
    ```

    ## Virtualization

    By default, all dropdown options render to the DOM at once. This works well for typical lists but degrades performance with very large option sets as the browser must lay out and paint thousands of elements. Pass `virtualize` to enable windowed rendering, which only renders the items currently visible in the scroll viewport plus a small overscan buffer.

    Consider enabling `virtualize` when the dropdown feels sluggish to open or keyboard navigation becomes laggy. These symptoms typically appear around 200–500 items depending on device performance and item complexity (e.g., items with descriptions are heavier). For smaller lists, the default rendering path is sufficient and avoids the overhead of virtual positioning.

    Virtualization works with all existing features including lazy loading, pinned options, grouping, and keyboard navigation.

    ```tsx theme={null}
    import { useState } from "react";
    import { SelectFieldSync } from "@servicetitan/anvil2/beta";

    const options = Array.from({ length: 5000 }, (_, i) => ({
      id: i,
      label: `Option ${i + 1}`,
    }));

    const ExampleComponent = () => {
      const [selectedOption, setSelectedOption] = useState(null);

      return (
        <SelectFieldSync
          virtualize
          label="Large dataset"
          placeholder="Search options..."
          options={options}
          value={selectedOption}
          onSelectedOptionChange={setSelectedOption}
        />
      );
    };
    ```

    ## Adding New Items

    Provide an `onAddNewItem` handler to render an "Add new item" button below the option list. This affordance lets users create a value that does not exist yet — a new customer, tag, or category — quickly from the field.

    The button sits in a footer region inside the menu. Clicking or activating it via keyboard closes the menu and invokes the `onAddNewItem` handler with the current search text.

    From here, it is your responsibility to launch a `Dialog` (or `Drawer`, or any other overlay) for collecting the new item's details, append the result to the option source, invalidate the cache, and update selection state.

    You may pair `onAddNewItem` with `addItemLabel` to control the button text. The label accepts a static node or a function of the current search text — for example, returning `Add "foo"` when the user has typed `foo`.

    <LiveCode showCode example="selectfield-add-new-item" screenshot fullWidth>
      ```tsx lines expandable theme={null}
      import { useEffect, useRef, useState } from "react";
      import {
        SelectField,
        SelectFieldHandle,
        SelectFieldOption,
      } from "@servicetitan/anvil2/beta";
      import { Button, Dialog, Flex, TextField } from "@servicetitan/anvil2";

      type Customer = SelectFieldOption & { phone: string };

      function App() {
        const [customers, setCustomers] = useState<Customer[]>([
          { id: "1", label: "Acme HVAC", phone: "(555) 123-4567" },
          { id: "2", label: "Bluebird Plumbing", phone: "(555) 234-5678" },
          { id: "3", label: "Carolina Electric", phone: "(555) 345-6789" },
        ]);
        const [selected, setSelected] = useState<Customer | null>(null);
        const selectRef = useRef<SelectFieldHandle>(null);
        const containerRef = useRef<HTMLDivElement>(null);

        const [dialogOpen, setDialogOpen] = useState(false);
        const [draftName, setDraftName] = useState("");
        const [draftPhone, setDraftPhone] = useState("");

        // Open the menu on mount so the Add-new button is visible in screenshots.
        useEffect(() => {
          const timer = setTimeout(() => {
            const trigger = containerRef.current?.querySelector("input");
            trigger?.focus();
            trigger?.click();
          }, 100);
          return () => clearTimeout(timer);
        }, []);

        const openCustomerDialog = (initialName: string) => {
          setDraftName(initialName);
          setDraftPhone("");
          setDialogOpen(true);
        };

        const saveCustomer = () => {
          const newCustomer: Customer = {
            id: crypto.randomUUID(),
            label: draftName,
            phone: draftPhone,
          };
          setCustomers((prev) => [...prev, newCustomer]);
          setSelected(newCustomer);
          selectRef.current?.invalidate();
          setDialogOpen(false);
        };

        return (
          <div ref={containerRef} style={{ minWidth: "384px", minHeight: "380px" }}>
            <SelectField
              ref={selectRef}
              label="Customer"
              placeholder="Search customers..."
              initialLoad="open"
              loadOptions={async (searchValue) =>
                searchValue
                  ? customers.filter((c) =>
                      c.label.toLowerCase().includes(searchValue.toLowerCase()),
                    )
                  : customers
              }
              value={selected}
              onSelectedOptionChange={(opt) => setSelected(opt as Customer | null)}
              addItemLabel={(searchText) =>
                searchText
                  ? `Add "${searchText}" as new customer`
                  : "Add new customer"
              }
              onAddNewItem={openCustomerDialog}
            />

            <Dialog open={dialogOpen} onClose={() => setDialogOpen(false)}>
              <Dialog.Header>Add new customer</Dialog.Header>
              <Dialog.Content>
                <Flex direction="column" gap="3">
                  <TextField
                    label="Name"
                    required
                    value={draftName}
                    onChange={(e) => setDraftName(e.target.value)}
                  />
                  <TextField
                    label="Phone"
                    placeholder="(555) 555-5555"
                    value={draftPhone}
                    onChange={(e) => setDraftPhone(e.target.value)}
                  />
                </Flex>
              </Dialog.Content>
              <Dialog.Footer>
                <Dialog.CancelButton appearance="ghost">Cancel</Dialog.CancelButton>
                <Button
                  appearance="primary"
                  onClick={saveCustomer}
                  disabled={!draftName.trim()}
                >
                  Save customer
                </Button>
              </Dialog.Footer>
            </Dialog>
          </div>
        );
      }

      export default App;
      ```
    </LiveCode>

    <Note>
      Call `invalidate()` on the imperative handle after appending the new option.
      This clears the cached search results and forces a refetch the next time the
      menu opens, so the new item appears immediately. `clearCache()` alone wipes
      the cache but keeps the previously displayed options on screen until
      something else triggers a reload.
    </Note>

    ## Disabling Search

    Pass `disableSearch` to replace the searchable text input with a non-editable select trigger. The field uses the listbox ARIA pattern instead of combobox, providing a simpler interaction when typing to filter adds no value.

    This is useful when the option list is short and well-known or you are integrating with an API that does not support search.

    ```tsx theme={null}
    import { useState } from "react";
    import { SelectFieldSync } from "@servicetitan/anvil2/beta";

    const options = [
      { id: 1, label: "Small" },
      { id: 2, label: "Medium" },
      { id: 3, label: "Large" },
    ];

    const ExampleComponent = () => {
      const [selectedOption, setSelectedOption] = useState(null);

      return (
        <SelectFieldSync
          disableSearch
          label="Size"
          placeholder="Select a size"
          options={options}
          value={selectedOption}
          onSelectedOptionChange={setSelectedOption}
        />
      );
    };
    ```
  </Tab>

  <Tab title="SelectField Props">
    ## `SelectField` Props

    ```tsx theme={null}
    import { useState } from "react";
    import { SelectField } from "@servicetitan/anvil2/beta";

    const ExampleComponent = () => {
      const [selectedOption, setSelectedOption] = useState(null);

      return (
        <SelectField
          label="Search users"
          loadOptions={async (searchValue) => {
            const response = await fetch(`/api/users?q=${searchValue}`);
            const users = await response.json();
            return users.map((user) => ({
              id: user.id,
              label: user.name,
            }));
          }}
          onSelectedOptionChange={setSelectedOption}
          value={selectedOption}
        />
      );
    };
    ```

    <ParamField path="label" type="string" required>
      The label of the select field. Supports inline markdown formatting.
    </ParamField>

    <ParamField path="labelAiMark" type={`boolean | AiMarkWithTooltipOrPopoverConfig`}>
      Displays an AI mark next to the label. Use `true` for the icon only, or pass tooltip or popover configuration. See [AI Marks](/docs/web/utilities/ai-marks).
    </ParamField>

    <ParamField path="loadOptions" type="function" required>
      Function to load options. The signature depends on the `lazy` mode:

      * **Non-lazy**: `(searchValue: string) => SelectFieldOption[] | Promise<SelectFieldOption[]>`
      * **Page-based**: `(searchValue: string, pageNumber: number, pageSize: number) => { options: SelectFieldOption[], hasMore?: boolean }`
      * **Offset-based**: `(searchValue: string, offset: number, limit: number) => { options: SelectFieldOption[], hasMore?: boolean }`
      * **Group-based**: `(searchValue: string, previousGroupKey: string | number | null) => { options: SelectFieldGroupedOption[], hasMore?: boolean }`
    </ParamField>

    <ParamField path="onSelectedOptionChange" type="(option: SelectFieldOption | null) => void" required>
      Callback fired when the selected option changes.
    </ParamField>

    <ParamField path="value" type="SelectFieldOption | null" required>
      The currently selected option. Must be controlled state.
    </ParamField>

    <ParamField path="addItemLabel" type="ReactNode | (searchText: string) => ReactNode">
      Label rendered inside the "Add new item" button. May be a static node or a callback that receives the current search text and returns a dynamic label (for example, `Add "${searchText}"`). Defaults to `"Add new item"` when omitted but `onAddNewItem` is provided.
    </ParamField>

    <ParamField path="cache" type="SelectFieldCacheOptions">
      Configuration for caching `loadOptions` results:

      * `enabled` — Whether caching is enabled (default: `true`)
      * `maxSize` — Maximum number of search values to cache before clearing (default: `15`)
    </ParamField>

    <ParamField path="className" type="string">
      Custom CSS class name for the wrapper element.
    </ParamField>

    <ParamField path="debounceMs" type="number" default="200">
      Milliseconds to debounce search input before calling `loadOptions`.
    </ParamField>

    <ParamField path="description" type="ReactElement | string">
      Description text displayed below the input field.
    </ParamField>

    <ParamField path="disableClearButton" type="boolean" default="false">
      Whether to hide the clear button.
    </ParamField>

    <ParamField path="disableSearch" type="boolean" default="false">
      Replaces the searchable text input with a non-editable select trigger. Uses the listbox ARIA pattern instead of combobox. When combined with `displayMenuAs="dialog"`, the search input in the dialog header is also removed.
    </ParamField>

    <ParamField path="disabled" type="boolean" default="false">
      Whether the field is disabled. When disabled, the field cannot be interacted with and the dropdown cannot open.
    </ParamField>

    <ParamField path="displayMenuAs" type="'auto' | 'popover' | 'dialog'" default="auto">
      How to display the options menu:

      * `auto` — Popover on desktop, dialog on mobile
      * `popover` — Always display as popover
      * `dialog` — Always display as dialog
    </ParamField>

    <ParamField path="error" type="ReactElement | string | boolean">
      Error state for the field. When `true`, shows error styling. When a string or ReactElement, also displays as an error message below the field.
    </ParamField>

    <ParamField path="errorAriaLive" type="'off' | 'assertive' | 'polite'" default="assertive">
      Controls the `aria-live` behavior for error messages. See [MDN docs](https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/Attributes/aria-live).
    </ParamField>

    <ParamField path="groupSorter" type="(a: SelectFieldGroupByValue, b: SelectFieldGroupByValue) => number">
      Function to compare two group values for sorting. When provided, groups are sorted using this comparator. Without this, groups appear in the order they are first encountered. Best used with `SelectFieldSync` or non-lazy `SelectField`. Avoid using with `lazy="group"` as it causes the menu to shift unexpectedly when new groups load.
    </ParamField>

    <ParamField path="groupToString" type="(groupValue: SelectFieldGroupByValue) => string">
      Function to convert group values to display labels. Only used with grouped options. `SelectFieldGroupByValue` is `string | number`.
    </ParamField>

    <ParamField path="hideLabel" type="boolean" default="false">
      Visually hides the label above the input while keeping it accessible to screen readers. Note: This does not affect the label displayed in the adaptive dialog view on mobile devices.
    </ParamField>

    <ParamField path="hint" type="ReactElement | string">
      Hint text displayed below the input field.
    </ParamField>

    <ParamField path="id" type="string">
      The id of the select field.
    </ParamField>

    <ParamField path="initialLoad" type="'auto' | 'immediate' | 'open'" default="auto">
      Controls when `loadOptions` is first called:

      * `auto` — Currently equivalent to `immediate`
      * `immediate` — Load on component mount
      * `open` — Load when dropdown is opened
    </ParamField>

    <ParamField path="lazy" type="'page' | 'offset' | 'group' | false" default="false">
      Enables lazy loading with the specified pagination strategy.
    </ParamField>

    <ParamField path="lazyOptions" type="object">
      Configuration for lazy loading:

      * **Page mode**: `{ pageSize?: number }` (default pageSize: 20)
      * **Offset mode**: `{ limit?: number }` (default limit: 20)
      * **Group mode**: `{}`
    </ParamField>

    <ParamField path="moreInfo" type="ReactNode">
      Additional information displayed in a tooltip next to the label. Renders an info icon button that shows the content on hover or focus.
    </ParamField>

    <ParamField path="onAddNewItem" type="(searchText: string) => void">
      Click handler for the "Add new item" button. The button is only rendered when this prop is provided. Receives the current search text. The menu closes synchronously on click and the handler runs immediately after — typically opening a Dialog or Drawer to collect the new item's details. Consumers own overlay presentation, focus return, option list updates, and selection state. See the [Adding New Items](#adding-new-items) section for a full example.
    </ParamField>

    <ParamField path="pinned" type="SelectFieldPinnedOptions">
      Options to pin to the top of the list. Accepts a single section object or an array of section objects. Each section requires:

      * `label` — Display label for the section header
      * `options` — Static array (`SelectFieldOption[]`) or dynamic loader function (`(searchValue: string) => SelectFieldOption[] | Promise<SelectFieldOption[]>`)
      * `searchReactive` — (Optional) Whether to re-call the loader when the search value changes. Defaults to `true`. Set to `false` to call the loader once and reuse the result.
      * `cacheSize` — (Optional) Maximum number of search results to cache per section. Defaults to `15`. Only applies when `searchReactive` is `true`.
    </ParamField>

    <ParamField path="placeholder" type="string">
      Placeholder text for the input field.
    </ParamField>

    <ParamField path="prefix" type="string | ReactElement">
      Content to display before the input field.
    </ParamField>

    <ParamField path="readOnly" type="boolean" default="false">
      Whether the field is read-only. When read-only, the dropdown can open to view options but selections cannot be changed.
    </ParamField>

    <ParamField path="required" type="boolean" default="false">
      Whether the field is required. Shows a red asterisk (\*) next to the label.
    </ParamField>

    <ParamField path="size" type="'small' | 'medium' | 'large'">
      The size of the select field.
    </ParamField>

    <ParamField path="style" type="CSSProperties">
      Custom inline styles for the wrapper element.
    </ParamField>

    <ParamField path="suffix" type="string | ReactElement">
      Content to display after the input field.
    </ParamField>

    <ParamField path="virtualize" type="boolean" default="false">
      Enables windowed rendering for the dropdown list. Improves performance for large option sets by only rendering visible items. Consider enabling when the dropdown feels sluggish to open or keyboard navigation becomes laggy, typically around 200–500+ items.
    </ParamField>
  </Tab>

  <Tab title="SelectFieldSync Props">
    ## `SelectFieldSync` Props

    `SelectFieldSync` accepts all props from `SelectField` except `loadOptions`, `lazy`, and `debounceMs`, plus the following:

    ```tsx theme={null}
    import { useState } from "react";
    import { SelectFieldSync } from "@servicetitan/anvil2/beta";

    const options = [
      { id: 1, label: "Option One" },
      { id: 2, label: "Option Two" },
      { id: 3, label: "Option Three" },
    ];

    const ExampleComponent = () => {
      const [selectedOption, setSelectedOption] = useState(null);

      return (
        <SelectFieldSync
          label="Select an option"
          onSelectedOptionChange={setSelectedOption}
          options={options}
          value={selectedOption}
        />
      );
    };
    ```

    <ParamField path="label" type="string" required>
      The label of the select field. Supports inline markdown formatting.
    </ParamField>

    <ParamField path="labelAiMark" type={`boolean | AiMarkWithTooltipOrPopoverConfig`}>
      Displays an AI mark next to the label. Use `true` for the icon only, or pass tooltip or popover configuration. See [AI Marks](/docs/web/utilities/ai-marks).
    </ParamField>

    <ParamField path="onSelectedOptionChange" type="(option: SelectFieldOption | null) => void" required>
      Callback fired when the selected option changes.
    </ParamField>

    <ParamField path="options" type="SelectFieldOption[]" required>
      The array of options to display in the select field.
    </ParamField>

    <ParamField path="value" type="SelectFieldOption | null" required>
      The currently selected option. Must be controlled state.
    </ParamField>

    <ParamField path="addItemLabel" type="ReactNode | (searchText: string) => ReactNode">
      Label rendered inside the "Add new item" button. May be a static node or a callback that receives the current search text and returns a dynamic label (for example, `Add "${searchText}"`). Defaults to `"Add new item"` when omitted but `onAddNewItem` is provided.
    </ParamField>

    <ParamField path="className" type="string">
      Custom CSS class name for the wrapper element.
    </ParamField>

    <ParamField path="description" type="ReactElement | string">
      Description text displayed below the input field.
    </ParamField>

    <ParamField path="disableClearButton" type="boolean" default="false">
      Whether to hide the clear button.
    </ParamField>

    <ParamField path="disableSearch" type="boolean" default="false">
      Replaces the searchable text input with a non-editable select trigger. Uses the listbox ARIA pattern instead of combobox. When combined with `displayMenuAs="dialog"`, the search input in the dialog header is also removed.
    </ParamField>

    <ParamField path="disabled" type="boolean" default="false">
      Whether the field is disabled. When disabled, the field cannot be interacted with and the dropdown cannot open.
    </ParamField>

    <ParamField path="displayMenuAs" type="'auto' | 'popover' | 'dialog'" default="auto">
      How to display the options menu.
    </ParamField>

    <ParamField path="error" type="ReactElement | string | boolean">
      Error state for the field. When `true`, shows error styling. When a string or ReactElement, also displays as an error message below the field.
    </ParamField>

    <ParamField path="errorAriaLive" type="'off' | 'assertive' | 'polite'" default="assertive">
      Controls the `aria-live` behavior for error messages.
    </ParamField>

    <ParamField path="filter" type="function | MatchSorterOptions">
      Controls how options are filtered and sorted when the user types a search value. Can be:

      * A function: `(options: SelectFieldOption[], searchValue: string) => SelectFieldOption[]` — receives all options and returns the filtered subset in the desired display order
      * A [match-sorter options object](https://github.com/kentcdodds/match-sorter#options) to customize the default match-sorter behavior (e.g., which keys to match against, ranking strategy)

      Before any search is performed, options appear in the order they are supplied. Default: Filters by `label` and `searchText` fields using match-sorter, with results ranked by match quality.
    </ParamField>

    <ParamField path="groupSorter" type="(a: SelectFieldGroupByValue, b: SelectFieldGroupByValue) => number">
      Function to compare two group values for sorting. When provided, options are sorted by group using this comparator, then by match-sort order within each group. Ungrouped options appear last.
    </ParamField>

    <ParamField path="groupToString" type="(groupValue: SelectFieldGroupByValue) => string">
      Function to convert group values to display labels. Only used with grouped options. `SelectFieldGroupByValue` is `string | number`.
    </ParamField>

    <ParamField path="hideLabel" type="boolean" default="false">
      Visually hides the label above the input while keeping it accessible to screen readers. Note: This does not affect the label displayed in the adaptive dialog view on mobile devices.
    </ParamField>

    <ParamField path="hint" type="ReactElement | string">
      Hint text displayed below the input field.
    </ParamField>

    <ParamField path="id" type="string">
      The id of the select field.
    </ParamField>

    <ParamField path="moreInfo" type="ReactNode">
      Additional information displayed in a tooltip next to the label. Renders an info icon button that shows the content on hover or focus.
    </ParamField>

    <ParamField path="onAddNewItem" type="(searchText: string) => void">
      Click handler for the "Add new item" button. The button is only rendered when this prop is provided. Receives the current search text. The menu closes synchronously on click and the handler runs immediately after — typically opening a Dialog or Drawer to collect the new item's details. Consumers own overlay presentation, focus return, option list updates, and selection state. See the [Adding New Items](#adding-new-items) section for a full example.
    </ParamField>

    <ParamField path="pinned" type="SelectFieldPinnedOptions">
      Options to pin to the top of the list. Accepts the same section object format as `SelectField`'s `pinned` prop.
    </ParamField>

    <ParamField path="placeholder" type="string">
      Placeholder text for the input field.
    </ParamField>

    <ParamField path="prefix" type="string | ReactElement">
      Content to display before the input field.
    </ParamField>

    <ParamField path="readOnly" type="boolean" default="false">
      Whether the field is read-only. When read-only, the dropdown can open to view options but selections cannot be changed.
    </ParamField>

    <ParamField path="required" type="boolean" default="false">
      Whether the field is required. Shows a red asterisk (\*) next to the label.
    </ParamField>

    <ParamField path="size" type="'small' | 'medium' | 'large'">
      The size of the select field.
    </ParamField>

    <ParamField path="style" type="CSSProperties">
      Custom inline styles for the wrapper element.
    </ParamField>

    <ParamField path="suffix" type="string | ReactElement">
      Content to display after the input field.
    </ParamField>

    <ParamField path="virtualize" type="boolean" default="false">
      Enables windowed rendering for the dropdown list. Improves performance for large option sets by only rendering visible items. Consider enabling when the dropdown feels sluggish to open or keyboard navigation becomes laggy, typically around 200–500+ items.
    </ParamField>
  </Tab>

  <Tab title="SelectFieldOption Type">
    ## `SelectFieldOption` Type

    Options passed to or returned from `SelectField` components must conform to this shape:

    ```tsx theme={null}
    type SelectFieldOption = {
      /** Unique identifier for the option */
      id: string | number;

      /** Display text for the option */
      label: string;

      /** Optional text used for search matching (in addition to label) */
      searchText?: string;

      /** Group key for grouped options (required when using lazy="group") */
      group?: string | number;

      /** Whether the option is disabled and cannot be selected */
      disabled?: boolean;

      /** Optional extra data to associate with the option */
      extra?: Record<string, unknown>;

      /** Optional rich content configuration */
      content?: {
        title?: string;
        description?: string;
        chips?: Pick<ChipProps, "label" | "color" | "textWrap">[];
        avatar?: Pick<AvatarProps, "name" | "status" | "color" | "image">;
        icon?: Pick<IconProps, "svg" | "color">;
      };
    };
    ```

    ### Content Properties

    The `content` object supports the following optional properties:

    * `title` — Replaces the `label` as the primary display text in the dropdown.
    * `description` — Secondary text displayed below the title.
    * `chips` — Array of chip configurations rendered below the title and description. Each chip accepts `label`, `color`, and `textWrap`.
    * `avatar` — Renders an avatar to the left of the option text. Accepts `name`, `status`, `color`, and `image`.
    * `icon` — Renders an icon to the right of the option text. Accepts `svg` and `color`.

    ### Example with Rich Content

    ```tsx theme={null}
    const options: SelectFieldOption[] = [
      {
        id: 1,
        label: "John Doe",
        searchText: "john.doe@example.com",
        content: {
          title: "John Doe",
          description: "Engineering",
        },
      },
      {
        id: 2,
        label: "Jane Smith",
        searchText: "jane.smith@example.com",
        content: {
          title: "Jane Smith",
          description: "Design",
        },
      },
    ];
    ```

    ### Example with Avatar, Icon, and Chips

    ```tsx theme={null}
    import GroupIcon from "@servicetitan/anvil2/assets/icons/material/round/groups.svg";

    const options: SelectFieldOption[] = [
      {
        id: 1,
        label: "Anvil Team",
        content: {
          title: "Anvil Team (4)",
          description: "Design system maintainer",
          icon: { svg: GroupIcon },
        },
      },
      {
        id: 2,
        label: "Ergonomic Chair",
        content: {
          title: "Ergonomic Chair",
          description: "Offers support and breathability",
          avatar: { name: "Ergonomic Chair", image: "/images/chair.png" },
          chips: [
            { label: "In Stock" },
            { label: "Popular", color: "#F97316" },
          ],
        },
      },
    ];
    ```

    ### Example with Disabled Options

    ```tsx theme={null}
    const options: SelectFieldOption[] = [
      {
        id: 1,
        label: "Available Option",
      },
      {
        id: 2,
        label: "Unavailable Option",
        disabled: true,
      },
      {
        id: 3,
        label: "Another Available Option",
      },
    ];
    ```
  </Tab>
</Tabs>
