UNPKG

uniforms

Version:
198 lines (197 loc) 6.7 kB
import clone from 'lodash/clone'; import get from 'lodash/get'; import omit from 'lodash/omit'; import setWith from 'lodash/setWith'; import React, { Component } from 'react'; import { changedKeys } from './changedKeys'; import { context } from './context'; import { randomIds } from './randomIds'; export class BaseForm extends Component { constructor(props) { super(props); // @ts-expect-error: State may be bigger, but it'll be covered by the subclasses. this.state = { changed: false, changedMap: Object.create(null), resetCount: 0, submitted: false, submitting: false, }; this.mounted = false; this.randomId = randomIds(this.props.id); this.onReset = this.reset = this.onReset.bind(this); this.onChange = this.change = this.onChange.bind(this); this.onSubmit = this.submit = this.onSubmit.bind(this); // TODO: It shouldn't be here const getModel = this.getModel.bind(this); this.getModel = (mode, model = getModel(mode)) => mode !== undefined && this.props.modelTransform ? this.props.modelTransform(mode, model) : model; } componentDidMount() { this.mounted = true; } // eslint-disable-next-line @typescript-eslint/no-unused-vars componentDidUpdate(prevProps, prevState, snapshot) { } componentWillUnmount() { this.mounted = false; if (this.delayId) { clearTimeout(this.delayId); } } getContext() { return { changed: this.state.changed, changedMap: this.state.changedMap, error: this.getContextError(), formRef: this, model: this.getContextModel(), name: this.getContextName(), onChange: this.getContextOnChange(), onSubmit: this.getContextOnSubmit(), randomId: this.randomId, schema: this.getContextSchema(), state: this.getContextState(), submitted: this.state.submitted, submitting: this.state.submitting, validating: false, }; } getContextName() { return []; } getContextError() { return this.props.error; } getContextModel() { return this.getModel('form'); } getContextState() { return { disabled: !!this.props.disabled, readOnly: !!this.props.readOnly, showInlineError: !!this.props.showInlineError, }; } getContextSchema() { return this.props.schema; } getContextOnChange() { // It's bound in constructor. // eslint-disable-next-line @typescript-eslint/unbound-method return this.onChange; } getContextOnSubmit() { // It's bound in constructor. // eslint-disable-next-line @typescript-eslint/unbound-method return this.onSubmit; } getModel(mode, model = this.props.model) { return model; } getNativeFormProps() { const props = omit(this.props, [ 'autosave', 'autosaveDelay', 'disabled', 'error', 'model', 'modelTransform', 'onChange', 'onSubmit', 'readOnly', 'schema', 'showInlineError', ]); return Object.assign(Object.assign({}, props), { // It's bound in constructor. // eslint-disable-next-line @typescript-eslint/unbound-method onSubmit: this.onSubmit, key: `reset-${this.state.resetCount}` }); } onChange(key, value) { // Do not set `changed` before componentDidMount if (this.mounted) { const keys = changedKeys(key, value, get(this.getModel(), key)); if (keys.length !== 0) { this.setState(state => // If all are already marked, we can skip the update completely. state.changed && keys.every(key => !!get(state.changedMap, key)) ? null : { changed: true, changedMap: keys.reduce((changedMap, key) => setWith(changedMap, key, {}, clone), clone(state.changedMap)), }); } } if (this.props.onChange) { this.props.onChange(key, value); } // Do not call `onSubmit` before componentDidMount if (this.mounted && this.props.autosave) { if (this.delayId) { clearTimeout(this.delayId); } // Delay autosave by `autosaveDelay` milliseconds... this.delayId = setTimeout(() => { // ...and wait for all scheduled `setState`s to commit. This is required // for AutoForm to validate correct model, waiting in `onChange`. if (this.mounted) { this.setState(() => null, () => { this.onSubmit(); }); } }, this.props.autosaveDelay); } } __reset(state) { return { changed: false, changedMap: Object.create(null), resetCount: state.resetCount + 1, submitted: false, submitting: false, }; } onReset() { if (this.mounted) { // @ts-expect-error // It's bound in constructor. // eslint-disable-next-line @typescript-eslint/unbound-method this.setState(this.__reset); } } onSubmit(event) { if (event) { event.preventDefault(); event.stopPropagation(); } if (this.mounted) { this.setState(state => (state.submitted ? null : { submitted: true })); } const result = this.props.onSubmit(this.getModel('submit')); if (!(result instanceof Promise)) { return Promise.resolve(); } if (this.mounted) { this.setState({ submitting: true }); } return result.finally(() => { if (this.mounted) { this.setState({ submitting: false }); } }); } render() { return (React.createElement(context.Provider, { value: this.getContext() }, React.createElement("form", Object.assign({}, this.getNativeFormProps())))); } } BaseForm.displayName = 'Form'; BaseForm.defaultProps = { autosave: false, autosaveDelay: 0, error: null, model: Object.create(null), noValidate: true, onSubmit() { }, };