Formule is a powerful, user-friendly, extensible and mobile-friendly form building library based on JSON Schema and RJSF, which aims to make form creation easier for both technical and non-technical people.
It originated from the need of a flexible tool for physicists at CERN to create their custom forms in the CERN Analysis Preservation application (a process that was originally done by the CAP team who had to manually define the JSON schemas for every member experiment) in a zero-code fashion. This tool proved to be very useful for us to more easily scalate and expand, reaching a wider audience here at CERN. So, we thought it could also be useful for other people and decided to decouple it from CAP and release it as an open source library.
Warning
react-formule has just come out and is undergoing active development, so please feel free to share any issue you find with us and/or to contribute!
A simple setup (see ./formule-demo
) could look like this:
Formule consists of the following main components:
FormuleContext
: Formule components need to be wrapped by a FormuleContext. It also allows you to provide an antd theme and your own custom fields and widgets.- The form editor, which has been split into three different components that work together for more flexibility:
SelectOrEdit
(or, separately,SelectFieldType
andPropertyEditor
): You can select fields to add to the form and customize their properties.SchemaPreview
: A tree view of the fields where you can rearrange or select fields to be edited.FormPreview
: A live, iteractive preview of the form which lets you toggle between the editable and the published version. If you only want to show the editable version, useEditablePreview
instead.
FormuleForm
: You can use it to display a form (JSON Schema) generated by Formule. The editable version will be displayed by default. You can passisPublished
if you want to see the published version.
It also exports the following functions:
initFormuleSchema
: Inits or resets the JSONSchema. You can also load an existing schema by passing it as an argument.getFormuleState
: Formule has its own internal redux state. You can retrieve it at any moment if you so require for more advanced use cases. If you want to continuosly synchronize the Formule state in your app, you can pass a callback function to FormuleContext instead (see below), which will be called every time the form state changes.
And the following utilities:
CodeEditor
: Useful if you want to edit the JSON schemas (or any other code) manually.CodeViewer
: Useful if you want to visualize the JSON schemas that are being generated (as you can see in the demo).CodeDiffViewer
: Useful if you want to compare two different JSON schemas, for example to see the changes since the last save.
As well as the following utility functions to handle saving and loading schemas from local storage if you need and for unsaved change detection:
getAllFromLocalStorage
saveToLocalStorage
deleteFromLocalStorage
loadFromLocalStorage
isUnsaved
Have a look at src/index.ts
to see all exported components and functions. You can also have a look at formule-demo
to see how they are used there.
Formule includes a variety of predefined field types, grouped in three categories:
- Simple fields:
Text
,Text area
,Number
,Checkbox
,Switch
,Radio
,Select
andDate
fields. - Collections:
Object
: Use it of you want to group fields or to add several of them inside of aList
.List
: It allows you to have as many instances of a field orObject
as you want.Accordion
: When containing aList
, it works as aList
with collapsible entries.Layer
: When containing aList
, it works as aList
whose entries will open in a dialog window.Tab
: It's commonly supposed to be used as a wrapper around the rest of the elements. You will normally want to add anObject
inside and you can use it to separate the form in different pages or sections.
- Advanced fields: More complex or situational fields such as
URI
,Rich/Latex editor
,Tags
,ID Fetcher
andCode Editor
.
You can freely remove some of these predefined fields and add your own custom fields and widgets following the JSON Schema specifications. More details below.
All of these items contain different settings that you can tinker with, separated into Schema Settings (generally affecting how the field works) and UI Schema Settings (generally affecting how the field looks like).
npm install react-formule
# or
yarn add react-formule
import {
FormuleContext,
SelectOrEdit,
SchemaPreview,
FormPreview,
} from "react-formule";
return (
<FormuleContext>
<SelectOrEdit />
<SchemaPreview />
<FormPreview />
</FormuleContext>
);
Override (if existing) or create your own field types (rjsf type definitions) similarly to how it's done in fieldTypes.jsx
, passing them as customFieldTypes
. Implement your own custom fields and widgets (react components) by passing them as customFields
and/or customWidgets
(see forms/fields/
and forms/widgets/
for examples). If you also want to use a different published version of a field or widget, pass the component in customPublishedFields
or customPublishedWidgets
.
const customFieldTypes = {
advanced: {
myfield: {
title: ...
...
}
}
}
const customFields: {
myfield: MyField // react component
}
<FormuleContext
theme={{token: {colorPrimary: "blue"}}} // antd theme
customFieldTypes={customFieldTypes}
customFields={customFields}
customWidgets={...}
customPublishedFields={...}
customPublishedWidgets={...}>
// ...
</FormuleContext>
If you use Formule to edit existing JSON schemas that include extra fields (e.g. metadata fields) that you don't want to show up in the Formule editor (i.e. in SchemaPreview
and SchemaTree
), you can use transformSchema
to exclude them:
const transformSchema = (schema) => {
// Remove properties here...
return transformedSchema;
};
<FormuleContext transformSchema={transformSchema}>/* ... */</FormuleContext>;
You can add a custom transformErrors
function to process, edit or filter the errors from RJSF in the way that best suits our needs:
const transformErrors = (errors) => {
return errors.filter(...)
};
<FormuleForm transformErrors={transformErrors} />
If you want to run some logic in your application every time the current Formule state changes in any way (e.g. to run some action every time a new field is added to the form) you can pass a function to be called back when that happens:
const handleFormuleStateChange = (newState) => {
// Do something when the state changes
};
<FormuleContext synchonizeState={handleFormuleStateChange}>
// ...
</FormuleContext>;
Alternatively, you can pull the current state on demand by calling getFormuleState
at any moment.
Tip
For more examples, feel free to browse around formule-demo and the CERN Analysis Preservation repository, where we use all the features mentioned above.
Apart from trying the online demo you can clone the repo and run formule-demo
to play around. Follow the instructions in its README: it will explain how to install react-formule
as a local dependency so that you can modify Formule and test the changes live in your host app, which will be ideal if you want to troubleshoot or contribute to the project. Your contributions are welcome! π