Start trial
Plans & PricingContact Us
Log InStart For Free

How to build a Workflow app with TinyMCE Self-hosted & Cloud

August 3rd, 2022

5 min read

The workflow config, self-hosted and cloud, in the cloud represented by arrow, gear, and star icons

Written by

Joe Robinson

Category

How-to Use TinyMCE

Attention to detail makes a workflow app better than your average Kanban board. It combines many small features and abilities – and adding up to something better, more useful, and productive. 

If you’re building a workflow app, which features should be combined together to make that better, useful, and productive experience?

There’s the big three priorities: User Interface (UI), User Experience (UX), and Information Architecture. Following that, there’s knowing what framework to build on.  Then there are the small components…

Don’t overlook the small components. 

If they’re really contributing to the bigger picture, those small components help to carry the larger priorities. This is the exact role that TinyMCE plays – where it supports a larger tech stack, as a rich text editing component within the workflow app.

TinyMCE’s rich text editing capabilities have maintained the trust of more than 1.5 million developers worldwide. One reason for that is TinyMCE’s flexibility. You can combine different TinyMCE configurations to suit your workflow project. If you’re using TinyMCE Self-hosted, for instance, but you need access to TinyMCE’s premium plugins, you can combine self-hosting with TinyMCE Cloud.

That’s what this article explains – how to set up a combined TinyMCE Cloud and Self-hosted configuration, using the TinyMCE Workflow starter config, which contains all of the plugins and features common to a market-leading workflow platform.

Getting started with a workflow app

To get started, what you’ll need is a TinyMCE API key. To get your API key:

For a workflow demo, this article adapts content from the TinyMCE workflow started config:

Adding your API key gives you a 14-day free trial of TinyMCE plugins. If you're accessing TinyMCE through Tiny Cloud, your API key is also required to prevent the editor changing into read-only mode. Check on the API key documentation for further information.

Locating your FREE API key

Once you’ve navigated to the sign up page, you can create a TinyMCE account. The account page has a number of useful features, including a getting started config and the API key. The key is available within the top section of the account page.

Why use an API key? The key gives you 14 days FREE access to TinyMCE premium plugins. It's important for accessing Premium plugins, even when self-hosting TinyMCE.

The free trial begins when you use the key in your editor, and load it into a browser with Premium plugins configured in the tiny.init script. The key also prevents errors appearing in the textarea.

Be aware that if you change your configuration from self-hosted to Tiny Cloud, you'll need your API key added to your TinyMCE configuration to prevent the editor changing to read-only mode.

Getting the TinyMCE SDK for self-hosting

Self-hosting TinyMCE requires a copy of the SDK. You can download a copy, and reference the TinyMCE JavaScript file, by completing the following:

  1. Head to the Get TinyMCE page, and click on the Download TinyMCE SDK Now button
  2. Unzip the downloaded folder
  3. Set up a new folder inside your workstation environment, and place the TinyMCE unzipped file within the new folder.

Create your demo workflow project

In a workflow app, large tasks break into smaller tasks, which have their own specific screens. The demo workflow has a UI that matches a task screen

The TinyMCE workflow demo config working in the browser

This would be something similar to what you would see in a Kanban board. There are two rich text editors in the config: a description text editor, and a comment box text editor.

The description editor has several functions included with TinyMCE’s premium plugins so your customers can express themselves. The comment editor has fewer toolbar options compared to the description editor to keep the experience minimal. 

The button elements on the page are not connected to any functions. For an app in production, you would need to program save functionality that runs when customers press the button.

The demo project comes pre-configured with a combination of TinyMCE Cloud and Self-hosted premium plugins. And, if you have an existing workflow project, you can adapt these instructions to add-in TinyMCE, as a component to yout project

To get started:

  1. Create an index.html file inside the new folder you created on your workstation in the previous step on getting the TinyMCE SDK folder
  2. Copy the following workflow demo HTML into your index file. The content is drawn directly from the TinyMCE workflow and collaboration platform solution:
<!doctype html>
<html>
<head>
  <meta charset="utf-8">
  <title>TinyMCE Workflow Starter Config</title>
  <meta name="viewport" content="width=device-width, initial-scale=1">
  <script>
    let commonConfig = {
      plugins: "advcode advtable autocorrect autolink checklist codesample editimage emoticons image link linkchecker lists media mediaembed powerpaste table tinymcespellchecker",
      menubar: false,
    }
      toolbar: "blocks | bold italic forecolor backcolor | numlist bullist checklist | link image emoticons table codesample hr blockquote | code ",
      setup: (editor) => {
        editor.on('focus', () => {
          document.getElementById('editor-description-wrap').classList.add('enabled');
        });
      },

      ...commonConfig,
    });

    tinymce.init({
      selector: "#editor-comment-content",
      toolbar: "bold italic forecolor backcolor | numlist bullist | link image emoticons codesample blockquote ",

      setup: (editor) => {
          editor.on('focus', () => {
            document.getElementById('editor-comment-wrap').classList.add('enabled');
          });
        },
      ...commonConfig,
    });

    function save(id) {
      document.getElementById(id).classList.remove('enabled');
    }

  </script>
  </head>
