Rich text editor in RW form

Wondering if anyone has figured out a good integration for a rich-text editor in the scaffolded RW form. I’m trying to figure out how best to add one but still utilize RW form functionality like validation.

Also, how to enable the data to be converted to “savable” information in the database.

I’ve been experimenting with Draft.js but not sure how to work through the above barriers.

1 Like

I have not tackled this problem myself and this may be a really silly suggestion, but awhile ago I discovered an MDN page that gives a really simple but functional rich text editor that’s done entirely with vanilla javascript.

<!doctype html>
<html>
<head>
<title>Rich Text Editor</title>
<script type="text/javascript">
var oDoc, sDefTxt;

function initDoc() {
  oDoc = document.getElementById("textBox");
  sDefTxt = oDoc.innerHTML;
  if (document.compForm.switchMode.checked) { setDocMode(true); }
}

function formatDoc(sCmd, sValue) {
  if (validateMode()) { document.execCommand(sCmd, false, sValue); oDoc.focus(); }
}

function validateMode() {
  if (!document.compForm.switchMode.checked) { return true ; }
  alert("Uncheck \"Show HTML\".");
  oDoc.focus();
  return false;
}

function setDocMode(bToSource) {
  var oContent;
  if (bToSource) {
    oContent = document.createTextNode(oDoc.innerHTML);
    oDoc.innerHTML = "";
    var oPre = document.createElement("pre");
    oDoc.contentEditable = false;
    oPre.id = "sourceText";
    oPre.contentEditable = true;
    oPre.appendChild(oContent);
    oDoc.appendChild(oPre);
    document.execCommand("defaultParagraphSeparator", false, "div");
  } else {
    if (document.all) {
      oDoc.innerHTML = oDoc.innerText;
    } else {
      oContent = document.createRange();
      oContent.selectNodeContents(oDoc.firstChild);
      oDoc.innerHTML = oContent.toString();
    }
    oDoc.contentEditable = true;
  }
  oDoc.focus();
}

