Setting a global footer on a Vaadin AppLayout

More recently I have started using the Vaadin Flow framework in version 24 to build a corporate application. One of the first challenges I stumbled upon when building a user interface was creating a global footer that is simply present on all other views.

The user interface in Vaadin is usually build from the common root AppLayout component which is to be extended from a main view class. This AppLayout provides a few helper methods to fill the predefined content areas for navigation, content and drawer. While e.g. the addToNavBar method can be used to create menus and alike and hence basically defining a navigation header section, there is no dedicated method for a footer.

A potential footer needs to be added to the content itself. Now theoretically we could set a footer on every child view, but that seems vastly repetitive. It would be much nicer to be able to define the footer in the main view of our app in addition to the header. And this seems easily achievable by using another convenience method setContent from the AppLayout component:

public class MainView extends AppLayout implements RouterLayout {
    @Override
    public void setContent(final Component content) {
        final Span footerVersionText = new Span("IAM A FOOTER");
        content.getElement().appendChild(footerVersionText.getElement());
        super.setContent(content);
    }
}

Indeed, this places our footer on every child view as long as it is a static route, and we are only adding content via the component constructors.

Unfortunately, this is not working for more sophisticated, dynamic routes on an application. Assume a first-oder child view that uses the main view as a layout via @Route(value="some-route", layout = MainView.class). Lets imagine this view is presenting a list of items. In addition to this view we want to implement a “details” view, only showing information of a particular item. In terms of routing we obviously want to go with the logical option some-route/12345 where 12345 represents an item id. This way URLs are comprehensible and all details views are easily accessible via a URL. If we now implement this details view we have to hook into the same routing as for some-route (see aforementioned @Route(...)). Additionally, we need to implement the HasUrlParameter and AfterNavigationObserver in order to resolve the appropriate id from th URL and act upon it. This means we are not building our view content from the constructor but from the afterNavigation method of the AfterNavigationObserver interface. This is because we first need to finish navigation and catch the id from the URL.

Hence, if we are now navigating to this dedicated details view, we can observe how our footer starts to be rendered on top of the details view’ content. This happens because Vaadin internally renders the setContent components before it renders specified content from the AfterNavigationObserver. Simply put, setContent content is added to the DOM at an earlier lifecycle stage than content from the AfterNavigationObserver and that of course makes sense.

Now to render our footer also correctly in these cases we simply do the following:

@Route(value = "some-route", layout = MainView.class)
public class SomeRouteDetailsView extends VerticalLayout implements HasUrlParameter<String>, AfterNavigationObserver {
    @Override
    public void afterNavigation(AfterNavigationEvent event) {
        final HorizontalLayout detailsViewContent = new HorizontalLayout();
        super.addComponentAsFirst(detailsViewContent);
    }
}

We have to add the content which we build in the afterNavigation method (from the AfterNavigationObserver) not via the typical add method of a Vaadin component but via addComponentAsFirst. This resolves the order issue for our footer, and it is displayed at the bottom of the page again. The method and logic behind it is pretty self-explanatory.