Skip to main content
Version: 1.157.0

Create a Custom Tool

This tutorial will get you started with your first custom tool using our JavaScript API.

Before you proceed, please make sure you have read the overview and understand the following:

Custom JavaScript

The following JavaScript has to be passed in the customJS parameter when initializing Unlayer. registerTool, registerPropertyEditor, createViewer and createWidget methods are only available there.

Learn More

Register your Tool

The first step is to register your custom tool using the registerTool method.

Register Tool Interface

unlayer.registerTool({
name: 'my_tool',
label: 'My Tool',
icon: 'fa-smile',
supportedDisplayModes: ['web', 'email'],
options: {},
values: {},
renderer: {
Viewer: unlayer.createViewer({
render(values) {
return '<div>I am a custom tool.</div>';
},
}),
exporters: {
web: function (values) {
return '<div>I am a custom tool.</div>';
},
email: function (values) {
return '<div>I am a custom tool.</div>';
},
},
head: {
css: function (values) {},
js: function (values) {},
},
},
validator(data) {
return [];
},
});

It needs some configuration options which are explained below:

OptionDescription
nameUnique name for your tool
labelLabel that users see in the tools panel
iconIcon that users see in the tools panel. The value can be class name of any fontawesome icon, or a URL for your icon's image
supportedDisplayModesWhich display modes will this tool work in. Default value is ['email', 'web']
usageLimitOptional. You can set a limit to the number of times this tool can be used in a design
optionsThis is where we define tool properties and property editors
valuesInitial values for our tool's properties
rendererThis is where we will add a Viewer , exporters and head
validatorYou can add a validator function to check for issues in the values. For example, you can check values of properties to make sure everything looks correct. These errors/warnings would show up in the Audit tab. Learn More
React: Recommended Usage

If you are using React to create a custom tool or property editor, you can reuse Unlayer's React. We have unlayer.React and unlayer.ReactDOM available globally. This is recommended to avoid loading React twice and run into version conflicts. Check our React Example.


Add Tool Properties

Every tool needs certain properties to create content. Properties are variables that the user can modify, such as colors, font sizes, text alignment, images, and more.

Each property is assigned a property editor which is a UI control widget used by the user to modify the property value. Check out the list of built-in property editors.

Color Picker Example

In this example, we will add 2 properties to our tool under a group called "Colors":

  1. Text Color textColor
  2. Background Color backgroundColor

Property Editors Interface

Let's define the property group and both the properties inside it. This will be done in the options object. Each property group should have a unique key in options so we will call this group colors.

We will name the properties textColor and backgroundColor and use the built-in color_picker editor widget for both of them. We will give each property a default value of #FF0000.

For the Renderer, we will use the textColor and backgroundColor in the HTML to apply those colors to our template. These properties will be available in the values object passed to the Viewer and the exporters.

unlayer.registerTool({
name: 'my_tool',
label: 'My Tool',
icon: 'fa-smile',
supportedDisplayModes: ['web', 'email'],
options: {
colors: {
// Property Group
title: 'Colors', // Title for Property Group
position: 1, // Position of Property Group
options: {
textColor: {
// Property: textColor
label: 'Text Color', // Label for Property
defaultValue: '#FF0000',
widget: 'color_picker', // Property Editor Widget: color_picker
},
backgroundColor: {
// Property: backgroundColor
label: 'Background Color', // Label for Property
defaultValue: '#FF0000',
widget: 'color_picker', // Property Editor Widget: color_picker
},
},
},
},
values: {},
renderer: {
Viewer: unlayer.createViewer({
render(values) {
return `<div style="color: ${values.textColor}; background-color: ${values.backgroundColor};">I am a custom tool.</div>`;
},
}),
exporters: {
web: function (values) {
return `<div style="color: ${values.textColor}; background-color: ${values.backgroundColor};">I am a custom tool.</div>`;
},
email: function (values) {
return `<div style="color: ${values.textColor}; background-color: ${values.backgroundColor};">I am a custom tool.</div>`;
},
},
head: {
css: function (values) {},
js: function (values) {},
},
},
validator(data) {
const { defaultErrors, values } = data;
return [];
},
});

Custom Property Editor

In the example above, we used a built-in color_picker property editor. We have many built-in property editors available but many times you may want to use a custom property editor that is tightly integrated with your application.

Here's a small example where we add a custom color picker property editor. We'll call it my_color_picker. We will use the registerPropertyEditor method to register it.

Custom Property Editor Interface

