Skip to content

Commit

Permalink
NEW HtmlEditorField component for react rich text
Browse files Browse the repository at this point in the history
This allows one to use a Rich Text field from within a react form, where
as previously it was only possible from server side PHP rendering
templates, and Entwine.

The downside to this is that although `@tinymce/tinymce-react` is the
official package for this, it is in turn only a very loose wrapper
around the window.TinyMCE global. It also posed problems with the
configuration generated by the PHP config class, and altering that would
destroy backwards compatiblity - which for now is critical for adoption.

So while there is now a React component for rendering TinyMCE, it is
only a bare bone component that still relies on Entwine to boot the
actual editor via the `data-config` attribute on the React output
`textarea` element.
  • Loading branch information
Dylan Wagstaff committed Sep 5, 2018
1 parent b812132 commit 0a95114
Show file tree
Hide file tree
Showing 9 changed files with 93 additions and 7 deletions.
2 changes: 1 addition & 1 deletion client/dist/js/bundle.js

Large diffs are not rendered by default.

2 changes: 1 addition & 1 deletion client/dist/js/vendor.js

Large diffs are not rendered by default.

2 changes: 2 additions & 0 deletions client/src/boot/registerComponents.js
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ import Loading from 'components/Loading/Loading';
import ViewModeToggle from 'components/ViewModeToggle/ViewModeToggle';
import Search from 'components/Search/Search';
import SearchToggle from 'components/Search/SearchToggle';
import HtmlEditorField from 'components/HtmlEditorField/HtmlEditorField';

export default () => {
Injector.component.registerMany({
Expand Down Expand Up @@ -76,5 +77,6 @@ export default () => {
ViewModeToggle,
Search,
SearchToggle,
HtmlEditorField,
});
};
8 changes: 6 additions & 2 deletions client/src/components/Form/Form.js
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,7 @@ class Form extends Component {
const fields = this.props.mapFieldsToComponents(this.props.fields);
const actions = this.props.mapActionsToComponents(this.props.actions);
const messages = this.renderMessages();
const FormTag = this.props.formTag;

const className = ['form'];
if (valid === false) {
Expand All @@ -57,9 +58,10 @@ class Form extends Component {
};

return (
<form
<FormTag
{...formProps}
ref={(form) => { this.form = form; this.props.setDOM(form); }}
role="form"
>
{fields &&
<fieldset>
Expand All @@ -74,7 +76,7 @@ class Form extends Component {
? <div className="btn-toolbar" role="group">{actions}</div>
: null
}
</form>
</FormTag>
);
}
}
Expand Down Expand Up @@ -102,10 +104,12 @@ Form.propTypes = {
value: PropTypes.any,
type: PropTypes.string,
})),
formTag: PropTypes.oneOfType([PropTypes.func, PropTypes.string]),
};

Form.defaultProps = {
setDOM: () => null,
formTag: 'form',
};

export default Form;
4 changes: 3 additions & 1 deletion client/src/components/FormBuilder/FormBuilder.js
Original file line number Diff line number Diff line change
Expand Up @@ -320,6 +320,7 @@ class FormBuilder extends Component {
form,
afterMessages,
autoFocus,
formTag,
} = this.props;

const props = {
Expand All @@ -344,7 +345,8 @@ class FormBuilder extends Component {
persistentSubmitErrors,
validate: this.validateForm,
autoFocus,
setDOM: (formDOM) => { this.formDOM = formDOM; }
setDOM: (formDOM) => { this.formDOM = formDOM; },
formTag,
};

return (
Expand Down
65 changes: 65 additions & 0 deletions client/src/components/HtmlEditorField/HtmlEditorField.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,65 @@
/* global window */
import React from 'react';
import Script from 'react-load-script';
import { Component as TextField } from 'components/TextField/TextField';
import fieldHolder from 'components/FieldHolder/FieldHolder';


class HtmlEditorField extends TextField {
/**
* sets initial state:
* if editorjs IS defined, we are NOT ready (must check dependency first).
* if editorjs is NOT defined, we ARE ready (no dependency).
*/
constructor(props) {
super(props);
this.state = {
isReady: !props.data.editorjs
};
this.handleReady = this.handleReady.bind(this);
}

getInputProps() {
return {
...super.getInputProps(),
...this.props.data.attributes,
};
}

/**
* Once the dependency script is loaded, updating the internal state
* will trigger a reload and present the editor to the user
*/
handleReady() {
if (!window.TinyMCE && window.tinymce) {
window.TinyMCE = window.tinymce;
}
this.setState({ isReady: true });
}

/**
* TinyMCE operates from a global script being loaded in first.
* We must ensure this dependency is loaded before proceeding to
* render the editor proper
*/
renderDependencyScript() {
return <Script url={this.props.data.editorjs} onLoad={this.handleReady} />;
}

/**
* Renders the rich text editor (TinyMCE)
* Happens only after the dependency script has been loaded
*/
renderRichTextEditor() {
const config = JSON.parse(this.props.data.attributes['data-config']);
return super.render(config);
}

render() {
return (this.state.isReady) ? this.renderRichTextEditor() : this.renderDependencyScript();
}
}

export { HtmlEditorField as Component };

export default fieldHolder(HtmlEditorField);
12 changes: 10 additions & 2 deletions client/src/legacy/HtmlEditorField.js
Original file line number Diff line number Diff line change
Expand Up @@ -288,7 +288,7 @@ jQuery.entwine('ss', function($) {
/**
* Constructor: onmatch
*/
onadd: function() {
onmatch: function() {
var edClass = this.data('editor') || 'default',
ed = ss.editorWrappers[edClass]();
this.setEditor(ed);
Expand All @@ -298,14 +298,22 @@ jQuery.entwine('ss', function($) {
this._super();
},

onadd: function() {
this.onmatch();
},

/**
* Destructor: onunmatch
*/
onremove: function() {
onunmatch: function() {
this.getEditor().destroy();
this._super();
},

onremove: function() {
this.onunmatch();
},

/**
* Make sure the editor has flushed all it's buffers before the form is submitted.
*/
Expand Down
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,7 @@
"react-dnd": "^2.2.3",
"react-dnd-html5-backend": "^2.2.3",
"react-dom": "15.3.1",
"react-load-script": "^0.0.6",
"react-redux": "^4.4.1",
"react-router": "^2.4.1",
"react-router-redux": "^4.0.5",
Expand Down
4 changes: 4 additions & 0 deletions yarn.lock
Original file line number Diff line number Diff line change
Expand Up @@ -7753,6 +7753,10 @@ react-inspector@^2.2.2:
babel-runtime "^6.26.0"
is-dom "^1.0.9"

react-load-script@^0.0.6:
version "0.0.6"
resolved "https://registry.yarnpkg.com/react-load-script/-/react-load-script-0.0.6.tgz#db6851236aaa25bb622677a2eb51dad4f8d2c258"

react-modal@^3.1.10:
version "3.2.1"
resolved "https://registry.yarnpkg.com/react-modal/-/react-modal-3.2.1.tgz#fa8f76aed55b67c22dcf1a1c15b05c8d11f18afe"
Expand Down

0 comments on commit 0a95114

Please sign in to comment.