Get the FREE Ultimate OpenClaw Setup Guide →

unlayer-integration

Scanned
npx machina-cli add skill unlayer/unlayer-skills/unlayer-integration --openclaw
Files (1)
SKILL.md
10.0 KB

Integrate Unlayer Editor

Overview

Unlayer provides official wrappers for React, Vue, and Angular, plus a plain JavaScript embed. All wrappers share the same underlying API — only the editor access pattern differs.

Which Framework?

FrameworkPackageInstallEditor Access
Reactreact-email-editornpm i react-email-editoruseRef<EditorRef>ref.current?.editor
Vuevue-email-editornpm i vue-email-editorthis.$refs.emailEditor.editor
Angularangular-email-editornpm i angular-email-editor@ViewChildthis.emailEditor.editor
Plain JSCDN script tag<script> embedGlobal unlayer object

⚠️ Before installing any Unlayer package, verify the version exists on npm:

npm view react-email-editor version   # check latest published version

Never pin a version number you haven't verified. Use npm install <package> --save without a version to get the latest, or run npm view <package> versions --json to see all available versions.


React (Complete Working Example)

npm install react-email-editor --save
import React, { useRef, useState } from 'react';
import EmailEditor, { EditorRef, EmailEditorProps } from 'react-email-editor';

const EmailBuilder = () => {
  const emailEditorRef = useRef<EditorRef>(null);
  const [saving, setSaving] = useState(false);

  // Save design JSON + export HTML to your backend
  const handleSave = () => {
    const unlayer = emailEditorRef.current?.editor;
    if (!unlayer) return;

    setSaving(true);
    unlayer.exportHtml(async (data) => {
      try {
        const response = await fetch('/api/templates', {
          method: 'POST',
          headers: { 'Content-Type': 'application/json' },
          body: JSON.stringify({
            design: data.design,  // Save this — needed to edit later
            html: data.html,      // The rendered HTML output
          }),
        });
        if (!response.ok) throw new Error('Save failed');
        console.log('Saved successfully');
      } catch (err) {
        console.error('Save error:', err);
      } finally {
        setSaving(false);
      }
    });
  };

  // Load a saved design when editor is ready
  const onReady: EmailEditorProps['onReady'] = async (unlayer) => {
    try {
      const response = await fetch('/api/templates/123');
      if (response.ok) {
        const saved = await response.json();
        unlayer.loadDesign(saved.design); // Pass the saved design JSON
      }
    } catch (err) {
      console.log('No saved design, starting blank');
    }
  };

  return (
    <div>
      <button onClick={handleSave} disabled={saving}>
        {saving ? 'Saving...' : 'Save'}
      </button>
      <EmailEditor
        ref={emailEditorRef}
        onReady={onReady}
        options={{
          projectId: 123456, // Dashboard > Project > Settings
          displayMode: 'email',
        }}
      />
    </div>
  );
};

export default EmailBuilder;

Your backend should accept and return:

// POST /api/templates — save
{ design: object, html: string }

// GET /api/templates/:id — load
{ design: object, html: string, updatedAt: string }

React Props:

PropTypeDefaultDescription
optionsObject{}All unlayer.init() options (projectId, displayMode, etc.)
toolsObject{}Per-tool configuration
appearanceObject{}Theme and panel settings
onReadyFunctionCalled when editor is ready (receives unlayer instance)
onLoadFunctionCalled when iframe loads (before ready)
styleObject{}Container inline styles
minHeightString'500px'Minimum editor height

Docs: https://docs.unlayer.com/builder/react-component GitHub: https://github.com/unlayer/react-email-editor


Vue (Complete Working Example)

npm install vue-email-editor --save
<template>
  <div id="app">
    <button v-on:click="handleSave" :disabled="saving">
      {{ saving ? 'Saving...' : 'Save' }}
    </button>
    <EmailEditor ref="emailEditor" v-on:load="editorLoaded" />
  </div>
</template>

<script>
import { EmailEditor } from 'vue-email-editor';

export default {
  components: { EmailEditor },
  data() {
    return { saving: false };
  },
  methods: {
    async editorLoaded() {
      try {
        const response = await fetch('/api/templates/123');
        if (response.ok) {
          const saved = await response.json();
          this.$refs.emailEditor.editor.loadDesign(saved.design);
        }
      } catch (err) {
        console.log('No saved design, starting blank');
      }
    },
    handleSave() {
      this.saving = true;
      this.$refs.emailEditor.editor.exportHtml(async (data) => {
        try {
          await fetch('/api/templates', {
            method: 'POST',
            headers: { 'Content-Type': 'application/json' },
            body: JSON.stringify({ design: data.design, html: data.html }),
          });
        } catch (err) {
          console.error('Save error:', err);
        } finally {
          this.saving = false;
        }
      });
    },
  },
};
</script>

Props: minHeight, options, tools, appearance, locale, projectId.

Docs: https://docs.unlayer.com/builder/vue-component GitHub: https://github.com/unlayer/vue-email-editor


Angular (Complete Working Example)

npm install angular-email-editor --save

Module (app.module.ts):

import { EmailEditorModule } from 'angular-email-editor';

@NgModule({ imports: [EmailEditorModule] })
export class AppModule {}

Component (app.component.ts):

import { Component, ViewChild } from '@angular/core';
import { EmailEditorComponent } from 'angular-email-editor';

@Component({
  selector: 'app-root',
  templateUrl: './app.component.html',
})
export class AppComponent {
  @ViewChild(EmailEditorComponent)
  private emailEditor: EmailEditorComponent;
  saving = false;