<body>
  <main>
  </main>
</body>
  1. Reference the TinyMCE folder in a script tag inside the document head section

This tag will get the self-hosted config started:

<script src="/path/to/tinymce/tinymce.min.js"></script>

Configure the CDN link and API key 

  1. Copy the config script, and paste it into the head section of your index.html file directly after the path to the tinymce.min.js file reference:

  <script>
    let commonConfig = {
      plugins: "advcode advtable autocorrect autolink checklist codesample editimage emoticons image link linkchecker lists media mediaembed powerpaste table tinymcespellchecker",
      menubar: false,
      inline: true,
      toolbar_persist: true,
      object_resizing: false,
      spellchecker_language: 'en',
      spellchecker_active: true
    }
      toolbar: "blocks | bold italic forecolor backcolor | numlist bullist checklist | link image emoticons table codesample hr blockquote | code ",
      fixed_toolbar_container: '#editor-description-toolbar',
      placeholder: "Add a description",
      editimage_toolbar: "rotateleft rotateright",
      setup: (editor) => {
        editor.on('focus', () => {
          document.getElementById('editor-description-wrap').classList.add('enabled');
        });
      },

      ...commonConfig,
    });

    tinymce.init({
      selector: "#editor-comment-content",
      toolbar: "bold italic forecolor backcolor | numlist bullist | link image emoticons codesample blockquote ",
      placeholder: "Add a comment...",
      fixed_toolbar_container: '#editor-comment-toolbar',

      setup: (editor) => {
          editor.on('focus', () => {
            document.getElementById('editor-comment-wrap').classList.add('enabled');
          });
        },
      ...commonConfig,
    });

    function save(id) {
      document.getElementById(id).classList.remove('enabled');
    }

  </script>

This TinyMCE init script has more than one editor – the description editor and the comment editor. At the same time, there are plugin features that both editors need. Instead of repeating the same plugin features, the config contains a specific set of methods that allow plugin features needed for both elements to load across both editors.

This is what’s called a common config.

The “commonConfig” option specified at the end of each individual tinymce init for the description and editor code loads the plugin features needed for both elements.

  1. Paste the CDN script before the tinymce.init script in your index.html file:

<script src="https://cdn.tiny.cloud/1/no-api-key/tinymce/6/plugins.min.js" referrerpolicy="origin"></script>
  1. Change the “your-api-key”, replacing it with your own API key.

  2. Save the changes.

Configure the workflow app interface

  1. Copy the following style tag content, and paste it into your index.html file after the TinyMCE Self host CDN scripts:

<style>
        body {
            font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen-Sans, Ubuntu, Cantarell, 'Helvetica Neue', Helvetica, Arial, sans-serif;
        }

        main {
            max-width: 720px;
            margin: auto;
        }

        .editor-wrap.enabled {
            border-radius: 3px;
            border: 1px solid #ccc;
        }

        .editor-content {
            transition: min-height .25s, padding-bottom .25s;
            min-height: 0;
            outline: none;
            border-radius: 1px;
            transition: box-shadow .15s, background-color .15s;
        }

        .editor-content>*:first-child {
            margin-top: 0;
        }

        .editor-content>*:last-child {
            margin-bottom: 0;
        }

        .editor-wrap#editor-description-wrap:hover:not(.enabled) .editor-content {
            background-color: #efefef;
            box-shadow: 0 0 0 6px #efefef;
        }

        .editor-wrap.enabled .editor-content {
            min-height: 140px;
            padding: 1rem;
        }

        .editor-toolbar {
            padding: 3px;
            display: none;
            border-bottom: 1px solid #ccc;
            z-index: 1;
        }

        .editor-wrap.enabled .editor-toolbar {
            display: block;
        }

        .editor-footer {
            margin: 1rem;
            display: none;
            padding-top: .5rem;
        }

        .editor-wrap.enabled .editor-footer {
            display: block;
        }

        .editor-save-btn {
            background: #207ab7;
            font-size: 16px;
            font-weight: bold;
            color: #fff;
            border: none;
            outline: none;
            height: 34px;
            line-height-step: 34px;
            padding: 0 16px;
            border-radius: 4px;
            margin: 0;
            -webkit-tap-highlight-color: rgba(0, 0, 0, 0);
            -webkit-user-select: none;
        }

        .editor-cancel-btn {
            background: #dfe3ec;
            font-size: 16px;
            font-weight: bold;
            color: #17224f;
            border: none;
            outline: none;
            height: 34px;
            line-height-step: 34px;
            padding: 0 16px;
            border-radius: 4px;
            margin: 0;
            -webkit-tap-highlight-color: rgba(0, 0, 0, 0);
            -webkit-user-select: none;
        }

        .editor-label {
            font-size: 19px;
            margin: 2rem 0 1rem;
            display: block;
            border-bottom: 1px solid #dfe3ec;
        }

        .comment {
            display: flex;
        }

        .avatar {
            background-color: #ff9999;
            color: #17224f;
            font-size: 18px;
            font-weight: bold;
            border-radius: 100px;
            width: 42px;
            height: 42px;
            line-height: 42px;
            text-align: center;
            margin-right: 8px;
        }

        .editor-wrap#editor-comment-wrap {
            padding: 1rem;
        }

        .editor-wrap#editor-comment-wrap .editor-content {
            padding: 1rem;
        }
        /* and not namespaced to `.editor-content` */

        .componentBorder {
            border: 1px solid #ccc;
            border-radius: 3px;
            padding: 0.01rem;
            flex-grow: 1;
            position: relative;
            width: 900px;
        }

        a,
        a:link {
            color: blue;
        }

        a:visited {
            color: purple;
        }

        a:hover {
            color: green;
        }

        a:active {
            color: red;
        }

        h1 {
            font-size: 1.75rem;
            font-weight: strong;
        }

        h2 {
            font-size: 1.5rem;
            font-weight: strong;
        }

        h3 {
            font-size: 1rem;
            font-weight: strong;
        }

        hr {
            margin: 1.5rem 0;
            padding: .5rem 0;
            border: 0;
        }

        hr::after {
            content: '';
            display: block;
            border-width: 0 0 1px 0;
            border-color: #ccc;
            border-style: solid;
        }

        img {
            max-width: 100%;
            height: auto;
        }

        .tox.tox-tinymce-aux .tox-toolbar__overflow {
            background-color: #fff !important;
        }
    </style>
  1. Update the HTML content by copying the following configuration into your demo file between the body tags:

