// Copyright 2021 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef UI_VIEWS_TEST_TEST_WIDGET_BUILDER_H_
#define UI_VIEWS_TEST_TEST_WIDGET_BUILDER_H_
#include <memory>
#include "build/build_config.h"
#include "ui/base/class_property.h"
#include "ui/base/mojom/window_show_state.mojom-forward.h"
#include "ui/base/ui_base_types.h"
#include "ui/gfx/geometry/rect.h"
#include "ui/gfx/native_ui_types.h"
#include "ui/views/widget/widget.h"
#if defined(USE_AURA)
#include "ui/aura/window.h"
#endif
namespace views::test {
struct WidgetBuilderParams {
#if defined(USE_AURA)
int window_id = aura::Window::kInitialId;
std::u16string window_title = std::u16string();
#endif
bool show = true;
};
// A builder to create a views::Widget for testing purpose. There are three
// ways to create a widget: `BuildOwnedByNativeWidget()`,
// `BuildOwnsNativeWidget()`, and `BuildClientOwnsWidget()`. Please refer
// to the documentation of each methods to find out which is better for your
// test cases as there are subtle differences.
class TestWidgetBuilder {
public:
explicit TestWidgetBuilder(WidgetBuilderParams params = {});
explicit TestWidgetBuilder(Widget::InitParams widget_init_params,
WidgetBuilderParams params = {});
TestWidgetBuilder(TestWidgetBuilder&& other);
TestWidgetBuilder(const TestWidgetBuilder& other) = delete;
TestWidgetBuilder& operator=(const TestWidgetBuilder& other) = delete;
~TestWidgetBuilder();
// Sets the property of Widget::InitParams to be used when creating
// a widget.
TestWidgetBuilder& SetWidgetType(Widget::InitParams::Type type);
TestWidgetBuilder& SetZOrderLevel(ui::ZOrderLevel z_order);
TestWidgetBuilder& SetBounds(const gfx::Rect& bounds);
TestWidgetBuilder& SetParent(gfx::NativeView parent);
TestWidgetBuilder& SetContext(gfx::NativeWindow context);
TestWidgetBuilder& SetActivatable(bool activatable);
TestWidgetBuilder& SetShowState(ui::mojom::WindowShowState show_state);
// Sets the window property to be set on the window of a widget.
template <typename T>
TestWidgetBuilder& SetWindowProperty(const ui::ClassProperty<T>* property,
T value) {
widget_init_params_.init_properties_container.SetProperty(property, value);
return *this;
}
#if defined(USE_AURA)
// Set the window id used on the window of a test widget.
TestWidgetBuilder& SetWindowId(int window_id);
// Having a non-empty title helps avoid accessibility paint check failures
// in tests. For instance, `WindowMiniView` gets its accessible name from
// the window title.
TestWidgetBuilder& SetWindowTitle(const std::u16string& title);
#endif
// A widget is shown when created by default. Use this if you want not
// to show when created.
TestWidgetBuilder& SetShow(bool show);
// Set the widget's delegate. It is not owned by the widget.
TestWidgetBuilder& SetDelegate(WidgetDelegate* delegate);
// Set the widget to be used for initialization.
TestWidgetBuilder& SetWidget(std::unique_ptr<Widget> widget);
// Set the native widget to be used for the widget.
TestWidgetBuilder& SetNativeWidget(NativeWidget* native_widet);
// Deprecated: Use `BuildClientOwnsWidget` instead.
// Creates a widget owned by a native window (aura::Window on ChromeOS) and
// returns a raw pointer. Use this if you want to create a widget that
// behaves like an application. You should not delete the widget directly but
// must call 'Widget::CloseWithReason' (recommended) or 'Widget::Close'. Note
// that this is just an request, and the widget may not be closed and deleted
// if the widget implementation rejected the request. This is also
// asynchronus, and instances of the widget and its window will be deleted in
// a posted task. There is a 'Widget::CloseNow' which forcibly and
// synchronously closes and delete the widget and its window, but this should
// not be used in normal situation.
Widget* BuildOwnedByNativeWidget();
// Deprecated: Use `BuildClientOwnsWidget` instead.
// Creates a widget that owns a native window (aura::Window on
// ChromeOS) and returns an unique pointer of the widget which owns a native
// window (aura::Window on ChromeOS). It will be closed and deleted
// immediately when the object exits its scope. The important difference is
// that a widget won't be deleted when the window is deleted first and
// Widget::GetNativeWindow() may return nullptr. Prefer
// BuildClientOwnsWidget() to this.
[[nodiscard]] std::unique_ptr<Widget> BuildOwnsNativeWidget();
// Creates a widget that can live independently of the native widget, and
// where the native widget can live independently of the widget. The native
// widget will be owned by the native window. When the widget is closed, it
// requests that the native widget also be closed and deleted, but that is
// allowed to happen asynchronously.
[[nodiscard]] std::unique_ptr<Widget> BuildClientOwnsWidget();
// Deprecated: Use `BuildClientOwnsWidget` instead.
// Create a widget using the ownership model given in the Widget::IntiParams.
[[nodiscard]] std::unique_ptr<Widget> BuildDeprecated();
private:
// Both BuildOwnsNativeWidget() and BuildClientOwnsWidget() are just
// wrappers around this.
[[nodiscard]] std::unique_ptr<Widget> BuildWidgetWithOwnership(
Widget::InitParams::Ownership ownership);
// Note: ownership is a placeholder. It is overridden when building
// a widget.
views::Widget::InitParams widget_init_params_{
views::Widget::InitParams::CLIENT_OWNS_WIDGET};
WidgetBuilderParams params_;
std::unique_ptr<Widget> widget_;
bool built_ = false;
};
} // namespace views::test
#endif // UI_VIEWS_TEST_TEST_WIDGET_BUILDER_H_