2016-03-22 12:25:23 +13:00
|
|
|
import React from 'react';
|
|
|
|
import { connect } from 'react-redux';
|
|
|
|
import { bindActionCreators } from 'redux';
|
2016-04-26 12:01:40 +12:00
|
|
|
import * as formActions from 'state/form/FormActions';
|
2016-04-21 21:59:44 +12:00
|
|
|
import * as schemaActions from 'state/schema/SchemaActions';
|
|
|
|
import SilverStripeComponent from 'lib/SilverStripeComponent';
|
|
|
|
import FormComponent from 'components/Form/Form';
|
2016-04-12 16:47:24 +12:00
|
|
|
import FormActionComponent from 'components/FormAction/FormAction';
|
2016-04-21 21:59:44 +12:00
|
|
|
import TextField from 'components/TextField/TextField';
|
|
|
|
import HiddenField from 'components/HiddenField/HiddenField';
|
|
|
|
import GridField from 'components/GridField/GridField';
|
2016-03-29 15:38:48 +13:00
|
|
|
import fetch from 'isomorphic-fetch';
|
2016-04-07 22:35:52 +12:00
|
|
|
import deepFreeze from 'deep-freeze';
|
2016-04-12 16:47:24 +12:00
|
|
|
import backend from 'lib/Backend';
|
2016-04-26 11:25:27 +12:00
|
|
|
import merge from 'merge';
|
2016-03-29 15:38:48 +13:00
|
|
|
|
|
|
|
import es6promise from 'es6-promise';
|
|
|
|
es6promise.polyfill();
|
2016-03-22 12:25:23 +13:00
|
|
|
|
|
|
|
// Using this to map field types to components until we implement dependency injection.
|
2016-03-31 10:45:54 +13:00
|
|
|
const fakeInjector = {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Components registered with the fake DI container.
|
|
|
|
*/
|
|
|
|
components: {
|
|
|
|
TextField,
|
|
|
|
GridField,
|
|
|
|
HiddenField,
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the component matching the passed component name.
|
|
|
|
* Used when a component type is provided bt the form schema.
|
|
|
|
*
|
|
|
|
* @param string componentName - The name of the component to get from the injector.
|
|
|
|
*
|
|
|
|
* @return object|null
|
|
|
|
*/
|
|
|
|
getComponentByName(componentName) {
|
|
|
|
return this.components[componentName];
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Default data type to component mappings.
|
|
|
|
* Used as a fallback when no component type is provided in the form schema.
|
|
|
|
*
|
|
|
|
* @param string dataType - The data type provided by the form schema.
|
|
|
|
*
|
|
|
|
* @return object|null
|
|
|
|
*/
|
|
|
|
getComponentByDataType(dataType) {
|
|
|
|
switch (dataType) {
|
2016-04-12 16:47:24 +12:00
|
|
|
case 'Text':
|
2016-03-31 10:45:54 +13:00
|
|
|
return this.components.TextField;
|
|
|
|
case 'Hidden':
|
|
|
|
return this.components.HiddenField;
|
|
|
|
case 'Custom':
|
|
|
|
return this.components.GridField;
|
|
|
|
default:
|
|
|
|
return null;
|
2016-03-22 12:25:23 +13:00
|
|
|
}
|
2016-03-31 10:45:54 +13:00
|
|
|
},
|
|
|
|
};
|
2016-03-22 12:25:23 +13:00
|
|
|
|
|
|
|
export class FormBuilderComponent extends SilverStripeComponent {
|
|
|
|
|
2016-03-31 10:45:54 +13:00
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
|
|
|
|
|
|
|
this.formSchemaPromise = null;
|
2016-04-07 17:15:14 +12:00
|
|
|
this.state = { isFetching: false };
|
2016-04-12 16:47:24 +12:00
|
|
|
|
|
|
|
this.mapActionsToComponents = this.mapActionsToComponents.bind(this);
|
2016-04-12 10:24:16 +12:00
|
|
|
this.mapFieldsToComponents = this.mapFieldsToComponents.bind(this);
|
2016-04-12 16:47:24 +12:00
|
|
|
this.handleFieldUpdate = this.handleFieldUpdate.bind(this);
|
|
|
|
this.handleSubmit = this.handleSubmit.bind(this);
|
|
|
|
this.removeForm = this.removeForm.bind(this);
|
2016-04-21 14:38:02 +12:00
|
|
|
this.getFormId = this.getFormId.bind(this);
|
|
|
|
this.getFormSchema = this.getFormSchema.bind(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the schema for this form
|
|
|
|
*
|
|
|
|
* @returns {array}
|
|
|
|
*/
|
|
|
|
getFormSchema() {
|
|
|
|
return this.props.schemas[this.props.schemaUrl];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the ID for this form
|
|
|
|
*
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
getFormId() {
|
|
|
|
const schema = this.getFormSchema();
|
|
|
|
if (schema) {
|
|
|
|
return schema.id;
|
|
|
|
}
|
|
|
|
return null;
|
2016-04-07 17:15:14 +12:00
|
|
|
}
|
2016-03-31 10:45:54 +13:00
|
|
|
|
2016-04-07 17:15:14 +12:00
|
|
|
componentDidMount() {
|
2016-03-31 10:45:54 +13:00
|
|
|
this.fetch();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fetches data used to generate a form. This can be form schema and or form state data.
|
|
|
|
* When the response comes back the data is saved to state.
|
|
|
|
*
|
|
|
|
* @param boolean schema - If form schema data should be returned in the response.
|
|
|
|
* @param boolean state - If form state data should be returned in the response.
|
|
|
|
*
|
|
|
|
* @return object - Promise from the AJAX request.
|
|
|
|
*/
|
2016-04-12 16:47:24 +12:00
|
|
|
fetch(schema = true, state = true) {
|
2016-03-31 10:45:54 +13:00
|
|
|
const headerValues = [];
|
|
|
|
|
2016-04-07 17:15:14 +12:00
|
|
|
if (this.state.isFetching === true) {
|
2016-03-31 10:45:54 +13:00
|
|
|
return this.formSchemaPromise;
|
2016-03-22 12:25:23 +13:00
|
|
|
}
|
|
|
|
|
2016-03-31 10:45:54 +13:00
|
|
|
if (schema === true) {
|
|
|
|
headerValues.push('schema');
|
2016-03-22 12:25:23 +13:00
|
|
|
}
|
|
|
|
|
2016-03-31 10:45:54 +13:00
|
|
|
if (state === true) {
|
|
|
|
headerValues.push('state');
|
2016-03-22 12:25:23 +13:00
|
|
|
}
|
|
|
|
|
2016-03-31 10:45:54 +13:00
|
|
|
this.formSchemaPromise = fetch(this.props.schemaUrl, {
|
|
|
|
headers: { 'X-FormSchema-Request': headerValues.join() },
|
|
|
|
credentials: 'same-origin',
|
|
|
|
})
|
|
|
|
.then(response => response.json())
|
|
|
|
.then(json => {
|
2016-04-12 16:47:24 +12:00
|
|
|
const formSchema = Object.assign({}, { id: json.id, schema: json.schema });
|
|
|
|
const formState = Object.assign({}, json.state);
|
|
|
|
|
2016-04-12 10:24:16 +12:00
|
|
|
// TODO See "Enable once <CampaignAdmin> ..." below
|
2016-04-12 16:47:24 +12:00
|
|
|
// this.setState({ isFetching: false });
|
|
|
|
|
|
|
|
if (typeof formSchema.id !== 'undefined') {
|
|
|
|
const defaultData = {
|
|
|
|
SecurityID: this.props.config.SecurityID,
|
|
|
|
};
|
|
|
|
|
|
|
|
if (formSchema.schema.actions.length > 0) {
|
|
|
|
defaultData[formSchema.schema.actions[0].name] = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.submitApi = backend.createEndpointFetcher({
|
|
|
|
url: formSchema.schema.attributes.action,
|
|
|
|
method: formSchema.schema.attributes.method,
|
|
|
|
defaultData,
|
|
|
|
});
|
|
|
|
|
|
|
|
this.props.schemaActions.setSchema(formSchema);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof formState.id !== 'undefined') {
|
2016-04-26 12:01:40 +12:00
|
|
|
this.props.formActions.addForm(formState);
|
2016-04-12 16:47:24 +12:00
|
|
|
}
|
2016-03-31 10:45:54 +13:00
|
|
|
});
|
|
|
|
|
2016-04-12 10:24:16 +12:00
|
|
|
// TODO Enable once <CampaignAdmin> is initialised via page.js route callbacks
|
|
|
|
// At the moment, it's running an Entwine onadd() rule which ends up
|
|
|
|
// rendering the index view, and only then calling route.start() to
|
|
|
|
// match the detail view (admin/campaigns/set/:id/show).
|
|
|
|
// This causes the form builder to be unmounted during a fetch() call.
|
|
|
|
// this.setState({ isFetching: true });
|
2016-03-31 10:45:54 +13:00
|
|
|
|
|
|
|
return this.formSchemaPromise;
|
|
|
|
}
|
|
|
|
|
2016-04-12 16:47:24 +12:00
|
|
|
/**
|
|
|
|
* Update handler passed down to each form field as a prop.
|
|
|
|
* Form fields call this method when their state changes.
|
|
|
|
*
|
|
|
|
* You can pass an optional callback as the third param. This can be used to
|
|
|
|
* implement custom behaviour. For example you can use `createFn` hook from
|
|
|
|
* your controller context like this.
|
|
|
|
*
|
|
|
|
* controller.js
|
|
|
|
* ...
|
|
|
|
* detailEditFormCreateFn(Component, props) {
|
|
|
|
* const extendedProps = Object.assign({}, props, {
|
|
|
|
* handleFieldUpdate: (event, updates) => {
|
|
|
|
* props.handleFieldUpdate(event, updates, (formId, updateFieldAction) => {
|
|
|
|
* const customUpdates = Object.assign({}, updates, {
|
|
|
|
* value: someCustomParsing(updates.value),
|
|
|
|
* };
|
|
|
|
*
|
|
|
|
* updateFieldAction(formId, customUpdates);
|
|
|
|
* });
|
|
|
|
* },
|
|
|
|
* });
|
|
|
|
*
|
|
|
|
* return <Component {...extendedProps} />;
|
|
|
|
* }
|
|
|
|
* ...
|
|
|
|
*
|
|
|
|
* @param {object} event - Change event from the form field component.
|
|
|
|
* @param {object} updates - Values to set in state.
|
|
|
|
* @param {string} updates.id - Field ID. Required to identify the field in the store.
|
|
|
|
* @param {function} [fn] - Optional function for custom behaviour. See example in description.
|
|
|
|
*/
|
|
|
|
handleFieldUpdate(event, updates, fn) {
|
|
|
|
if (typeof fn !== 'undefined') {
|
2016-04-26 12:01:40 +12:00
|
|
|
fn(this.getFormId(), this.props.formActions.updateField);
|
2016-04-12 16:47:24 +12:00
|
|
|
} else {
|
2016-04-26 12:01:40 +12:00
|
|
|
this.props.formActions.updateField(this.getFormId(), updates);
|
2016-04-12 16:47:24 +12:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Form submission handler passed to the Form Component as a prop.
|
|
|
|
* Provides a hook for controllers to access for state and provide custom functionality.
|
|
|
|
*
|
|
|
|
* For example:
|
|
|
|
*
|
|
|
|
* controller.js
|
|
|
|
* ```
|
|
|
|
* constructor(props) {
|
|
|
|
* super(props);
|
|
|
|
* this.handleSubmit = this.handleSubmit.bind(this);
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* handleSubmit(event, fieldValues, submitFn) {
|
|
|
|
* event.preventDefault();
|
|
|
|
*
|
|
|
|
* // Apply custom validation.
|
|
|
|
* if (!this.validate(fieldValues)) {
|
|
|
|
* return;
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* submitFn();
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* render() {
|
|
|
|
* return <FormBuilder handleSubmit={this.handleSubmit} />
|
|
|
|
* }
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* @param {Object} event
|
|
|
|
*/
|
|
|
|
handleSubmit(event) {
|
|
|
|
const schemaFields = this.props.schemas[this.props.schemaUrl].schema.fields;
|
2016-04-26 12:01:40 +12:00
|
|
|
const fieldValues = this.props.form[this.getFormId()].fields
|
2016-04-12 16:47:24 +12:00
|
|
|
.reduce((prev, curr) => Object.assign({}, prev, {
|
|
|
|
[schemaFields.find(schemaField => schemaField.id === curr.id).name]: curr.value,
|
|
|
|
}), {});
|
|
|
|
|
|
|
|
const submitFn = () => {
|
2016-04-26 12:01:40 +12:00
|
|
|
this.props.formActions.submitForm(
|
2016-04-12 16:47:24 +12:00
|
|
|
this.submitApi,
|
2016-04-21 14:38:02 +12:00
|
|
|
this.getFormId(),
|
2016-04-12 16:47:24 +12:00
|
|
|
fieldValues
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
if (typeof this.props.handleSubmit !== 'undefined') {
|
|
|
|
this.props.handleSubmit(event, fieldValues, submitFn);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
submitFn();
|
|
|
|
}
|
|
|
|
|
2016-03-31 10:45:54 +13:00
|
|
|
/**
|
|
|
|
* Maps a list of schema fields to their React Component.
|
|
|
|
* Only top level form fields are handled here, composite fields (TabSets etc),
|
|
|
|
* are responsible for mapping and rendering their children.
|
|
|
|
*
|
|
|
|
* @param array fields
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
mapFieldsToComponents(fields) {
|
2016-04-12 10:24:16 +12:00
|
|
|
const createFn = this.props.createFn;
|
2016-04-12 16:47:24 +12:00
|
|
|
const handleFieldUpdate = this.handleFieldUpdate;
|
2016-04-12 10:24:16 +12:00
|
|
|
|
2016-03-31 10:45:54 +13:00
|
|
|
return fields.map((field, i) => {
|
|
|
|
const Component = field.component !== null
|
|
|
|
? fakeInjector.getComponentByName(field.component)
|
|
|
|
: fakeInjector.getComponentByDataType(field.type);
|
|
|
|
|
|
|
|
if (Component === null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Props which every form field receives.
|
2016-04-07 22:35:52 +12:00
|
|
|
// Leave it up to the schema and component to determine
|
|
|
|
// which props are required.
|
2016-04-12 16:47:24 +12:00
|
|
|
const props = deepFreeze(Object.assign({}, field, { handleFieldUpdate }));
|
2016-03-31 10:45:54 +13:00
|
|
|
|
2016-04-12 10:24:16 +12:00
|
|
|
// Provides container components a place to hook in
|
|
|
|
// and apply customisations to scaffolded components.
|
|
|
|
if (typeof createFn === 'function') {
|
|
|
|
return createFn(Component, props);
|
|
|
|
}
|
|
|
|
|
2016-03-31 10:45:54 +13:00
|
|
|
return <Component key={i} {...props} />;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2016-04-12 16:47:24 +12:00
|
|
|
/**
|
|
|
|
* Maps a list of form actions to their React Component.
|
|
|
|
*
|
|
|
|
* @param array actions
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
mapActionsToComponents(actions) {
|
|
|
|
const createFn = this.props.createFn;
|
|
|
|
|
|
|
|
return actions.map((action, i) => {
|
2016-04-20 13:43:13 +12:00
|
|
|
let props = deepFreeze(action);
|
2016-04-12 16:47:24 +12:00
|
|
|
|
2016-04-20 16:51:00 +12:00
|
|
|
// Add sensible defaults for common actions.
|
|
|
|
switch (props.name) {
|
|
|
|
case 'action_save':
|
|
|
|
props = deepFreeze(Object.assign({}, props, {
|
|
|
|
type: 'submit',
|
|
|
|
label: props.title,
|
|
|
|
icon: 'save',
|
|
|
|
}));
|
|
|
|
break;
|
|
|
|
case 'action_cancel':
|
|
|
|
props = deepFreeze(Object.assign({}, props, {
|
|
|
|
type: 'button',
|
|
|
|
label: props.title,
|
|
|
|
icon: 'cancel',
|
|
|
|
}));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2016-04-12 16:47:24 +12:00
|
|
|
}
|
|
|
|
|
2016-04-20 16:51:00 +12:00
|
|
|
if (typeof createFn === 'function') {
|
|
|
|
return createFn(FormActionComponent, props);
|
2016-04-20 13:43:13 +12:00
|
|
|
}
|
|
|
|
|
2016-04-12 16:47:24 +12:00
|
|
|
return <FormActionComponent key={i} {...props} />;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Merges the structural and state data of a form field.
|
|
|
|
* The structure of the objects being merged should match the structures
|
|
|
|
* generated by the SilverStripe FormSchema.
|
|
|
|
*
|
2016-04-26 11:25:27 +12:00
|
|
|
* @param {object} structure - Structural data for a single field.
|
|
|
|
* @param {object} state - State data for a single field.
|
|
|
|
* @return {object}
|
2016-04-12 16:47:24 +12:00
|
|
|
*/
|
|
|
|
mergeFieldData(structure, state) {
|
2016-04-26 11:25:27 +12:00
|
|
|
return merge.recursive(true, structure, {
|
|
|
|
data: state.data,
|
2016-04-12 16:47:24 +12:00
|
|
|
messages: state.messages,
|
|
|
|
valid: state.valid,
|
|
|
|
value: state.value,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Cleans up Redux state used by the form when the Form component is unmonuted.
|
|
|
|
*
|
|
|
|
* @param {string} formId - ID of the form to clean up.
|
|
|
|
*/
|
|
|
|
removeForm(formId) {
|
2016-04-26 12:01:40 +12:00
|
|
|
this.props.formActions.removeForm(formId);
|
2016-04-12 16:47:24 +12:00
|
|
|
}
|
|
|
|
|
2016-03-31 10:45:54 +13:00
|
|
|
render() {
|
2016-04-21 14:38:02 +12:00
|
|
|
const formId = this.getFormId();
|
|
|
|
if (!formId) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
const formSchema = this.getFormSchema();
|
2016-04-26 12:01:40 +12:00
|
|
|
const formState = this.props.form[formId];
|
2016-03-31 10:45:54 +13:00
|
|
|
|
|
|
|
// If the response from fetching the initial data
|
|
|
|
// hasn't come back yet, don't render anything.
|
2016-04-07 21:29:52 +12:00
|
|
|
if (!formSchema) {
|
2016-03-31 10:45:54 +13:00
|
|
|
return null;
|
|
|
|
}
|
2016-03-22 12:25:23 +13:00
|
|
|
|
2016-04-11 09:16:32 +12:00
|
|
|
// Map form schema to React component attribute names,
|
|
|
|
// which requires renaming some of them (by unsetting the original keys)
|
|
|
|
const attributes = Object.assign({}, formSchema.schema.attributes, {
|
|
|
|
class: null,
|
|
|
|
className: formSchema.schema.attributes.class,
|
|
|
|
enctype: null,
|
|
|
|
encType: formSchema.schema.attributes.enctype,
|
|
|
|
});
|
|
|
|
|
2016-04-12 16:47:24 +12:00
|
|
|
// If there is structural and state data availabe merge those data for each field.
|
|
|
|
// Otherwise just use the structural data.
|
|
|
|
const fieldData = formSchema.schema && formState
|
|
|
|
? formSchema.schema.fields.map((f, i) => this.mergeFieldData(f, formState.fields[i]))
|
|
|
|
: formSchema.schema.fields;
|
|
|
|
|
2016-03-31 10:45:54 +13:00
|
|
|
const formProps = {
|
2016-04-07 21:29:52 +12:00
|
|
|
actions: formSchema.schema.actions,
|
2016-04-11 09:16:32 +12:00
|
|
|
attributes,
|
2016-04-12 16:47:24 +12:00
|
|
|
componentWillUnmount: this.removeForm,
|
2016-04-07 21:29:52 +12:00
|
|
|
data: formSchema.schema.data,
|
2016-04-12 16:47:24 +12:00
|
|
|
fields: fieldData,
|
2016-04-21 14:38:02 +12:00
|
|
|
formId,
|
2016-04-12 16:47:24 +12:00
|
|
|
handleSubmit: this.handleSubmit,
|
|
|
|
mapActionsToComponents: this.mapActionsToComponents,
|
2016-03-31 10:45:54 +13:00
|
|
|
mapFieldsToComponents: this.mapFieldsToComponents,
|
|
|
|
};
|
2016-03-22 12:25:23 +13:00
|
|
|
|
2016-03-31 10:45:54 +13:00
|
|
|
return <FormComponent {...formProps} />;
|
|
|
|
}
|
2016-03-22 12:25:23 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
FormBuilderComponent.propTypes = {
|
2016-04-12 16:47:24 +12:00
|
|
|
config: React.PropTypes.object,
|
2016-04-12 10:24:16 +12:00
|
|
|
createFn: React.PropTypes.func,
|
2016-04-26 12:01:40 +12:00
|
|
|
form: React.PropTypes.object.isRequired,
|
|
|
|
formActions: React.PropTypes.object.isRequired,
|
2016-04-12 16:47:24 +12:00
|
|
|
handleSubmit: React.PropTypes.func,
|
2016-03-31 10:45:54 +13:00
|
|
|
schemas: React.PropTypes.object.isRequired,
|
2016-04-12 16:47:24 +12:00
|
|
|
schemaActions: React.PropTypes.object.isRequired,
|
|
|
|
schemaUrl: React.PropTypes.string.isRequired,
|
2016-03-22 12:25:23 +13:00
|
|
|
};
|
|
|
|
|
|
|
|
function mapStateToProps(state) {
|
2016-03-31 10:45:54 +13:00
|
|
|
return {
|
2016-04-12 16:47:24 +12:00
|
|
|
config: state.config,
|
2016-04-26 12:01:40 +12:00
|
|
|
form: state.form,
|
2016-03-31 10:45:54 +13:00
|
|
|
schemas: state.schemas,
|
|
|
|
};
|
2016-03-22 12:25:23 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
function mapDispatchToProps(dispatch) {
|
2016-03-31 10:45:54 +13:00
|
|
|
return {
|
2016-04-26 12:01:40 +12:00
|
|
|
formActions: bindActionCreators(formActions, dispatch),
|
2016-04-12 16:47:24 +12:00
|
|
|
schemaActions: bindActionCreators(schemaActions, dispatch),
|
2016-03-31 10:45:54 +13:00
|
|
|
};
|
2016-03-22 12:25:23 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
export default connect(mapStateToProps, mapDispatchToProps)(FormBuilderComponent);
|