// Copyright 2018 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chromeos/ui/frame/frame_utils.h"
#include "chromeos/constants/chromeos_features.h"
#include "chromeos/ui/base/chromeos_ui_constants.h"
#include "chromeos/ui/base/display_util.h"
#include "chromeos/ui/base/window_properties.h"
#include "chromeos/ui/base/window_state_type.h"
#include "ui/aura/env.h"
#include "ui/aura/window.h"
#include "ui/base/hit_test.h"
#include "ui/display/screen.h"
#include "ui/gfx/geometry/point.h"
#include "ui/views/widget/widget.h"
#include "ui/views/widget/widget_delegate.h"
#include "ui/views/window/hit_test_utils.h"
#include "ui/views/window/non_client_view.h"
namespace chromeos {
using WindowOpacity = views::Widget::InitParams::WindowOpacity;
int FrameBorderNonClientHitTest(views::NonClientFrameView* view,
const gfx::Point& point_in_widget) {
gfx::Rect expanded_bounds = view->bounds();
int outside_bounds = chromeos::kResizeOutsideBoundsSize;
if (aura::Env::GetInstance()->is_touch_down())
outside_bounds *= chromeos::kResizeOutsideBoundsScaleForTouch;
expanded_bounds.Inset(-outside_bounds);
if (!expanded_bounds.Contains(point_in_widget))
return HTNOWHERE;
// Check the frame first, as we allow a small area overlapping the contents
// to be used for resize handles.
views::Widget* widget = view->GetWidget();
bool in_tablet_mode = display::Screen::GetScreen()->InTabletMode();
// Ignore the resize border when maximized or full screen or in (split view)
// tablet mode.
const bool has_resize_border =
!widget->IsMaximized() && !widget->IsFullscreen() && !in_tablet_mode;
const int resize_border_size =
has_resize_border ? chromeos::kResizeInsideBoundsSize : 0;
int frame_component = view->GetHTComponentForFrame(
point_in_widget, gfx::Insets(resize_border_size),
chromeos::kResizeAreaCornerSize, chromeos::kResizeAreaCornerSize,
has_resize_border);
if (frame_component != HTNOWHERE)
return frame_component;
int client_component =
widget->client_view()->NonClientHitTest(point_in_widget);
if (client_component != HTNOWHERE)
return client_component;
// Check if it intersects with children (frame caption button, back button,
// etc.).
int hit_test_component =
views::GetHitTestComponent(widget->non_client_view(), point_in_widget);
if (hit_test_component != HTNOWHERE)
return hit_test_component;
// Caption is a safe default.
return HTCAPTION;
}
void ResolveInferredOpacity(views::Widget::InitParams* params) {
DCHECK_EQ(params->opacity, WindowOpacity::kInferred);
if (params->type == views::Widget::InitParams::TYPE_WINDOW &&
params->layer_type == ui::LAYER_TEXTURED) {
// A framed window may have a rounded corner which requires the
// window to be transparent. WindowManager controls the actual
// opaque-ness of the window depending on its window state.
params->init_properties_container.SetProperty(
chromeos::kWindowManagerManagesOpacityKey, true);
params->opacity = WindowOpacity::kTranslucent;
} else {
params->opacity = WindowOpacity::kOpaque;
}
}
bool ShouldUseRestoreFrame(const views::Widget* widget) {
aura::Window* window = widget->GetNativeWindow();
// This is true when dragging a maximized window in ash. During this phase,
// the window should look as if it was restored, but keep its maximized state.
if (window->GetProperty(chromeos::kFrameRestoreLookKey))
return true;
// Maximized and fullscreen windows should use the maximized frame.
if (widget->IsMaximized() || widget->IsFullscreen())
return false;
return true;
}
SnapDirection GetSnapDirectionForWindow(aura::Window* window, bool left_top) {
const bool is_primary_display_layout = chromeos::IsDisplayLayoutPrimary(
display::Screen::GetScreen()->GetDisplayNearestWindow(window));
if (left_top) {
return is_primary_display_layout ? SnapDirection::kPrimary
: SnapDirection::kSecondary;
} else {
return is_primary_display_layout ? SnapDirection::kSecondary
: SnapDirection::kPrimary;
}
}
int GetFrameCornerRadius(const aura::Window* native_window) {
if (!ShouldWindowHaveRoundedCorners(native_window)) {
return 0;
}
const WindowStateType window_state =
native_window->GetProperty(kWindowStateTypeKey);
if (window_state == WindowStateType::kPip) {
return kPipRoundedCornerRadius;
}
return features::IsRoundedWindowsEnabled() ? features::RoundedWindowsRadius()
: kTopCornerRadiusWhenRestored;
}
bool CanPropertyEffectFrameRadius(const void* class_property_key) {
return class_property_key == kIsShowingInOverviewKey ||
class_property_key == kWindowStateTypeKey;
}
bool ShouldWindowStateHaveRoundedCorners(WindowStateType type) {
return IsNormalWindowStateType(type) || type == WindowStateType::kFloated ||
type == WindowStateType::kPip;
}
bool ShouldWindowHaveRoundedCorners(const aura::Window* native_window) {
const WindowStateType window_state =
native_window->GetProperty(kWindowStateTypeKey);
// In overview mode, the native window is displayed in `ash::WindowMiniView`
// with its own `ash::WindowMiniViewHeaderView`. This mini view has its own
// rounded corners. Therefore we do not need to round the native window.
// Apart from redundant rounding, rounding the native frame is problematic for
// browsers. For packaged apps, we hide the frame header but for browsers, we
// still show the header since the tab strip is rendered over the header. In
// overview mode, the header becomes a part of contents of WindowMiniView and
// rounding the header ends up rounding the top corners of the contents.
const bool in_overview = native_window->GetProperty(kIsShowingInOverviewKey);
return ShouldWindowStateHaveRoundedCorners(window_state) && !in_overview;
}
} // namespace chromeos