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
|
// Copyright 2025 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "ash/system/accessibility/mouse_keys/mouse_keys_bubble_controller.h"
#include "ash/accessibility/mouse_keys/mouse_keys_controller.h"
#include "ash/shell.h"
#include "ash/system/accessibility/mouse_keys/mouse_keys_bubble_view.h"
#include "ash/wm/collision_detection/collision_detection_utils.h"
namespace ash {
MouseKeysBubbleController::MouseKeysBubbleController() = default;
MouseKeysBubbleController::~MouseKeysBubbleController() {
if (widget_ && !widget_->IsClosed()) {
widget_->CloseNow();
}
StopTimer();
}
void MouseKeysBubbleController::StopTimer() {
if (timer_.IsRunning()) {
timer_.Stop();
}
}
void MouseKeysBubbleController::UpdateMouseKeysBubblePosition(
gfx::Point position) {
if (mouse_keys_bubble_view_) {
mouse_keys_bubble_view_->SetAnchorRect(gfx::Rect(position, gfx::Size()));
}
}
void MouseKeysBubbleController::UpdateBubble(
bool visible,
MouseKeysBubbleIconType icon,
const std::optional<std::u16string>& text) {
EnsureInitialize();
gfx::Point bubble_position =
Shell::Get()->mouse_keys_controller()->GetLastMousePositionDips();
bubble_position.Offset(16, 16);
UpdateMouseKeysBubblePosition(bubble_position);
Update(icon, text);
widget_->SetVisible(visible);
if (timer_.IsRunning()) {
timer_.Reset();
} else {
timer_.Start(
FROM_HERE, base::Seconds(2),
base::BindRepeating(&MouseKeysBubbleController::HideWidgetAfterDelay,
GetWeakPtr()));
}
}
void MouseKeysBubbleController::HideWidgetAfterDelay() {
if (widget_) {
widget_->Hide();
}
}
void MouseKeysBubbleController::OnViewIsDeleting(views::View* observed_view) {
if (observed_view != mouse_keys_bubble_view_) {
return;
}
StopTimer();
mouse_keys_bubble_view_->views::View::RemoveObserver(this);
mouse_keys_bubble_view_ = nullptr;
widget_ = nullptr;
}
void MouseKeysBubbleController::EnsureInitialize() {
if (widget_) {
return;
}
mouse_keys_bubble_view_ = new MouseKeysBubbleView();
mouse_keys_bubble_view_->views::View::AddObserver(this);
widget_ =
views::BubbleDialogDelegateView::CreateBubble(mouse_keys_bubble_view_);
widget_->SetZOrderLevel(ui::ZOrderLevel::kFloatingUIElement);
CollisionDetectionUtils::MarkWindowPriorityForCollisionDetection(
widget_->GetNativeWindow(),
CollisionDetectionUtils::RelativePriority::kMouseKeysBubble);
}
void MouseKeysBubbleController::Update(
MouseKeysBubbleIconType icon,
const std::optional<std::u16string>& text) {
DCHECK(mouse_keys_bubble_view_);
DCHECK(widget_);
// Update `mouse_keys_bubble_view_`.
mouse_keys_bubble_view_->Update(icon, text);
// Update the bounds to fit entirely within the screen.
gfx::Rect new_bounds = widget_->GetWindowBoundsInScreen();
gfx::Rect display_bounds =
display::Screen::GetScreen()->GetDisplayMatching(new_bounds).bounds();
new_bounds.AdjustToFit(display_bounds);
// Update the preferred bounds based on other system windows.
gfx::Rect resting_bounds = CollisionDetectionUtils::AvoidObstacles(
display::Screen::GetScreen()->GetDisplayNearestWindow(
widget_->GetNativeWindow()),
new_bounds, CollisionDetectionUtils::RelativePriority::kMouseKeysBubble);
widget_->SetBounds(resting_bounds);
}
} // namespace ash
|