  async editorLoaded() {
    try {
      const response = await fetch('/api/templates/123');
      if (response.ok) {
        const saved = await response.json();
        this.emailEditor.editor.loadDesign(saved.design);
      }
    } catch (err) {
      console.log('No saved design');
    }
  }

  handleSave() {
    this.saving = true;
    this.emailEditor.editor.exportHtml(async (data) => {
      try {
        await fetch('/api/templates', {
          method: 'POST',
          headers: { 'Content-Type': 'application/json' },
          body: JSON.stringify({ design: data.design, html: data.html }),
        });
      } catch (err) {
        console.error('Save error:', err);
      } finally {
        this.saving = false;
      }
    });
  }
}

Template (app.component.html):

<div>
  <button (click)="handleSave()" [disabled]="saving">
    {{ saving ? 'Saving...' : 'Save' }}
  </button>
  <email-editor (loaded)="editorLoaded($event)"></email-editor>
</div>

Docs: https://docs.unlayer.com/builder/angular-component


Plain JavaScript

<script src="https://editor.unlayer.com/embed.js"></script>
<div id="editor-container" style="height: 700px;"></div>

<script>
  unlayer.init({
    id: 'editor-container',
    projectId: 1234,
    displayMode: 'email',
  });

  unlayer.addEventListener('editor:ready', async function () {
    try {
      const response = await fetch('/api/templates/123');
      if (response.ok) {
        const saved = await response.json();
        unlayer.loadDesign(saved.design);
      }
    } catch (err) {
      console.log('Starting blank');
    }
  });

  function handleSave() {
    unlayer.exportHtml(async function (data) {
      await fetch('/api/templates', {
        method: 'POST',
        headers: { 'Content-Type': 'application/json' },
        body: JSON.stringify({ design: data.design, html: data.html }),
      });
    });
  }
</script>

Docs: https://docs.unlayer.com/builder/installation


Multiple Editor Instances

Use createEditor() instead of init():

const editor1 = unlayer.createEditor({ id: 'email-editor', displayMode: 'email' });
const editor2 = unlayer.createEditor({ id: 'web-editor', displayMode: 'web' });

editor1.loadDesign(emailDesign);
editor2.loadDesign(webDesign);

Common Mistakes

MistakeFix
Container too smallMinimum 1024px wide x 700px tall. Editor fills its container.
Calling methods before readyAlways wait for editor:ready event or onReady callback
Using init() for multiple editorsUse unlayer.createEditor() instead
Missing projectIdGet it from Dashboard > Project > Settings
Only saving HTML, not design JSONAlways save both — design JSON lets users edit later
No loading state while savingDisable the save button to prevent double saves
Pinning a non-existent package versionRun npm view <package> version to verify before pinning. Use npm install <package> --save without a version to get the latest.

Troubleshooting

ErrorCauseFix
Editor shows blank white spaceContainer div has 0 heightSet explicit height: min-height: 700px
editor:ready never firesScript not loaded or wrong idCheck id matches an existing div, check network tab for embed.js
ref.current?.editor is undefinedAccessed before mountOnly use inside onReady callback
Design doesn't loadMalformed JSONValidate JSON, check schemaVersion field

Resources

Source

git clone https://github.com/unlayer/unlayer-skills/blob/main/unlayer-integration/SKILL.mdView on GitHub

Overview

This skill covers integrating the Unlayer editor using official wrappers for React, Vue, Angular, or plain JavaScript. All wrappers share the same underlying API, with editor access patterns varying by framework, and support for multiple editor instances on a page.

How This Skill Works

Choose a wrapper (React, Vue, Angular, or plain JS) and obtain the editor instance through the framework-specific ref. Then use the shared API (e.g., exportHtml, loadDesign) to save and load designs. The setup enables multiple editor instances and consistent editor access across frameworks.

When to Use It

  • I’m building a React app and need an embeddable email editor with exportable HTML/design data.
  • I’m using Vue and want a component that exposes the editor via refs for programmatic control.
  • I’m working in Angular and need to access the editor instance with ViewChild.
  • I’m adding a plain JavaScript embed via a CDN script tag for a lightweight page.
  • I need to run multiple Unlayer editor instances on the same page.

Quick Start

  1. Step 1: Install the wrapper for your framework (e.g., npm i react-email-editor).
  2. Step 2: Create a component and hold a reference to the editor (e.g., useRef in React).
  3. Step 3: On events like Save or Ready, access the editor via ref and call methods like exportHtml or loadDesign.

Best Practices

  • Verify the npm version exists before installing any Unlayer package to avoid pinning a non-existent version.
  • Use the official wrappers (React, Vue, Angular) or the plain JS embed to match your project.
  • Access the editor via the framework’s recommended ref mechanism (useRef, this.$refs, @ViewChild) and avoid direct DOM manipulation.
  • Leverage onReady and exportHtml/loadDesign to persist and restore designs safely.
  • Plan for multiple editors by managing separate refs and isolated state per instance.

Example Use Cases

  • Embed a React EmailEditor with a ref, call exportHtml on Save, and POST design + HTML to the backend.
  • In Vue, load a saved design on editor ready by calling unlayer.loadDesign(saved.design).
  • An Angular page uses @ViewChild to access this.emailEditor.editor and export HTML for storage.
  • A plain JS page uses the global unlayer object to initialize and interact with the editor without a framework.
  • Place multiple email editors on a single dashboard page to edit different templates concurrently.

Frequently Asked Questions

Add this skill to your agents
Sponsor this space

Reach thousands of developers