function printDoc() {
  if (!validateMode()) { return; }
  var oPrntWin = window.open("","_blank","width=450,height=470,left=400,top=100,menubar=yes,toolbar=no,location=no,scrollbars=yes");
  oPrntWin.document.open();
  oPrntWin.document.write("<!doctype html><html><head><title>Print<\/title><\/head><body onload=\"print();\">" + oDoc.innerHTML + "<\/body><\/html>");
  oPrntWin.document.close();
}
</script>
<style type="text/css">
.intLink { cursor: pointer; }
img.intLink { border: 0; }
#toolBar1 select { font-size:10px; }
#textBox {
  width: 540px;
  height: 200px;
  border: 1px #000000 solid;
  padding: 12px;
  overflow: scroll;
}
#textBox #sourceText {
  padding: 0;
  margin: 0;
  min-width: 498px;
  min-height: 200px;
}
#editMode label { cursor: pointer; }
</style>
</head>
<body onload="initDoc();">
<form name="compForm" method="post" action="sample.php" onsubmit="if(validateMode()){this.myDoc.value=oDoc.innerHTML;return true;}return false;">
<input type="hidden" name="myDoc">
<div id="toolBar1">
<select onchange="formatDoc('formatblock',this[this.selectedIndex].value);this.selectedIndex=0;">
<option selected>- formatting -</option>
<option value="h1">Title 1 &lt;h1&gt;</option>
<option value="h2">Title 2 &lt;h2&gt;</option>
<option value="h3">Title 3 &lt;h3&gt;</option>
<option value="h4">Title 4 &lt;h4&gt;</option>
<option value="h5">Title 5 &lt;h5&gt;</option>
<option value="h6">Subtitle &lt;h6&gt;</option>
<option value="p">Paragraph &lt;p&gt;</option>
<option value="pre">Preformatted &lt;pre&gt;</option>
</select>
<select onchange="formatDoc('fontname',this[this.selectedIndex].value);this.selectedIndex=0;">
<option class="heading" selected>- font -</option>
<option>Arial</option>
<option>Arial Black</option>
<option>Courier New</option>
<option>Times New Roman</option>
</select>
<select onchange="formatDoc('fontsize',this[this.selectedIndex].value);this.selectedIndex=0;">
<option class="heading" selected>- size -</option>
<option value="1">Very small</option>
<option value="2">A bit small</option>
<option value="3">Normal</option>
<option value="4">Medium-large</option>
<option value="5">Big</option>
<option value="6">Very big</option>
<option value="7">Maximum</option>
</select>
<select onchange="formatDoc('forecolor',this[this.selectedIndex].value);this.selectedIndex=0;">
<option class="heading" selected>- color -</option>
<option value="red">Red</option>
<option value="blue">Blue</option>
<option value="green">Green</option>
<option value="black">Black</option>
</select>
<select onchange="formatDoc('backcolor',this[this.selectedIndex].value);this.selectedIndex=0;">
<option class="heading" selected>- background -</option>
<option value="red">Red</option>
<option value="green">Green</option>
<option value="black">Black</option>
</select>
</div>
<div id="toolBar2">
<img class="intLink" title="Clean" onclick="if(validateMode()&&confirm('Are you sure?')){oDoc.innerHTML=sDefTxt};" src="" />
<img class="intLink" title="Print" onclick="printDoc();" src="">
<img class="intLink" title="Undo" onclick="formatDoc('undo');" src="" />
<img class="intLink" title="Redo" onclick="formatDoc('redo');" src="" />
<img class="intLink" title="Remove formatting" onclick="formatDoc('removeFormat')" src="">
<img class="intLink" title="Bold" onclick="formatDoc('bold');" src="" />
<img class="intLink" title="Italic" onclick="formatDoc('italic');" src="" />
<img class="intLink" title="Underline" onclick="formatDoc('underline');" src="" />
<img class="intLink" title="Left align" onclick="formatDoc('justifyleft');" src="" />
<img class="intLink" title="Center align" onclick="formatDoc('justifycenter');" src="" />
<img class="intLink" title="Right align" onclick="formatDoc('justifyright');" src="" />
<img class="intLink" title="Numbered list" onclick="formatDoc('insertorderedlist');" src="" />
<img class="intLink" title="Dotted list" onclick="formatDoc('insertunorderedlist');" src="" />
<img class="intLink" title="Quote" onclick="formatDoc('formatblock','blockquote');" src="" />
<img class="intLink" title="Delete indentation" onclick="formatDoc('outdent');" src="" />
<img class="intLink" title="Add indentation" onclick="formatDoc('indent');" src="" />
<img class="intLink" title="Hyperlink" onclick="var sLnk=prompt('Write the URL here','http:\/\/');if(sLnk&&sLnk!=''&&sLnk!='http://'){formatDoc('createlink',sLnk)}" src="" />
<img class="intLink" title="Cut" onclick="formatDoc('cut');" src="" />
<img class="intLink" title="Copy" onclick="formatDoc('copy');" src="" />
<img class="intLink" title="Paste" onclick="formatDoc('paste');" src="" />
</div>
<div id="textBox" contenteditable="true"><p>Lorem ipsum</p></div>
<p id="editMode"><input type="checkbox" name="switchMode" id="switchBox" onchange="setDocMode(this.checked);" /> <label for="switchBox">Show HTML</label></p>
<p><input type="submit" value="Send" /></p>
</form>
</body>
</html>

I played around with it just a bit and wrote a little blog post about it. It seemed like a pretty cool way to get around having to use any third party library, but I’m sure it’s very limited.

2 Likes

Have you looked at the Trix editor by 37signals/Basecamp? https://trix-editor.org/

3 Likes

@ajcwebdev and @rob thanks for the quick replies! It seems like each option has it’s merits so I’ll be exploring both.

I’m trying to build my current project fairly quickly so Trix is probably a good solution for me to just drop in.

For anyone who finds this down the line and is searching for a good rich-text editor, I’d recommend also looking into react-quill:

I ended up migrating to it from Trix, since there were two areas that I found Trix falling short: (1) the ability to easily add H2’s in content and (2) the ability to add tables easily. react-quill comes with H2’s out of the box, can support tables with another module, and maintains state in HTML, which makes creating, updating, and displaying rich text from your database trivial.

1 Like

How did you bind the ReactQuill-component to the props.post.body? I can’t figure it out.
The validation works, but getting/setting the value does not.

In PostForm

<WyiswygField label="body" name="body" content={props.post?.body} validation={{ required: true }} />

Custom Field

const WyiswygField = ({ label, name, content, validation }) => {
  const [value, setValue] = useState(content)

  const register = useRegister({
    name,
    validation: {...validation }
  })

  const { className: labelClassName, style: labelStyle } = useErrorStyles({
    className: `my-label-class`,
    errorClassName: `my-label-error-class`,
    name,
  })

  const { className: inputClassName, style: inputStyle } = useErrorStyles({
    className: `my-input-class`,
    errorClassName: `my-input-error-class`,
    name,
  })

  return (
    <>
      <label className={labelClassName} style={labelStyle}>{label}</label>

      <ReactQuill
            {...register}
            defaultValue={value} onChange={setValue}
          />
      <FieldError name={name}/>
    </>
  )
}

