Custom element conversion
There are three levels on which elements can be converted:
- By using the two-way converter:
conversion.elementToElement()
.
This is a fully declarative API. It is the least powerful option but it is the easiest one to use. - By using one-way converters: for example
conversion.for( 'downcast' ).elementToElement()
andconversion.for( 'upcast' ).elementToElement()
.
In this case, you need to define at least two converters (for upcast and downcast), but the “how” part becomes a callback, and hence you gain more control over it. - Finally, by using event-based converters.
In this case, you need to listen to events fired byDowncastDispatcher
andUpcastDispatcher
. This method has full access to every bit of logic that a converter needs to implement and therefore it can be used to write the most complex conversion methods.
This guide explains how to migrate from a simple two-way converter to an event-based converter as the requirements regarding the feature get more complex.
# Introduction
Let us assume that the content in your application contains “info boxes”. As for now, it was only required to wrap a part of the content in a <div>
element that would look like this in the data and editing views:
<div class="info-box">
<!-- Any editable content. -->
<p>This is <strong>important!</strong></p>
</div>
The data is represented in the model as the following structure:
<infoBox>
<!-- Any $block content. -->
<paragraph><$text>This is </$text><$text bold="true">important!</$text></paragraph>
</infoBox>
This can be easily done with the below schema and converters in a simple InfoBox
plugin:
class InfoBox {
constructor( editor ) {
// 1. Define infoBox as an object that can contain any other content.
editor.model.schema.register( 'infoBox', {
allowWhere: '$block',
allowContentOf: '$root',
isObject: true
} );
// 2. The conversion is straightforward:
editor.conversion.elementToElement( {
model: 'infoBox',
view: {
name: 'div',
classes: 'info-box'
}
} );
}
}
# Migrating to an event-based converter
Let us now assume that the requirements have changed and there is a need for adding an additional element in the data and editing views that will display the type of the info box (warning, error, info, etc.).
The new info box structure:
<div class="info-box info-box-warning">
<div class="info-box-title">Warning</div>
<div class="info-box-content">
<!-- Any editable content. -->
<p>This is <strong>important!</strong></p>
</div>
</div>
The “Warning” part should not be editable. It defines the type of the info box so you can store this bit of information as an attribute of the <infoBox>
element:
<infoBox infoBoxType="warning">
<!-- Any $block content. -->
<paragraph><$text>This is </$text><$text bold="true">important!</$text></paragraph>
</infoBox>
Let us see how to update the basic implementation to cover these requirements.
# Demo
Below is a demo of the editor with a sample info box.
# Schema
The type of the box is defined by an additional class on the main <div>
but it is also represented as text in <div class="info-box-title">
. All the info box content must now be placed inside <div class="info-box-content">
instead of the main wrapper.
For the above requirements you can see that the model structure of the infoBox
does not need to change much. You can still use a single element in the model. The only addition to the model is an attribute that will store information about the info box type:
editor.model.schema.register( 'infoBox', {
allowWhere: '$block',
allowContentOf: '$root',
isObject: true,
allowAttributes: [ 'infoBoxType' ] // Added.
} );
# Event-based upcast converter
The conversion of the type of the box itself can be achieved by using attributeToAttribute()
(info-box-*
CSS classes to the infoBoxType
model attribute). However, two more changes were made to the data format that you need to handle:
- There is a new
<div class="info-box-title">
element that should be ignored during the upcast conversion as it duplicates the information conveyed by the main element’s CSS class. - The content of the info box is now located inside another element. Previously it was located directly in the main wrapper.
Neither two-way nor one-way converters can handle such conversion. Therefore, you need to use an event-based converter with the following behavior:
- Create a model
<infoBox>
element with theinfoBoxType
attribute. - Skip the conversion of
<div class="info-box-title">
as the information about type can be obtained from the wrapper’s CSS classes. - Convert the children of
<div class="info-box-content">
and insert them directly into<infoBox>
.
function upcastConverter( event, data, conversionApi ) {
const viewInfoBox = data.viewItem;
// Check whether the view element is an info box <div>.
// Otherwise, it should be handled by another converter.
if ( !viewInfoBox.hasClass( 'info-box' ) ) {
return;
}
// Create the model structure.
const modelElement = conversionApi.writer.createElement( 'infoBox', {
infoBoxType: getTypeFromViewElement( viewInfoBox )
} );
// Try to safely insert the element into the model structure.
// If `safeInsert()` returns `false`, the element cannot be safely inserted
// into the content and the conversion process must stop.
// This may happen if the data that you are converting has an incorrect structure
// (e.g. it was copied from an external website).
if ( !conversionApi.safeInsert( modelElement, data.modelCursor ) ) {
return;
}
// Mark the info box <div> as handled by this converter.
conversionApi.consumable.consume( viewInfoBox, { name: true } );
// Let us assume that the HTML structure is always the same.
// Note: For full bulletproofing this converter, you should also check
// whether these elements are the right ones.
const viewInfoBoxTitle = viewInfoBox.getChild( 0 );
const viewInfoBoxContent = viewInfoBox.getChild( 1 );
// Mark info box inner elements (title and content <div>s) as handled by this converter.
conversionApi.consumable.consume( viewInfoBoxTitle, { name: true } );
conversionApi.consumable.consume( viewInfoBoxContent, { name: true } );
// Let the editor handle the children of <div class="info-box-content">.
conversionApi.convertChildren( viewInfoBoxContent, modelElement );
// Finally, update the conversion's modelRange and modelCursor.
conversionApi.updateConversionResult( modelElement, data );
}
// A helper function to read the type from the view classes.
function getTypeFromViewElement( viewElement ) {
if ( viewElement.hasClass( 'info-box-info' ) ) {
return 'Info';
}
if ( viewElement.hasClass( 'info-box-warning' ) ) {
return 'Warning';
}
return 'None';
}
This upcast converter callback can now be plugged by adding a listener to the UpcastDispatcher#element
event. You will listen to element:div
to ensure that the callback is called only for <div>
elements.
editor.conversion.for( 'upcast' )
.add( dispatcher => dispatcher.on( 'element:div', upcastConverter ) );
# Event-based downcast converter
The missing bits are the downcast converters for the editing and data pipelines.
You will want to use the widget system to make the info box behave like an “object”. Another aspect that you need to take care of is the fact that the view structure has more elements than the model structure. In this case, you could actually use one-way converters. However, this tutorial will showcase how an event-based converter would look.
See the Implementing a block widget guide to learn about the widget system.
The remaining downcast converters:
function editingDowncastConverter( event, data, conversionApi ) {
let { infoBox, infoBoxContent, infoBoxTitle } = createViewElements( data, conversionApi );
// Decorate view items as a widget and widget editable area.
infoBox = toWidget( infoBox, conversionApi.writer, { label: 'info box widget' } );
infoBoxContent = toWidgetEditable( infoBoxContent, conversionApi.writer );
insertViewElements( data, conversionApi, infoBox, infoBoxTitle, infoBoxContent );
}
function dataDowncastConverter( event, data, conversionApi ) {
const { infoBox, infoBoxContent, infoBoxTitle } = createViewElements( data, conversionApi );
insertViewElements( data, conversionApi, infoBox, infoBoxTitle, infoBoxContent );
}
function createViewElements( data, conversionApi ) {
const type = data.item.getAttribute( 'infoBoxType' );
const infoBox = conversionApi.writer.createContainerElement( 'div', {
class: `info-box info-box-${ type.toLowerCase() }`
} );
const infoBoxContent = conversionApi.writer.createEditableElement( 'div', {
class: 'info-box-content'
} );
const infoBoxTitle = conversionApi.writer.createUIElement( 'div',
{ class: 'info-box-title' },
function( domDocument ) {
const domElement = this.toDomElement( domDocument );
domElement.innerText = type;
return domElement;
} );
return { infoBox, infoBoxContent, infoBoxTitle };
}
function insertViewElements( data, conversionApi, infoBox, infoBoxTitle, infoBoxContent ) {
conversionApi.consumable.consume( data.item, 'insert' );
conversionApi.writer.insert(
conversionApi.writer.createPositionAt( infoBox, 0 ),
infoBoxTitle
);
conversionApi.writer.insert(
conversionApi.writer.createPositionAt( infoBox, 1 ),
infoBoxContent
);
// The default mapping between the model <infoBox> and its view representation.
conversionApi.mapper.bindElements( data.item, infoBox );
// However, since the model <infoBox> content needs to end up in the inner
// <div class="info-box-content">, you need to bind one with another overriding
// a part of the default binding.
conversionApi.mapper.bindElements( data.item, infoBoxContent );
conversionApi.writer.insert(
conversionApi.mapper.toViewPosition( data.range.start ),
infoBox
);
}
These two converters need to be plugged as listeners to the DowncastDispatcher#insert
event:
editor.conversion.for( 'editingDowncast' )
.add( dispatcher => dispatcher.on( 'insert:infoBox', editingDowncastConverter ) );
editor.conversion.for( 'dataDowncast' )
.add( dispatcher => dispatcher.on( 'insert:infoBox', dataDowncastConverter ) );
# Updated plugin code
The updated InfoBox
plugin that glues the event-based converters together:
class InfoBox {
constructor( editor ) {
// Schema definition.
editor.model.schema.register( 'infoBox', {
allowWhere: '$block',
allowContentOf: '$root',
isObject: true,
allowAttributes: [ 'infoBoxType' ]
} );
// Upcast converter.
editor.conversion.for( 'upcast' )
.add( dispatcher => dispatcher.on( 'element:div', upcastConverter ) );
// The downcast conversion must be split as you need a widget in the editing pipeline.
editor.conversion.for( 'editingDowncast' )
.add( dispatcher => dispatcher.on( 'insert:infoBox', editingDowncastConverter ) );
editor.conversion.for( 'dataDowncast' )
.add( dispatcher => dispatcher.on( 'insert:infoBox', dataDowncastConverter ) );
}
}
function upcastConverter() {
// ...
}
function editingDowncastConverter() {
// ...
}
function dataDowncastConverter() {
// ...
}