unlayer.registerPropertyEditor({
name: 'my_color_picker',
Widget: unlayer.createWidget({
render(value, updateValue, data) {
return `
<input class="color-value" value=${value} />
<button class="red">Red</button>
<button class="green">Green</button>
<button class="blue">Blue</button>
`;
},
mount(node, value, updateValue, data) {
var input = node.getElementsByClassName('color-value')[0];
input.onchange = function (event) {
updateValue(event.target.value);
};

var redButton = node.getElementsByClassName('red')[0];
redButton.onclick = function () {
updateValue('#f00');
};

var greenButton = node.getElementsByClassName('green')[0];
greenButton.onclick = function () {
updateValue('#0f0');
};

var blueButton = node.getElementsByClassName('blue')[0];
blueButton.onclick = function () {
updateValue('#00f');
};
},
}),
});

Once our new property editor is registered, you can use it just like the built-in property editors when registering your tool. The configuration options are explained below:

OptionDescription
nameUnique slug for your property editor widget. You will have to provide this in options as a widget when registering your tool.
WidgetThis is the UI widget. If you are using React, this would be a Component. If you are using vanilla JavaScript, you will use createWidget.

Widget

For each property editor, you have to pass a Widget. If you are using React, this would be a Component. If you are using vanilla JavaScript, you will use the createWidget function.

JavaScript

The unlayer.createWidget function would be passed an object with the following attributes. You can check the full example.

AttributeDescription
renderThis is the HTML of your widget. It is passed the following arguments:

value is the current value of this property
updateValue is the function to update the value of this property. The first argument has to be the new value, and the second argument is optional data that can be used in the transformer to update values for other properties.
data is the optional data passed by your application
mountThis gets called when your widget is mounted. You can attach events to the HTML elements here. It is passed the following arguments:

node is the DOM ref to the HTML element that you created in the render method.
value is the current value of this property
updateValue is the function to update the value of this property. The first argument has to be the new value, and the second argument is optional data that can be used in the transformer to update values for other properties.
data is the optional data passed by your application

React

If you are using React, you can create the same widget as a Component. You can check the full example here.

import React from 'react';

const MyColorPicker = (props) => {
const { label, value, updateValue, data } = props;

return (
<div>
<input
class="color-value"
defaultValue={value}
onChange={(e) => updateValue(e.target.value)}
/>
<button class="red" onClick={() => updateValue('#f00')}>
Red
</button>
<button class="green" onClick={() => updateValue('#0f0')}>
Green
</button>
<button class="blue" onClick={() => updateValue('#00f')}>
Blue
</button>
</div>
);
};

unlayer.registerPropertyEditor({
name: 'my_color_picker',
Widget: MyColorPicker,
});
React: Recommended Usage

If you are using React to create a custom tool or property editor, you can reuse Unlayer's React. We have unlayer.React and unlayer.ReactDOM available globally. This is recommended to avoid loading React twice and run into version conflicts. Check our React Example.


Pass Data to Custom Tools

In some cases, your custom tool will need data from your application instance. You can pass that data when the Unlayer editor is initialized. Inside the tools object, you can add the data that your tool needs.

In this example, we will pass user's information. This data will be available in the values parameter to your tool's renderer and exporters. If your tool's name is my_tool, you will add an object called custom#my_tool and pass the data object in it. Replace NAME with the name of your tool.

You can see a full running example here: Live Demo

unlayer.init({
...
tools: {
'custom#NAME': {
data: {
name: 'John Doe',
age: '27',
photo: 'http://via.placeholder.com/350x150'
}
}
}
});

Pass Data to Property Editors

Some property editors need data from your application programmatically. You can pass this data during the initialization.

Replace NAME with your tool's name and PROPERTY-NAME with the property's name. For example, if your tool's name is my_color_picker and property name is colorOptions, you will use custom#my_color_picker as the key and colorOptions as the property name.

You can see a full running example here: Live Demo

unlayer.init({
...
tools: {
'custom#NAME': {
properties: {
PROPERTY-NAME: {
editor: {
data: {
"hello": "world"
}
}
}
}
}
}
});

Let's say your custom tool has a property that needs to use our built-in dropdown property editor. You can use the following example to pass options to the dropdown property editor.

In this example, our tool's name is contact_form and our property's name is department which is using a dropdown editor. We want the user to pick Sales, Customer Support, HR or Other from the dropdown.

You can see a full running example here: Live Demo

unlayer.init({
...
tools: {
'custom#contact_form': {
properties: {
department: {
editor: {
data: {
options: [
{label: "Custom Support", value: "Customer Support"},
{label: "Sales", value: "Sales"},
{label: "HR", value: "HR"},
{label: "Other", value: "Other"}
]
}
}
}
}
}
}
});

Full Example

We have a full example of custom tools and custom property editors with the JavaScript API in our playgrounds. Check it out here.