<div class="componentBorder">
    <main>
        <label class="editor-label">Description</label>
        <div class="editor-wrap" id="editor-description-wrap">
            <div class="editor-toolbar" id="editor-description-toolbar"></div>
            <div class="editor-content" id="editor-description-content">
                <h2><strong>What’s your rich text editor </strong><strong>workflow</strong><strong> project? </strong></h2>
                <p>Are you…</p>
                <ul>
                    <li><strong>Building a new workflow or project management platform</strong> and need to add rich text editing functionality?</li>
                    <li><strong>Extending your existing workflow or project management app</strong> and need to enhance your default editor?</li>
                </ul>
                <p>Then use the only WYSIWYG editor that’s trusted by 1.5M developers.</p>
                <h3><strong>Curious about TinyMCE?</strong></h3>
                <p>Click inside this box to activate the editor and see how it works!</p>
            </div>
            <footer class="editor-footer">
                <button type="button" id="save-description-btn" onclick="save('editor-description-wrap')" class="editor-save-btn">Save</button>
                <button type="button" onclick="alert('Your own custom save function')" class="editor-cancel-btn">Cancel</button>
            </footer>
        </div>

        <label class="editor-label">Comments</label>
        <div class="comment">
            <div class="avatar">FD</div>
            <div class="editor-wrap" id="editor-comment-wrap">
                <div class="editor-toolbar" id="editor-comment-toolbar"></div>
                <div class="editor-content" id="editor-comment-content"></div>
                <footer class="editor-footer">
                    <button type="button" id="save-description-btn" onclick="save('editor-comment-wrap')" class="editor-save-btn">Save</button>
                    <button type="button" onclick="alert('Your own custom cancel function')" class="editor-cancel-btn">Cancel</button>
                </footer>
            </div>
        </div>
    </main>
   </div>
</body>

Save the changes, and load the demo in your browser:

TinyMCE workflow app config working in the browser

Need more workflow information?

To find more information, you can read through the workflow page sections. The workflow starter config comes with HTML comments, to explain what each aspect of the configuration does to support the larger workflow app.

Signing up for a FREE rich text editor API key gives you access to Premium plugin functionality for 14 days. After this time, you’ll need to consider which plan is suitable for your uses, and if your workflow application would benefit from the Premium plugin functionality. And, while you’re in your 14-day free trial period (or after you’ve signed up for a paid plan), you can contact support if you have any questions on adding TinyMCE to your workflow app.

TinyMCESelf HostingWorkflowConfiguration
byJoe Robinson

Technical and creative writer, editor, and a TinyMCE advocate. An enthusiast for teamwork, open source software projects, and baking. Can often be found puzzling over obscure history, cryptic words, and lucid writing.

Related Articles

  • How-to Use TinyMCEDec 12th, 2024

    Bootstrap Inline Forms: Step-by-Step Instructions | TinyMCE

Join 100,000+ developers who get regular tips & updates from the Tiny team.

This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.

Tiny logo

Stay Connected

SOC2 compliance badge

Products

TinyMCEDriveMoxieManager
© Copyright 2025 Tiny Technologies Inc.

TinyMCE® and Tiny® are registered trademarks of Tiny Technologies, Inc.