UNPKG

ra-core

Version:

Core components of react-admin, a frontend Framework for building admin applications on top of REST services, using ES6, React

91 lines 5.81 kB
import * as React from 'react'; import { createContext, isValidElement, useContext, useRef, useState, } from 'react'; import { useTranslate } from "../../i18n/useTranslate.js"; import set from 'lodash/set.js'; /** * This hook provides support for suggestion creation in inputs which have choices. * * @param options The hook option * @param {ReactElement} options.create A react element which will be rendered when users choose to create a new choice. This component must call the `useCreateSuggestionContext` hook which provides `onCancel`, `onCreate` and `filter`. See the examples. * @param {React.ReactNode|string} options.createLabel Optional. The label for the choice item allowing users to create a new choice. Can be a translation key. Defaults to `ra.action.create`. * @param {React.ReactNode|string} options.createItemLabel Optional. The label for the choice item allowing users to create a new choice when they already entered a filter. Can be a translation key. The function and ttranslation will receive an `item` parameter. Providing this option will turn the create label when there is no filter to be a hint (i.e. a disabled item). * @param {any} options.createValue Optional. The value for the choice item allowing users to create a new choice. Defaults to `@@ra-create`. * @param {any} options.createHintValue Optional. The value for the (disabled) item hinting users on how to create a new choice. Defaults to `@@ra-create-hint`. * @param {String} options.filter Optional. The filter users may have already entered. Useful for autocomplete inputs for example. * @param {OnCreateHandler} options.onCreate Optional. A function which will be called when users choose to create a new choice, if the `create` option wasn't provided. * @param {Function} options.handleChange A function to pass to the input. Receives the same parameter as the original event handler and an additional newItem parameter if a new item was create. * * @returns {UseSupportCreateValue} An object with the following properties: * - getCreateItem: a function which will return the label of the choice for create a new choice. * - handleChange: a function which should be called when the input value changes. It will call the `onCreate` function if the value is the createValue. * - createElement: a React element to render after the input. It will be rendered when users choose to create a new choice. It renders null otherwise. * - getOptionDisabled: a function which should be passed to the input to disable the create choice when the filter is empty (to make it a hint). */ export const useSupportCreateSuggestion = (options) => { const { create, createLabel = 'ra.action.create', createItemLabel, createValue = '@@ra-create', createHintValue = '@@ra-create-hint', optionText = 'name', filter, handleChange, onCreate, } = options; const translate = useTranslate(); const [renderOnCreate, setRenderOnCreate] = useState(false); const filterRef = useRef(filter); return { createId: createValue, createHintId: createHintValue, getCreateItem: (filter) => { filterRef.current = filter; return set({ id: createItemLabel && !filter ? createHintValue : createValue, }, typeof optionText === 'string' ? optionText : 'name', filter && createItemLabel ? typeof createItemLabel === 'string' ? translate(createItemLabel, { item: filter, _: createItemLabel, }) : createItemLabel(filter) : typeof createLabel === 'string' ? translate(createLabel, { _: createLabel }) : createLabel); }, handleChange: async (eventOrValue) => { const value = eventOrValue?.target?.value || eventOrValue; const finalValue = Array.isArray(value) ? [...value].pop() : value; if (finalValue?.id === createValue || finalValue === createValue) { if (!isValidElement(create)) { if (!onCreate) { // this should never happen because the createValue is only added if a create function is provided // @see AutocompleteInput:filterOptions throw new Error('To create a new option, you must pass an onCreate function or a create element.'); } const newSuggestion = await onCreate(filter); if (newSuggestion) { handleChange(newSuggestion); return; } } else { setRenderOnCreate(true); return; } } handleChange(eventOrValue); }, createElement: renderOnCreate && isValidElement(create) ? (React.createElement(CreateSuggestionContext.Provider, { value: { filter: filterRef.current, onCancel: () => setRenderOnCreate(false), onCreate: item => { setRenderOnCreate(false); handleChange(item); }, } }, create)) : null, getOptionDisabled: option => option?.id === createHintValue || option === createHintValue, }; }; const CreateSuggestionContext = createContext(undefined); export const useCreateSuggestionContext = () => { const context = useContext(CreateSuggestionContext); if (!context) { throw new Error('useCreateSuggestionContext must be used inside a CreateSuggestionContext.Provider'); } return context; }; //# sourceMappingURL=useSupportCreateSuggestion.js.map