1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127
|
// Copyright 2023 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "ui/ozone/platform/wayland/host/xdg_activation.h"
#include "base/functional/bind.h"
#include "base/functional/callback_helpers.h"
#include "base/test/bind.h"
#include "base/test/mock_callback.h"
#include "base/time/time.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/ozone/platform/wayland/host/wayland_seat.h"
#include "ui/ozone/platform/wayland/test/mock_surface.h"
#include "ui/ozone/platform/wayland/test/mock_wayland_platform_window_delegate.h"
#include "ui/ozone/platform/wayland/test/test_keyboard.h"
#include "ui/ozone/platform/wayland/test/wayland_test.h"
using ::testing::_;
using ::testing::StrEq;
using ::testing::Values;
namespace ui {
namespace {
constexpr gfx::Rect kDefaultBounds(0, 0, 100, 100);
const char kMockStaticTestToken[] = "CHROMIUM_MOCK_XDG_ACTIVATION_TOKEN";
} // namespace
using XdgActivationTest = WaylandTestSimple;
// Tests that XdgActivation uses the proper surface to request token.
TEST_F(XdgActivationTest, RequestNewToken) {
MockWaylandPlatformWindowDelegate window_delegate1;
MockWaylandPlatformWindowDelegate window_delegate2;
PostToServerAndWait([](wl::TestWaylandServerThread* server) {
wl_seat_send_capabilities(server->seat()->resource(),
WL_SEAT_CAPABILITY_KEYBOARD);
});
ASSERT_TRUE(connection_->seat()->keyboard());
window_.reset();
auto window1 = CreateWaylandWindowWithParams(
PlatformWindowType::kWindow, kDefaultBounds, &window_delegate1);
auto window2 = CreateWaylandWindowWithParams(
PlatformWindowType::kWindow, kDefaultBounds, &window_delegate2);
ActivateSurface(window_delegate1);
MapSurface(window_delegate1);
ActivateSurface(window_delegate2);
MapSurface(window_delegate2);
ASSERT_TRUE(window1->IsSurfaceConfigured());
ASSERT_TRUE(window2->IsSurfaceConfigured());
// When window is shown, it automatically gets keyboard focus. Reset it
connection_->window_manager()->SetKeyboardFocusedWindow(nullptr);
PostToServerAndWait([surface_id2 = window2->root_surface()->get_surface_id()](
wl::TestWaylandServerThread* server) {
auto* const keyboard = server->seat()->keyboard()->resource();
auto* const xdg_activation = server->xdg_activation_v1();
auto* const surface2 =
server->GetObject<wl::MockSurface>(surface_id2)->resource();
wl::ScopedWlArray empty({});
wl_keyboard_send_enter(keyboard, server->GetNextSerial(), surface2,
empty.get());
// The following should be called once for the initial request and then
// again when the second request is sent after the initial one completes.
EXPECT_CALL(*xdg_activation, TokenSetSurface(_, _, surface2)).Times(2);
EXPECT_CALL(*xdg_activation, TokenCommit(_, _)).Times(2);
});
// Expect a successful token request.
{
::testing::StrictMock<
base::MockCallback<base::nix::XdgActivationTokenCallback>>
callback;
EXPECT_CALL(callback, Run(std::string(kMockStaticTestToken)));
connection_->xdg_activation()->RequestNewToken(callback.Get());
PostToServerAndWait(
[](wl::TestWaylandServerThread* server) {
auto* const xdg_activation = server->xdg_activation_v1();
ASSERT_TRUE(xdg_activation);
ASSERT_TRUE(xdg_activation->get_token());
xdg_activation_token_v1_send_done(
xdg_activation->get_token()->resource(), kMockStaticTestToken);
},
true);
}
// Emulate a timeout.
{
::testing::StrictMock<
base::MockCallback<base::nix::XdgActivationTokenCallback>>
callback;
EXPECT_CALL(callback, Run(std::string()));
connection_->xdg_activation()->RequestNewToken(callback.Get());
task_environment_.FastForwardBy(base::Milliseconds(600));
}
}
// Tests that with too many requests at some point the request queue will be
// full and the subsequent request callbacks will be run immediately with an
// empty token instead of adding them to the queue for sending to the server
// later.
TEST_F(XdgActivationTest, RequestNewToken_TooManyRequests) {
// The first 100 requests should just be queued.
for (int i = 0; i < 100; ++i) {
connection_->xdg_activation()->RequestNewToken(
base::BindOnce([](std::string _) {}));
}
// The next request should result in the callback being run immediately.
::testing::StrictMock<
base::MockCallback<base::nix::XdgActivationTokenCallback>>
callback;
EXPECT_CALL(callback, Run(std::string()));
connection_->xdg_activation()->RequestNewToken(callback.Get());
}
} // namespace ui
|