In this guide, we learn the basics of styling a Vaadin application using Cascading Style Sheets (CSS).
While it’s possible to style some parts of a Vaadin application using the Java API, this approach tends to be rather limited for mid- to large-sized applications. So, it is recommended to do the styling of a Vaadin application by adding CSS inside standard style sheets. This guide focuses only on the recommended style-sheets approach.
The following topics are discussed in this guide:
- Create a custom theme
- Where to place your CSS
- How to override Lumo values
- How to selectively style views and components
- When to add CSS under the /components directory
- How to selectively style the internals of a Vaadin component
- How to selectively style a sub-component of a Vaadin component
Create a custom theme
To start, we need to inform Vaadin about the location in which we will place our CSS. The easiest way to do this is to add the @Theme
annotation on a class that implements AppShellConfigurator
.
For example, in a Spring Boot application, you could have an application configuration class such as the following:
@Theme("myapp")
public class Application extends SpringBootServletInitializer
implements AppShellConfigurator {
public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
}
Enter fullscreen mode Exit fullscreen mode
By default, projects downloaded from start.vaadin.com come with the @Theme
annotation added to the Application
class.
Where to place your CSS
Custom CSS should be placed inside a specific folder located under the frontend/themes
directory in your project. This directory has something like the following minimal structure:
frontend
└── themes
└── myapp
├── components/
└── styles.css
Enter fullscreen mode Exit fullscreen mode
NOTE: the myapp
sub-folder might be named differently in your case. This folder should have the same name as the one used for the custom theme using the @Theme
annotation.
Custom CSS should be placed in one of the following locations:
- Inside the
styles.css
file - As a separate CSS file, which is imported from within
styles.css
- Under the
components/
directory in order to style the (local CSS) internals of Vaadin components (such as a Vaadin Button)
The first and simplest option is to place the custom CSS directly inside the styles.css
file. For example, adding the following snippet to styles.css
would change the font family and size of the whole Vaadin application.
* {
font-family: "Lucida Console", "Courier New", monospace;
font-size: 1rem;
}
Enter fullscreen mode Exit fullscreen mode
However, as the number of custom stylings increases, it becomes convenient to organize the styles into separate .css
files. In order to do this, simply create a new CSS file and import that from within styles.css
.
For example, if we create a new CSS file called main-view.css
at frontend/themes/myapp/views/main-view.css
, then we would import this file by adding the following line to styles.css
:
@import url('./views/main-view.css');
Enter fullscreen mode Exit fullscreen mode
Further below, we will discuss the scenarios in which we should add our CSS to the components/
directory.
How to override Lumo values
The default look and feel of a Vaadin application is based on a built-in theme, called Lumo. This theme is essentially a bunch of CSS custom properties (such as colors, typography, and sizes) that each has a CSS variable assigned to it.
One easy way to customize the look and feel of our application requires us to override the default values of the Lumo variables.
For example, suppose our application has a bunch of TextField, ComboBox, DatePicker, and Button components. By default, they will look as follows:
Suppose that we want to increase the roundedness of their corners. By default, the Lumo theme provides these components with a small rounded corner whose value is defined in the --lumo-border-radius-m
variable. In order to increase the roundedness of the corners, we can override the default value of this variable. Specifically, we can add the following inside the styles.css
file in order to increase the default roundedness value:
html {
--lumo-border-radius-m: 1em;
}
Enter fullscreen mode Exit fullscreen mode
This style will increase the corner roundedness of many components at once, so that they will look similar to the following screenshot:
But what if one wants to override the defaults for only a subset of components? No problem; simply use the name of the components as the CSS selector.
For example, if one wants to override the rounded corner defaults for only the TextField and ComboBox components, then the following should be added inside the styles.css
file:
vaadin-text-field, vaadin-combo-box {
--lumo-border-radius-m: 1em;
}
Enter fullscreen mode Exit fullscreen mode
This will change the defaults for the TextField and ComboBox only, leaving other components, such as the DatePicker and Button, with their default values.
How to selectively style views and components
So far, we have been discussing ways to style all components belonging to a certain kind in the same way. For example, if we want to set the width of all Button components within our application, we could add something like the following to the styles.css
file:
vaadin-button {
width: 140px;
}
Enter fullscreen mode Exit fullscreen mode
However, we often want to style the same kind of component differently, depending on its function within our application. For example, we might want to style the Button component differently in our application, depending on whether it is a “Signup” button or a “Delete account!” button.
The easiest way to style views and components selectively is by providing them with a CSS class name from the Java API. All Vaadin components and views have an addClassNames()
method that can be used for this purpose.
To illustrate, the following view makes use of the addClassNames()
method to assign a CSS class name to the view itself, as well as to the components that it includes.
import com.vaadin.flow.component.button.Button;
import com.vaadin.flow.component.html.H1;
import com.vaadin.flow.component.html.Paragraph;
import com.vaadin.flow.component.orderedlayout.VerticalLayout;
import com.vaadin.flow.router.Route;
@Route("")
public class MyView extends VerticalLayout {
public MyView() {
addClassNames("my-view"); //CSS classname for the whole view
H1 heading = new H1("My View Title");
Paragraph text = new Paragraph("Thanks for shopping with us. Click the button to submit your order.");
Button submitButton = new Button("Submit");
submitButton.addClassNames("submit-button", "wide-button"); //CSS classnames for the submit button
Button cancelButton = new Button("Cancel");
cancelButton.addClassNames("cancel-button", "wide-button"); //CSS classnames for the cancel button
add(heading, text, submitButton, cancelButton);
}
}
Enter fullscreen mode Exit fullscreen mode
These CSS class names can then be used to selectively style this view and its components. For example, adding the following to styles.css
demonstrates how selective styling can be achieved.
.my-view {
padding: 1em;
}
.my-view h1 {
/*Only applies to the h1 headers inside my-view*/
margin-top: 0;
}
p {
/*Applies to all p elements regardless of whether they are located inside my-view*/
font-size: 1.2em;
}
.submit-button {
color: green;
}
.cancel-button {
color: red;
}
.wide-button {
width: 120px;
}
Enter fullscreen mode Exit fullscreen mode
Now after the selective styling is applied, MyView
will look as follows:
When to add CSS under the /components
directory
Customizing styling using the two aforementioned options (namely by placing styles inside styles.css
, or inside a separate file imported from within styles.css
) should work for views and layouts created within the project. However, these two options are not guaranteed to work if one aims to style the internals of a Vaadin component.
Custom styling of the internals of Vaadin components, such as the Grid or ComboBox components, needs to be placed under the frontend/themes/myapp/components/
directory. (Note again that the myapp
folder name might be different in your case.)
For example, let’s assume we want to increase the size of the toggle icon that opens the dropdown menu of a ComboBox.
If we inspect this toggle in Chrome (right-click on the toggle and select the Inspect option), we will see that it has an attribute called part
whose value is equal to toggle-button
.
To style this part, we need to create a file called vaadin-combo-box.css
and place it under the frontend/themes/myapp/components/
directory. In this file, we can do something like the following to increase the size of the ComboBox toggle.
[part="toggle-button"] {
font-size: 2em;
}
Enter fullscreen mode Exit fullscreen mode
Note that the files placed under the /components
directory have to exactly match the Vaadin component names as they appear in the DOM. For example, the ComboBox component appears in the DOM as vaadin-combo-box
. Accordingly, the styling file of the ComboBox has to be named exactly vaadin-combo-box.css
.
How to selectively style the internals of a Vaadin component
We just saw how to style the internals of a Vaadin component by using the part
property and placing the CSS inside a specific file placed under the frontend/themes/myapp/components/
directory. Specifically, we saw how to style the ComboBox toggle using the [part="toggle-button"]
selector and placing the CSS in a file called vaadin-combo-box.css
that resides under the frontend/themes/myapp/components/
directory.
However, this approach will style all toggle elements of all the ComboBox components in our application in the same way. What if we want our styling to apply to only some toggle elements but not to others?
Here we can follow the same approach as we did above to selectively style views and components. That is, we can provide the component with a CSS class name from the Java API using the addClassNames()
method. Then we can use this CSS class name to selectively style the internal part of the component.
For example, suppose that our application has the following two ComboBox components:
ComboBox<String> firstCombo = new ComboBox<>();
comboBox.setItems("Earth", "Mars");
ComboBox<String> secondCombo = new ComboBox<>();
comboBox.setItems("Mercury", "Venus");
Enter fullscreen mode Exit fullscreen mode
If we want to style the toggle part of these two components differently, we can do this by providing each ComboBox with a different classname as follows
firstCombo.addClassNames("first-combo");
secondCombo.addClassNames("second-combo");
Enter fullscreen mode Exit fullscreen mode
Then, in frontend/themes/myapp/components/vaadin-combo-box.css
, we can selectively style the toggle element of the two components differently by doing something like:
:host(.first-combo) [part="toggle-button"] {
font-size: 2em;
}
:host(.second-combo) [part="toggle-button"] {
font-size: 1em;
}
Enter fullscreen mode Exit fullscreen mode
Notice that we have to use here the :host()
shadow function because the toggle-button
part is located inside the shadow DOM of the ComboBox.
How to selectively style a sub-component of a Vaadin component
There are cases in which we cannot use CSS classnames to do the selective styling. Specifically, we cannot rely on CSS classnames when we want to style the sub-components of a Vaadin component.
What do I mean by a sub-component? Some Vaadin components create sub-components that exist in the DOM independently from the Vaadin component itself. This is noticeably the case for the Vaadin components, such as the DatePicker, Dialog, and ComboBox, that open an overlay during user interaction. For example, a ComboBox adds a vaadin-combo-box-overlay
element in the DOM when the user clicks on the toggle button to see the list of the available options. This vaadin-combo-box-overlay
element exists in the DOM independently from the vaadin-combo-box
element itself.
The fact that they exist in the DOM independently from their parent component means that a CSS classname that is given to the parent element is not propagated down to the sub-component. For example, if we give a ComboBox a classname from the Java API, only the vaadin-combo-box
element will get this classname in the DOM, not the vaadin-combo-box-overlay
element.
Instead of classnames, we can rely on the theme
attribute as a selector in order to selectively style a sub-component of a Vaadin component. Unlike the classnames, the theme
attribute has the benefit of propagating through to the sub-components.
We can provide a component with a theme attribute from the Java API using the addThemeName()
method.
For example, like before suppose that our application has the following two ComboBox components:
ComboBox<String> firstCombo = new ComboBox<>();
firstCombo.setItems("Earth", "Mars");
ComboBox<String> secondCombo = new ComboBox<>();
secondCombo.setItems("Mercury", "Venus");
Enter fullscreen mode Exit fullscreen mode
Suppose that we want to change the background color of the drop-down list of items that the user sees when they open the ComboBox. This background color is controlled by the background-color
property of the overlay
part of the vaadin-combo-box-overlay
element. This overlay
part in the DOM is highlighted in the following screenshot.
To selectively style the background of the overlay of the two ComboBox components, we can first give each one of them a theme name using the Java API as follows:
firstCombo.addThemeName("first-combo");
secondCombo.addThemeName("second-combo");
Enter fullscreen mode Exit fullscreen mode
Then, we need to create a new file called vaadin-combo-box-overlay.css
and place it under the frontend/themes/myapp/components
directory.
In this file, we can selectively style the overlay part of the vaadin-combo-box-overlay
element of the two ComboBoxes by doing something like:
:host([theme="first-combo"]) [part="overlay"] {
background-color: rosybrown;
}
:host([theme="second-combo"]) [part="overlay"] {
background-color: aquamarine;
}
Enter fullscreen mode Exit fullscreen mode
Here again we are using the :host()
shadow function because the overlay part is located inside the shadow DOM of the vaadin-combo-box-overlay
element.
Conclusion
In this guide, we learned the basics of styling a Vaadin application using CSS. Be sure to check the official Vaadin documentation for more advanced use cases.
暂无评论内容