Skip to content
This repository has been archived by the owner on Jun 26, 2020. It is now read-only.

Latest commit

 

History

History
128 lines (88 loc) · 6.72 KB

conversion-introduction.md

File metadata and controls

128 lines (88 loc) · 6.72 KB
category menu-title order
framework-deep-dive-conversion
Advanced concepts
10

Advanced conversion concepts

This guide extends the {@link framework/guides/architecture/editing-engine introduction to CKEditor 5 editing engine architecture}. Therefore, we highly recommend reading the former guide first.

In this guide we will dive deeper into some of the conversion concepts.

Inline and block content

Generally speaking, there are two main types of the content in the editor view and data output: inline and block.

The inline content means elements like <strong>, <a> or <span>. Unlike <p>, <blockquote> or <div>, inline elements do not structure the data. Instead, they mark some text in a specific (visual and semantical) way. These elements are a characteristic of a text. For instance, you could say that some part of the text is bold, or is linked, etc.. This concept has its reflection in the model of the rich-text editor where <a> or <strong> are not represented as elements. Instead, they are attributes of the text.

For example — in the model, you might have a <paragraph> element with the "Foo bar" text, where "bar" has the bold attribute set to true. A pseudo–code of this model data structure could look as follows:

<paragraph>
	"Foo "			// no attributes
	"bar"			// bold=true
</paragraph>
Throughout the rest of this guide the following, shorter convention will be used to represent model text attributes:
```html
<paragraph>Foo <$text bold="true">bar</$text></paragraph>
```

Note that there is no <strong> or any other additional element there, it is just some text with an attribute.

So, when does this text become wrapped with a <strong> element? This happens during the conversion to the view. It is also important to know which type of a view element needs to be used. In case of elements that represent inline formatting, this should be an {@link module:engine/view/attributeelement~AttributeElement}.

Conversion of multiple text attributes

A model text node may have multiple attributes (e.g. be bolded and linked) and all of them are converted to their respective view elements by independent converters.

Keep in mind that in the model, attributes do not have any specific order. This is contrary to the editor view or HTML output, where inline elements are nested in one another. Fortunately, the nesting happens automatically during the conversion from the model to the view. This makes working in the model simpler, as features do not need to take care of breaking or rearranging elements in the model.

For instance, consider the following model structure:

<paragraph>
	<$text bold="true" linkHref="url">Foo </$text>
	<$text linkHref="url">bar</$text>
	<$text bold="true"> baz</$text>
</paragraph>

During the conversion, it will be converted to the following view structure:

<p>
	<a href="url"><strong>Foo </strong>bar</a><strong> baz</strong>
</p>

Note that the <a> element is converted in such way that it always becomes the "topmost" element. This is intentional so that no element ever breaks a link, which would otherwise look as follows:

<p>
	<strong><a href="url">Foo </a></strong><a href="url">bar</a><strong> baz</strong>
</p>

There are two links with the same href attribute next to each other in the generated view (editor output), which is semantically wrong. To make sure that it never happens, the view element that represents a link must have a priority defined. Most elements, like for instance <strong>, do not care about it and stick to the default priority (10). The {@link features/link link feature} ensures that all view <a> elements have the priority set to 5 so they are kept outside other elements.

Merging attribute elements during conversion

Most of the simple view inline elements like <strong> or <em> do not have any attributes. Some of them have just one, for instance <a> has its href.

But it is easy to come up with features that style a part of a text in a more complex way. An example would be the {@link features/font font family feature}. When used, it adds the fontFamily attribute to the text in the model, which is later converted to a <span> element with a corresponding style attribute.

So what would happen if several attributes were set on the same part of the text? Take this model example where fontSize is used next to fontFamily:

<paragraph>
	<$text fontFamily="Tahoma" fontSize="big">foo</$text>
</paragraph>

CKEditor 5 features are implemented in a granular way, which means that e.g. the font size converter is completely independent from the font family converter. This means that the above example is converted as follows:

  • fontFamily="value" converts to <span style="font-family: value;">,
  • fontSize="value" converts to <span class="text-value">.

And, in theory, you could expect the following HTML as a result:

<p>
	<span style="font-family: Tahoma;">
		<span class="text-big">foo</span>
	</span>
</p>

But this is not the most optimal output you can get from the rich-text editor. Why not have just one <span> element instead?

<p>
	<span style="font-family: Tahoma;" class="text-big">foo</span>
</p>

Obviously a single <span> makes more sense. And thanks to the merging mechanism built into the conversion process, this would be the actual result of the conversion.

Why is it so? In the above scenario, two model attributes are converted to <span> elements. When the first attribute (say, fontFamily) is converted, there is no <span> in the view yet. So the first <span> is added with the style attribute. But then, when fontSize is converted, the <span> is already in the view. The {@link module:engine/view/downcastwriter~DowncastWriter downcast writer} recognizes it and checks whether the elements can be merged, following these 3 rules:

  1. Both elements must have the same {@link module:engine/view/element~Element#name name}.
  2. Both elements must have the same {@link module:engine/view/attributeelement~AttributeElement#priority priority}.
  3. Neither can have an {@link module:engine/view/attributeelement~AttributeElement#id ID}.

Examples

Once you understand more about the conversion of model attributes, you can check some examples of:

  • {@link framework/guides/deep-dive/conversion-extending-output Extending the editor output} — How to extend the output of existing CKEditor 5 features.
  • {@link framework/guides/deep-dive/conversion-preserving-custom-content Preserving custom content} — How to make CKEditor 5 accept more content.