// Copyright 2024 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/picker/views/picker_widget.h"
#include "ash/bubble/bubble_event_filter.h"
#include "ash/picker/views/picker_positioning.h"
#include "ash/picker/views/picker_style.h"
#include "ash/picker/views/picker_view.h"
#include "ash/public/cpp/shell_window_ids.h"
#include "ash/shell.h"
#include "base/functional/bind.h"
#include "base/functional/callback.h"
#include "base/memory/ptr_util.h"
#include "ui/display/screen.h"
#include "ui/events/event.h"
#include "ui/gfx/geometry/insets.h"
#include "ui/gfx/geometry/rect.h"
#include "ui/gfx/geometry/size.h"
#include "ui/views/view_utils.h"
#include "ui/views/widget/unique_widget_ptr.h"
#include "ui/views/widget/widget.h"
namespace ash {
namespace {
// This is the maximum size of PickerView, including emoji bar.
constexpr int kPickerViewMaxHeight = 356;
// Gets the preferred layout to use given `anchor_bounds` in screen coordinates.
PickerLayoutType GetLayoutType(const gfx::Rect& anchor_bounds,
PickerPositionType position_type) {
return position_type == PickerPositionType::kCentered ||
anchor_bounds.bottom() + kPickerViewMaxHeight <=
display::Screen::GetScreen()
->GetDisplayMatching(anchor_bounds)
.work_area()
.bottom()
? PickerLayoutType::kMainResultsBelowSearchField
: PickerLayoutType::kMainResultsAboveSearchField;
}
views::Widget::InitParams CreateInitParams(
PickerViewDelegate* delegate,
const gfx::Rect& anchor_bounds,
PickerPositionType position_type,
const base::TimeTicks trigger_event_timestamp) {
auto picker_view = std::make_unique<PickerView>(
delegate, anchor_bounds, GetLayoutType(anchor_bounds, position_type),
position_type, trigger_event_timestamp);
views::Widget::InitParams params(
views::Widget::InitParams::NATIVE_WIDGET_OWNS_WIDGET,
views::Widget::InitParams::TYPE_BUBBLE);
params.parent = Shell::GetContainer(
Shell::GetRootWindowForDisplayId(
display::Screen::GetScreen()->GetDisplayMatching(anchor_bounds).id()),
kShellWindowId_FloatContainer);
params.activatable = views::Widget::InitParams::Activatable::kYes;
params.shadow_type = views::Widget::InitParams::ShadowType::kNone;
params.opacity = views::Widget::InitParams::WindowOpacity::kTranslucent;
params.z_order = ui::ZOrderLevel::kFloatingUIElement;
// TODO(b/309706053): Replace this with the finalized string.
params.name = "Picker";
params.delegate = picker_view.release();
return params;
}
} // namespace
views::UniqueWidgetPtr PickerWidget::Create(
PickerViewDelegate* delegate,
const gfx::Rect& anchor_bounds,
base::TimeTicks trigger_event_timestamp) {
return base::WrapUnique(new PickerWidget(delegate, anchor_bounds,
PickerPositionType::kNearAnchor,
trigger_event_timestamp));
}
views::UniqueWidgetPtr PickerWidget::CreateCentered(
PickerViewDelegate* delegate,
const gfx::Rect& anchor_bounds,
base::TimeTicks trigger_event_timestamp) {
return base::WrapUnique(new PickerWidget(delegate, anchor_bounds,
PickerPositionType::kCentered,
trigger_event_timestamp));
}
void PickerWidget::OnNativeBlur() {
SetVisibilityAnimationTransition(
views::Widget::VisibilityTransition::ANIMATE_NONE);
// TODO: b/322280416 - Add a close reason here for metrics.
Close();
}
PickerWidget::PickerWidget(PickerViewDelegate* delegate,
const gfx::Rect& anchor_bounds,
PickerPositionType position_type,
base::TimeTicks trigger_event_timestamp)
: views::Widget(CreateInitParams(delegate,
anchor_bounds,
position_type,
trigger_event_timestamp)),
bubble_event_filter_(/*widget=*/this) {
SetVisibilityAnimationTransition(
views::Widget::VisibilityTransition::ANIMATE_HIDE);
}
PickerWidget::~PickerWidget() = default;
} // namespace ash