In this chapter we are going to create a provider for entity arrays which can be passed as a prop to table, dropdown and other components, which require a list of entities. We should work out a standardized way for the process of fetching entities, no matter where they are fetched from: Either from an API endpoint or just from a simple array or from somewhere else.
First, I'd like to introduce some terms of the Java world in relation to an array of objects:
The Collection in Java is a framework that provides an architecture to store and manipulate the group of objects. Java Collections can achieve all the operations that you perform on a data such as searching, sorting, insertion, manipulation, and deletion.
https://www.javatpoint.com/collections-in-java, 2022-06-21
As of this, I think it's a better idea to call the data itself - without any functionality - the "collection". The processor of operations like fetching, creating etc. could be called "collection manager" or in case of just providing collections, the "collection provider". Java does not make a difference about these two. The distinction between these two does also align with the naming in the React-Admin framework.
💡 Marmelab's React-Admin uses a similar pattern called data providers. Even creation, updating and deletion of one or multiple entities is supported.
I suggest only supporting the fetch of entities, with filter, sorting and pagination params by such a "collection provider". As of my experience in frontend programming, writing and storing data is often an individually designed process. Because of this we'll not support write operations in our collection providers, or at least for now.
Let's try to create an interface for the collection provider which takes a query holding the information about filtering, sorting and pagination params.
//src/packages/core/collection/collection.ts
export type Sorting = {
field: string;
direction: 'asc' | 'desc';
}[];
export type Filters = {
[field: string]: string | number | boolean;
};
// Let's wrap the data in an "entry" holding a recognizable "key", because
// not every entity should be forced to hold a primary key called "id".
// Therefore we need to be able to create a custom and recognizable key, directly after
// a fetch process has been finished.
export type Entry<Data = any> = {
key: string;
data: Data;
};
// Furthermore we should support pagination by a "limit" and "offset" param.
// This keeps the door open to reload some entries for every constellation.
export type CollectionQuery = {
search?: string;
offset: number;
limit: number;
filters: Filters;
sorting: Sorting;
};
// Let's provide a reusable default query
export const defaultQuery: CollectionQuery = {
offset: 0,
limit: 10,
filters: {},
sorting: [],
};
// Furthermore we should provide a collection information object holding at least all params
// of the query itself.
// With this we store the information for which filters, sorting and pagination params the delivered entries
// were received. This should normally correspond with the params of the query, but HAS NOT TO!
export type CollectionInfo = CollectionQuery & {
totalCount: number;
filteredCount: number;
};
export function createQuery(provider: CollectionProvider): CollectionQuery {
const latestQueryInfo = provider.latestQueryInfo ?? {};
return {
...defaultQuery,
...latestQueryInfo,
};
}
// To display useful information we should support things like "isFetching" in the
// state of a collection provider. With this we e.g. could show a loader icon or so.
export type CollectionProviderState<D = any> = {
key: string;
isFetching: boolean;
hasInitialFetchBeenDone: boolean;
entries: Entry<D>[];
latestQueryInfo?: CollectionInfo;
};
export type EntriesOperation = 'append' | 'replace';
export type CollectionProvider<D = any> = CollectionProviderState<D> & {
fetch: (query?: CollectionQuery, op?: EntriesOperation) => Promise<any>;
};
So now that we have an interface defined for a collection provider, let's create an implementation in the next step.
It is always useful to be able to provide entries from a simple array. But we don't know the format of an array entity yet. We only know it in the place where we want to implement the provider: In the component who is using the provider. So let's create a hook which does create us the desired provider from an array:
// src/packages/core/collection/arrayProvider.tsx
import { useEffect, useRef, useState } from 'react';
import {
CollectionProvider,
CollectionProviderState,
CollectionQuery,
defaultQuery,
EntriesOperation,
Entry,
} from './collection';
import { v4 } from 'uuid';
function getPaginatedEntries(entries: Entry[], query: CollectionQuery) {
return entries.slice(query.offset, query.offset + query.limit);
}
type EntriesToShowConfig = {
currentEntries: Entry[];
availableEntries: Entry[];
query?: CollectionQuery;
op?: EntriesOperation;
};
function createEntriesToShow(props: EntriesToShowConfig): Entry[] {
const query = props.query ?? defaultQuery;
let entriesToShow = props.availableEntries;
// todo: support filtering (not part of the tutorial yet)
// todo: support sorting (not part of the tutorial yet)
entriesToShow = getPaginatedEntries(entriesToShow, query);
entriesToShow =
props.op === 'append' && props.currentEntries ? [...props.currentEntries, ...entriesToShow] : entriesToShow;
return entriesToShow;
}
export type ArrayCollectionProviderProps<D = any> = {
dataArray: D[];
createEntryKey: (data: D) => string;
};
export function useArrayCollectionProvider<D = any>(props: ArrayCollectionProviderProps<D>): CollectionProvider<D> {
const availableEntriesRef = useRef<Entry<D>[]>(
props.dataArray.map((data) => ({
key: props.createEntryKey(data),
data,
}))
);
const [state, setState] = useState<CollectionProviderState>({
key: v4(),
isFetching: false,
entries: [],
hasInitialFetchBeenDone: false,
});
useEffect(() => {
availableEntriesRef.current = props.dataArray.map((data) => ({
key: props.createEntryKey(data),
data,
}));
setState({
...state,
entries: createEntriesToShow({
availableEntries: availableEntriesRef.current,
query: state.latestQueryInfo,
currentEntries: state.entries,
}),
hasInitialFetchBeenDone: true,
});
}, [JSON.stringify(props.dataArray), setState, availableEntriesRef]);
function fetch(query: CollectionQuery = defaultQuery, op: EntriesOperation = 'replace') {
return new Promise<void>((resolve) => {
resolve();
setState({
...state,
entries: createEntriesToShow({
availableEntries: availableEntriesRef.current,
query: query,
currentEntries: state.entries,
op,
}),
hasInitialFetchBeenDone: true,
});
});
}
return { ...state, fetch };
}
On the other hand, if we'd like to have a provider which should provide a collection from an API endpoint,
we need to write another implementation for this, which is called ApiCollectionProvider
or so.
As for now, we have no api package yet, so let's continue with the ArrayCollectionProvider
in the next steps.
Don't forget to export everything in an index.ts
:
// src/packages/core/collection/index.ts
export * from './collection';
export * from './arrayProvider';
Before we can benefit from the advantages of our ArrayCollectionProvider
we must create a component
which requires a CollectionProvider
property to work with a collection.
I think a dropdown component is a common use case for that.
So let's first write a selection component. This one does not use the collection provider yet, but is more an adapter for some MUI components:
// src/packages/core/form/CoreSelection.tsx
import React, { FC, ReactNode, useState } from 'react';
import { MenuItem, Select, InputLabel, FormControl } from '@mui/material';
import { v4 } from 'uuid';
import { useTranslator } from '@packages/core/i18n';
import { Entry } from '@packages/core/collection';
export type CoreSelectionProps<Data = any> = {
variant?: 'standard' | 'outlined' | 'filled';
margin?: 'dense' | 'normal' | 'none';
label?: ReactNode;
options: Entry<Data>[];
chosenOption: null | Entry<Data>;
renderOption: (entry: Entry<Data>) => ReactNode;
canChooseNone?: boolean;
onChange?: (option: null | Entry<Data>) => void;
disabled?: boolean;
readOnly?: boolean;
error?: boolean;
errorSection?: ReactNode;
fullWidth?: boolean;
size?: 'small' | 'medium';
};
export const CoreSelection: FC<CoreSelectionProps> = (props) => {
const { t } = useTranslator();
const [labelId] = useState(v4());
const shouldChosenOptionBeAddedToOptions =
props.chosenOption && !props.options.find((o) => o.key === props.chosenOption?.key);
const options: Entry[] =
props.chosenOption && shouldChosenOptionBeAddedToOptions
? [props.chosenOption, ...props.options]
: props.options;
const shouldNoneOptionBeShown = !props.chosenOption || props.canChooseNone;
function getEntryByKeyOrNull(key: string): null | Entry {
const entry = options.find((e) => e.key === key);
if (!entry) {
return null;
}
return entry;
}
return (
<FormControl
margin={props.margin}
variant={props.variant}
error={props.error}
fullWidth={props.fullWidth}
size={props.size}>
{props.label && <InputLabel id={labelId}>{props.label}</InputLabel>}
<Select
readOnly={props.readOnly}
disabled={props.disabled}
labelId={labelId}
value={props.chosenOption?.key ?? ''}
onChange={(event) => {
if (props.onChange) {
props.onChange(getEntryByKeyOrNull(event.target.value));
}
}}
label={props.label}>
{shouldNoneOptionBeShown && (
<MenuItem value="">
<em>{t('core.form.selection.choose')}</em>
</MenuItem>
)}
{options.map((o) => (
<MenuItem key={o?.key} value={o?.key}>
{props.renderOption(o)}
</MenuItem>
))}
</Select>
{props.errorSection}
</FormControl>
);
};
Next, let's support a new form element type SINGLE_SELECTION
, by doing the following steps:
// src/packages/core/form/formElementState.ts
// Add following import statement:
import { Entry } from '@packages/core/collection';
// Make the FormElementTypes enum look like so:
export enum FormElementTypes {
TEXT_FIELD = 'textField-c615d5de',
CHECKBOX = 'checkbox-c615d5de',
SINGLE_SELECTION = 'singleSelection-c615d5de',
}
// Create a new form element state for a single selection:
export type SingleSelectionState<D = any> = GenericFormElementState<
FormElementTypes.SINGLE_SELECTION,
{ chosenOption: null | Entry<D>; messages: Message[] }
>;
// Add the state to the FormElementState union type:
export type FormElementState = TextFieldState | CheckboxState | SingleSelectionState;
// Write a single state selection factory function, to reuse it later:
export function createSingleSelectionState<D = any>(
partial: Partial<Omit<SingleSelectionState, 'type'>> = {}
): SingleSelectionState<D> {
return {
messages: [],
chosenOption: null,
...partial,
type: FormElementTypes.SINGLE_SELECTION,
};
}
As a next step, we should create the SingleSelection
component which is using the CoreSelection
:
// src/packages/core/form/SingleSelection.tsx
import React, { FC, ReactNode } from 'react';
import { CollectionProvider, Entry } from '@packages/core/collection';
import { CoreSelection } from './CoreSelection';
import { Messages } from './Messages';
import { SingleSelectionState } from './formElementState';
export type SingleSelectionProps<Data = any> = {
provider: CollectionProvider<Data>;
data: SingleSelectionState<Data>;
renderOption: (entry: Entry<Data>) => ReactNode;
onChangeData?: (data: SingleSelectionState<Data>) => void;
variant?: 'standard' | 'outlined' | 'filled';
margin?: 'dense' | 'normal' | 'none';
label?: ReactNode;
canChooseNone?: boolean;
disabled?: boolean;
readOnly?: boolean;
fullWidth?: boolean;
size?: 'small' | 'medium';
};
export const SingleSelection: FC<SingleSelectionProps> = (props) => {
const hasErrorMessages = !!props.data.messages.find((m) => m.severity === 'error');
return (
<CoreSelection
variant={props.variant}
margin={props.margin}
label={props.label}
canChooseNone={props.canChooseNone}
disabled={props.disabled}
readOnly={props.readOnly}
fullWidth={props.fullWidth}
size={props.size}
options={props.provider.entries}
chosenOption={props.data.chosenOption}
onChange={(chosenOption) => {
if (props.onChangeData) {
props.onChangeData({ ...props.data, chosenOption });
}
}}
renderOption={props.renderOption}
error={hasErrorMessages}
errorSection={props.data.messages.length ? <Messages messages={props.data.messages} /> : undefined}
/>
);
};
Don't forget to support the automatic form element message enrichment by adding the
FormElementTypes.SINGLE_SELECTION
in the switch statement of formElementStatesEnrichment.ts
,
like we have done it for the TextField
and the Checkbox
.
Finally, let's add the translation key for core.form.selection.choose
:
// src/components/translations/deCH.json
"core": {
// other keys...
"form": {
"selection": {
"choose": "Bitte wählen"
}
}
// other keys...
}
and
// src/components/translations/enUS.json
"core": {
// other keys...
"form": {
"selection": {
"choose": "Please choose"
}
}
}
Now, we are able to benefit from the work we have done so far. Let's create a gender selection in the user registration form with no effort.
// src/pages/auth/RegisterPage.tsx
// Additionally import the single selection state and its factory function
import {
createSingleSelectionState,
SingleSelectionState,
// others...
} from '@packages/core/form';
// Add the following import statements
import { SingleSelection } from '@packages/core/form/SingleSelection';
import { Entry, useArrayCollectionProvider } from '@packages/core/collection';
// Create the genders array
type GenderId = 'f' | 'm' | 'o';
const genderIds: GenderId[] = ['f', 'm', 'o'];
// Extend the registration form state with the gender selection:
type RegistrationFormState = {
genderSelection: SingleSelectionState<GenderId>;
// others...
};
// Extend the factory function as well:
function createRegistrationFormState(): RegistrationFormState {
return {
genderSelection: createSingleSelectionState(),
// others...
};
}
// Insert the following hook in the top of the RegistrationForm component:
const genderIdsProvider = useArrayCollectionProvider<GenderId>({
dataArray: genderIds,
createEntryKey: (gId) => gId,
});
// Add the gender selection component as the first child of the <Form> component like so:
<SingleSelection
data={props.data.genderSelection}
onChangeData={(data) => props.onChangeData({ ...props.data, genderSelection: data })}
provider={genderIdsProvider}
renderOption={(e: Entry<GenderId>) => {
switch (e.data) {
case 'f':
return t('pages.registerPage.genderOptions.female');
case 'm':
return t('pages.registerPage.genderOptions.male');
case 'o':
return t('pages.registerPage.genderOptions.other');
default:
console.error(`genderId "${e.data}" is not supported!`);
return null;
}
}}
label={t('pages.registerPage.gender')}
variant="outlined"
margin="dense"
canChooseNone
fullWidth
/>
Don't forget to add the translation keys:
// src/components/translations/deCH.json
"pages": {
// other keys...
"registerPage": {
"gender": "Geschlecht",
"genderOptions": {
"female": "Weiblich",
"male": "Männlich",
"other": "Andere"
},
// other keys...
}
// other keys...
}
and
// src/components/translations/enUS.json
"pages": {
// other keys...
"registerPage": {
"gender": "Gender",
"genderOptions": {
"female": "Female",
"male": "Male",
"other": "Other"
},
// other keys...
}
// other keys...
}
Cool, let's check it in the browser!