Skip to content

vojtechportes/react-query-builder

Repository files navigation

React Query Builder

Simple, highly configurable query builder
for React written in TypeScript

npm version License: MIT

Documentation · Demo

React Query Builder


React Query Builder is a TypeScript library for building nested filter editors, formatting them into external query syntaxes, and parsing supported expressions back into builder state.

It also supports an optional SQL text mode for Builder, with built-in syntax and semantic validation, plus an optional Monaco-based advanced editor integration for protected locked ranges.

Full documentation, API reference, and the interactive demo are available on the project website:

Installation

npm install @vojtechportes/react-query-builder

React Query Builder supports React 18+.

Example

import React, { useState } from 'react';
import {
  Builder,
  type DenormalizedQuery,
  type IBuilderFieldProps,
} from '@vojtechportes/react-query-builder';

const fields: IBuilderFieldProps[] = [
  {
    field: 'STATE',
    label: 'State',
    type: 'LIST',
    operators: ['EQUAL', 'NOT_EQUAL'],
    value: [
      { value: 'CZ', label: 'Czech Republic' },
      { value: 'SK', label: 'Slovakia' },
    ],
  },
  {
    field: 'IS_IN_EU',
    label: 'Is in EU',
    type: 'BOOLEAN',
  },
];

const initialData: DenormalizedQuery = [
  {
    type: 'GROUP',
    value: 'AND',
    isNegated: false,
    children: [
      {
        field: 'STATE',
        operator: 'EQUAL',
        value: 'CZ',
        readOnly: true,
      },
      {
        field: 'IS_IN_EU',
        operator: 'EQUAL',
        value: true,
      },
    ],
  },
];

export const MyBuilder = () => {
  const [data, setData] = useState(initialData);

  return <Builder fields={fields} data={data} onChange={setData} />;
};

Validation and Usage Limits

Field metadata can define both value validation and structural usage limits. Use validation for rule-local checks like required values or numeric bounds, and use usageLimit when a field or shared field bucket should only appear a limited number of times globally or within the same parent group.

const fields: IBuilderFieldProps[] = [
  {
    field: 'PRIMARY_EMAIL',
    label: 'Primary email',
    type: 'TEXT',
    operators: ['EQUAL', 'CONTAINS'],
    usageLimit: {
      max: 1,
      scope: 'global',
    },
    validation: {
      required: true,
    },
  },
  {
    field: 'BILLING_CONTACT',
    label: 'Billing contact',
    type: 'TEXT',
    operators: ['EQUAL'],
    usageLimit: {
      key: 'contact-field',
      max: 1,
      scope: 'parent',
    },
  },
  {
    field: 'SHIPPING_CONTACT',
    label: 'Shipping contact',
    type: 'TEXT',
    operators: ['EQUAL'],
    usageLimit: {
      key: 'contact-field',
      max: 1,
      scope: 'parent',
    },
  },
];

When a usage limit is exhausted, the field becomes disabled in the selector, and the built-in Add Rule action is disabled when no selectable fields remain in the current scope.

Dynamic Field Options

When list options depend on other rules or on external application state, keep the fields array stable and update options through the imperative builderRef API instead of replacing the full field definition.

This is especially useful for dependent selects, async option loading, and integrations with tools like TanStack React Query.

Use field-level methods when every rule of the same field should share one runtime option set:

  • isFieldInUse(field)
  • setFieldOptions(field, options)
  • setFieldOptionsStatus(field, status)
  • invalidateFieldOptions(field)
  • reloadFieldOptions(field)
  • clearFieldOptions(field)

Use rule-level methods when options depend on other rules or on surrounding app state:

  • bindRuleOptions(field, config)
  • useBuilderRuleDependencies(builderRef, field, dependencyFields)
  • subscribeToRuleDependencies(field, dependencyFields, listener)
  • setRuleOptions(ruleId, options)
  • setRuleOptionsStatus(ruleId, status)
  • reconcileRuleValueWithOptions(ruleId, { strategy: 'clear-if-missing' })

See the Dynamic Field Options documentation for the full concept, live example, and React Query integration example:

UI Adapters

The package also exposes ready-made component mappings through versioned subpath exports:

  • @vojtechportes/react-query-builder/mui/v9 for new Material UI projects
  • @vojtechportes/react-query-builder/mui/v7 for applications still on MUI 7
  • @vojtechportes/react-query-builder/bootstrap/v5 for Bootstrap 5 projects
  • @vojtechportes/react-query-builder/antd/v6 for new Ant Design projects
  • @vojtechportes/react-query-builder/antd/v5 for applications still on Ant Design 5
  • @vojtechportes/react-query-builder/mantine/v9 for new Mantine projects
  • @vojtechportes/react-query-builder/mantine/v8 for applications still on Mantine 8
  • @vojtechportes/react-query-builder/fluentui/v8 for Fluent UI React 8 projects
  • @vojtechportes/react-query-builder/radix/v1 for Radix Themes 1 projects

