#include "ash/wm/splitview/split_view_controller.h"
#include <algorithm>
#include <cmath>
#include <memory>
#include "ash/accessibility/accessibility_controller_impl.h"
#include "ash/constants/app_types.h"
#include "ash/constants/ash_features.h"
#include "ash/display/screen_orientation_controller.h"
#include "ash/keyboard/ui/keyboard_ui_controller.h"
#include "ash/public/cpp/metrics_util.h"
#include "ash/public/cpp/window_properties.h"
#include "ash/root_window_controller.h"
#include "ash/root_window_settings.h"
#include "ash/screen_util.h"
#include "ash/session/session_controller_impl.h"
#include "ash/shell.h"
#include "ash/style/ash_color_provider.h"
#include "ash/wm/desks/desks_controller.h"
#include "ash/wm/float/float_controller.h"
#include "ash/wm/mru_window_tracker.h"
#include "ash/wm/overview/delayed_animation_observer_impl.h"
#include "ash/wm/overview/overview_controller.h"
#include "ash/wm/overview/overview_grid.h"
#include "ash/wm/overview/overview_item.h"
#include "ash/wm/overview/overview_types.h"
#include "ash/wm/overview/overview_utils.h"
#include "ash/wm/snap_group/snap_group_controller.h"
#include "ash/wm/splitview/split_view_constants.h"
#include "ash/wm/splitview/split_view_divider.h"
#include "ash/wm/splitview/split_view_metrics_controller.h"
#include "ash/wm/splitview/split_view_observer.h"
#include "ash/wm/splitview/split_view_utils.h"
#include "ash/wm/tablet_mode/tablet_mode_controller.h"
#include "ash/wm/tablet_mode/tablet_mode_window_state.h"
#include "ash/wm/window_positioning_utils.h"
#include "ash/wm/window_properties.h"
#include "ash/wm/window_resizer.h"
#include "ash/wm/window_restore/window_restore_controller.h"
#include "ash/wm/window_state.h"
#include "ash/wm/window_transient_descendant_iterator.h"
#include "ash/wm/window_util.h"
#include "ash/wm/wm_event.h"
#include "base/auto_reset.h"
#include "base/containers/contains.h"
#include "base/containers/cxx20_erase.h"
#include "base/containers/flat_set.h"
#include "base/functional/bind.h"
#include "base/memory/raw_ptr.h"
#include "base/metrics/histogram_functions.h"
#include "base/metrics/histogram_macros.h"
#include "base/metrics/user_metrics.h"
#include "base/ranges/algorithm.h"
#include "chromeos/ui/base/window_properties.h"
#include "chromeos/ui/frame/caption_buttons/snap_controller.h"
#include "components/app_restore/desk_template_read_handler.h"
#include "components/app_restore/window_properties.h"
#include "third_party/abseil-cpp/absl/types/optional.h"
#include "ui/aura/client/aura_constants.h"
#include "ui/aura/window_delegate.h"
#include "ui/base/class_property.h"
#include "ui/base/hit_test.h"
#include "ui/base/ime/ash/ime_bridge.h"
#include "ui/base/ime/input_method.h"
#include "ui/compositor/layer.h"
#include "ui/compositor/layer_animator.h"
#include "ui/compositor/presentation_time_recorder.h"
#include "ui/compositor/throughput_tracker.h"
#include "ui/display/types/display_constants.h"
#include "ui/gfx/animation/slide_animation.h"
#include "ui/gfx/animation/tween.h"
#include "ui/gfx/geometry/rect.h"
#include "ui/gfx/geometry/transform_util.h"
#include "ui/views/animation/compositor_animation_runner.h"
#include "ui/views/widget/widget.h"
#include "ui/wm/core/coordinate_conversion.h"
#include "ui/wm/core/shadow_controller.h"
#include "ui/wm/core/window_util.h"
#include "ui/wm/public/activation_client.h"
namespace ash {
namespace {
using ::chromeos::WindowStateType;
constexpr float kFixedPositionRatios[] = {0.f, chromeos::kDefaultSnapRatio,
1.0f};
constexpr float kBlackScrimFadeInRatio = 0.1f;
constexpr float kBlackScrimOpacity = 0.4f;
constexpr int kSplitViewThresholdPixelsPerSec = 72;
constexpr base::TimeDelta kSplitViewChunkTime = base::Milliseconds(500);
constexpr char kDividerAnimationSmoothness[] =
"Ash.SplitViewResize.AnimationSmoothness.DividerAnimation";
constexpr char kClamshellSplitViewResizeSingleHistogram[] =
"Ash.SplitViewResize.PresentationTime.ClamshellMode.SingleWindow";
constexpr char kClamshellSplitViewResizeWithOverviewHistogram[] =
"Ash.SplitViewResize.PresentationTime.ClamshellMode.WithOverview";
constexpr char kTabletSplitViewResizeSingleHistogram[] =
"Ash.SplitViewResize.PresentationTime.TabletMode.SingleWindow";
constexpr char kClamshellSplitViewResizeMultiHistogram[] =
"Ash.SplitViewResize.PresentationTime.ClamshellMode.MultiWindow";
constexpr char kTabletSplitViewResizeMultiHistogram[] =
"Ash.SplitViewResize.PresentationTime.TabletMode.MultiWindow";
constexpr char kTabletSplitViewResizeWithOverviewHistogram[] =
"Ash.SplitViewResize.PresentationTime.TabletMode.WithOverview";
constexpr char kClamshellSplitViewResizeSingleMaxLatencyHistogram[] =
"Ash.SplitViewResize.PresentationTime.MaxLatency.ClamshellMode."
"SingleWindow";
constexpr char kClamshellSplitViewResizeWithOverviewMaxLatencyHistogram[] =
"Ash.SplitViewResize.PresentationTime.MaxLatency.ClamshellMode."
"WithOverview";
constexpr char kTabletSplitViewResizeSingleMaxLatencyHistogram[] =
"Ash.SplitViewResize.PresentationTime.MaxLatency.TabletMode.SingleWindow";
constexpr char kTabletSplitViewResizeMultiMaxLatencyHistogram[] =
"Ash.SplitViewResize.PresentationTime.MaxLatency.TabletMode.MultiWindow";
constexpr char kTabletSplitViewResizeWithOverviewMaxLatencyHistogram[] =
"Ash.SplitViewResize.PresentationTime.MaxLatency.TabletMode.WithOverview";
constexpr char kSplitViewSwapWindowsSource[] = "Ash.SplitView.SwapWindowSource";
base::Time g_multi_display_split_view_start_time;
bool IsExactlyOneRootInSplitView() {
const aura::Window::Windows all_root_windows = Shell::GetAllRootWindows();
return 1 ==
base::ranges::count_if(
all_root_windows, [](aura::Window* root_window) {
return SplitViewController::Get(root_window)->InSplitViewMode();
});
}
gfx::Point GetBoundedPosition(const gfx::Point& location_in_screen,
const gfx::Rect& bounds_in_screen) {
return gfx::Point(std::clamp(location_in_screen.x(), bounds_in_screen.x(),
bounds_in_screen.right() - 1),
std::clamp(location_in_screen.y(), bounds_in_screen.y(),
bounds_in_screen.bottom() - 1));
}
ui::InputMethod* GetCurrentInputMethod() {
if (auto* bridge = IMEBridge::Get()) {
if (auto* handler = bridge->GetInputContextHandler())
return handler->GetInputMethod();
}
return nullptr;
}
WindowStateType GetStateTypeFromSnapPosition(
SplitViewController::SnapPosition snap_position) {
DCHECK(snap_position != SplitViewController::SnapPosition::kNone);
if (snap_position == SplitViewController::SnapPosition::kPrimary)
return WindowStateType::kPrimarySnapped;
if (snap_position == SplitViewController::SnapPosition::kSecondary)
return WindowStateType::kSecondarySnapped;
NOTREACHED();
return WindowStateType::kDefault;
}
int GetMinimumWindowLength(aura::Window* window, bool horizontal) {
int minimum_width = 0;
if (window && window->delegate()) {
gfx::Size minimum_size = window->delegate()->GetMinimumSize();
minimum_width = horizontal ? minimum_size.width() : minimum_size.height();
}
return minimum_width;
}
int GetWindowLength(aura::Window* window, bool horizontal) {
const auto& bounds = window->bounds();
return horizontal ? bounds.width() : bounds.height();
}
bool IsSnapped(aura::Window* window) {
if (!window)
return false;
return WindowState::Get(window)->IsSnapped();
}
bool IsInTabletMode() {
TabletModeController* tablet_mode_controller =
Shell::Get()->tablet_mode_controller();
return tablet_mode_controller && tablet_mode_controller->InTabletMode();
}
bool IsInOverviewSession() {
OverviewController* overview_controller = Shell::Get()->overview_controller();
return overview_controller && overview_controller->InOverviewSession();
}
OverviewSession* GetOverviewSession() {
return IsInOverviewSession()
? Shell::Get()->overview_controller()->overview_session()
: nullptr;
}
bool ShouldShowOverviewInClamshellOnWindowSnapped() {
auto window_list =
Shell::Get()->mru_window_tracker()->BuildMruWindowList(kActiveDesk);
base::EraseIf(window_list, window_util::ShouldExcludeForOverview);
return IsSnapGroupEnabledInClamshellMode() &&
Shell::Get()
->snap_group_controller()
->IsArm1AutomaticallyLockEnabled() &&
!window_list.empty();
}
void RemoveSnappingWindowFromOverviewIfApplicable(
OverviewSession* overview_session,
aura::Window* window) {
if (!overview_session) {
return;
}
OverviewItem* item = overview_session->GetOverviewItemForWindow(window);
if (!item) {
return;
}
item->EnsureVisible();
item->RestoreWindow(false);
overview_session->RemoveItem(item);
}
void TriggerWMEventToSnapWindow(WindowState* window_state,
WMEventType event_type) {
CHECK(event_type == WM_EVENT_SNAP_PRIMARY ||
event_type == WM_EVENT_SNAP_SECONDARY);
const WMEvent window_event(event_type, window_state->snap_ratio().value_or(
chromeos::kDefaultSnapRatio));
window_state->OnWMEvent(&window_event);
}
}
class SplitViewController::DividerSnapAnimation
: public gfx::SlideAnimation,
public gfx::AnimationDelegate {
public:
DividerSnapAnimation(SplitViewController* split_view_controller,
int starting_position,
int ending_position)
: gfx::SlideAnimation(this),
split_view_controller_(split_view_controller),
starting_position_(starting_position),
ending_position_(ending_position) {
SetSlideDuration(base::Milliseconds(300));
SetTweenType(gfx::Tween::EASE_IN);
aura::Window* window = split_view_controller->primary_window()
? split_view_controller->primary_window()
: split_view_controller->secondary_window();
DCHECK(window);
views::Widget* widget = views::Widget::GetWidgetForNativeWindow(window);
if (!widget)
return;
gfx::AnimationContainer* container = new gfx::AnimationContainer();
container->SetAnimationRunner(
std::make_unique<views::CompositorAnimationRunner>(widget, FROM_HERE));
SetContainer(container);
tracker_.emplace(widget->GetCompositor()->RequestNewThroughputTracker());
tracker_->Start(
metrics_util::ForSmoothness(base::BindRepeating([](int smoothness) {
UMA_HISTOGRAM_PERCENTAGE(kDividerAnimationSmoothness, smoothness);
})));
}
DividerSnapAnimation(const DividerSnapAnimation&) = delete;
DividerSnapAnimation& operator=(const DividerSnapAnimation&) = delete;
~DividerSnapAnimation() override = default;
int ending_position() const { return ending_position_; }
private:
void AnimationEnded(const gfx::Animation* animation) override {
DCHECK(split_view_controller_->InSplitViewMode());
DCHECK(!split_view_controller_->is_resizing_with_divider_);
DCHECK_EQ(ending_position_, split_view_controller_->divider_position_);
split_view_controller_->EndResizeWithDividerImpl();
split_view_controller_->EndSplitViewAfterResizingAtEdgeIfAppropriate();
if (tracker_)
tracker_->Stop();
}
void AnimationProgressed(const gfx::Animation* animation) override {
DCHECK(split_view_controller_->InSplitViewMode());
DCHECK(!split_view_controller_->is_resizing_with_divider_);
split_view_controller_->divider_position_ =
CurrentValueBetween(starting_position_, ending_position_);
split_view_controller_->NotifyDividerPositionChanged();
split_view_controller_->UpdateSnappedWindowsAndDividerBounds();
if (is_animating()) {
split_view_controller_->UpdateResizeBackdrop();
split_view_controller_->SetWindowsTransformDuringResizing();
}
}
void AnimationCanceled(const gfx::Animation* animation) override {
if (tracker_)
tracker_->Cancel();
}
raw_ptr<SplitViewController, ExperimentalAsh> split_view_controller_;
int starting_position_;
int ending_position_;
absl::optional<ui::ThroughputTracker> tracker_;
};
class SplitViewController::AutoSnapController
: public wm::ActivationChangeObserver,
public aura::WindowObserver {
public:
explicit AutoSnapController(SplitViewController* split_view_controller)
: split_view_controller_(split_view_controller) {
Shell::Get()->activation_client()->AddObserver(this);
AddWindow(split_view_controller->root_window());
for (auto* window :
Shell::Get()->mru_window_tracker()->BuildMruWindowList(kActiveDesk)) {
AddWindow(window);
}
}
~AutoSnapController() override {
for (auto* window : observed_windows_)
window->RemoveObserver(this);
Shell::Get()->activation_client()->RemoveObserver(this);
}
AutoSnapController(const AutoSnapController&) = delete;
AutoSnapController& operator=(const AutoSnapController&) = delete;
void OnWindowActivated(ActivationReason reason,
aura::Window* gained_active,
aura::Window* lost_active) override {
if (!gained_active)
return;
if (reason == ActivationReason::WINDOW_DISPOSITION_CHANGED)
return;
AutoSnapWindowIfNeeded(gained_active);
}
void OnWindowVisibilityChanging(aura::Window* window, bool visible) override {
if (!visible)
return;
if (auto* window_state = WindowState::Get(window);
!window_state || !window_state->IsMinimized()) {
return;
}
if (window->GetProperty(kHideDuringWindowDragging))
return;
AutoSnapWindowIfNeeded(window);
}
void OnWindowAddedToRootWindow(aura::Window* window) override {
AddWindow(window);
}
void OnWindowRemovingFromRootWindow(aura::Window* window,
aura::Window* new_root) override {
RemoveWindow(window);
}
void OnWindowDestroying(aura::Window* window) override {
RemoveWindow(window);
}
private:
void AutoSnapWindowIfNeeded(aura::Window* window) {
DCHECK(window);
if (window->GetRootWindow() != split_view_controller_->root_window())
return;
if (!split_view_controller_->InSplitViewMode()) {
return;
}
if (WindowState::Get(window)->IsFloated() &&
split_view_controller_->BothSnapped()) {
return;
}
if (DesksController::Get()->AreDesksBeingModified()) {
return;
}
if (split_view_controller_->IsWindowInSplitView(window) ||
!base::Contains(
Shell::Get()->mru_window_tracker()->BuildMruWindowList(kActiveDesk),
window)) {
return;
}
if (split_view_controller_->InClamshellSplitViewMode()) {
if (split_view_controller_->IsWindowInTransitionalState(window)) {
return;
}
Shell::Get()->overview_controller()->EndOverview(
OverviewEndAction::kSplitView);
return;
}
DCHECK(split_view_controller_->InTabletSplitViewMode());
if (WindowState::Get(window)->is_dragged()) {
return;
}
if (split_view_controller_->IsDividerAnimating()) {
if (WindowState::Get(window)->IsUserPositionable())
split_view_controller_->EndSplitView(
EndReason::kUnsnappableWindowActivated);
return;
}
absl::optional<float> snap_ratio =
split_view_controller_->ComputeSnapRatio(window);
if (!snap_ratio) {
if (WindowState::Get(window)->IsUserPositionable()) {
split_view_controller_->EndSplitView(
EndReason::kUnsnappableWindowActivated);
ShowAppCannotSnapToast();
}
return;
}
WindowState::Get(window)->set_snap_action_source(
WindowSnapActionSource::kAutoSnapBySplitview);
split_view_controller_->SnapWindow(
window,
(split_view_controller_->default_snap_position() ==
SnapPosition::kPrimary)
? SnapPosition::kSecondary
: SnapPosition::kPrimary,
false, *snap_ratio);
}
void AddWindow(aura::Window* window) {
if (split_view_controller_->root_window() != window->GetRootWindow())
return;
if (!window->HasObserver(this))
window->AddObserver(this);
observed_windows_.insert(window);
}
void RemoveWindow(aura::Window* window) {
window->RemoveObserver(this);
observed_windows_.erase(window);
}
raw_ptr<SplitViewController, ExperimentalAsh> split_view_controller_;
base::flat_set<aura::Window*> observed_windows_;
};
class SplitViewController::ToBeSnappedWindowsObserver
: public aura::WindowObserver,
public WindowStateObserver {
public:
explicit ToBeSnappedWindowsObserver(
SplitViewController* split_view_controller)
: split_view_controller_(split_view_controller) {}
ToBeSnappedWindowsObserver(const ToBeSnappedWindowsObserver&) = delete;
ToBeSnappedWindowsObserver& operator=(const ToBeSnappedWindowsObserver&) =
delete;
~ToBeSnappedWindowsObserver() override {
for (auto& to_be_snapped_window : to_be_snapped_windows_) {
aura::Window* window = to_be_snapped_window.second;
if (window) {
window->RemoveObserver(this);
WindowState::Get(window)->RemoveObserver(this);
}
}
to_be_snapped_windows_.clear();
}
void AddToBeSnappedWindow(aura::Window* window,
SplitViewController::SnapPosition snap_position) {
if (split_view_controller_->IsWindowInSplitView(window)) {
if (WindowState::Get(window)->GetStateType() !=
GetStateTypeFromSnapPosition(snap_position)) {
split_view_controller_->AttachSnappingWindow(window, snap_position);
}
return;
}
aura::Window* old_window = to_be_snapped_windows_[snap_position];
if (old_window == window)
return;
if (old_window) {
to_be_snapped_windows_[snap_position] = nullptr;
WindowState::Get(old_window)->RemoveObserver(this);
old_window->RemoveObserver(this);
}
if (WindowState::Get(window)->GetStateType() ==
GetStateTypeFromSnapPosition(snap_position)) {
split_view_controller_->AttachSnappingWindow(window, snap_position);
split_view_controller_->OnWindowSnapped(window,
absl::nullopt);
} else {
to_be_snapped_windows_[snap_position] = window;
WindowState::Get(window)->AddObserver(this);
window->AddObserver(this);
}
}
bool IsObserving(const aura::Window* window) const {
return FindWindow(window) != to_be_snapped_windows_.end();
}
void OnWindowDestroying(aura::Window* window) override {
auto iter = FindWindow(window);
DCHECK(iter != to_be_snapped_windows_.end());
window->RemoveObserver(this);
WindowState::Get(window)->RemoveObserver(this);
to_be_snapped_windows_.erase(iter);
}
void OnPreWindowStateTypeChange(WindowState* window_state,
WindowStateType old_type) override {
auto iter = FindWindow(window_state->window());
DCHECK(iter != to_be_snapped_windows_.end());
SnapPosition snap_position = iter->first;
if (window_state->GetStateType() ==
GetStateTypeFromSnapPosition(snap_position)) {
to_be_snapped_windows_.erase(iter);
window_state->RemoveObserver(this);
window_state->window()->RemoveObserver(this);
split_view_controller_->AttachSnappingWindow(window_state->window(),
snap_position);
}
}
private:
std::map<SnapPosition, aura::Window*>::const_iterator FindWindow(
const aura::Window* window) const {
for (auto iter = to_be_snapped_windows_.begin();
iter != to_be_snapped_windows_.end(); iter++) {
if (iter->second == window)
return iter;
}
return to_be_snapped_windows_.end();
}
const raw_ptr<SplitViewController, ExperimentalAsh> split_view_controller_;
std::map<SnapPosition, aura::Window*> to_be_snapped_windows_;
};
SplitViewController* SplitViewController::Get(const aura::Window* window) {
DCHECK(window);
DCHECK(window->GetRootWindow());
DCHECK(RootWindowController::ForWindow(window));
return RootWindowController::ForWindow(window)->split_view_controller();
}
bool SplitViewController::IsLayoutHorizontal(aura::Window* window) {
return IsLayoutHorizontal(
display::Screen::GetScreen()->GetDisplayNearestWindow(window));
}
bool SplitViewController::IsLayoutHorizontal(const display::Display& display) {
if (IsInTabletMode()) {
return IsCurrentScreenOrientationLandscape();
}
DCHECK(display.is_valid());
return chromeos::IsLandscapeOrientation(GetSnapDisplayOrientation(display));
}
bool SplitViewController::IsLayoutPrimary(aura::Window* window) {
return IsLayoutPrimary(
display::Screen::GetScreen()->GetDisplayNearestWindow(window));
}
bool SplitViewController::IsLayoutPrimary(const display::Display& display) {
if (IsInTabletMode()) {
return IsCurrentScreenOrientationPrimary();
}
DCHECK(display.is_valid());
return chromeos::IsPrimaryOrientation(GetSnapDisplayOrientation(display));
}
bool SplitViewController::IsPhysicalLeftOrTop(SnapPosition position,
aura::Window* window) {
DCHECK_NE(SnapPosition::kNone, position);
return position == (IsLayoutPrimary(window) ? SnapPosition::kPrimary
: SnapPosition::kSecondary);
}
bool SplitViewController::IsPhysicalLeftOrTop(SnapPosition position,
const display::Display& display) {
DCHECK_NE(SnapPosition::kNone, position);
return position == (IsLayoutPrimary(display) ? SnapPosition::kPrimary
: SnapPosition::kSecondary);
}
SplitViewController::SplitViewController(aura::Window* root_window)
: root_window_(root_window),
to_be_snapped_windows_observer_(
std::make_unique<ToBeSnappedWindowsObserver>(this)),
split_view_metrics_controller_(
std::make_unique<SplitViewMetricsController>(this)) {
Shell::Get()->accessibility_controller()->AddObserver(this);
Shell::Get()->tablet_mode_controller()->AddObserver(this);
if (SnapGroupController* snap_group_controller =
Shell::Get()->snap_group_controller()) {
snap_group_controller->AddObserver(this);
}
split_view_type_ = IsInTabletMode() ? SplitViewType::kTabletType
: SplitViewType::kClamshellType;
}
SplitViewController::~SplitViewController() {
if (Shell::Get()->tablet_mode_controller()) {
Shell::Get()->tablet_mode_controller()->RemoveObserver(this);
}
if (Shell::Get()->accessibility_controller()) {
Shell::Get()->accessibility_controller()->RemoveObserver(this);
}
if (SnapGroupController* snap_group_controller =
Shell::Get()->snap_group_controller()) {
snap_group_controller->RemoveObserver(this);
}
EndSplitView(EndReason::kRootWindowDestroyed);
}
bool SplitViewController::InSplitViewMode() const {
return state_ != State::kNoSnap;
}
bool SplitViewController::BothSnapped() const {
return state_ == State::kBothSnapped;
}
bool SplitViewController::InClamshellSplitViewMode() const {
return InSplitViewMode() && split_view_type_ == SplitViewType::kClamshellType;
}
bool SplitViewController::InTabletSplitViewMode() const {
return InSplitViewMode() && split_view_type_ == SplitViewType::kTabletType;
}
bool SplitViewController::CanSnapWindow(aura::Window* window) const {
return CanSnapWindow(window, chromeos::kDefaultSnapRatio);
}
bool SplitViewController::CanSnapWindow(aura::Window* window,
float snap_ratio) const {
if (!ShouldAllowSplitView())
return false;
if (!WindowState::Get(window)->CanSnapOnDisplay(
display::Screen::GetScreen()->GetDisplayNearestWindow(
const_cast<aura::Window*>(root_window_.get())))) {
return false;
}
const bool is_to_be_restored_window =
window == WindowRestoreController::Get()->to_be_snapped_window();
if (!is_to_be_restored_window && !wm::CanActivateWindow(window))
return false;
return GetMinimumWindowLength(window, IsLayoutHorizontal(window)) <=
GetDividerEndPosition() * snap_ratio -
kSplitviewDividerShortSideLength / 2;
}
absl::optional<float> SplitViewController::ComputeSnapRatio(
aura::Window* window) {
aura::Window* default_window = GetDefaultSnappedWindow();
absl::optional<float> default_window_snap_ratio =
default_window ? WindowState::Get(default_window)->snap_ratio()
: absl::nullopt;
if (!default_window_snap_ratio) {
return CanSnapWindow(window)
? absl::make_optional(chromeos::kDefaultSnapRatio)
: absl::nullopt;
}
static constexpr auto kOppositeRatiosMap =
base::MakeFixedFlatMap<float, float>(
{{chromeos::kOneThirdSnapRatio, chromeos::kTwoThirdSnapRatio},
{chromeos::kDefaultSnapRatio, chromeos::kDefaultSnapRatio},
{chromeos::kTwoThirdSnapRatio, chromeos::kOneThirdSnapRatio}});
auto* it = kOppositeRatiosMap.find(*default_window_snap_ratio);
if (it == kOppositeRatiosMap.end()) {
return CanSnapWindow(window)
? absl::make_optional(chromeos::kDefaultSnapRatio)
: absl::nullopt;
}
float snap_ratio = it->second;
if (CanSnapWindow(window, snap_ratio)) {
return snap_ratio;
}
if (snap_ratio == chromeos::kOneThirdSnapRatio && CanSnapWindow(window) &&
CanSnapWindow(default_window)) {
return chromeos::kDefaultSnapRatio;
}
return absl::nullopt;
}
void SplitViewController::SnapWindow(aura::Window* window,
SnapPosition snap_position,
bool activate_window,
float snap_ratio) {
DCHECK(window && CanSnapWindow(window));
DCHECK_NE(snap_position, SnapPosition::kNone);
DCHECK(!is_resizing_with_divider_);
DCHECK(!IsDividerAnimating());
OverviewSession* overview_session = GetOverviewSession();
if (activate_window ||
(overview_session &&
overview_session->IsWindowActiveWindowBeforeOverview(window))) {
to_be_activated_window_ = window;
}
to_be_snapped_windows_observer_->AddToBeSnappedWindow(window, snap_position);
if (root_window_ != window->GetRootWindow()) {
window_util::MoveWindowToDisplay(window,
display::Screen::GetScreen()
->GetDisplayNearestWindow(root_window_)
.id());
}
const WMEvent event(snap_position == SnapPosition::kPrimary
? WM_EVENT_SNAP_PRIMARY
: WM_EVENT_SNAP_SECONDARY,
snap_ratio);
WindowState::Get(window)->OnWMEvent(&event);
base::RecordAction(base::UserMetricsAction("SplitView_SnapWindow"));
}
void SplitViewController::OnWMEvent(aura::Window* window,
WMEventType event_type) {
DCHECK(event_type == WM_EVENT_SNAP_PRIMARY ||
event_type == WM_EVENT_SNAP_SECONDARY);
if (!ShouldAllowSplitView())
return;
const bool in_overview = IsInOverviewSession();
if (split_view_type_ == SplitViewType::kClamshellType &&
!(in_overview || IsSnapGroupEnabledInClamshellMode())) {
return;
}
const int32_t window_id =
window->GetProperty(app_restore::kRestoreWindowIdKey);
if (in_overview &&
window == WindowRestoreController::Get()->to_be_snapped_window() &&
app_restore::DeskTemplateReadHandler::Get()->GetWindowInfo(window_id)) {
return;
}
if (to_be_snapped_windows_observer_->IsObserving(window)) {
return;
}
const SnapPosition to_snap_position = event_type == WM_EVENT_SNAP_PRIMARY
? SnapPosition::kPrimary
: SnapPosition::kSecondary;
absl::optional<float> new_snap_ratio = WindowState::Get(window)->snap_ratio();
if (new_snap_ratio) {
divider_position_ = GetDividerPosition(to_snap_position, *new_snap_ratio);
if (split_view_divider_) {
split_view_divider_->UpdateDividerBounds();
}
if (!primary_window_ != !secondary_window_) {
NotifyDividerPositionChanged();
} else if (primary_window_ && secondary_window_) {
auto* other_window = event_type == WM_EVENT_SNAP_PRIMARY
? secondary_window_.get()
: primary_window_.get();
DCHECK(other_window);
const int other_window_length =
GetMinimumWindowLength(other_window, IsLayoutHorizontal(window));
const int work_area_size = GetDividerEndPosition();
const int window_length = event_type == WM_EVENT_SNAP_PRIMARY
? divider_position_
: work_area_size - divider_position_ -
kSplitviewDividerShortSideLength;
if (window_length + other_window_length +
kSplitviewDividerShortSideLength >
work_area_size) {
Shell::Get()->overview_controller()->StartOverview(
OverviewStartAction::kSplitView,
OverviewEnterExitType::kImmediateEnter);
} else {
UpdateSnappedBounds(other_window);
}
}
}
to_be_snapped_windows_observer_->AddToBeSnappedWindow(window,
to_snap_position);
}
void SplitViewController::AttachSnappingWindow(aura::Window* window,
SnapPosition snap_position) {
UpdateSnappingWindowTransformedBounds(window);
OverviewSession* overview_session = GetOverviewSession();
RemoveSnappingWindowFromOverviewIfApplicable(overview_session, window);
if (state_ == State::kNoSnap) {
Shell::Get()->AddShellObserver(this);
Shell::Get()->overview_controller()->AddObserver(this);
if (features::IsAdjustSplitViewForVKEnabled()) {
keyboard::KeyboardUIController::Get()->AddObserver(this);
Shell::Get()->activation_client()->AddObserver(this);
}
auto_snap_controller_ = std::make_unique<AutoSnapController>(this);
absl::optional<float> snap_ratio = WindowState::Get(window)->snap_ratio();
divider_position_ =
(divider_position_ < 0)
? GetDividerPosition(
snap_position,
snap_ratio ? *snap_ratio : chromeos::kDefaultSnapRatio)
: divider_position_;
default_snap_position_ = snap_position;
if (split_view_type_ == SplitViewType::kTabletType) {
split_view_divider_ = std::make_unique<SplitViewDivider>(this);
}
splitview_start_time_ = base::Time::Now();
if (IsExactlyOneRootInSplitView()) {
base::RecordAction(
base::UserMetricsAction("SplitView_MultiDisplaySplitView"));
g_multi_display_split_view_start_time = splitview_start_time_;
}
}
aura::Window* previous_snapped_window = nullptr;
if (snap_position == SnapPosition::kPrimary) {
if (primary_window_ != window) {
previous_snapped_window = primary_window_;
StopObserving(SnapPosition::kPrimary);
primary_window_ = window;
}
if (secondary_window_ == window) {
secondary_window_ = nullptr;
default_snap_position_ = SnapPosition::kPrimary;
}
} else if (snap_position == SnapPosition::kSecondary) {
if (secondary_window_ != window) {
previous_snapped_window = secondary_window_;
StopObserving(SnapPosition::kSecondary);
secondary_window_ = window;
}
if (primary_window_ == window) {
primary_window_ = nullptr;
default_snap_position_ = SnapPosition::kSecondary;
}
}
StartObserving(window);
DCHECK_EQ(overview_session, GetOverviewSession());
if (previous_snapped_window && overview_session) {
InsertWindowToOverview(previous_snapped_window);
overview_session->GetOverviewItemForWindow(previous_snapped_window)
->OnSelectorItemDragEnded(true);
}
if (split_view_type_ == SplitViewType::kTabletType) {
divider_position_ = GetClosestFixedDividerPosition();
split_view_divider_->UpdateDividerBounds();
}
base::RecordAction(base::UserMetricsAction("SplitView_SnapWindow"));
}
void SplitViewController::SwapWindows(SwapWindowsSource swap_windows_source) {
DCHECK(InSplitViewMode());
if (IsDividerAnimating()) {
return;
}
SnapGroupController* snap_group_controller =
Shell::Get()->snap_group_controller();
if (snap_group_controller && snap_group_controller->AreWindowsInSnapGroup(
primary_window_, secondary_window_)) {
snap_group_controller->RemoveSnapGroupContainingWindow(primary_window_);
}
SwapWindowsAndUpdateBounds();
if (IsSnapped(primary_window_)) {
TriggerWMEventToSnapWindow(WindowState::Get(primary_window_),
WM_EVENT_SNAP_PRIMARY);
}
if (IsSnapped(secondary_window_)) {
TriggerWMEventToSnapWindow(WindowState::Get(secondary_window_),
WM_EVENT_SNAP_SECONDARY);
}
if (!primary_window_ || !secondary_window_) {
default_snap_position_ =
primary_window_ ? SnapPosition::kPrimary : SnapPosition::kSecondary;
}
divider_position_ = GetClosestFixedDividerPosition();
UpdateStateAndNotifyObservers();
NotifyWindowSwapped();
switch (swap_windows_source) {
case SwapWindowsSource::kDoubleTap: {
base::RecordAction(
base::UserMetricsAction("SplitView_DoubleTapDividerSwapWindows"));
break;
}
case SwapWindowsSource::kSnapGroupSwapWindowsButton: {
base::RecordAction(
base::UserMetricsAction("SplitView_SwapWindowsButtonSwapWindows"));
}
}
base::UmaHistogramEnumeration(kSplitViewSwapWindowsSource,
swap_windows_source);
}
SplitViewController::SnapPosition
SplitViewController::GetPositionOfSnappedWindow(
const aura::Window* window) const {
DCHECK(IsWindowInSplitView(window));
return window == primary_window_ ? SnapPosition::kPrimary
: SnapPosition::kSecondary;
}
aura::Window* SplitViewController::GetSnappedWindow(SnapPosition position) {
DCHECK_NE(SnapPosition::kNone, position);
return position == SnapPosition::kPrimary ? primary_window_.get()
: secondary_window_.get();
}
aura::Window* SplitViewController::GetDefaultSnappedWindow() {
if (default_snap_position_ == SnapPosition::kPrimary)
return primary_window_;
if (default_snap_position_ == SnapPosition::kSecondary)
return secondary_window_;
return nullptr;
}
gfx::Rect SplitViewController::GetSnappedWindowBoundsInParent(
SnapPosition snap_position,
aura::Window* window_for_minimum_size,
float snap_ratio) {
gfx::Rect bounds = GetSnappedWindowBoundsInScreen(
snap_position, window_for_minimum_size, snap_ratio);
wm::ConvertRectFromScreen(root_window_, &bounds);
return bounds;
}
gfx::Rect SplitViewController::GetSnappedWindowBoundsInParent(
SnapPosition snap_position,
aura::Window* window_for_minimum_size) {
return GetSnappedWindowBoundsInParent(snap_position, window_for_minimum_size,
chromeos::kDefaultSnapRatio);
}
gfx::Rect SplitViewController::GetSnappedWindowBoundsInScreen(
SnapPosition snap_position,
aura::Window* window_for_minimum_size,
float snap_ratio) {
const gfx::Rect work_area_bounds_in_screen =
screen_util::GetDisplayWorkAreaBoundsInScreenForActiveDeskContainer(
root_window_);
if (snap_position == SnapPosition::kNone) {
return work_area_bounds_in_screen;
}
if (window_for_minimum_size && ShouldUseWindowBoundsDuringFastResize()) {
gfx::Rect bounds = window_for_minimum_size->bounds();
wm::ConvertRectToScreen(window_for_minimum_size->parent(), &bounds);
return bounds;
}
const bool horizontal = IsLayoutHorizontal(root_window_);
const bool snap_left_or_top =
IsPhysicalLeftOrTop(snap_position, root_window_);
const bool in_tablet_mode = IsInTabletMode();
const int work_area_size = GetDividerEndPosition();
int divider_position = divider_position_ < 0
? GetDividerPosition(snap_position, snap_ratio)
: divider_position_;
if (divider_position_ < 0 && !in_tablet_mode) {
if (auto* window = WindowRestoreController::Get()->to_be_snapped_window()) {
app_restore::WindowInfo* window_info =
window->GetProperty(app_restore::kWindowInfoKey);
if (window_info && window_info->snap_percentage) {
const int snap_percentage = *window_info->snap_percentage;
divider_position = snap_percentage * work_area_size / 100;
if (!snap_left_or_top)
divider_position = work_area_size - divider_position;
}
}
}
int window_size;
if (snap_left_or_top) {
window_size = divider_position;
} else {
window_size = work_area_size - divider_position;
if (in_tablet_mode ||
(split_view_divider_ && IsSnapGroupEnabledInClamshellMode())) {
window_size -= kSplitviewDividerShortSideLength;
}
}
const int minimum =
GetMinimumWindowLength(window_for_minimum_size, horizontal);
DCHECK(window_for_minimum_size || minimum == 0);
if (window_size < minimum) {
if (in_tablet_mode && !is_resizing_with_divider_) {
window_size = work_area_size / 2 - kSplitviewDividerShortSideLength / 2;
if (!snap_left_or_top && work_area_size % 2 == 1)
++window_size;
} else {
window_size = minimum;
}
}
if (window_for_minimum_size && !in_tablet_mode) {
const gfx::Size* preferred_size =
window_for_minimum_size->GetProperty(kUnresizableSnappedSizeKey);
if (preferred_size &&
!WindowState::Get(window_for_minimum_size)->CanResize()) {
if (horizontal && preferred_size->width() > 0)
window_size = preferred_size->width();
if (!horizontal && preferred_size->height() > 0)
window_size = preferred_size->height();
}
}
int left = work_area_bounds_in_screen.x();
int top = work_area_bounds_in_screen.y();
int right = work_area_bounds_in_screen.right();
int bottom = work_area_bounds_in_screen.bottom();
int& left_or_top = horizontal ? left : top;
int& right_or_bottom = horizontal ? right : bottom;
if (snap_left_or_top) {
right_or_bottom = left_or_top + window_size;
} else {
left_or_top = right_or_bottom - window_size;
}
gfx::Rect snapped_window_bounds_in_screen;
snapped_window_bounds_in_screen.SetByBounds(left, top, right, bottom);
return snapped_window_bounds_in_screen;
}
gfx::Rect SplitViewController::GetSnappedWindowBoundsInScreen(
SnapPosition snap_position,
aura::Window* window_for_minimum_size) {
return GetSnappedWindowBoundsInScreen(snap_position, window_for_minimum_size,
chromeos::kDefaultSnapRatio);
}
bool SplitViewController::ShouldUseWindowBoundsDuringFastResize() {
return is_resizing_with_divider_ &&
tablet_resize_mode_ == TabletResizeMode::kFast;
}
int SplitViewController::GetDefaultDividerPosition() const {
return GetDividerPosition(SnapPosition::kPrimary,
chromeos::kDefaultSnapRatio);
}
int SplitViewController::GetDividerPosition(SnapPosition snap_position,
float snap_ratio) const {
int divider_end_position = GetDividerEndPosition();
const float snap_width = divider_end_position * snap_ratio;
int next_divider_position = snap_position == SnapPosition::kPrimary
? snap_width
: divider_end_position - snap_width;
if (split_view_type_ == SplitViewType::kTabletType ||
IsSnapGroupEnabledInClamshellMode()) {
next_divider_position -= kSplitviewDividerShortSideLength / 2;
}
return next_divider_position;
}
bool SplitViewController::IsDividerAnimating() const {
return divider_snap_animation_ && divider_snap_animation_->is_animating();
}
void SplitViewController::StartResizeWithDivider(
const gfx::Point& location_in_screen) {
DCHECK(InSplitViewMode());
if (is_resizing_with_divider_ || IsDividerAnimating()) {
return;
}
is_resizing_with_divider_ = true;
split_view_divider_->UpdateDividerBounds();
previous_event_location_ = location_in_screen;
accumulated_drag_time_ticks_ = base::TimeTicks::Now();
accumulated_drag_distance_ = 0;
tablet_resize_mode_ = TabletResizeMode::kNormal;
for (aura::Window* window : {primary_window_, secondary_window_}) {
if (window == nullptr) {
continue;
}
WindowState* window_state = WindowState::Get(window);
gfx::Point location_in_parent(location_in_screen);
::wm::ConvertPointFromScreen(window->parent(), &location_in_parent);
int window_component = GetWindowComponentForResize(window);
window_state->CreateDragDetails(gfx::PointF(location_in_parent),
window_component,
::wm::WINDOW_MOVE_SOURCE_TOUCH);
window_state->OnDragStarted(window_component);
}
base::RecordAction(base::UserMetricsAction("SplitView_ResizeWindows"));
if (state_ == State::kBothSnapped) {
presentation_time_recorder_ = CreatePresentationTimeHistogramRecorder(
split_view_divider_->divider_widget()->GetCompositor(),
kTabletSplitViewResizeMultiHistogram,
kTabletSplitViewResizeMultiMaxLatencyHistogram);
return;
}
DCHECK(GetOverviewSession());
if (GetOverviewSession()->GetGridWithRootWindow(root_window_)->empty()) {
presentation_time_recorder_ = CreatePresentationTimeHistogramRecorder(
split_view_divider_->divider_widget()->GetCompositor(),
kTabletSplitViewResizeSingleHistogram,
kTabletSplitViewResizeSingleMaxLatencyHistogram);
} else {
presentation_time_recorder_ = CreatePresentationTimeHistogramRecorder(
split_view_divider_->divider_widget()->GetCompositor(),
kTabletSplitViewResizeWithOverviewHistogram,
kTabletSplitViewResizeWithOverviewMaxLatencyHistogram);
}
}
void SplitViewController::ResizeWithDivider(
const gfx::Point& location_in_screen) {
DCHECK(InSplitViewMode());
if (!is_resizing_with_divider_) {
return;
}
base::AutoReset<bool> auto_reset(&processing_resize_event_, true);
presentation_time_recorder_->RequestNext();
const gfx::Rect work_area_bounds =
screen_util::GetDisplayWorkAreaBoundsInScreenForActiveDeskContainer(
root_window_);
gfx::Point modified_location_in_screen =
GetBoundedPosition(location_in_screen, work_area_bounds);
UpdateTabletResizeMode(base::TimeTicks::Now(), modified_location_in_screen);
if (tablet_resize_mode_ == TabletResizeMode::kFast) {
resize_timer_.Start(FROM_HERE, kSplitViewChunkTime, this,
&SplitViewController::OnResizeTimer);
}
UpdateDividerPosition(modified_location_in_screen);
NotifyDividerPositionChanged();
UpdateSnappedWindowsAndDividerBounds();
UpdateResizeBackdrop();
UpdateBlackScrim(modified_location_in_screen);
SetWindowsTransformDuringResizing();
previous_event_location_ = modified_location_in_screen;
}
void SplitViewController::EndResizeWithDivider(
const gfx::Point& location_in_screen) {
presentation_time_recorder_.reset();
DCHECK(InSplitViewMode());
if (!is_resizing_with_divider_) {
return;
}
black_scrim_layer_.reset();
resize_timer_.Stop();
tablet_resize_mode_ = TabletResizeMode::kNormal;
is_resizing_with_divider_ = false;
const gfx::Rect work_area_bounds =
screen_util::GetDisplayWorkAreaBoundsInScreenForActiveDeskContainer(
root_window_);
gfx::Point modified_location_in_screen =
GetBoundedPosition(location_in_screen, work_area_bounds);
UpdateDividerPosition(modified_location_in_screen);
NotifyDividerPositionChanged();
UpdateSnappedWindowsAndDividerBounds();
NotifyWindowResized();
const int target_divider_position = GetClosestFixedDividerPosition();
if (divider_position_ == target_divider_position ||
IsSnapGroupEnabledInClamshellMode()) {
EndResizeWithDividerImpl();
EndSplitViewAfterResizingAtEdgeIfAppropriate();
} else {
divider_snap_animation_ = std::make_unique<DividerSnapAnimation>(
this, divider_position_, target_divider_position);
divider_snap_animation_->Show();
}
}
void SplitViewController::EndSplitView(EndReason end_reason) {
if (!InSplitViewMode()) {
return;
}
end_reason_ = end_reason;
const bool is_divider_animating = IsDividerAnimating();
if ((is_resizing_with_divider_ || is_divider_animating) &&
end_reason != EndReason::kRootWindowDestroyed) {
is_resizing_with_divider_ = false;
if (is_divider_animating) {
divider_snap_animation_->Stop();
divider_position_ = divider_snap_animation_->ending_position();
}
EndResizeWithDividerImpl();
}
presentation_time_recorder_.reset();
Shell::Get()->RemoveShellObserver(this);
Shell::Get()->overview_controller()->RemoveObserver(this);
if (features::IsAdjustSplitViewForVKEnabled()) {
keyboard::KeyboardUIController::Get()->RemoveObserver(this);
Shell::Get()->activation_client()->RemoveObserver(this);
}
auto_snap_controller_.reset();
if (IsSnapGroupEnabledInClamshellMode()) {
Shell::Get()->snap_group_controller()->RemoveSnapGroupContainingWindow(
primary_window_ == nullptr ? secondary_window_.get()
: primary_window_.get());
}
StopObserving(SnapPosition::kPrimary);
StopObserving(SnapPosition::kSecondary);
black_scrim_layer_.reset();
default_snap_position_ = SnapPosition::kNone;
divider_position_ = -1;
divider_closest_ratio_ = std::numeric_limits<float>::quiet_NaN();
snapping_window_transformed_bounds_map_.clear();
UpdateStateAndNotifyObservers();
split_view_divider_.reset();
base::RecordAction(base::UserMetricsAction("SplitView_EndSplitView"));
const base::Time now = base::Time::Now();
UMA_HISTOGRAM_LONG_TIMES("Ash.SplitView.TimeInSplitView",
now - splitview_start_time_);
if (IsExactlyOneRootInSplitView()) {
UMA_HISTOGRAM_LONG_TIMES("Ash.SplitView.TimeInMultiDisplaySplitView",
now - g_multi_display_split_view_start_time);
}
}
bool SplitViewController::IsWindowInSplitView(
const aura::Window* window) const {
return window && (window == primary_window_ || window == secondary_window_);
}
void SplitViewController::InitDividerPositionForTransition(
int divider_position) {
DCHECK(!InSplitViewMode());
DCHECK_EQ(divider_position_, -1);
divider_position_ = divider_position;
}
bool SplitViewController::IsWindowInTransitionalState(
const aura::Window* window) const {
return to_be_snapped_windows_observer_->IsObserving(window);
}
void SplitViewController::OnOverviewButtonTrayLongPressed(
const gfx::Point& event_location) {
if (!ShouldAllowSplitView())
return;
MruWindowTracker::WindowList mru_window_list =
Shell::Get()->mru_window_tracker()->BuildWindowForCycleList(kActiveDesk);
if (mru_window_list.empty())
return;
auto* overview_controller = Shell::Get()->overview_controller();
aura::Window* target_window = mru_window_list[0];
if (InSplitViewMode()) {
DCHECK(IsWindowInSplitView(target_window));
DCHECK(target_window);
EndSplitView();
overview_controller->EndOverview(
OverviewEndAction::kOverviewButtonLongPress);
MaximizeIfSnapped(target_window);
wm::ActivateWindow(target_window);
base::RecordAction(
base::UserMetricsAction("Tablet_LongPressOverviewButtonExitSplitView"));
return;
}
if (!CanSnapWindow(target_window)) {
ShowAppCannotSnapToast();
return;
}
overview_controller->StartOverview(
OverviewStartAction::kOverviewButtonLongPress,
OverviewEnterExitType::kImmediateEnter);
WindowState::Get(target_window)
->set_snap_action_source(
WindowSnapActionSource::kLongPressOverviewButtonToSnap);
SnapWindow(target_window, SnapPosition::kPrimary,
true);
base::RecordAction(
base::UserMetricsAction("Tablet_LongPressOverviewButtonEnterSplitView"));
}
void SplitViewController::OnWindowDragStarted(aura::Window* dragged_window) {
DCHECK(dragged_window);
if (IsWindowInSplitView(dragged_window)) {
OnSnappedWindowDetached(dragged_window,
WindowDetachedReason::kWindowDragged);
}
if (split_view_divider_) {
split_view_divider_->OnWindowDragStarted(dragged_window);
}
}
void SplitViewController::OnWindowDragEnded(
aura::Window* dragged_window,
SnapPosition desired_snap_position,
const gfx::Point& last_location_in_screen) {
DCHECK(!window_util::IsDraggingTabs(dragged_window));
EndWindowDragImpl(dragged_window, dragged_window->is_destroying(),
desired_snap_position, last_location_in_screen);
}
void SplitViewController::OnWindowDragCanceled() {
if (split_view_divider_)
split_view_divider_->OnWindowDragEnded();
}
SplitViewController::SnapPosition SplitViewController::ComputeSnapPosition(
const gfx::Point& last_location_in_screen) {
const int divider_position = InSplitViewMode() ? this->divider_position()
: GetDefaultDividerPosition();
const int position = IsLayoutHorizontal(root_window_)
? last_location_in_screen.x()
: last_location_in_screen.y();
return (position <= divider_position) == IsLayoutPrimary(root_window_)
? SnapPosition::kPrimary
: SnapPosition::kSecondary;
}
bool SplitViewController::BoundsChangeIsFromVKAndAllowed(
aura::Window* window) const {
return features::IsAdjustSplitViewForVKEnabled() && changing_bounds_by_vk_ &&
window == (IsLayoutPrimary(window) ? secondary_window_.get()
: primary_window_.get());
}
void SplitViewController::AddObserver(SplitViewObserver* observer) {
observers_.AddObserver(observer);
}
void SplitViewController::RemoveObserver(SplitViewObserver* observer) {
observers_.RemoveObserver(observer);
}
void SplitViewController::MaybeDetachWindow(aura::Window* dragged_window) {
auto* snap_group_controller = Shell::Get()->snap_group_controller();
if (snap_group_controller && InSplitViewMode() &&
!is_resizing_with_divider_ &&
!Shell::Get()->overview_controller()->InOverviewSession()) {
OnSnappedWindowDetached(dragged_window,
WindowDetachedReason::kWindowDragged);
}
}
void SplitViewController::OpenOverviewOnTheOtherSideOfTheScreen(
SnapPosition snap_position) {
default_snap_position_ = snap_position;
if (!IsInOverviewSession() && !DesksController::Get()->animation() &&
ShouldShowOverviewInClamshellOnWindowSnapped()) {
Shell::Get()->snap_group_controller()->RemoveSnapGroupContainingWindow(
primary_window_);
split_view_divider_.reset();
Shell::Get()->overview_controller()->StartOverview(
OverviewStartAction::kSplitView, OverviewEnterExitType::kNormal);
}
}
void SplitViewController::OnWindowPropertyChanged(aura::Window* window,
const void* key,
intptr_t old) {
if (key != aura::client::kResizeBehaviorKey)
return;
if (window->GetProperty(aura::client::kResizeBehaviorKey) ==
static_cast<int>(old)) {
return;
}
if (CanSnapWindow(window))
return;
EndSplitView();
Shell::Get()->overview_controller()->EndOverview(
OverviewEndAction::kSplitView);
ShowAppCannotSnapToast();
}
void SplitViewController::OnWindowBoundsChanged(
aura::Window* window,
const gfx::Rect& old_bounds,
const gfx::Rect& new_bounds,
ui::PropertyChangeReason reason) {
DCHECK_EQ(root_window_, window->GetRootWindow());
if (InTabletSplitViewMode() && is_resizing_with_divider_) {
if (!processing_resize_event_)
SetWindowsTransformDuringResizing();
return;
}
if (!InClamshellSplitViewMode() || IsSnapGroupEnabledInClamshellMode()) {
return;
}
WindowState* window_state = WindowState::Get(window);
if (window_state->is_dragged()) {
DCHECK_NE(WindowResizer::kBoundsChange_None,
window_state->drag_details()->bounds_change);
if (window_state->drag_details()->bounds_change ==
WindowResizer::kBoundsChange_Repositions) {
Shell::Get()->overview_controller()->EndOverview(
OverviewEndAction::kSplitView);
return;
}
DCHECK(window_state->drag_details()->bounds_change &
WindowResizer::kBoundsChange_Resizes);
DCHECK(presentation_time_recorder_);
presentation_time_recorder_->RequestNext();
}
const gfx::Rect work_area =
screen_util::GetDisplayWorkAreaBoundsInScreenForActiveDeskContainer(
root_window_);
if (IsLayoutHorizontal(window)) {
divider_position_ = window == primary_window_
? new_bounds.width()
: work_area.width() - new_bounds.width();
} else {
divider_position_ = window == primary_window_
? new_bounds.height()
: work_area.height() - new_bounds.height();
}
NotifyDividerPositionChanged();
}
void SplitViewController::OnWindowDestroyed(aura::Window* window) {
DCHECK(InSplitViewMode());
DCHECK(IsWindowInSplitView(window));
auto iter = snapping_window_transformed_bounds_map_.find(window);
if (iter != snapping_window_transformed_bounds_map_.end())
snapping_window_transformed_bounds_map_.erase(iter);
OnSnappedWindowDetached(window, WindowDetachedReason::kWindowDestroyed);
if (to_be_activated_window_ == window) {
to_be_activated_window_ = nullptr;
}
}
void SplitViewController::OnResizeLoopStarted(aura::Window* window) {
if (!InClamshellSplitViewMode()) {
return;
}
if (WindowState::Get(window)->drag_details()->window_component !=
GetWindowComponentForResize(window)) {
Shell::Get()->overview_controller()->EndOverview(
OverviewEndAction::kSplitView);
return;
}
if (IsSnapGroupEnabledInClamshellMode() && state_ == State::kBothSnapped) {
presentation_time_recorder_ = CreatePresentationTimeHistogramRecorder(
window->layer()->GetCompositor(),
kClamshellSplitViewResizeMultiHistogram,
kClamshellSplitViewResizeSingleMaxLatencyHistogram);
return;
}
DCHECK(GetOverviewSession());
if (GetOverviewSession()->GetGridWithRootWindow(root_window_)->empty()) {
presentation_time_recorder_ = CreatePresentationTimeHistogramRecorder(
window->layer()->GetCompositor(),
kClamshellSplitViewResizeSingleHistogram,
kClamshellSplitViewResizeSingleMaxLatencyHistogram);
} else {
presentation_time_recorder_ = CreatePresentationTimeHistogramRecorder(
window->layer()->GetCompositor(),
kClamshellSplitViewResizeWithOverviewHistogram,
kClamshellSplitViewResizeWithOverviewMaxLatencyHistogram);
}
}
void SplitViewController::OnResizeLoopEnded(aura::Window* window) {
if (!InClamshellSplitViewMode())
return;
presentation_time_recorder_.reset();
NotifyWindowResized();
if (divider_position_ <
GetDividerEndPosition() * chromeos::kOneThirdSnapRatio ||
divider_position_ >
GetDividerEndPosition() * chromeos::kTwoThirdSnapRatio) {
Shell::Get()->overview_controller()->EndOverview(
OverviewEndAction::kSplitView);
WindowState::Get(window)->Maximize();
}
}
void SplitViewController::OnPostWindowStateTypeChange(
WindowState* window_state,
WindowStateType old_type) {
DCHECK_EQ(
window_state->GetDisplay().id(),
display::Screen::GetScreen()->GetDisplayNearestWindow(root_window_).id());
aura::Window* window = window_state->window();
if (window_state->IsSnapped()) {
bool do_divider_spawn_animation = false;
if (state_ == State::kNoSnap &&
split_view_type_ == SplitViewType::kTabletType &&
old_type != WindowStateType::kMinimized &&
!window->transform().IsIdentity()) {
do_divider_spawn_animation = true;
}
OnWindowSnapped(window, old_type);
if (do_divider_spawn_animation)
DoSplitDividerSpawnAnimation(window);
} else if (window_state->IsNormalStateType() || window_state->IsMaximized() ||
window_state->IsFullscreen() || window_state->IsFloated()) {
EndSplitView();
Shell::Get()->overview_controller()->EndOverview(
OverviewEndAction::kSplitView);
} else if (window_state->IsMinimized()) {
OnSnappedWindowDetached(window, WindowDetachedReason::kWindowMinimized);
if (!InSplitViewMode()) {
if (split_view_type_ == SplitViewType::kTabletType) {
InsertWindowToOverview(window);
} else {
Shell::Get()->overview_controller()->EndOverview(
OverviewEndAction::kSplitView);
}
}
}
}
void SplitViewController::OnPinnedStateChanged(aura::Window* pinned_window) {
if (WindowState::Get(pinned_window)->IsPinned() && InSplitViewMode())
EndSplitView(EndReason::kUnsnappableWindowActivated);
}
void SplitViewController::OnOverviewModeStarting() {
DCHECK(InSplitViewMode());
if (default_snap_position_ == SnapPosition::kPrimary) {
StopObserving(SnapPosition::kSecondary);
} else if (default_snap_position_ == SnapPosition::kSecondary) {
StopObserving(SnapPosition::kPrimary);
}
UpdateStateAndNotifyObservers();
}
void SplitViewController::OnOverviewModeEnding(
OverviewSession* overview_session) {
DCHECK(InSplitViewMode());
if (state_ == State::kBothSnapped)
overview_session->SetWindowListNotAnimatedWhenExiting(root_window_);
if (split_view_type_ == SplitViewType::kClamshellType)
return;
if (state_ == State::kBothSnapped) {
return;
}
OverviewGrid* current_grid =
overview_session->GetGridWithRootWindow(root_window_);
if (!current_grid || current_grid->empty()) {
return;
}
for (const auto& overview_item : current_grid->window_list()) {
aura::Window* window = overview_item->GetWindow();
if (window != GetDefaultSnappedWindow()) {
absl::optional<float> snap_ratio = ComputeSnapRatio(window);
if (snap_ratio) {
overview_item->RestoreWindow(false);
overview_session->RemoveItem(overview_item.get());
WindowState::Get(window)->set_snap_action_source(
WindowSnapActionSource::kAutoSnapBySplitview);
SnapWindow(window,
(default_snap_position_ == SnapPosition::kPrimary)
? SnapPosition::kSecondary
: SnapPosition::kPrimary,
false, *snap_ratio);
overview_session->SetWindowListNotAnimatedWhenExiting(root_window_);
return;
}
}
}
if (DesksController::Get()->AreDesksBeingModified()) {
return;
}
EndSplitView();
ShowAppCannotSnapToast();
}
void SplitViewController::OnOverviewModeEnded() {
DCHECK(InSplitViewMode());
if (split_view_type_ == SplitViewType::kClamshellType &&
!IsSnapGroupEnabledInClamshellMode()) {
EndSplitView();
}
}
void SplitViewController::OnDisplayRemoved(
const display::Display& old_display) {
DCHECK(!(InClamshellSplitViewMode() && IsSnapGroupEnabledInClamshellMode()));
if (GetRootWindowSettings(root_window_)->display_id ==
display::kInvalidDisplayId) {
return;
}
if (state_ == State::kPrimarySnapped || state_ == State::kSecondarySnapped) {
Shell::Get()->overview_controller()->StartOverview(
OverviewStartAction::kSplitView,
OverviewEnterExitType::kImmediateEnter);
}
}
void SplitViewController::OnDisplayMetricsChanged(
const display::Display& display,
uint32_t metrics) {
if (GetRootWindowSettings(root_window_)->display_id != display.id())
return;
const bool is_previous_layout_right_side_up =
is_previous_layout_right_side_up_;
is_previous_layout_right_side_up_ = IsLayoutPrimary(display);
if (!InSplitViewMode())
return;
if ((primary_window_ && !CanSnapWindow(primary_window_)) ||
(secondary_window_ && !CanSnapWindow(secondary_window_))) {
if (!Shell::Get()->session_controller()->IsUserSessionBlocked())
EndSplitView();
return;
}
if (split_view_type_ == SplitViewType::kClamshellType)
return;
if (IsDividerAnimating()) {
StopAndShoveAnimatedDivider();
EndResizeWithDividerImpl();
}
if ((metrics & display::DisplayObserver::DISPLAY_METRIC_ROTATION) ||
(metrics & display::DisplayObserver::DISPLAY_METRIC_WORK_AREA)) {
if (std::isnan(divider_closest_ratio_))
divider_closest_ratio_ = kFixedPositionRatios[1];
if (is_previous_layout_right_side_up != IsLayoutPrimary(display))
divider_closest_ratio_ = 1.f - divider_closest_ratio_;
divider_position_ =
static_cast<int>(divider_closest_ratio_ * GetDividerEndPosition()) -
kSplitviewDividerShortSideLength / 2;
}
if (!is_resizing_with_divider_) {
divider_position_ = GetClosestFixedDividerPosition();
}
EndSplitViewAfterResizingAtEdgeIfAppropriate();
if (!InSplitViewMode())
return;
NotifyDividerPositionChanged();
UpdateSnappedWindowsAndDividerBounds();
}
void SplitViewController::OnTabletModeStarting() {
split_view_type_ = SplitViewType::kTabletType;
}
void SplitViewController::OnTabletModeStarted() {
is_previous_layout_right_side_up_ = IsCurrentScreenOrientationPrimary();
if (InSplitViewMode()) {
divider_position_ = GetClosestFixedDividerPosition();
if (!split_view_divider_) {
split_view_divider_ = std::make_unique<SplitViewDivider>(this);
}
UpdateSnappedWindowsAndDividerBounds();
NotifyDividerPositionChanged();
}
}
void SplitViewController::OnTabletModeEnding() {
split_view_type_ = SplitViewType::kClamshellType;
const bool is_divider_animating = IsDividerAnimating();
if (is_resizing_with_divider_ || is_divider_animating) {
is_resizing_with_divider_ = false;
if (is_divider_animating) {
StopAndShoveAnimatedDivider();
}
EndResizeWithDividerImpl();
}
if (state_ != State::kBothSnapped) {
split_view_divider_.reset();
}
}
void SplitViewController::OnTabletModeEnded() {
is_previous_layout_right_side_up_ = true;
}
void SplitViewController::OnAccessibilityStatusChanged() {
if (Shell::Get()->accessibility_controller()->spoken_feedback().enabled())
EndSplitView();
}
void SplitViewController::OnAccessibilityControllerShutdown() {
Shell::Get()->accessibility_controller()->RemoveObserver(this);
}
void SplitViewController::OnKeyboardOccludedBoundsChanged(
const gfx::Rect& screen_bounds) {
if (!features::IsAdjustSplitViewForVKEnabled())
return;
if (IsLayoutHorizontal(root_window_))
return;
aura::Window* bottom_window = GetPhysicalRightOrBottomWindow();
if (!bottom_window &&
!bottom_window->Contains(window_util::GetActiveWindow())) {
return;
}
if (screen_bounds.IsEmpty()) {
UpdateSnappedWindowsAndDividerBounds();
return;
}
auto* text_input_client = GetCurrentInputMethod()->GetTextInputClient();
if (!text_input_client)
return;
const gfx::Rect caret_bounds = text_input_client->GetCaretBounds();
if (caret_bounds == gfx::Rect())
return;
const int keyboard_occluded_y = screen_bounds.y();
if (keyboard_occluded_y - caret_bounds.bottom() > kMinCaretKeyboardDist)
return;
gfx::Rect bottom_bounds = bottom_window->GetBoundsInScreen();
const gfx::Rect work_area =
screen_util::GetDisplayWorkAreaBoundsInScreenForActiveDeskContainer(
root_window_);
const int y =
std::max(keyboard_occluded_y - bottom_bounds.height(),
static_cast<int>(work_area.y() +
work_area.height() * kMinDividerPositionRatio));
bottom_bounds.set_y(y);
bottom_bounds.set_height(keyboard_occluded_y - y);
int divider_position = y - kSplitviewDividerShortSideLength;
{
base::AutoReset<bool> enable_bounds_change(&changing_bounds_by_vk_, true);
bottom_window->SetBoundsInScreen(
bottom_bounds,
display::Screen::GetScreen()->GetDisplayNearestWindow(root_window_));
}
split_view_divider_->divider_widget()->SetBounds(
SplitViewDivider::GetDividerBoundsInScreen(work_area, false,
divider_position,
false));
split_view_divider_->SetAdjustable(false);
}
void SplitViewController::OnWindowActivated(ActivationReason reason,
aura::Window* gained_active,
aura::Window* lost_active) {
if (!features::IsAdjustSplitViewForVKEnabled()) {
return;
}
if (!split_view_divider_ || split_view_divider_->IsAdjustable()) {
return;
}
if (IsLayoutHorizontal(root_window_)) {
return;
}
aura::Window* bottom_window = GetPhysicalRightOrBottomWindow();
if (!bottom_window)
return;
if (bottom_window->Contains(lost_active) &&
!bottom_window->Contains(gained_active)) {
UpdateSnappedWindowsAndDividerBounds();
}
}
void SplitViewController::OnSnapGroupCreated() {
CHECK(IsSnapGroupEnabledInClamshellMode());
CreateSplitViewDividerInClamshell();
}
void SplitViewController::OnSnapGroupRemoved() {
CHECK(IsSnapGroupEnabledInClamshellMode());
split_view_divider_.reset();
}
aura::Window* SplitViewController::GetPhysicalLeftOrTopWindow() {
DCHECK(root_window_);
return IsLayoutPrimary(root_window_) ? primary_window_.get()
: secondary_window_.get();
}
aura::Window* SplitViewController::GetPhysicalRightOrBottomWindow() {
DCHECK(root_window_);
return IsLayoutPrimary(root_window_) ? secondary_window_.get()
: primary_window_.get();
}
void SplitViewController::StartObserving(aura::Window* window) {
if (window && !window->HasObserver(this)) {
Shell::Get()->shadow_controller()->UpdateShadowForWindow(window);
window->AddObserver(this);
WindowState::Get(window)->AddObserver(this);
if (split_view_divider_) {
split_view_divider_->AddObservedWindow(window);
}
}
}
void SplitViewController::StopObserving(SnapPosition snap_position) {
aura::Window* window = GetSnappedWindow(snap_position);
if (window == primary_window_) {
primary_window_ = nullptr;
} else {
secondary_window_ = nullptr;
}
if (window && window->HasObserver(this)) {
window->RemoveObserver(this);
WindowState::Get(window)->RemoveObserver(this);
if (split_view_divider_)
split_view_divider_->RemoveObservedWindow(window);
Shell::Get()->shadow_controller()->UpdateShadowForWindow(window);
RestoreTransformIfApplicable(window);
}
}
void SplitViewController::UpdateStateAndNotifyObservers() {
State previous_state = state_;
if (IsSnapped(primary_window_) && IsSnapped(secondary_window_)) {
state_ = State::kBothSnapped;
} else if (IsSnapped(primary_window_)) {
state_ = State::kPrimarySnapped;
} else if (IsSnapped(secondary_window_)) {
state_ = State::kSecondarySnapped;
} else {
state_ = State::kNoSnap;
}
DCHECK(previous_state != State::kNoSnap || state_ != State::kNoSnap);
for (auto& observer : observers_) {
observer.OnSplitViewStateChanged(previous_state, state_);
}
}
void SplitViewController::NotifyDividerPositionChanged() {
for (auto& observer : observers_)
observer.OnSplitViewDividerPositionChanged();
}
void SplitViewController::NotifyWindowResized() {
for (auto& observer : observers_)
observer.OnSplitViewWindowResized();
}
void SplitViewController::NotifyWindowSwapped() {
for (auto& observer : observers_)
observer.OnSplitViewWindowSwapped();
}
void SplitViewController::CreateSplitViewDividerInClamshell() {
CHECK(InClamshellSplitViewMode());
divider_position_ = GetClosestFixedDividerPosition();
split_view_divider_ = std::make_unique<SplitViewDivider>(this);
UpdateSnappedWindowsAndDividerBounds();
NotifyDividerPositionChanged();
}
void SplitViewController::UpdateBlackScrim(
const gfx::Point& location_in_screen) {
DCHECK(InSplitViewMode());
if (!black_scrim_layer_) {
black_scrim_layer_ = std::make_unique<ui::Layer>(ui::LAYER_SOLID_COLOR);
black_scrim_layer_->SetColor(AshColorProvider::Get()->GetBackgroundColor());
auto* divider_layer =
split_view_divider_->divider_widget()->GetNativeWindow()->layer();
auto* divider_parent_layer = divider_layer->parent();
divider_parent_layer->Add(black_scrim_layer_.get());
divider_parent_layer->StackBelow(black_scrim_layer_.get(), divider_layer);
}
SnapPosition position = GetBlackScrimPosition(location_in_screen);
if (position == SnapPosition::kNone) {
black_scrim_layer_.reset();
return;
}
black_scrim_layer_->SetBounds(GetSnappedWindowBoundsInScreen(
position, nullptr));
const int location = IsLayoutHorizontal(root_window_)
? location_in_screen.x()
: location_in_screen.y();
gfx::Rect work_area_bounds =
screen_util::GetDisplayWorkAreaBoundsInScreenForActiveDeskContainer(
root_window_);
if (!IsLayoutHorizontal(root_window_))
work_area_bounds.Transpose();
float opacity = kBlackScrimOpacity;
const float ratio = chromeos::kOneThirdSnapRatio - kBlackScrimFadeInRatio;
const int distance = std::min(std::abs(location - work_area_bounds.x()),
std::abs(work_area_bounds.right() - location));
if (distance > work_area_bounds.width() * ratio) {
opacity -= kBlackScrimOpacity *
(distance - work_area_bounds.width() * ratio) /
(work_area_bounds.width() * kBlackScrimFadeInRatio);
opacity = std::max(opacity, 0.f);
}
black_scrim_layer_->SetOpacity(opacity);
}
void SplitViewController::UpdateResizeBackdrop() {
auto create_backdrop = [](aura::Window* window) {
auto resize_backdrop_layer =
std::make_unique<ui::Layer>(ui::LAYER_SOLID_COLOR);
ui::Layer* parent = window->layer()->parent();
ui::Layer* stacking_target = window->layer();
parent->Add(resize_backdrop_layer.get());
parent->StackBelow(resize_backdrop_layer.get(), stacking_target);
return resize_backdrop_layer;
};
auto update_backdrop = [this](SnapPosition position, aura::Window* window,
ui::Layer* backdrop) {
backdrop->SetBounds(GetSnappedWindowBoundsInParent(position, nullptr));
backdrop->SetColor(window->GetProperty(
wm::IsActiveWindow(window) ? chromeos::kFrameActiveColorKey
: chromeos::kFrameInactiveColorKey));
};
if (state_ == State::kPrimarySnapped || state_ == State::kBothSnapped) {
if (!left_resize_backdrop_layer_)
left_resize_backdrop_layer_ = create_backdrop(primary_window_);
update_backdrop(SnapPosition::kPrimary, primary_window_,
left_resize_backdrop_layer_.get());
}
if (state_ == State::kSecondarySnapped || state_ == State::kBothSnapped) {
if (!right_resize_backdrop_layer_)
right_resize_backdrop_layer_ = create_backdrop(secondary_window_);
update_backdrop(SnapPosition::kSecondary, secondary_window_,
right_resize_backdrop_layer_.get());
}
}
void SplitViewController::UpdateSnappedWindowsAndDividerBounds() {
if (IsSnapped(primary_window_)) {
UpdateSnappedBounds(primary_window_);
}
if (IsSnapped(secondary_window_)) {
UpdateSnappedBounds(secondary_window_);
}
if (split_view_divider_) {
split_view_divider_->UpdateDividerBounds();
if (features::IsAdjustSplitViewForVKEnabled()) {
split_view_divider_->SetAdjustable(true);
}
}
}
void SplitViewController::UpdateSnappedBounds(aura::Window* window) {
DCHECK(IsWindowInSplitView(window));
WindowState* window_state = WindowState::Get(window);
const bool in_tablet_mode = IsInTabletMode();
if (window->GetProperty(aura::client::kAppType) ==
static_cast<int>(AppType::ARC_APP)) {
const gfx::Rect requested_bounds =
in_tablet_mode
? TabletModeWindowState::GetBoundsInTabletMode(window_state)
: GetSnappedWindowBoundsInScreen(GetPositionOfSnappedWindow(window),
window);
const SetBoundsWMEvent event(requested_bounds,
true);
window_state->OnWMEvent(&event);
return;
}
if (in_tablet_mode) {
TabletModeWindowState::UpdateWindowPosition(
window_state, WindowState::BoundsChangeAnimationType::kAnimate);
return;
} else if (IsSnapGroupEnabledInClamshellMode()) {
const gfx::Rect requested_bounds = GetSnappedWindowBoundsInScreen(
GetPositionOfSnappedWindow(window), window);
const SetBoundsWMEvent event(requested_bounds, true);
window_state->OnWMEvent(&event);
return;
}
window_state->UpdateSnappedBounds();
}
SplitViewController::SnapPosition SplitViewController::GetBlackScrimPosition(
const gfx::Point& location_in_screen) {
const gfx::Rect work_area_bounds =
screen_util::GetDisplayWorkAreaBoundsInScreenForActiveDeskContainer(
root_window_);
if (!work_area_bounds.Contains(location_in_screen))
return SnapPosition::kNone;
gfx::Size primary_window_min_size, secondary_window_min_size;
if (primary_window_ && primary_window_->delegate())
primary_window_min_size = primary_window_->delegate()->GetMinimumSize();
if (secondary_window_ && secondary_window_->delegate())
secondary_window_min_size = secondary_window_->delegate()->GetMinimumSize();
bool right_side_up = IsLayoutPrimary(root_window_);
int divider_end_position = GetDividerEndPosition();
int primary_window_distance = 0, secondary_window_distance = 0;
int min_left_length = 0, min_right_length = 0;
if (IsLayoutHorizontal(root_window_)) {
int left_distance = location_in_screen.x() - work_area_bounds.x();
int right_distance = work_area_bounds.right() - location_in_screen.x();
primary_window_distance = right_side_up ? left_distance : right_distance;
secondary_window_distance = right_side_up ? right_distance : left_distance;
min_left_length = primary_window_min_size.width();
min_right_length = secondary_window_min_size.width();
} else {
int top_distance = location_in_screen.y() - work_area_bounds.y();
int bottom_distance = work_area_bounds.bottom() - location_in_screen.y();
primary_window_distance = right_side_up ? top_distance : bottom_distance;
secondary_window_distance = right_side_up ? bottom_distance : top_distance;
min_left_length = primary_window_min_size.height();
min_right_length = secondary_window_min_size.height();
}
if (primary_window_distance <
divider_end_position * chromeos::kOneThirdSnapRatio ||
primary_window_distance < min_left_length) {
return SnapPosition::kPrimary;
}
if (secondary_window_distance <
divider_end_position * chromeos::kOneThirdSnapRatio ||
secondary_window_distance < min_right_length) {
return SnapPosition::kSecondary;
}
return SnapPosition::kNone;
}
void SplitViewController::UpdateDividerPosition(
const gfx::Point& location_in_screen) {
if (IsLayoutHorizontal(root_window_))
divider_position_ += location_in_screen.x() - previous_event_location_.x();
else
divider_position_ += location_in_screen.y() - previous_event_location_.y();
divider_position_ = std::max(0, divider_position_);
}
int SplitViewController::GetClosestFixedDividerPosition() {
int divider_end_position = GetDividerEndPosition();
divider_closest_ratio_ = FindClosestPositionRatio(
divider_position_ + kSplitviewDividerShortSideLength / 2,
divider_end_position);
int fixed_position = divider_end_position * divider_closest_ratio_;
if (divider_closest_ratio_ > 0.f && divider_closest_ratio_ < 1.f) {
fixed_position -= kSplitviewDividerShortSideLength / 2;
}
return fixed_position;
}
void SplitViewController::StopAndShoveAnimatedDivider() {
DCHECK(IsDividerAnimating());
divider_snap_animation_->Stop();
divider_position_ = divider_snap_animation_->ending_position();
NotifyDividerPositionChanged();
UpdateSnappedWindowsAndDividerBounds();
}
bool SplitViewController::ShouldEndSplitViewAfterResizingAtEdge() {
DCHECK(InTabletSplitViewMode() || IsSnapGroupEnabledInClamshellMode());
return divider_position_ == 0 || divider_position_ == GetDividerEndPosition();
}
void SplitViewController::EndSplitViewAfterResizingAtEdgeIfAppropriate() {
if (!ShouldEndSplitViewAfterResizingAtEdge()) {
return;
}
aura::Window* active_window = GetActiveWindowAfterResizingUponExit();
aura::Window* insert_overview_window = nullptr;
if (IsInOverviewSession()) {
insert_overview_window = GetDefaultSnappedWindow();
}
EndSplitView();
if (active_window) {
Shell::Get()->overview_controller()->EndOverview(
OverviewEndAction::kSplitView);
wm::ActivateWindow(active_window);
} else if (insert_overview_window) {
InsertWindowToOverview(insert_overview_window, false);
}
}
aura::Window* SplitViewController::GetActiveWindowAfterResizingUponExit() {
DCHECK(InSplitViewMode());
if (!ShouldEndSplitViewAfterResizingAtEdge()) {
return nullptr;
}
return divider_position_ == 0 ? GetPhysicalRightOrBottomWindow()
: GetPhysicalLeftOrTopWindow();
}
int SplitViewController::GetDividerEndPosition() const {
const gfx::Rect work_area_bounds =
screen_util::GetDisplayWorkAreaBoundsInScreenForActiveDeskContainer(
root_window_);
return IsLayoutHorizontal(root_window_) ? work_area_bounds.width()
: work_area_bounds.height();
}
void SplitViewController::OnWindowSnapped(
aura::Window* window,
absl::optional<chromeos::WindowStateType> previous_state) {
RestoreTransformIfApplicable(window);
UpdateStateAndNotifyObservers();
SnapGroupController* snap_group_controller =
Shell::Get()->snap_group_controller();
if (state_ == State::kBothSnapped && IsSnapGroupEnabledInClamshellMode() &&
snap_group_controller->IsArm1AutomaticallyLockEnabled()) {
snap_group_controller->AddSnapGroup(primary_window_, secondary_window_);
DCHECK(snap_group_controller->AreWindowsInSnapGroup(primary_window_,
secondary_window_));
}
if (to_be_activated_window_ == window) {
to_be_activated_window_ = nullptr;
wm::ActivateWindow(window);
}
if (previous_state &&
*previous_state == chromeos::WindowStateType::kFloated &&
IsInTabletMode()) {
auto mru_windows =
Shell::Get()->mru_window_tracker()->BuildWindowForCycleList(
kActiveDesk);
for (aura::Window* mru_window : mru_windows) {
auto* window_state = WindowState::Get(mru_window);
if (mru_window != window && !window_state->IsMinimized() &&
window_state->CanSnap()) {
const SnapPosition snap_position =
GetPositionOfSnappedWindow(window) == SnapPosition::kPrimary
? SnapPosition::kSecondary
: SnapPosition::kPrimary;
WMEvent event(snap_position == SnapPosition::kPrimary
? WM_EVENT_SNAP_PRIMARY
: WM_EVENT_SNAP_SECONDARY);
WindowState::Get(mru_window)->OnWMEvent(&event);
return;
}
}
}
if (!IsInOverviewSession() &&
(split_view_type_ == SplitViewType::kTabletType ||
ShouldShowOverviewInClamshellOnWindowSnapped()) &&
(state_ == State::kPrimarySnapped ||
state_ == State::kSecondarySnapped)) {
if (!DesksController::Get()->animation()) {
Shell::Get()->overview_controller()->StartOverview(
OverviewStartAction::kSplitView, OverviewEnterExitType::kNormal);
}
return;
}
if (primary_window_ && secondary_window_) {
UpdateSnappedBounds(window == primary_window_ ? secondary_window_.get()
: primary_window_.get());
}
}
void SplitViewController::OnSnappedWindowDetached(aura::Window* window,
WindowDetachedReason reason) {
const bool is_window_destroyed =
reason == WindowDetachedReason::kWindowDestroyed;
const SnapPosition position_of_snapped_window =
GetPositionOfSnappedWindow(window);
if (is_window_destroyed) {
StopObserving(position_of_snapped_window);
}
const bool is_divider_animating = IsDividerAnimating();
if (is_resizing_with_divider_ || is_divider_animating) {
is_resizing_with_divider_ = false;
if (is_divider_animating) {
StopAndShoveAnimatedDivider();
}
EndResizeWithDividerImpl();
}
if (!is_window_destroyed) {
StopObserving(position_of_snapped_window);
}
if ((!primary_window_ && !secondary_window_) ||
(IsSnapGroupEnabledInClamshellMode() &&
(!primary_window_ || !secondary_window_))) {
EndSplitView(reason == WindowDetachedReason::kWindowDragged
? EndReason::kWindowDragStarted
: EndReason::kNormal);
} else {
DCHECK(InTabletSplitViewMode());
if (reason == WindowDetachedReason::kWindowFloated && IsInTabletMode()) {
aura::Window* other_window =
GetSnappedWindow(position_of_snapped_window == SnapPosition::kPrimary
? SnapPosition::kSecondary
: SnapPosition::kPrimary);
WMEvent event(WM_EVENT_MAXIMIZE);
WindowState::Get(other_window)->OnWMEvent(&event);
return;
}
default_snap_position_ =
primary_window_ ? SnapPosition::kPrimary : SnapPosition::kSecondary;
UpdateStateAndNotifyObservers();
Shell::Get()->overview_controller()->StartOverview(
OverviewStartAction::kSplitView,
reason == WindowDetachedReason::kWindowDragged
? OverviewEnterExitType::kImmediateEnter
: OverviewEnterExitType::kNormal);
}
}
float SplitViewController::FindClosestPositionRatio(float distance,
float length) {
const float current_ratio = distance / length;
float closest_ratio = 0.f;
std::vector<float> position_ratios(
kFixedPositionRatios,
kFixedPositionRatios + sizeof(kFixedPositionRatios) / sizeof(float));
ModifyPositionRatios(&position_ratios);
for (const float ratio : position_ratios) {
if (std::abs(current_ratio - ratio) <
std::abs(current_ratio - closest_ratio)) {
closest_ratio = ratio;
}
}
return closest_ratio;
}
void SplitViewController::ModifyPositionRatios(
std::vector<float>* out_position_ratios) {
const bool landscape = IsCurrentScreenOrientationLandscape();
const int min_left_size =
GetMinimumWindowLength(GetPhysicalLeftOrTopWindow(), landscape);
const int min_right_size =
GetMinimumWindowLength(GetPhysicalRightOrBottomWindow(), landscape);
const int divider_end_position = GetDividerEndPosition();
const float min_size_left_ratio =
static_cast<float>(min_left_size) / divider_end_position;
const float min_size_right_ratio =
static_cast<float>(min_right_size) / divider_end_position;
if (min_size_left_ratio <= chromeos::kOneThirdSnapRatio) {
out_position_ratios->push_back(chromeos::kOneThirdSnapRatio);
}
if (min_size_right_ratio <= chromeos::kOneThirdSnapRatio) {
out_position_ratios->push_back(chromeos::kTwoThirdSnapRatio);
}
if (min_size_left_ratio > chromeos::kDefaultSnapRatio ||
min_size_right_ratio > chromeos::kDefaultSnapRatio) {
base::Erase(*out_position_ratios, chromeos::kDefaultSnapRatio);
}
}
int SplitViewController::GetWindowComponentForResize(aura::Window* window) {
DCHECK(IsWindowInSplitView(window));
return window == primary_window_ ? HTRIGHT : HTLEFT;
}
gfx::Point SplitViewController::GetEndDragLocationInScreen(
aura::Window* window,
const gfx::Point& location_in_screen) {
gfx::Point end_location(location_in_screen);
if (!IsWindowInSplitView(window))
return end_location;
const gfx::Rect bounds = GetSnappedWindowBoundsInScreen(
GetPositionOfSnappedWindow(window), window);
if (IsLayoutHorizontal(window)) {
end_location.set_x(window == GetPhysicalLeftOrTopWindow() ? bounds.right()
: bounds.x());
} else {
end_location.set_y(window == GetPhysicalLeftOrTopWindow() ? bounds.bottom()
: bounds.y());
}
return end_location;
}
void SplitViewController::RestoreTransformIfApplicable(aura::Window* window) {
auto iter = snapping_window_transformed_bounds_map_.find(window);
if (iter == snapping_window_transformed_bounds_map_.end())
return;
const gfx::Rect item_bounds = iter->second;
snapping_window_transformed_bounds_map_.erase(iter);
if (!window->layer()->GetTargetTransform().IsIdentity()) {
const gfx::Rect snapped_bounds = GetSnappedWindowBoundsInScreen(
GetPositionOfSnappedWindow(window), window);
const gfx::Transform starting_transform = gfx::TransformBetweenRects(
gfx::RectF(snapped_bounds), gfx::RectF(item_bounds));
SetTransformWithAnimation(window, starting_transform, gfx::Transform());
}
}
void SplitViewController::SetWindowsTransformDuringResizing() {
DCHECK(InTabletSplitViewMode() || IsSnapGroupEnabledInClamshellMode());
DCHECK_GE(divider_position_, 0);
const bool horizontal = IsLayoutHorizontal(root_window_);
aura::Window* left_or_top_window = GetPhysicalLeftOrTopWindow();
aura::Window* right_or_bottom_window = GetPhysicalRightOrBottomWindow();
gfx::Transform left_or_top_transform;
if (left_or_top_window) {
const int left_size = divider_position_;
const int distance =
left_size - GetWindowLength(left_or_top_window, horizontal);
if (distance < 0) {
left_or_top_transform.Translate(horizontal ? distance : 0,
horizontal ? 0 : distance);
}
SetTransform(left_or_top_window, left_or_top_transform);
}
gfx::Transform right_or_bottom_transform;
if (right_or_bottom_window) {
const int right_size = GetDividerEndPosition() - divider_position_ -
kSplitviewDividerShortSideLength;
const int distance =
right_size - GetWindowLength(right_or_bottom_window, horizontal);
if (distance < 0) {
right_or_bottom_transform.Translate(horizontal ? -distance : 0,
horizontal ? 0 : -distance);
}
SetTransform(right_or_bottom_window, right_or_bottom_transform);
}
}
void SplitViewController::RestoreWindowsTransformAfterResizing() {
DCHECK(InSplitViewMode());
if (primary_window_)
SetTransform(primary_window_, gfx::Transform());
if (secondary_window_)
SetTransform(secondary_window_, gfx::Transform());
if (black_scrim_layer_.get())
black_scrim_layer_->SetTransform(gfx::Transform());
}
void SplitViewController::SetTransformWithAnimation(
aura::Window* window,
const gfx::Transform& start_transform,
const gfx::Transform& target_transform) {
const gfx::PointF target_origin = GetTargetBoundsInScreen(window).origin();
for (auto* window_iter : GetTransientTreeIterator(window)) {
aura::Window* parent_window = window_iter->parent();
gfx::RectF original_bounds(window_iter->GetTargetBounds());
::wm::TranslateRectToScreen(parent_window, &original_bounds);
const gfx::PointF pivot(target_origin.x() - original_bounds.x(),
target_origin.y() - original_bounds.y());
const gfx::Transform new_start_transform =
TransformAboutPivot(pivot, start_transform);
const gfx::Transform new_target_transform =
TransformAboutPivot(pivot, target_transform);
if (new_start_transform != window_iter->layer()->GetTargetTransform())
window_iter->SetTransform(new_start_transform);
std::vector<ui::ImplicitAnimationObserver*> animation_observers;
if (window_iter == window) {
animation_observers.push_back(
new WindowTransformAnimationObserver(window));
OverviewController* overview_controller =
Shell::Get()->overview_controller();
if (overview_controller->IsCompletingShutdownAnimations() ||
(overview_controller->overview_session() &&
overview_controller->overview_session()->is_shutting_down() &&
overview_controller->overview_session()
->enter_exit_overview_type() !=
OverviewEnterExitType::kImmediateExit)) {
auto overview_exit_animation_observer =
std::make_unique<ExitAnimationObserver>();
animation_observers.push_back(overview_exit_animation_observer.get());
overview_controller->AddExitAnimationObserver(
std::move(overview_exit_animation_observer));
}
}
DoSplitviewTransformAnimation(window_iter->layer(),
SPLITVIEW_ANIMATION_SET_WINDOW_TRANSFORM,
new_target_transform, animation_observers);
}
}
void SplitViewController::UpdateSnappingWindowTransformedBounds(
aura::Window* window) {
if (!window->layer()->GetTargetTransform().IsIdentity()) {
snapping_window_transformed_bounds_map_[window] = gfx::ToEnclosedRect(
window_util::GetTransformedBounds(window, 0));
}
}
void SplitViewController::InsertWindowToOverview(aura::Window* window,
bool animate) {
if (!window || !GetOverviewSession())
return;
GetOverviewSession()->AddItemInMruOrder(window, true, animate,
true,
false);
}
void SplitViewController::FinishWindowResizing(aura::Window* window) {
if (window != nullptr) {
WindowState* window_state = WindowState::Get(window);
window_state->OnCompleteDrag(gfx::PointF(
GetEndDragLocationInScreen(window, previous_event_location_)));
window_state->DeleteDragDetails();
}
}
void SplitViewController::EndResizeWithDividerImpl() {
DCHECK(InSplitViewMode());
DCHECK(!is_resizing_with_divider_);
left_resize_backdrop_layer_.reset();
right_resize_backdrop_layer_.reset();
resize_timer_.Stop();
presentation_time_recorder_.reset();
RestoreWindowsTransformAfterResizing();
FinishWindowResizing(primary_window_);
FinishWindowResizing(secondary_window_);
}
void SplitViewController::OnResizeTimer() {
if (InSplitViewMode())
ResizeWithDivider(previous_event_location_);
}
void SplitViewController::UpdateTabletResizeMode(
base::TimeTicks event_time_ticks,
const gfx::Point& event_location) {
if (IsLayoutHorizontal(root_window_)) {
accumulated_drag_distance_ +=
std::abs(event_location.x() - previous_event_location_.x());
} else {
accumulated_drag_distance_ +=
std::abs(event_location.y() - previous_event_location_.y());
}
const base::TimeDelta chunk_time_ticks =
event_time_ticks - accumulated_drag_time_ticks_;
if (chunk_time_ticks >= kSplitViewChunkTime) {
int drag_per_second =
accumulated_drag_distance_ / chunk_time_ticks.InSecondsF();
tablet_resize_mode_ = drag_per_second > kSplitViewThresholdPixelsPerSec
? TabletResizeMode::kFast
: TabletResizeMode::kNormal;
accumulated_drag_time_ticks_ = event_time_ticks;
accumulated_drag_distance_ = 0;
}
}
void SplitViewController::EndWindowDragImpl(
aura::Window* window,
bool is_being_destroyed,
SnapPosition desired_snap_position,
const gfx::Point& last_location_in_screen) {
if (split_view_divider_)
split_view_divider_->OnWindowDragEnded();
if (is_being_destroyed)
return;
if (GetOverviewSession() && GetOverviewSession()->IsWindowInOverview(window))
return;
if (WindowState::Get(window)->IsFloated()) {
return;
}
DCHECK_EQ(root_window_, window->GetRootWindow());
const bool was_splitview_active = InSplitViewMode();
if (desired_snap_position == SnapPosition::kNone) {
if (was_splitview_active) {
SnapWindow(window, ComputeSnapPosition(last_location_in_screen),
true);
} else {
SetTransformWithAnimation(window, window->layer()->GetTargetTransform(),
gfx::Transform());
OverviewSession* overview_session = GetOverviewSession();
if (overview_session) {
overview_session->SetWindowListNotAnimatedWhenExiting(root_window_);
overview_session->set_enter_exit_overview_type(
OverviewEnterExitType::kImmediateExit);
}
wm::ActivateWindow(window);
Shell::Get()->overview_controller()->EndOverview(
OverviewEndAction::kSplitView);
TabletModeWindowState::UpdateWindowPosition(
WindowState::Get(window),
WindowState::BoundsChangeAnimationType::kAnimate);
}
} else {
SnapWindow(window, desired_snap_position, true);
}
}
void SplitViewController::DoSplitDividerSpawnAnimation(aura::Window* window) {
DCHECK(window->layer()->GetAnimator()->GetTargetTransform().IsIdentity());
SnapPosition snap_position = GetPositionOfSnappedWindow(window);
const gfx::Rect bounds =
GetSnappedWindowBoundsInScreen(snap_position, window);
gfx::Point p = IsPhysicalLeftOrTop(snap_position, window)
? bounds.bottom_right()
: bounds.origin();
static const double value = gfx::Tween::CalculateValue(
gfx::Tween::FAST_OUT_SLOW_IN,
kSplitviewDividerSpawnDelay / kSplitviewWindowTransformDuration);
p = gfx::TransformAboutPivot(
gfx::PointF(bounds.origin()),
gfx::Tween::TransformValueBetween(value, window->transform(),
gfx::Transform()))
.MapPoint(p);
split_view_divider_->DoSpawningAnimation(IsLayoutHorizontal(window) ? p.x()
: p.y());
}
void SplitViewController::SwapWindowsAndUpdateBounds() {
gfx::Rect primary_window_bounds =
primary_window_ ? primary_window_->GetBoundsInScreen() : gfx::Rect();
gfx::Rect secondary_window_bounds =
secondary_window_ ? secondary_window_->GetBoundsInScreen() : gfx::Rect();
aura::Window* cached_window = primary_window_;
primary_window_ = secondary_window_;
secondary_window_ = cached_window;
const auto dst_display =
display::Screen::GetScreen()->GetDisplayNearestWindow(root_window_);
if (primary_window_) {
primary_window_->SetBoundsInScreen(secondary_window_bounds, dst_display);
}
if (secondary_window_) {
secondary_window_->SetBoundsInScreen(primary_window_bounds, dst_display);
}
}
}