chromium/ash/bubble/bubble_event_filter.cc

// 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 "ash/bubble/bubble_event_filter.h"

#include "ash/bubble/bubble_utils.h"
#include "ash/shelf/shelf.h"
#include "ash/shell.h"
#include "ash/wm/container_finder.h"
#include "base/functional/callback.h"
#include "ui/events/event.h"
#include "ui/gfx/geometry/point.h"
#include "ui/gfx/geometry/rect.h"
#include "ui/views/widget/widget.h"

namespace ash {

BubbleEventFilter::BubbleEventFilter(views::Widget* bubble_widget,
                                     views::View* button,
                                     OnClickedOutsideCallback on_click_outside)
    : bubble_widget_(bubble_widget),
      button_(button),
      on_click_outside_(on_click_outside) {
  Shell::Get()->AddPreTargetHandler(this);
}

BubbleEventFilter::~BubbleEventFilter() {
  Shell::Get()->RemovePreTargetHandler(this);
}

void BubbleEventFilter::SetButton(views::View* button) {
  button_ = button;
}

void BubbleEventFilter::OnMouseEvent(ui::MouseEvent* event) {
  if (event->type() == ui::EventType::kMousePressed) {
    ProcessPressedEvent(*event);
  }
}

void BubbleEventFilter::OnTouchEvent(ui::TouchEvent* event) {
  if (event->type() == ui::EventType::kTouchPressed) {
    ProcessPressedEvent(*event);
  }
}

bool BubbleEventFilter::ShouldRunOnClickOutsideCallback(
    const ui::LocatedEvent& event) {
  if (!bubble_widget_) {
    return false;
  }

  // Check the general rules for closing bubbles.
  if (!bubble_utils::ShouldCloseBubbleForEvent(event)) {
    return false;
  }

  gfx::Point event_location = event.target()
                                  ? event.target()->GetScreenLocation(event)
                                  : event.root_location();
  // Ignore clicks inside the bubble widget.
  if (bubble_widget_->GetWindowBoundsInScreen().Contains(event_location)) {
    return false;
  }

  // Ignore clicks that hit the button (which usually spawned the widget).
  // Note that we need to use `HitTestPoint()` because certain button (i.e. the
  // shelf home button) have a custom view targeter that extends its hit test
  // bounds beyond the button bounds, so when deciding whether or not to close
  // the bubble we need to do a real hit test against the button, not just check
  // if the click point is inside its bounds.
  if (button_) {
    gfx::Point point_in_button = event_location;
    views::View::ConvertPointFromScreen(button_, &point_in_button);
    if (button_->HitTestPoint(point_in_button)) {
      return false;
    }
  }

  return true;
}

void BubbleEventFilter::ProcessPressedEvent(const ui::LocatedEvent& event) {
  if (ShouldRunOnClickOutsideCallback(event)) {
    on_click_outside_.Run(event);
  }
}

}  // namespace ash