Contribute to this guide

guideMulti-root editor

Multi-root editor is an editor type that features multiple, separate editable areas.

The main difference between using a multi-root editor and using multiple separate editors (like in the inline editor demo) is the fact that in a multi-root editor all editable areas belong to the same editor instance share the same configuration, toolbar and the undo stack, and produce one document.

Destination of the Month

Picture of a sunlit facade of a Maltan building.
It's siesta time in Valletta.

Valletta

The capital city of Malta is the top destination this summer. It’s home to cutting-edge contemporary architecture, baroque masterpieces, delicious local cuisine, and at least 8 months of sun. It’s also a top destination for filmmakers, so you can take a tour through locations familiar to you from Game of Thrones, Gladiator, Troy, and many more.

The three greatest things you learn from traveling

Find out more

Walking the capitals of Europe: Warsaw

Find out more

# Editor example configuration

Check out the Quick start guide to learn more about implementing this kind of editor. You will find implementation steps there. You can see this example editor’s code below.

View editor configuration script
import MultiRootEditor from '@ckeditor/ckeditor5-build-multi-root';

MultiRootEditor
    .create(
        // Define roots / editable areas:
        {
            header: document.querySelector( '#header' ),
            content: document.querySelector( '#content' ),
            leftSide: document.querySelector( '#left-side' ),
            rightSide: document.querySelector( '#right-side' )
        },
        // Editor configration:
        {
            cloudServices: {
                // All predefined builds include the Easy Image feature.
                // Provide correct configuration values to use it.
                tokenUrl: 'https://example.com/cs-token-endpoint',
                uploadUrl: 'https://your-organization-id.cke-cs.com/easyimage/upload/'
                // Read more about Easy Image - https://ckeditor.com/docs/ckeditor5/latest/features/images/image-upload/easy-image.html.
                // For other image upload methods see the guide - https://ckeditor.com/docs/ckeditor5/latest/features/images/image-upload/image-upload.html.
            },
            toolbar: {
                items: [
                    'undo', 'redo',
                    '|', 'heading',
                    '|', 'bold', 'italic',
                    '|', 'link', 'insertImage', 'insertTable', 'mediaEmbed',
                    '|', 'bulletedList', 'numberedList', 'outdent', 'indent'
                ]
            }
        }
    )
    .then( editor => {
        window.editor = editor;

        // Append toolbar to a proper container.
        const toolbarContainer = document.querySelector( '#toolbar' );
        toolbarContainer.appendChild( editor.ui.view.toolbar.element );

        // Make toolbar sticky when the editor is focused.
        editor.ui.focusTracker.on( 'change:isFocused', () => {
            if ( editor.ui.focusTracker.isFocused ) {
                toolbarContainer.classList.add( 'sticky' );
            } else {
                toolbarContainer.classList.remove( 'sticky' );
            }
        } );
    } )
    .catch( error => {
        console.error( 'There was a problem initializing the editor.', error );
    } );

View editor content listing
<div id="toolbar"></div>
<!--
    Wrapping the structure inside a pair of
    contenteditable="true" + contenteditable="false" elements
    is required to provide proper caret handling when
    using arrow keys at the start and end of an editable area.
    
    You can skip them if you don't want to move the
    caret between editable areas using arrow keys.
!--> 
<div contenteditable="true">
    <div contenteditable="false">
        <div id="header">
            Header content is inserted here.
        </div>
        <div id="content">
            Main content is inserted here.
        </div>
        <div class="boxes">
            <div class="box box-left editor">
                <div id="left-side">
                    Left-side box content is inserted here.
                </div>
            </div>
            <div class="box box-right editor">
                <div id="right-side">
                    Right-side box content is inserted here.
                </div>
            </div>
        </div>
    </div>
</div>

<style>
    .box {
        width: 50%;
    }

    .box-left {
        margin-right: 10px;
    }

    /*
        Make the editable "fill" the whole box.
        The box will grow if the other box grows too.
        This makes the whole box "clickable".
     */
    .box .ck-editor__editable {
        height: 100%;
    }

    /*
        When toolbar receives this class, it becomes sticky.
        If the toolbar would be scrolled outside of the visible area,
        instead it is kept at the top edge of the window.
     */
    #toolbar.sticky {
        position: sticky;
        top: 0px;
        z-index: 10;
    }
</style>

# Setting and reading editor data

Please note that setting and reading the editor data is different for multi-root editor.

Pass an object when setting the editor data

Setting the data using editor.setData():

    editor.setData( {
        header: '<p>Content for header part.</p>',
        content: '<p>Content for main part.</p>',
        leftSide: '<p>Content for left-side box.</p>',
        rightSide: '<p>Content for right-side box.</p>'
    } );

Setting the data through config.initialData:

    MultiRootEditor.create(
        {
            header: document.querySelector( '#header' ),
            content: document.querySelector( '#content' ),
            leftSide: document.querySelector( '#left-side' ),
            rightSide: document.querySelector( '#right-side' )
        },
        {
            initialData: {
                header: '<p>Content for header part.</p>',
                content: '<p>Content for main part.</p>',
                leftSide: '<p>Content for left-side box.</p>',
                rightSide: '<p>Content for right-side box.</p>'
            }
        }
    );
Specify root name when obtaining the data
    editor.getData( { rootName: 'leftSide' } ); // -> '<p>Content for left-side box.</p>'

Learn more about using the multi-root editor in its API documentation.