@rjsf/utils
Version:
Utility functions for @rjsf/core
105 lines (100 loc) • 4.28 kB
text/typescript
import get from 'lodash/get';
import { CONST_KEY, DEFAULT_KEY, PROPERTIES_KEY } from './constants';
import getDiscriminatorFieldFromSchema from './getDiscriminatorFieldFromSchema';
import getUiOptions from './getUiOptions';
import toConstant from './toConstant';
import { RJSFSchema, EnumOptionsType, EnumValue, StrictRJSFSchema, FormContextType, UiSchema } from './types';
/** Reorders `options` according to `order`, which may contain a `'*'` wildcard representing all
* remaining options in their original order. Options not listed in `order` (and not covered by
* a wildcard) are dropped.
*/
function applyEnumOrder<S extends StrictRJSFSchema = RJSFSchema>(
options: EnumOptionsType<S>[],
order: EnumValue[],
): EnumOptionsType<S>[] {
const optionsByValue = new Map(options.map((opt) => [String(opt.value), opt]));
const orderedKeys = new Set(order.filter((v) => v !== '*').map(String));
const rest = options.filter((opt) => !orderedKeys.has(String(opt.value)));
return order.flatMap((entry) => {
if (entry === '*') {
return rest;
}
const opt = optionsByValue.get(String(entry));
return opt ? [opt] : [];
});
}
/** Gets the list of options from the `schema`. If the schema has an enum list, then those enum values are returned. The
* label will be the same as the `value`.
*
* If the schema has a `oneOf` or `anyOf`, then the value is the list of either:
* - The `const` values from the schema if present
* - If the schema has a discriminator and the label using either the `schema.title` or the value. If a `uiSchema` is
* provided, and it has the `ui:enumNames` matched with `enum` or it has an associated `oneOf` or `anyOf` with a list of
* objects containing `ui:title` then the UI schema values will replace the values from the schema.
*
* @param schema - The schema from which to extract the options list
* @param [uiSchema] - The optional uiSchema from which to get alternate labels for the options
* @returns - The list of options from the schema
*/
export default function optionsList<T = any, S extends StrictRJSFSchema = RJSFSchema, F extends FormContextType = any>(
schema: S,
uiSchema?: UiSchema<T, S, F>,
): EnumOptionsType<S>[] | undefined {
if (schema.enum) {
let enumNames: string[] | Record<string | number, string> | undefined;
let enumOrder: EnumValue[] | undefined;
if (uiSchema) {
const { enumNames: uiEnumNames, enumOrder: uiEnumOrder } = getUiOptions<T, S, F>(uiSchema);
enumNames = uiEnumNames;
enumOrder = uiEnumOrder;
}
let options = schema.enum.map((value, i) => {
const label = Array.isArray(enumNames)
? enumNames[i] || String(value)
: enumNames?.[String(value)] || String(value);
return { label, value };
});
if (enumOrder) {
options = applyEnumOrder(options, enumOrder);
}
return options;
}
let altSchemas: S['anyOf'] | S['oneOf'] = undefined;
let altUiSchemas: UiSchema<T, S, F> | undefined = undefined;
if (schema.anyOf) {
altSchemas = schema.anyOf;
altUiSchemas = uiSchema?.anyOf;
} else if (schema.oneOf) {
altSchemas = schema.oneOf;
altUiSchemas = uiSchema?.oneOf;
}
// See if there is a discriminator path specified in the schema, and if so, use it as the selectorField, otherwise
// pull one from the uiSchema
let selectorField = getDiscriminatorFieldFromSchema<S>(schema);
if (uiSchema) {
const { optionsSchemaSelector = selectorField } = getUiOptions<T, S, F>(uiSchema);
selectorField = optionsSchemaSelector;
}
return (
altSchemas &&
altSchemas.map((aSchemaDef, index) => {
const { title } = getUiOptions<T, S, F>(altUiSchemas?.[index]);
const aSchema = aSchemaDef as S;
let value: EnumOptionsType<S>['value'];
let label = title;
if (selectorField) {
const innerSchema: S = get(aSchema, [PROPERTIES_KEY, selectorField], {}) as S;
value = get(innerSchema, DEFAULT_KEY, get(innerSchema, CONST_KEY));
label = label || innerSchema?.title || aSchema.title || String(value);
} else {
value = toConstant(aSchema);
label = label || aSchema.title || String(value);
}
return {
schema: aSchema,
label,
value,
};
})
);
}