Skip to main content
Skip table of contents

Switching renderers

Introduction

The choice of Renderer has an impact on how publication content is displayed on the screen. This tutorial will teach you how you can switch between the different renderers in the Colibrio reading system. There are renderers that show a single page at a time, renderers that show a spread of two pages and a scrolling renderer.

Objective

Learn how to set up multiple renderers and switching between them, including an introduction to automatically switching renderers using ResponsiveViewRules.

Audience

A developer who knows Typescript and a little bit of HTML and CSS.

Prerequisites

Full code example

For reference, you can find the full code example in the Web framework tutorial repository.

In order to set up the developer environment including where to put your credentials, please follow the steps described in Set up tutorial environment.

Tutorial

index.html

Add a new <div> to index.html, with a radio button for each renderer (and one for setting ReaderView.setResponsiveRendererSelectionEnabled()). These radio buttons will be used to switch between the different renderers or to enable responsive renderer selection.

HTML
<div id="renderer-switcher">
    <p>Try resizing the window with "Responsive" selected to see how the responsive view
        rules affect the renderer choice.</p>
    <p>Active renderer: <span id="renderer-active-name"></span></p>
    <label>
        Responsive
        <input type="radio" name="renderer-select" value="Responsive" checked>
    </label>
    <label>
        Stack
        <input type="radio" name="renderer-select" value="Stack">
    </label>
    <label>
        FlipBook
        <input type="radio" name="renderer-select" value="FlipBook">
    </label>
    <label>
        SinglePageSwipe
        <input type="radio" name="renderer-select" value="SinglePageSwipe">
    </label>
    <label>
        SpreadSwipe
        <input type="radio" name="renderer-select" value="SpreadSwipe">
    </label>
    <label>
        SingleDocumentScroll
        <input type="radio" name="renderer-select" value="SingleDocumentScroll">
    </label>
</div>

AppRendererSwitcher.ts

The AppRendererSwitcher class is responsible for adding all the different renderers to the ReaderView and for the UI related to switching between the renderers.

Create a new file called AppRendererSwitcher.ts and in that file, create a class with the same name. Go ahead and declare all the properties and create placeholders for all the methods that will be implemented in this tutorial.

TYPESCRIPT
export class AppRendererSwitcher {
    constructor(private readerVIew: IReaderView, private containerElem: HTMLElement) {
    }
    private activateRenderer(rendererName: AppRendererName): void {
    }
    private setupRadioButtons(): void {
    }
    private setupActiveRendererNameElement(): HTMLParagraphElement {
    }
}

Renderer Names

In the global scope in AppRendererSwitcher.ts, define an enum with the names to use for the renderers. In this code example, it is important that these names are the same as the values for the radio buttons defined in index.html, but in general you can use whatever name you want.

The names can be used to identify the renderer and to get the renderer from the ReaderView. Responsive is not a name for a specific renderer, but will be used to enable responsive renderer selection, which allows the ReaderView to automatically select a renderer based on the rules that were provided when adding the renderers and which renderer best fits the viewport.

TYPESCRIPT
enum RendererName {
    RESPONSIVE = 'Responsive',
    STACK = 'Stack',
    FLIP_BOOK = 'FlipBook',
    SINGLE_PAGE_SWIPE = 'SinglePageSwipe',
    SPREAD_SWIPE = 'SpreadSwipe',
    SINGLE_DOCUMENT_SCROLL = 'SingleDocumentScroll'
}

constructor

In the constructor, start by creating new instances of all the renderers. Note the options parameter, setting a name for each renderer. The names will be used when setting up the radio buttons to switch renderers, as well as to display which renderer is currently active. For other renderer options, take a look at the API documentation for each renderer.

TYPESCRIPT
constructor(private readerView: IReaderView, private containerElem: HTMLElement) {
        const stackRenderer = new StackRenderer({name: RendererName.STACK});
        const flipBookRenderer = new FlipBookRenderer({name: RendererName.FLIP_BOOK});
        const singlePageSwipeRenderer = new SinglePageSwipeRenderer({name: RendererName.SINGLE_PAGE_SWIPE});
        const spreadSwipeRenderer = new SpreadSwipeRenderer({name: RendererName.SPREAD_SWIPE});
        const scrollRenderer = new SingleDocumentScrollRenderer({name: RendererName.SINGLE_DOCUMENT_SCROLL});
    // ... 
}

After creating the renderers, you also need to add them to the ReaderView before they can be used. For the Stack and the FlipBook renderers, also add a responsive view rule. These rules define when the ReaderView should use these renderers when responsive renderer selection is enabled. It is enabled by default, but will be disabled if you call ReaderView.setActiveRenderer(). To enable it again, call ReaderView.setResponsiveRendererSelection(true).

TYPESCRIPT
this.readerView.addRenderer(stackRenderer, "(orientation: portrait)");
this.readerView.addRenderer(flipBookRenderer, "(orientation: landscape)");

