Making content editable

Warning: As the execCommand() spec warns, its features "are not implemented consistently or fully by user agents", and in addition, it is marked as deprecated on the Document.execCommand() reference page. Therefore, much of the content on this page cannot be trusted for use in production code.

In HTML, any element can be editable. By using some JavaScript event handlers, you can transform your web page into a full and fast rich text editor. This article provides some information about this functionality.

Note: In Firefox 63 Beta/Dev edition, some of the rich-text editing features have been disabled by default, for better cross-browser compatibility. These are object resizing on <img>, <table>, and absolutely-positioned elements; inline table editing to add or remove rows and columns; and the grabber that allows moving of absolutely-positioned elements. See bug 1449564 for additional details.

How does it work?

All you have to do is set the contenteditable attribute on nearly any HTML element to make it editable.

Here's a simple example which creates a <div> element whose contents the user can edit.

<div contenteditable="true">This text can be edited by the user.</div>

Here's the above HTML in action:

Executing commands

When an HTML element has contenteditable set to true, the document.execCommand() method is made available. This lets you run commands to manipulate the contents of the editable region. Most commands affect the document's selection by, for example, applying a style to the text (bold, italics, etc.), while others insert new elements (like adding a link) or affect an entire line (indenting). When using contentEditable, calling execCommand() will affect the currently active editable element.

Differences in markup generation

Use of contenteditable across different browsers has been painful for a long time because of the differences in generated markup between browsers. For example, even something as simple as what happens when you press Enter/Return to create a new line of text inside an editable element was handled differently across the major browsers (Firefox inserted <br> elements, IE/Opera used <p>, Chrome/Safari used <div>).

Fortunately, in modern browsers things are somewhat more consistent. As of Firefox 60, Firefox will be updated to wrap the separate lines in <div> elements, matching the behavior of Chrome, modern Opera, Edge, and Safari.

Try it out in the above example.

Note: Internet Explorer, which is no longer being developed, uses <p> elements instead of <div>.

If you want to use a different paragraph separator, the above browsers all support document.execCommand, which provides a defaultParagraphSeparator command to allow you to change it. For example, to use <p> elements:

document.execCommand("defaultParagraphSeparator", false, "p");

Additionally, Firefox supports the non-standard argument, br, for defaultParagraphSeparator since Firefox 55. This is useful if your web application expects the older Firefox behavior, and you don't want to or don't have the time to update it to use the new behavior. You can use the older Firefox behavior with this line:

document.execCommand("defaultParagraphSeparator", false, "br");

Security

For security reasons, Firefox doesn't let JavaScript code use clipboard related features (copy, paste, etc.) by default. You can enable them by setting the preferences shown below using about:config:

user_pref("capability.policy.policynames", "allowclipboard");
user_pref("capability.policy.allowclipboard.sites", "https://www.mozilla.org");
user_pref("capability.policy.allowclipboard.Clipboard.cutcopy", "allAccess");
user_pref("capability.policy.allowclipboard.Clipboard.paste", "allAccess");

Example: A simple but complete rich text editor

<!DOCTYPE html>
<html lang="en-US">
  <head>
    <meta charset="utf-8" />
    <meta name="viewport" content="width=device-width" />
    <title>Rich Text Editor</title>
    <script>
      let doc, defTxt;

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

      function formatDoc(cmd, value) {
        if (validateMode()) {
          document.execCommand(cmd, false, value);
          doc.focus();
        }
      }

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

      function setDocMode(toSource) {
        let content;
        if (toSource) {
          content = document.createTextNode(doc.innerHTML);
          doc.innerHTML = "";
          const pre = document.createElement("pre");
          doc.contentEditable = false;
          pre.id = "sourceText";
          pre.contentEditable = true;
          pre.appendChild(content);
          doc.appendChild(pre);
          document.execCommand("defaultParagraphSeparator", false, "div");
        } else {
          if (document.all) {
            doc.innerHTML = doc.innerText;
          } else {
            content = document.createRange();
            content.selectNodeContents(doc.firstChild);
            doc.innerHTML = content.toString();
          }
          doc.contentEditable = true;
        }
        doc.focus();
      }

      function printDoc() {
        if (!validateMode()) {
          return;
        }
        const printWin = window.open(
          "",
          "_blank",
          "width=450,height=470,left=400,top=100,menubar=yes,toolbar=no,location=no,scrollbars=yes"
        );
        printWin.document.open();
        printWin.document.write(
          '<!doctype html><html><head><title>Print<\/title><\/head><body onload="print();">' +
            doc.innerHTML +
            "<\/body><\/html>"
        );
        printWin.document.close();
      }
    </script>
    <style>
      .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 = doc.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?')) { doc.innerHTML = defTxt };"
          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="const lnk = prompt('Write the URL here','http:\/\/');if (lnk && lnk != '' && lnk != '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>

See also