Install the peer dependencies that match the adapter you want to use and pass the exported components object to Builder.

Bootstrap 5 example:

import React from 'react';
import 'bootstrap/dist/css/bootstrap.min.css';
import { Builder } from '@vojtechportes/react-query-builder';
import { components } from '@vojtechportes/react-query-builder/bootstrap/v5';

export const MyBootstrapBuilder = () => {
  return (
    <Builder
      fields={fields}
      data={data}
      components={components}
      onChange={setData}
    />
  );
};

Material UI example:

npm install @mui/material@^9.0.1 @mui/icons-material@^9.0.1 @emotion/react @emotion/styled
import React from 'react';
import { Builder } from '@vojtechportes/react-query-builder';
import { components } from '@vojtechportes/react-query-builder/mui/v9';

export const MyMuiBuilder = () => {
  return (
    <Builder
      fields={fields}
      data={data}
      components={components}
      onChange={setData}
    />
  );
};

Ant Design example:

npm install antd@^6.0.0 @ant-design/icons@^6.0.0
import React from 'react';
import { Builder } from '@vojtechportes/react-query-builder';
import { components } from '@vojtechportes/react-query-builder/antd/v6';

export const MyAntdBuilder = () => {
  return (
    <Builder
      fields={fields}
      data={data}
      components={components}
      onChange={setData}
    />
  );
};

Mantine example:

npm install @mantine/core@^9.0.0 @mantine/hooks@^9.0.0
import React from 'react';
import '@mantine/core/styles.css';
import { MantineProvider } from '@mantine/core';
import { Builder } from '@vojtechportes/react-query-builder';
import { components } from '@vojtechportes/react-query-builder/mantine/v9';

export const MyMantineBuilder = () => {
  return (
    <MantineProvider>
      <Builder
        fields={fields}
        data={data}
        components={components}
        onChange={setData}
      />
    </MantineProvider>
  );
};

Fluent UI example:

npm install @fluentui/react@^8.125.6
import React from 'react';
import { Builder } from '@vojtechportes/react-query-builder';
import { components } from '@vojtechportes/react-query-builder/fluentui/v8';

export const MyFluentUiBuilder = () => {
  return (
    <Builder
      fields={fields}
      data={data}
      components={components}
      onChange={setData}
    />
  );
};

Radix example:

npm install @radix-ui/themes@^1.1.2 @radix-ui/react-icons@^1.3.2

The Radix adapter also uses @radix-ui/react-icons, so install it alongside @radix-ui/themes.

import React from 'react';
import '@radix-ui/themes/styles.css';
import { Theme } from '@radix-ui/themes';
import { Builder } from '@vojtechportes/react-query-builder';
import { components } from '@vojtechportes/react-query-builder/radix/v1';

export const MyRadixBuilder = () => {
  return (
    <Theme>
      <Builder
        fields={fields}
        data={data}
        components={components}
        onChange={setData}
      />
    </Theme>
  );
};

ThemeProvider customizes the built-in default component set. It does not theme the MUI, ANTD, Mantine, Fluent UI, or Radix adapters, which keep their styling in their host UI libraries.

More adapter details:

Text Mode

Builder can optionally switch between the visual query UI and a SQL text editor view.

<Builder
  fields={fields}
  data={data}
  textMode
  defaultMode="text"
  onChange={setData}
/>;

For advanced text editing, the package also exposes @vojtechportes/react-query-builder/monaco. The built-in editor is lightweight and works without extra dependencies, while the Monaco integration is the recommended path when locked query segments must stay protected in text mode.

Read-only and Protected Editing

Rules and groups can be fully read-only or partially read-only through readOnly.targets.

  • Rule targets: field, operator, value
  • Group targets: combinator, negation

Targeted read-only controls stay visible but become non-editable. When using the Monaco text editor integration, protected SQL fragments remain locked while users can still edit the unlocked parts of the query.

By default, deleting a group is also blocked when that delete would indirectly remove read-only protected descendants. Set readOnlyProtectsDelete={false} on Builder if you want to disable that subtree delete protection.

Query Conversion

The library also provides parser and formatter helpers through subpath exports:

  • @vojtechportes/react-query-builder/formatQuery
  • @vojtechportes/react-query-builder/parseQuery

Supported formats are documented on the website:

Responsive Behavior

The default builder components include a compact responsive layout for medium-width screens.

  • Rule rows reflow to preserve field, operator, action, and value legibility when horizontal space gets tighter.
  • Multi-select values use a summarized closed state to avoid chip overflow.
  • The default responsive behavior applies automatically when you use the built-in components.
  • If you replace layout containers such as components.Rule or components.Group, your custom components are responsible for their own responsive behavior.

Responsive behavior is documented in more detail on the website: