| name | ui-unit-testing |
| description | Guide Claude on writing fast, browser-free tests for Vaadin 25 Flow views with the Browserless Testing framework (formerly UI Unit Testing). This skill should be used when the user asks to "write a UI test", "unit test a view", "test without a browser", "use BrowserlessTest", "use UIUnitTest", "test a Vaadin component", "browser-free testing", "browserless testing", or needs help with the Vaadin browserless testing framework, component testers, navigation in tests, or mocking Spring beans in Vaadin view tests.
|
| version | 0.2.0 |
Browserless Testing in Vaadin 25
Use the Vaadin MCP tools (search_vaadin_docs) to look up the latest documentation whenever uncertain about a specific API detail. Always set vaadin_version to "25" and ui_language to "java".
What Browserless Testing Is
Browserless tests (previously called "UI Unit Tests") run server-side Java code without a browser or servlet container. You interact directly with your server-side view classes and Vaadin components. The BrowserlessTest / SpringBrowserlessTest base classes set up the Vaadin session, UI, and routing — all in the same JVM as your JUnit tests.
This makes tests fast (milliseconds, not seconds), stable (no browser flakiness), and easy to run in CI.
Free for all users since Vaadin 25.1. Earlier versions required a commercial TestBench subscription.
When to Use Browserless Tests vs. End-to-End Tests
Browserless tests — the default choice for most view testing:
- Testing view logic, navigation, form validation, component state
- TDD workflows where you run tests on every save
- Verifying that clicking a button shows the right notification
- Testing data binding and Binder behavior
End-to-end tests (TestBench) — for critical paths and client-side behavior:
- Login flows, checkout processes
- Testing JavaScript/client-side functionality
- Visual regression testing
- Cross-browser compatibility
Write many browserless tests and few end-to-end tests.
Setup
Maven dependency
The browserless testing framework ships in browserless-test-junit6. Vaadin 25 + Spring Boot 4 default to JUnit 6, which is API-compatible with JUnit 5.
<dependency>
<groupId>com.vaadin</groupId>
<artifactId>browserless-test-junit6</artifactId>
<scope>test</scope>
</dependency>
Migrating from UI Unit Testing? Replace the older vaadin-testbench-unit (JUnit 4) or vaadin-testbench-unit-junit5 artifact with browserless-test-junit6, and rename the base classes (UIUnitTest → BrowserlessTest, SpringUIUnitTest → SpringBrowserlessTest). The testing API is otherwise unchanged.
Writing Your First Test
Extend SpringBrowserlessTest for Spring Boot projects (and add @SpringBootTest), or BrowserlessTest otherwise:
@SpringBootTest
class HelloWorldViewTest extends SpringBrowserlessTest {
@Test
void clickButton_showsNotification() {
HelloWorldView view = navigate(HelloWorldView.class);
test(view.nameField).setValue("Marcus");
test(view.sayHelloButton).click();
Notification notification = $(Notification.class).first();
assertEquals("Hello Marcus", test(notification).getText());
}
}
Key points:
navigate(ViewClass.class) navigates to a route and returns the view instance
test(component) returns a type-specific tester with simulated user actions
$(ComponentClass.class) queries for components in the current UI (like jQuery for Vaadin)
- Fields on the view should be package-private (not private) so tests in the same package can access them
Navigation
MyView view = navigate(MyView.class);
DetailView view = navigate(DetailView.class, "123");
TemplateView view = navigate(TemplateView.class, Map.of("id", "456"));
MyView view = navigate("my-view", MyView.class);
Component Testers
The test() method returns a tester that simulates user interaction. Testers check that the component is visible, enabled, attached, and not behind a modal before allowing interaction.
test(textField).setValue("hello");
String value = test(textField).getValue();
test(button).click();
test(checkbox).setValue(true);
test(comboBox).selectItem("Option A");
GridTester<Person> gridTester = test(grid);
gridTester.clickRow(0);
Notification n = $(Notification.class).first();
String text = test(n).getText();
For commercial components (Chart, etc.), implement CommercialTesterWrappers on your test class.
Component Queries with $()
Find components in the current UI:
Button btn = $(Button.class).first();
TextField field = $(TextField.class).id("email");
List<Button> buttons = $(Button.class).all();
boolean hasGrid = $(Grid.class).exists();
VerticalLayout layout = $(VerticalLayout.class).id("content");
Button innerBtn = layout.$(Button.class).first();
Restricting Package Scanning
By default, browserless tests scan the entire classpath for routes. For faster startup, restrict to specific packages with @ViewPackages. Prefer the classes() array — it survives IDE refactors when classes move:
@SpringBootTest
@ViewPackages(classes = {MyView.class, OtherView.class})
class MyViewTest extends SpringBrowserlessTest {
}
@SpringBootTest
@ViewPackages(packages = {"com.example.app.feature1", "com.example.app.feature2"})
class MyViewTest extends SpringBrowserlessTest { }
Non-Spring Projects
If you aren't using Spring, extend BrowserlessTest directly and drop @SpringBootTest:
class MyViewTest extends BrowserlessTest {
@Test
void test() {
MyView view = navigate(MyView.class);
}
}
For Quarkus, see the Vaadin docs for the Quarkus-specific browserless base class and dependency.
Common Testing Patterns
Test form validation
@Test
void submitEmptyForm_showsValidationErrors() {
EditView view = navigate(EditView.class);
test(view.saveButton).click();
assertTrue(view.nameField.isInvalid());
}
Test navigation after action
@Test
void saveSuccess_navigatesToList() {
EditView view = navigate(EditView.class);
test(view.nameField).setValue("Test");
test(view.saveButton).click();
assertTrue(getCurrentView() instanceof ListView);
}
Test dialog content
@Test
void deleteButton_showsConfirmDialog() {
ListView view = navigate(ListView.class);
test(view.deleteButton).click();
ConfirmDialog dialog = $(ConfirmDialog.class).first();
assertNotNull(dialog);
}
Best Practices
- Make view fields package-private — not private, so tests in the same package can access them directly. This avoids fragile reflection-based lookups.
- Use
@ViewPackages to limit scanning — speeds up test initialization significantly in large projects.
- One assertion focus per test — test one behavior per method. Name tests descriptively:
action_expectedResult.
- Use
test() for interaction, not direct method calls — test(button).click() checks visibility and enabled state; button.click() bypasses those checks.
- Prefer browserless tests over end-to-end — they're 10-100x faster and more stable. Reserve end-to-end tests for client-side behavior and critical paths.
- Test the user's perspective — verify what the user sees (notification text, navigation result, field errors) rather than internal implementation details.