Hey @betty – apologies but I unfortunately haven’t worked with react-quill much at all since I posted this :point_up:, so I’m not sure what the issue would be. Sorry!

I was able to make the binding work by passing formMethods’s setValue to the rich text editor component.

You need to pass useForm’s return to the <FormProvider>, so in the form:

const StoreForm = (props) => {
  const formMethods = useForm()
  return (
    <div className="rw-form-wrapper">
      <Form
        onSubmit={onSubmit}
        error={props.error}
        formMethods={formMethods}
      >
          <RichTextEditor
          label="Description"
          name="description"
          content={props.store?.description}
          validation={{ required: true }}
          updateFn={formMethods.setValue}
        />
    </form>

And in your rich text component:

const RichTextEditor = ({ label, name, content, validation, updateFn }) => {
  return (
    <>
      <ReactQuill
        defaultValue={content}
        {...register}
        onChange={(val) => updateFn(name, val)}
        onBlur={(val) => updateFn(name, val)}
      />
    </>
  )
}
1 Like

Mantine also uses Quill and has done so quite succesfully. There are some limitations though ~ (like clearing the content works only with ref etc.) you can check it out here: mantine/src/mantine-rte at master · mantinedev/mantine · GitHub

1 Like

Got it working, thanks so much!

You’re welcome!

For WIW, I’ve been using Mantine Rich text editor and it’s been incredible.

@david, I really appreciate your insightful input!

I would like to add that instead of passing the setValue reference via props, you can utilize the useFormContext hook to access it directly inside the component. This allows for greater independence of the component from the form code.

For instance:

import { useRegister, useFormContext } from '@redwoodjs/forms'

const RichTextEditor = ({ label, name, content, validation}) => {
  const { setValue } = useFormContext()
  const register = useRegister({ name, validation  })
  return (
    <>
      <ReactQuill
        defaultValue={content}
        {...register}
        onChange={(val) => setValue(name, val)}
        onBlur={(val) => setValue(name, val)}
      />
    </>
  )
}

Another editor I’ve been using: https://tiptap.dev/

Thought I’d chime in too, since @morganmspencer did.

I’m using a custom build of Plate, which is built on top of Slate. Slate is an editor framework with a plugin architecture. Plate adds an opinionated set of functionality to it (like bold, quotes, etc.).

Quill serializes to HTML so you need to sanitize its output server side. You can capture the raw internal data structure Quill uses (Delta) and serialize it, but it also includes the editing history.

Slate was built based on ideas from Quill, Draft, and ProseMirror. PM has its own virtual DOM and isn’t natural to use with React, but it’s author is highly competent (he wrote and maintains CodeMirror also). Slate has its own JSON serialization format that is customizable. I’m serializing it to Portable Text, a format Sanity (a headless CMS vendor) created and maintains open source tooling around. You can serialize to Markdown easily (or HTML if you really want), which is nice.

My reasons for using Slate are:

  • It’s really easy to turn React components into plugins and then wire them into the editor, for example you could make an e-mail composer by copying the basic Plate editor plugins and changing them to output mjml markup, or to add embedded content shortcodes (like WordPress) that users can enter in longform text.
  • It was designed to support collaboration from the outset, so having several people working on one document works really well, including the diff algorithm.
  • It’s native React. It’s not trying to support a bunch of different view engines or bundle its own virtual DOM.
  • It handles nested content well, including programmatic transformations of nested content.
  • It only bundles what you need, since you’re manually adding editor features as plugins (or using an opinionated build like Plate). A lot of other editors include all kinds of things built-in.

Slate has a learning curve, but so do Quill and ProseMirror. Someone else mentioned Manitine - it’s pretty simple to get going and is built on TipTap. The latter is a freemium model, with a commercial Pro plan, so it doesn’t work for my project (open source library). You have to pay per seat with TipTap for things like the ability to add your own extensions and collaboration.

DraftJS was a second choice to Slate for me. It’s Facebook’s in-house library. I went with Slate because I felt like the architecture was better designed (from the viewpoint of the collaboration and plugin API), that Draft is tailored to Facebook’s unique needs rather than a wider developer audience’s needs, and that it’s possible Facebook loses interest in maintaining it or has internal chaos like they did with Flow.