For the remaining renderers, just pass a function that always returns false as the ResponsiveViewRule. This means that they will not be automatically picked by the ReaderView, but will still be available if you explicitly call ReaderView.setActiveRenderer().

TYPESCRIPT
this.readerView.addRenderer(singlePageSwipeRenderer, () => false);
this.readerView.addRenderer(spreadSwipeRenderer, () => false);
this.readerView.addRenderer(scrollRenderer, () => false);

Then call the method to set up the element that will be used to display the currently active renderer.

TYPESCRIPT
this.setupActiveRendererNameElement();

Finally, call the method to set up the radio buttons that will be used to switch between the different renderers.

TYPESCRIPT
this.setupRadioButtons();

That’s all for the constructor. Now on to the other methods!

setupActiveRendererNameElement

A method to update the UI whenever the ReaderView’s active renderer has changed. Add an engine event listerer, listening to the activeRendererChanged event and then get the new renderer using ReaderView.getActiveRenderer(). For a list of all engine events, take a look at IEngineEventTypeMap in the API documentation.

TYPESCRIPT
private setupActiveRendererNameElement() {
    const activeRendererNameElement = this.containerElem.querySelector<HTMLSpanElement>('#renderer-active-name');

    if (activeRendererNameElement) {
        activeRendererNameElement.innerText = this.readerView.getActiveRenderer()?.getName() ?? 'none';

        this.readerView.addEngineEventListener('activeRendererChanged', () => {
            activeRendererNameElement.innerText = this.readerView.getActiveRenderer()?.getName() ?? 'none';
        });
    } else {
        Logger.logError('Unable to find element with id #renderer-active-name');
    }
}

setupRadioButtons

In the setupRadioButtons method, set up the event listeners for switching renderers when a radio button has been selected.

TYPESCRIPT
private setupRadioButtons(): void {
    for (let rendererName of Object.values(RendererName)) {
        const valueSelector = "[value=" + rendererName + "]";
        const radioButton = this.containerElem.querySelector<HTMLInputElement>(valueSelector);
        if (radioButton) {
            radioButton.addEventListener('change', () => {
                if (radioButton.checked) {
                    this.activateRenderer(rendererName);
                }
            });
        } else {
            Logger.logError('Unable to find radio button matching: ' + valueSelector)
        }
    }
}

activateRenderer

activateRenderer is the method that is called whenever a radio button has been checked. Calling this.readerView.setResponsiveRendererSelectionEnabled(true) will allow the ReaderView to automatically select renderers, based on the responsive view rules that were passed in when adding the renderer to the ReaderView.

TYPESCRIPT
private activateRenderer(rendererName: RendererName): void {
    if (rendererName === RendererName.RESPONSIVE) {
        this.readerView.setResponsiveRendererSelectionEnabled(true);
    } else {
        const renderer = this.readerView.getRendererByName(rendererName);
        if (renderer) {
            this.readerView.setActiveRenderer(renderer);
        } else {
            Logger.logError('Unable to find renderer with name: ', rendererName);
        }
    }
}

Integrating with the App

Changes to App.ts

Add a method in the App class to create the new RendererSwitcherView instance.

TYPESCRIPT
createRendererSwitcherView(rendererSwitcherContainer: HTMLElement): void {
     new RendererSwitcherView(this.readerView, rendererSwitcherContainer);
}

index.ts

In index.ts, after creating the app, call the new createRendererSwitcherView method.

TYPESCRIPT
// After const app = new App(...)
const rendererSwitcherElement = document.getElementById('renderer-switcher');
if (rendererSwitcherElement) {
    app.createRendererSwitcherView(rendererSwitcherElement);
} else {
    Logger.logError('Unable to find #renderer-switcher');
}

CSS

Add the following CSS to index.css to properly position the radio buttons and their labels.

CSS
#renderer-switcher {
    width: 100%;
}

#renderer-switcher label {
    display: block;
}

#renderer-switcher input {
    float: left;
}

Concepts

Engine Event

Engine events are generated by the Colibrio Reading System whenever something happens that you, as a developer, might want to know about. It could be something like a pointer event, an event telling you that new content has been rendered, or many other things. For a full list of all possible events, please refer to the API documentation.

ReaderView

The ReaderView’s main responsibilities are rendering publication content and navigating in the publication.

Renderer

Renderers are used by a ReaderView to display publication content.

Responsive Renderer Selection

When responsive renderer selections is enabled, the rule provided when calling ReaderView.addRenderer() is used to evaluate when the renderer should be active. The rule can either a CSS media query, an implementation of the IResponsiveViewRule interface or a callback function.

If multiple renderers have been added with rules evaluate to true with the current configuration, the ReaderView will choose the renderer from those renderers that can use as much area of the viewport as possible.

If none of the renderers added to the ReaderView have matching rules, the ReaderView will choose the best fit from all renderers that have been added.

Calling ReaderView.setActiveRenderer() will cause responsive renderer selection to be disabled.

See IReaderView.addRenderer() in the API documentation for more information.

JavaScript errors detected

Please note, these errors can depend on your browser setup.

If this problem persists, please contact our support.