DataContext
DataContext builds a surrounding React context that binds an entire source dataset together with the fields placed within. It enables fields and other components to retrieve data from the source data using path parameters that identify where in the source data the target value is located, and the same components will report changes to the data back so the context can update the dataset.
Example of using the DataContext.Provider:
<DataContext.Provider data={data} onChange={handleChange}><Field.String path="/firstName" /></DataContext.Provider>
For a more complete feature set tailored to building forms, please use Form.Handler. It uses DataContext internally.
Components
At
DataContext.At makes it possible to dig into a data set to set a pointer as the root for sub components, as well as iterate array-data.
Context
The context object used in DataContext.Provider.
Provider
DataContext.Provider is the context provider that has to wrap the features if components of Field and Value is to be used with a common source instead of distributing values and events individually.
More details
If you don't want to repeat all the logic that drills down to values in the source data, and ensure that changes are sent to the right place, you can surround the components with a DataContext.Provider component. This means that you feed the form with source data in one place, and give it only one onChange callback. Then you only send the individual fields instructions about where in the data set the value that field is to process is located. The components then communicate internally and ensure that the values are retrieved and sent to the correct location.
The reference to a specific field's value in the dataset is given with a property called path. Paths are defined in a syntax called JSON Pointer, which is basically a slash-separated string that can go several levels, and consist of both object-properties and array indexes. Examples of paths are: /firstName, /nested/path/to/value and /list/2/keyInThirdObject. More information about JSON Pointers can be found on the website of JSON Schema.
In practice, this means that you can go from:
const handleChange = useCallback((path, value) => {// Update external state})return (<div id="my-form"><Field.Stringvalue={data.firstName}onChange={(value) => handleChange('firstName', value)}/><Field.Stringvalue={data.lastName}onChange={(value) => handleChange('lastName', value)}/><Field.Emailvalue={data.email}onChange={(value) => handleChange('email', value)}/><Field.Stringlabel="Special non-standardized value"value={data.specialValue}onChange={(value) => handleChange('specialValue', value)}/></div>)
to:
const handleChange = useCallback((path, value) => {// Update external state})return (<DataContext.Provider data={data} onChange={handleChange}><Field.String path="/firstName" /><Field.String path="/lastName" /><Field.Email path="/email" /><Field.Stringpath="/specialValue"label="Special non-standardized value"/></DataContext.Provider>)
This abstracts away some logic that many are used to having available for debugging and adjustments, which can be unfamiliar and difficult to get used to. The goal of the way this is designed is for it to be well tested and predictable, so that you don't need to have this boilerplate logic available. In addition, properties from the individual components make them flexible in use, and this can be continuously expanded to cover recurring needs from implementations.
Error handling
Besides how the forms are built up and the link to the surrounding data flow, these form components must ensure that the user experience is as much as possible in line with the way we have defined that it should work in practice. An example of this is when the error messages appear on the screen. Both the individual input component and any surrounding DataContext.Provider component hold an internal state that says whether the value in the field has an error or not. In addition, it has a separate state that states whether error messages should be displayed or not.
An example of what this leads to is when a field has an invalid value, for example because the field starts empty but is required. Or if the field requires a given syntax (such as national identity number), then the error message is not displayed before or at the same time as the user fills in the field in question. However, when the user jumps out of the field, the error message will appear if the value is still not valid based on the validation properties the component has received. When the user then starts to adjust the field in question, the error message is hidden again until they jump out of the field. In addition, a surrounding DataContext.Provider will check all the fields for errors, so that you do not get to the next step in a step-divided form, or can send the form and trigger onSubmit if there are still fields on the screen that have errors.
In the case of forms divided into several wizard steps, the combination of the components DataContext.Provider and Wizard.Step will also ensure that only the fields that are visible on the screen (for the relevant step, or based on what is hidden or shown via the Visibility component) provide a basis for whether one can proceed in the process or not.
Hierarchically overridable properties
Configuration of the form functionality through properties for all components can be hierarchically overridden. This means that the further into the component structure you get, the higher priority properties have. For example, a component that is given a path to retrieve data from the DataContext.Provider will rather prioritize a value property that the component receives directly if both parts are available:
<DataContext.Provider data={{ foo: 'I am the chosen one!' }}><Value.String path="/foo" /></DataContext.Provider>
<DataContext.Provider data={{ foo: 'I am not chosen :-(' }}><Value.String path="/foo" value="I am the one!" /></DataContext.Provider>
In the same way, components that have text properties built in, such as field label and error message for required field on Field.Email, will choose what it receives instead of the default values if both are available:
<Field.Email />// Gets the default label, and the email-pattern validation.
<Field.Email label="Send me e-mail on this address" />// Gets the custom label, but still the default email-pattern validation.