User-friendly query builder for React. Demo: https://ukrbublik.github.io/react-awesome-query-builder
npm install react-awesome-query-builder-pd
npm i react-awesome-query-builder --save
`
For AntDesign widgets only:
`
npm i antd --save
`
For Material-UI widgets v5 only:
`
npm i @mui/material @mui/lab @mui/icons-material --save
`
See basic usage for minimum code example.
See API and config for documentation.
Demo apps:
- npm start - demo app with hot reload of demo code and local library code, uses TS, uses complex config to demonstrate anvanced usage.
- npm run sandbox-ts - demo app with hot reload of only demo code (uses latest version of library from npm), uses TS, uses AntDesign widgets.
- npm run sandbox-js - demo app with hot reload of only demo code (uses latest version of library from npm), not uses TS, uses vanilla widgets.
Usage
#### Minimal Javascript example with class component
`javascript
import React, {Component} from 'react';
import {Query, Builder, BasicConfig, Utils as QbUtils} from 'react-awesome-query-builder';
// For AntDesign widgets only:
import AntdConfig from 'react-awesome-query-builder/lib/config/antd';
import 'antd/dist/antd.css'; // or import "react-awesome-query-builder/css/antd.less";
// For Material-UI widgets only:
import MaterialConfig from 'react-awesome-query-builder/lib/config/material';
import 'react-awesome-query-builder/lib/css/styles.css';
import 'react-awesome-query-builder/lib/css/compact_styles.css'; //optional, for more compact styles
// Choose your skin (ant/material/vanilla):
const InitialConfig = AntdConfig; // or MaterialConfig or BasicConfig
// You need to provide your own config. See below 'Config format'
const config = {
...InitialConfig,
fields: {
qty: {
label: 'Qty',
type: 'number',
fieldSettings: {
min: 0,
},
valueSources: ['value'],
preferWidgets: ['number'],
},
price: {
label: 'Price',
type: 'number',
valueSources: ['value'],
fieldSettings: {
min: 10,
max: 100,
},
preferWidgets: ['slider', 'rangeslider'],
},
color: {
label: 'Color',
type: 'select',
valueSources: ['value'],
fieldSettings: {
listValues: [
{ value: 'yellow', title: 'Yellow' },
{ value: 'green', title: 'Green' },
{ value: 'orange', title: 'Orange' }
],
}
},
is_promotion: {
label: 'Promo?',
type: 'boolean',
operators: ['equal'],
valueSources: ['value'],
},
}
};
// You can load query value from your backend storage (for saving see Query.onChange())
const queryValue = {"id": QbUtils.uuid(), "type": "group"};
class DemoQueryBuilder extends Component {
state = {
tree: QbUtils.checkTree(QbUtils.loadTree(queryValue), config),
config: config
};
render = () => (
{...config}
value={this.state.tree}
onChange={this.onChange}
renderBuilder={this.renderBuilder}
/>
{this.renderResult(this.state)}
)
renderBuilder = (props) => (
)
renderResult = ({tree: immutableTree, config}) => (
Query string: {JSON.stringify(QbUtils.queryString(immutableTree, config))}
MongoDb query: {JSON.stringify(QbUtils.mongodbFormat(immutableTree, config))}
SQL where: {JSON.stringify(QbUtils.sqlFormat(immutableTree, config))}
JsonLogic: {JSON.stringify(QbUtils.jsonLogicFormat(immutableTree, config))}
)
onChange = (immutableTree, config) => {
// Tip: for better performance you can apply throttle - see examples/demo
this.setState({tree: immutableTree, config: config});
const jsonTree = QbUtils.getTree(immutableTree);
console.log(jsonTree);
// jsonTree can be saved to backend, and later loaded to queryValue
}
}
`
#### Minimal TypeScript example with function component
(Codesandbox)
`typescript
import React, { useState } from "react";
import { Query, Builder, Utils as QbUtils } from "react-awesome-query-builder";
// types
import {
JsonGroup,
Config,
ImmutableTree,
BuilderProps
} from "react-awesome-query-builder";
// For AntDesign widgets only:
import AntdConfig from "react-awesome-query-builder/lib/config/antd";
import "antd/dist/antd.css"; // or import "react-awesome-query-builder/css/antd.less";
// For Material-UI widgets only:
//import MaterialConfig from "react-awesome-query-builder/lib/config/material";
import "react-awesome-query-builder/lib/css/styles.css";
import "react-awesome-query-builder/lib/css/compact_styles.css"; //optional, for more compact styles
// Choose your skin (ant/material/vanilla):
const InitialConfig = AntdConfig; // or MaterialConfig or BasicConfig
// You need to provide your own config. See below 'Config format'
const config: Config = {
...InitialConfig,
fields: {
qty: {
label: "Qty",
type: "number",
fieldSettings: {
min: 0
},
valueSources: ["value"],
preferWidgets: ["number"]
},
price: {
label: "Price",
type: "number",
valueSources: ["value"],
fieldSettings: {
min: 10,
max: 100
},
preferWidgets: ["slider", "rangeslider"]
},
color: {
label: "Color",
type: "select",
valueSources: ["value"],
fieldSettings: {
listValues: [
{ value: "yellow", title: "Yellow" },
{ value: "green", title: "Green" },
{ value: "orange", title: "Orange" }
]
}
},
is_promotion: {
label: "Promo?",
type: "boolean",
operators: ["equal"],
valueSources: ["value"]
}
}
};
// You can load query value from your backend storage (for saving see Query.onChange())
const queryValue: JsonGroup = { id: QbUtils.uuid(), type: "group" };
export const Demo: React.FC = () => {
const [state, setState] = useState({
tree: QbUtils.checkTree(QbUtils.loadTree(queryValue), config),
config: config
});
const onChange = (immutableTree: ImmutableTree, config: Config) => {
// Tip: for better performance you can apply throttle - see examples/demo
setState({ tree: immutableTree, config: config });
const jsonTree = QbUtils.getTree(immutableTree);
console.log(jsonTree);
// jsonTree can be saved to backend, and later loaded to queryValue
};
const renderBuilder = (props: BuilderProps) => (
);
return (
{...config}
value={state.tree}
onChange={onChange}
renderBuilder={renderBuilder}
/>
Query string:{" "}
{JSON.stringify(QbUtils.queryString(state.tree, state.config))}
MongoDb query:{" "}
{JSON.stringify(QbUtils.mongodbFormat(state.tree, state.config))}
SQL where:{" "}
{JSON.stringify(QbUtils.sqlFormat(state.tree, state.config))}
JsonLogic:{" "}
{JSON.stringify(QbUtils.jsonLogicFormat(state.tree, state.config))}
);
};
`
API
$3
Props:
- {...config} - destructured query CONFIG
- value - query value in internal Immutable format
- onChange - callback when value changed. Params: value (in Immutable format), config.
- renderBuilder - function to render query builder itself. Takes 1 param props you need to pass into .
Notes:
- If you put query builder component inside Material-UI's or , please:
- use prop disableEnforceFocus={true} for dialog or popver
- set css .MuiPopover-root, .MuiDialog-root { z-index: 900 !important; } (or 1000 for AntDesign v3)
$3
Render this component only inside Query.renderBuilder() like in example above:
`js
renderBuilder = (props) => (
)
`
Wrapping in div.query-builder is necessary.
Optionally you can add class .qb-lite to it for showing action buttons (like delete rule/group, add, etc.) only on hover, which will look cleaner.
Wrapping in div.query-builder-container is necessary if you put query builder inside scrollable block.
$3
- Save, load:
#### getTree (immutableValue, light = true) -> Object
Convert query value from internal Immutable format to JS format.
You can use it to save value on backend in onChange callback of .
Tip: Use light = false in case if you want to store query value in your state in JS format and pass it as value of after applying loadTree() (which is not recommended because of double conversion). See issue #190
#### loadTree (jsValue, config) -> Immutable
Convert query value from JS format to internal Immutable format.
You can use it to load saved value from backend and pass as value prop to (don't forget to also apply checkTree()).
#### checkTree (immutableValue, config) -> Immutable
Validate query value corresponding to config.
Invalid parts of query (eg. if field was removed from config) will be always deleted.
Invalid values (values not passing validateValue in config, bad ranges) will be deleted if showErrorMessage is false OR marked with errors if showErrorMessage is true.
#### isValidTree (immutableValue) -> Boolean
If showErrorMessage in config.settings is true, use this method to check is query has bad values.
- Export:
#### queryString (immutableValue, config, isForDisplay = false) -> String
Convert query value to custom string representation. isForDisplay = true can be used to make string more "human readable".
#### mongodbFormat (immutableValue, config) -> Object
Convert query value to MongoDb query object.
#### sqlFormat (immutableValue, config) -> String
Convert query value to SQL where string.
#### elasticSearchFormat (immutableValue, config) -> Object
Convert query value to ElasticSearch query object.
#### jsonLogicFormat (immutableValue, config) -> {logic, data, errors}
Convert query value to JsonLogic format.
If there are no errors, logic will be rule object and data will contain all used fields with null values ("template" data).
- Import:
#### loadFromJsonLogic (jsonLogicObject, config) -> Immutable
Convert query value from JsonLogic format to internal Immutable format.
$3
This library uses configarion driven aprroach.
Config defines what value types, operators are supported, how they are rendered, imported, exported.
At minimum, you need to provide your own set of fields as in basic usage.
See CONFIG for full documentation.
Versions
Versions 4.x are backward-compatible with 2.x and 3.x.
It's recommended to update your version.
$3
| Version | Supported |
| ------- | ------------------ |
| 4.x | :white_check_mark: |
| 3.x | :white_check_mark: |
| 2.x | :white_check_mark: |
| 1.x | :warning: |
| 0.x | :x: |
$3
See CHANGELOG
$3
From v2.0 of this lib AntDesign is now optional (peer) dependency, so you need to explicitly include antd (4.x) in package.json of your project if you want to use AntDesign UI.
Please import AntdConfig from react-awesome-query-builder/lib/config/antd and use it as base for your config (see below in usage).
Alternatively you can use BasicConfig for simple vanilla UI, which is by default.
Support of other UI frameworks (like Bootstrap) are planned for future, see Other UI frameworks.
Development
Clone this repo and run npm start.
Open http://localhost:3001/ in a browser.
You will see demo app with hot reload of demo code and local library code.
$3
- modules - Main source code of library
- components - Core React components
- widgets - Components to render list of fields, operators, values of different types. Built with UI frameworks
- config - Basic config lives here. See CONFIG docs.
- export - Code for export to JsonLogic, MongoDb, SQL, ElasticSearch, plain string
- import - Code for import from JsonLogic
- actions - Redux actions
- stores/tree.js - Redux store
- index.d.ts - TS definitions
- css - Styles for query builder
- examples - Demo app with hot reload of demo code and local library code, uses TS, uses complex config to demonstrate anvanced usage.
- sandbox - Demo app with hot reload of only demo code (uses latest version of library from npm), uses TS, uses AntDesign widgets.
- sandbox_simple - Demo app with hot reload of only demo code (uses latest version of library from npm), not uses TS, uses vanilla widgets.
- tests - All tests are here. Uses Karma, Mocha, Chai, Enzyme
$3
- npm run install-all - Install npm packages in root, examples, sandboxes. Required for other scripts!
- npm test - Run tests with Karma and update coverage. Requires Node.js v10+
- npm run lint - Run ESLint and TSC (in root, tests, examples, sandboxes)
- npm run lint-fix - Run ESLint with --fix option (in root, tests, examples, sandboxes)
- npm run clean - Clean all data that can be re-generated (like node_modules, build, coverage)
- npm run smoke - Run tests, lint, build lib, build examples, build sandboxes. Recommended before making PR
- npm run build - Build npm module to lib, build minified production package to build
- npm run build-examples - Build demo with webpack to examples/build
Feel free to open PR to add new reusable types/widgets/operators (eg., regex operator for string, IP type & widget).
Pull Requests are always welcomed :)
$3
Currently there are 3 collections of widgets:
- antdesign widgets
- material widgets
- vanilla widgets
Let's say you want to create new collection of Bootstrap widgets to be used in this lib (and submit PR which is always welcomed!).
You can use vanilla widgets as skeleton.
Then to enable new widgets you need to create config overrides like this:
material config
Contributors
$3
This project exists thanks to all the people who contribute. [Contribute].
$3
Become a financial contributor and help us sustain our community. [Contribute]
#### Individuals
#### Organizations
Support this project with your organization. Your logo will show up here with a link to your website. [Contribute]
react-awesome-query-builder is being sponsored by the following tool; please help to support us by taking a look and signing up to a free trial
License
MIT. See also LICENSE.txt`