Hooks
Refer to the TypeScript reference page for information about the types and interfaces referenced below.
These hooks are used internally by React Query Builder and exported for use in custom components.
State access
React Query Builder uses Redux within a custom React context to manage state without interfering with existing Redux stores.
useQueryBuilderQuery
Retrieves the complete, current query object for the nearest ancestor QueryBuilder
component.
The optional parameter should only be used when retrieving a query object from a different QueryBuilder
than the nearest ancestor. It can be either a complete props object as passed to a custom component or any object matching the interface { schema: { qbId: string } }
.
function useQueryBuilderQuery(props?: { schema: { qbId: string } }): RuleGroupTypeAny;
As a React hook, this function must follow the appropriate rules. It provides access to the current query during the render phase, not in event handlers.
To access the current query from an event handler, use props.schema.getQuery()
instead.
useQueryBuilderSelector
Prefer useQueryBuilderQuery
if you only need the query object for the nearest ancestor QueryBuilder
component.
Returns the current query from the Redux store when used with getQuerySelectorById
(see example below).
function useQueryBuilderSelector(selector: (state: RqbState) => RuleGroupTypeAny): RuleGroupTypeAny;
Example:
const CustomValueEditor = (props: ValueEditorProps) => {
const fullQuery = useQueryBuilderSelector(getQuerySelectorById(props.schema.qbId));
// Here you can use utilities like `findPath(getParentPath(props.path), fullQuery)`.
// This allows you to, for example, inspect the parent group of the current rule.
// You can then count sibling rules, check for unique `field` selections, etc.
// That information can be used for validation, information to the user, or
// anything else in your render function.
};
Component logic
The core logic of each component is encapsulated in a reusable hook. Each main component is little more than a call to its respective hook plus the JSX that uses the properties returned from that hook. This enables creating a custom presentation layer without copying logic code from the default components.
The @react-querybuilder/native
package demonstrates this concept well. It calls the hooks from its own query builder, rule group, and rule components, but nests the sub-components within React Native View
elements instead of HTML div
elements used by react-querybuilder
components.
useRule
Called by the Rule
component. See source code for returned properties.
function useRule(props: RuleProps): {
// See source code for returned properties
};
useRuleGroup
Called by the RuleGroup
component. See source code for returned properties.
function useRuleGroup(props: RuleGroupProps): {
// See source code for returned properties
};
useValueEditor
Called by the ValueEditor
component. Accepts the same ValueEditorProps
as the component and returns an object with these properties: the value as an array, a multi-value handler, a processed version of the parseNumbers
prop, and the classname(s) to be applied to each editor in editor series.
function useValueEditor(props: ValueEditorProps): {
valueAsArray: any[];
multiValueHandler: (val: string, idx: number) => void;
parseNumberMethod: ParseNumberMethod;
valueListItemClassName: string;
};
This hook updates the value
as a side effect when these conditions are true:
skipHook
isfalse
(the value editors in the compatibility packages set this totrue
to avoid infinite loops)inputType
is"number"
operator
is something other than"between"
,"notBetween"
,"in"
, or"notIn"
valueEditorType
is not "multiselect"value
is an array or a string containing a comma (,
) and at least one non-whitespace character on either side.
If all of these conditions are met, handleOnChange
will be called with the first element of the array, or any characters before the first comma if value
is a string.
useValueSelector
Called by the ValueSelector
component. Returns the given value as an array (unchanged if already an array) and a memoized change handler.
function useValueSelector(
props: Pick<ValueSelectorProps, 'handleOnChange' | 'listsAsArrays' | 'multiple' | 'value'>
): {
onChange: (v: string | string[]) => void;
val?: string | any[];
};
useSelectElementChangeHandler
Used by the ValueSelector
component. Returns a memoized change handler designed specifically for HTML <select />
elements.
function useSelectElementChangeHandler(props: {
multiple?: boolean;
onChange: (v: string | string[]) => void;
}): (e: ChangeEvent<HTMLSelectElement>) => void;
useShiftActions
Used by the ShiftActions
component. Generates shiftUp
and shiftDown
methods to move a rule/group up or down in the query hierarchy, plus shiftUpDisabled
/shiftDownDisabled
to indicate whether either button should be disabled (shiftUpDisabled
is true
for the first rule/group in the root group; shiftDownDisabled
is true
for the last rule/group in the root group).
function useShiftActions(
props: { path: Path } & Pick<Schema, 'combinators' | 'dispatchQuery' | 'getQuery'>
): {
shiftDown: () => void;
shiftDownDisabled: boolean;
shiftUp: () => void;
shiftUpDisabled: boolean;
};
useStopEventPropagation
Used by the default Rule
and RuleGroup
components to prevent default behavior and stop event propagation (e.g., a MouseEvent
after clicking a <button>
). Takes a function that accepts a MouseEvent
and context parameters, then returns a new function that calls event.preventDefault()
and event.stopPropagation()
before calling the original function with the same arguments.
function useStopEventPropagation(
method: (event: React.MouseEvent, context: any) => void
): (event: React.MouseEvent, context: any) => void;
This hook is not used in RuleNative
and RuleGroupNative
—the Rule
and RuleGroup
components for @react-querybuilder/native
.
useQueryBuilder
Returns everything needed to render a QueryBuilder
component. Internally, this hook passes the result of useQueryBuilderSetup
to useQueryBuilderSchema
and returns the result of useQueryBuilderSchema
. The returned object combines the results of useQueryBuilderSchema
and useQueryBuilderSetup
(see below).
As with useQueryBuilderSchema
, this Hook must be called from a descendant component of QueryBuilderStateProvider
. See QueryBuilder
source code for an example.
This hook is unlikely to be necessary unless you're reimplementing the entire QueryBuilder
component structure.
useQueryBuilderSetup
Called by the internal component rendered by QueryBuilder
. Merges props and context values with the defaults and generates actions.
This hook is unlikely to be necessary unless you're reimplementing the entire QueryBuilder
component structure.
function useQueryBuilderSetup(props: QueryBuilderProps): {
qbId: qbId.current;
rqbContext: ReturnType<typeof useMergedContext>;
fields: OptionList<Field>;
fieldMap: Record<string, Field>;
combinators: OptionList<Combinator>;
getOperatorsMain: (field: string) => OptionList<Operator>;
getRuleDefaultOperator: (field: string) => string;
getValueEditorTypeMain: (field: string, operator: string) => ValueEditorType;
getValueSourcesMain: (field: string, operator: string) => ValueSources;
getValuesMain: (field: string, operator: string) => OptionList;
getRuleDefaultValue: (rule: RuleType) => any;
getInputTypeMain: (field: string, operator: string) => string;
createRule: () => RuleType;
createRuleGroup: () => RuleGroupTypeAny;
};
useQueryBuilderSchema
Called by the internal component rendered by QueryBuilder
. Returns everything needed to render a wrapper element (e.g., <div>
) and the root RuleGroup
element based on the provided props and the result from useQueryBuilderSetup
.
This Hook must be called from a descendant component of QueryBuilderStateProvider
. See QueryBuilder
source code for an example.
This hook is unlikely to be necessary unless you're reimplementing the entire QueryBuilder
component structure.
function useQueryBuilderSchema(
props: QueryBuilderProps,
setup: ReturnType<typeof useQueryBuilderSetup>
): QueryBuilderProps & {
actions: QueryActions;
rootGroup: RuleGroupTypeAny;
rootGroupDisabled: RuleGroupTypeAny;
queryDisabled: boolean;
rqbContext: ReturnType<typeof useMergedContext>;
schema: Schema;
translations: TranslationsFull;
wrapperClassName: string;
dndEnabledAttr: 'enabled' | 'disabled';
inlineCombinatorsAttr: 'enabled' | 'disabled';
combinatorPropObject: Pick<RuleGroupProps, 'combinator'>;
};
Other utilities
useMergedContext
Merges the values inherited from the nearest ancestor QueryBuilderContext.Provider
with the current component's props. For controlClassnames
, controlElements
, and translations
, options that are not defined through either context or props will fall back to the defaults.
function useMergedContext(props: QueryBuilderContextProps): QueryBuilderContextProps;
usePreferProp
Given a default value, a prop value, and a context value (all boolean
or undefined
), returns the first one that is not undefined
in the order of (1) prop, (2) context, (3) default.
function usePreferProp(default: boolean, prop?: boolean, context?: boolean): boolean;
usePrevious
Returns the value of a prop or state variable from the previous render.
function usePrevious<T>(prop: T): T | null;
Internal
These hooks log error messages to the console in certain situations (only in "development" mode). They encourage correct usage of React Query Builder and aren't intended for use in custom components.
useControlledOrUncontrolled
Logs an error to the console if any of the following are true:
- Both
query
anddefaultQuery
props are defined. - The
query
prop is defined during one render and undefined in a subsequent render. - The
query
prop is undefined during one render and defined in a subsequent render.
useDeprecatedProps
Logs an error to the console if any of the following are true:
QueryBuilder
is rendered withindependentCombinators
prop (see Independent combinators)RuleGroup
is rendered withcombinator
orrules
props (deprecated in favor ofruleGroup
)Rule
is rendered withfield
,operator
, orvalue
props (deprecated in favor ofrule
)
useReactDndWarning
Logs an error to the console if the enableDragAndDrop
prop is true
but the react-dnd
and react-dnd-html5-backend
dependencies are not loaded.