Commit ff9d9941 authored by estade's avatar estade Committed by Commit bot

Apply new MD shadows to CrOS tray bubbles.

* affects system menu, message center, palette bubble, ime bubble, etc.
* WM shadow aperture calculation is adjusted to work better for small
  windows (like the message center when it's empty). There's a TODO
  here which will have to wait for Sebastien to get back so he can give
  feedback on what to do in this case.
* fix alignment of NO_ASSETS bubble --- don't account for
  non-existent stroke.
* existing BubbleBorder tests applied to more different kinds of bubble

BUG=608852

Review-Url: https://codereview.chromium.org/2555373004
Cr-Commit-Position: refs/heads/master@{#438196}
parent 479f44fc
...@@ -534,20 +534,12 @@ views::View* TrayBackgroundView::GetBubbleAnchor() const { ...@@ -534,20 +534,12 @@ views::View* TrayBackgroundView::GetBubbleAnchor() const {
gfx::Insets TrayBackgroundView::GetBubbleAnchorInsets() const { gfx::Insets TrayBackgroundView::GetBubbleAnchorInsets() const {
gfx::Insets anchor_insets = GetBubbleAnchor()->GetInsets(); gfx::Insets anchor_insets = GetBubbleAnchor()->GetInsets();
gfx::Insets tray_bg_insets = GetInsets(); gfx::Insets tray_bg_insets = GetInsets();
// TODO(estade): for reasons I don't understand, BubbleBorder distances the
// bubble by the arrow's "interior" thickness even when the paint type is
// PAINT_NONE.
const int kBigShadowArrowInteriorThickness = 9;
if (GetAnchorAlignment() == TrayBubbleView::ANCHOR_ALIGNMENT_BOTTOM) { if (GetAnchorAlignment() == TrayBubbleView::ANCHOR_ALIGNMENT_BOTTOM) {
return gfx::Insets(kBigShadowArrowInteriorThickness - tray_bg_insets.top(), return gfx::Insets(-tray_bg_insets.top(), anchor_insets.left(),
anchor_insets.left(), -tray_bg_insets.bottom(), -tray_bg_insets.bottom(), anchor_insets.right());
anchor_insets.right());
} else { } else {
return gfx::Insets( return gfx::Insets(anchor_insets.top(), -tray_bg_insets.left(),
anchor_insets.top(), anchor_insets.bottom(), -tray_bg_insets.right());
kBigShadowArrowInteriorThickness - tray_bg_insets.left(),
anchor_insets.bottom(),
kBigShadowArrowInteriorThickness - tray_bg_insets.right());
} }
} }
......
...@@ -255,10 +255,12 @@ gfx::Rect BubbleBorder::GetBounds(const gfx::Rect& anchor_rect, ...@@ -255,10 +255,12 @@ gfx::Rect BubbleBorder::GetBounds(const gfx::Rect& anchor_rect,
int h = anchor_rect.height(); int h = anchor_rect.height();
const gfx::Size size(GetSizeForContentsSize(contents_size)); const gfx::Size size(GetSizeForContentsSize(contents_size));
const int arrow_offset = GetArrowOffset(size); const int arrow_offset = GetArrowOffset(size);
const int stroke_width = shadow_ == NO_ASSETS ? 0 : kStroke;
// |arrow_shift| is necessary to visually align the tip of the bubble arrow // |arrow_shift| is necessary to visually align the tip of the bubble arrow
// with the anchor point. This shift is an inverse of the shadow thickness. // with the anchor point. This shift is an inverse of the shadow thickness.
int arrow_shift = UseMd() ? 0 : int arrow_shift = UseMd() ? 0
images_->arrow_interior_thickness + kStroke - images_->arrow_thickness; : images_->arrow_interior_thickness + stroke_width -
images_->arrow_thickness;
// When arrow is painted transparently the visible border of the bubble needs // When arrow is painted transparently the visible border of the bubble needs
// to be positioned at the same bounds as when the arrow is shown. // to be positioned at the same bounds as when the arrow is shown.
if (arrow_paint_type_ == PAINT_TRANSPARENT) if (arrow_paint_type_ == PAINT_TRANSPARENT)
...@@ -268,12 +270,13 @@ gfx::Rect BubbleBorder::GetBounds(const gfx::Rect& anchor_rect, ...@@ -268,12 +270,13 @@ gfx::Rect BubbleBorder::GetBounds(const gfx::Rect& anchor_rect,
// Calculate the bubble coordinates based on the border and arrow settings. // Calculate the bubble coordinates based on the border and arrow settings.
if (is_arrow_on_horizontal(arrow_)) { if (is_arrow_on_horizontal(arrow_)) {
if (is_arrow_on_left(arrow_)) { if (is_arrow_on_left(arrow_)) {
x += mid_anchor ? w / 2 - arrow_offset : kStroke - GetBorderThickness(); x += mid_anchor ? w / 2 - arrow_offset
: stroke_width - GetBorderThickness();
} else if (is_arrow_at_center(arrow_)) { } else if (is_arrow_at_center(arrow_)) {
x += w / 2 - arrow_offset; x += w / 2 - arrow_offset;
} else { } else {
x += mid_anchor ? w / 2 + arrow_offset - size.width() : x += mid_anchor ? w / 2 + arrow_offset - size.width()
w - size.width() + GetBorderThickness() - kStroke; : w - size.width() + GetBorderThickness() - stroke_width;
} }
y += is_arrow_on_top(arrow_) ? h + arrow_shift y += is_arrow_on_top(arrow_) ? h + arrow_shift
: -arrow_shift - size.height(); : -arrow_shift - size.height();
...@@ -281,12 +284,13 @@ gfx::Rect BubbleBorder::GetBounds(const gfx::Rect& anchor_rect, ...@@ -281,12 +284,13 @@ gfx::Rect BubbleBorder::GetBounds(const gfx::Rect& anchor_rect,
x += is_arrow_on_left(arrow_) ? w + arrow_shift x += is_arrow_on_left(arrow_) ? w + arrow_shift
: -arrow_shift - size.width(); : -arrow_shift - size.width();
if (is_arrow_on_top(arrow_)) { if (is_arrow_on_top(arrow_)) {
y += mid_anchor ? h / 2 - arrow_offset : kStroke - GetBorderThickness(); y += mid_anchor ? h / 2 - arrow_offset
: stroke_width - GetBorderThickness();
} else if (is_arrow_at_center(arrow_)) { } else if (is_arrow_at_center(arrow_)) {
y += h / 2 - arrow_offset; y += h / 2 - arrow_offset;
} else { } else {
y += mid_anchor ? h / 2 + arrow_offset - size.height() : y += mid_anchor ? h / 2 + arrow_offset - size.height()
h - size.height() + GetBorderThickness() - kStroke; : h - size.height() + GetBorderThickness() - stroke_width;
} }
} else { } else {
x += (w - size.width()) / 2; x += (w - size.width()) / 2;
......
...@@ -100,6 +100,8 @@ class VIEWS_EXPORT BubbleBorder : public Border { ...@@ -100,6 +100,8 @@ class VIEWS_EXPORT BubbleBorder : public Border {
NO_SHADOW_OPAQUE_BORDER, NO_SHADOW_OPAQUE_BORDER,
BIG_SHADOW, BIG_SHADOW,
SMALL_SHADOW, SMALL_SHADOW,
// NO_ASSETS borders don't draw a stroke or a shadow. This is used for
// platforms that provide their own shadows.
NO_ASSETS, NO_ASSETS,
SHADOW_COUNT, SHADOW_COUNT,
}; };
......
This diff is collapsed.
...@@ -196,7 +196,7 @@ TrayBubbleView::TrayBubbleView(View* anchor, ...@@ -196,7 +196,7 @@ TrayBubbleView::TrayBubbleView(View* anchor,
delegate_(delegate), delegate_(delegate),
preferred_width_(init_params.min_width), preferred_width_(init_params.min_width),
bubble_border_(new BubbleBorder(arrow(), bubble_border_(new BubbleBorder(arrow(),
BubbleBorder::BIG_SHADOW, BubbleBorder::NO_ASSETS,
init_params.bg_color)), init_params.bg_color)),
owned_bubble_border_(bubble_border_), owned_bubble_border_(bubble_border_),
is_gesture_dragging_(false), is_gesture_dragging_(false),
...@@ -272,6 +272,8 @@ void TrayBubbleView::OnBeforeBubbleWidgetInit(Widget::InitParams* params, ...@@ -272,6 +272,8 @@ void TrayBubbleView::OnBeforeBubbleWidgetInit(Widget::InitParams* params,
Widget* bubble_widget) const { Widget* bubble_widget) const {
if (delegate_) if (delegate_)
delegate_->OnBeforeBubbleWidgetInit(anchor_widget(), bubble_widget, params); delegate_->OnBeforeBubbleWidgetInit(anchor_widget(), bubble_widget, params);
// Apply a WM-provided shadow (see ui/wm/core/).
params->shadow_type = Widget::InitParams::SHADOW_TYPE_DROP;
} }
NonClientFrameView* TrayBubbleView::CreateNonClientFrameView(Widget* widget) { NonClientFrameView* TrayBubbleView::CreateNonClientFrameView(Widget* widget) {
......
...@@ -132,15 +132,6 @@ void Shadow::RecreateShadowLayer() { ...@@ -132,15 +132,6 @@ void Shadow::RecreateShadowLayer() {
const ShadowDetails& details = GetDetailsForElevation(ElevationForStyle()); const ShadowDetails& details = GetDetailsForElevation(ElevationForStyle());
shadow_layer_->UpdateNinePatchLayerImage(details.ninebox_image); shadow_layer_->UpdateNinePatchLayerImage(details.ninebox_image);
// The ninebox grid is defined in terms of the image size. The shadow blurs in
// both inward and outward directions from the edge of the contents, so the
// aperture goes further inside the image than the shadow margins (which
// represent exterior blur).
gfx::Rect aperture(details.ninebox_image.size());
gfx::Insets blur_region = gfx::ShadowValue::GetBlurRegion(details.values) +
gfx::Insets(kRoundedCornerRadius);
aperture.Inset(blur_region);
shadow_layer_->UpdateNinePatchLayerAperture(aperture);
UpdateLayerBounds(); UpdateLayerBounds();
} }
...@@ -194,6 +185,21 @@ void Shadow::UpdateLayerBounds() { ...@@ -194,6 +185,21 @@ void Shadow::UpdateLayerBounds() {
int border_y = border_h * blur_region.top() / blur_region.height(); int border_y = border_h * blur_region.top() / blur_region.height();
shadow_layer_->UpdateNinePatchLayerBorder( shadow_layer_->UpdateNinePatchLayerBorder(
gfx::Rect(border_x, border_y, border_w, border_h)); gfx::Rect(border_x, border_y, border_w, border_h));
// The ninebox grid is defined in terms of the image size. The shadow blurs in
// both inward and outward directions from the edge of the contents, so the
// aperture goes further inside the image than the shadow margins (which
// represent exterior blur).
gfx::Rect aperture(details.ninebox_image.size());
// The insets for the aperture are nominally |blur_region| but we need to
// resize them if the contents are too small.
// TODO(estade): by cutting out parts of ninebox, we lose the smooth
// horizontal or vertical transition. This isn't very noticeable, but we may
// need to address it by using a separate shadow layer for each ShadowValue,
// by adjusting the shadow for very small windows, or other means.
aperture.Inset(gfx::Insets(border_y, border_x, border_h - border_y,
border_w - border_x));
shadow_layer_->UpdateNinePatchLayerAperture(aperture);
} }
int Shadow::ElevationForStyle() { int Shadow::ElevationForStyle() {
......
...@@ -21,6 +21,7 @@ namespace wm { ...@@ -21,6 +21,7 @@ namespace wm {
// Simple class that draws a drop shadow around content at given bounds. // Simple class that draws a drop shadow around content at given bounds.
class WM_EXPORT Shadow : public ui::ImplicitAnimationObserver { class WM_EXPORT Shadow : public ui::ImplicitAnimationObserver {
public: public:
// TODO(estade): remove this enum and instead set elevation directly.
enum Style { enum Style {
// Active windows have more opaque shadows, shifted down to make the window // Active windows have more opaque shadows, shifted down to make the window
// appear "higher". // appear "higher".
......
...@@ -45,20 +45,22 @@ ShadowType GetShadowTypeFromWindow(aura::Window* window) { ...@@ -45,20 +45,22 @@ ShadowType GetShadowTypeFromWindow(aura::Window* window) {
return SHADOW_TYPE_NONE; return SHADOW_TYPE_NONE;
} }
bool ShouldUseSmallShadowForWindow(aura::Window* window) { Shadow::Style GetShadowStyleForWindow(aura::Window* window) {
switch (window->type()) { switch (window->type()) {
case ui::wm::WINDOW_TYPE_MENU: case ui::wm::WINDOW_TYPE_MENU:
case ui::wm::WINDOW_TYPE_TOOLTIP: case ui::wm::WINDOW_TYPE_TOOLTIP:
return true; return Shadow::STYLE_SMALL;
// System tray bubbles render like active windows. TODO(estade): this
// mechanism will need to be revisited for applying WM shadows to other
// types of bubbles which don't want to render such large shadows.
case ui::wm::WINDOW_TYPE_POPUP:
return Shadow::STYLE_ACTIVE;
default: default:
break; return IsActiveWindow(window) ? Shadow::STYLE_ACTIVE
: Shadow::STYLE_INACTIVE;
} }
return false;
}
Shadow::Style GetShadowStyleForWindow(aura::Window* window) {
return ShouldUseSmallShadowForWindow(window) ? Shadow::STYLE_SMALL :
(IsActiveWindow(window) ? Shadow::STYLE_ACTIVE : Shadow::STYLE_INACTIVE);
} }
// Returns the shadow style to be applied to |losing_active| when it is losing // Returns the shadow style to be applied to |losing_active| when it is losing
...@@ -193,12 +195,13 @@ void ShadowController::Impl::OnWindowActivated(ActivationReason reason, ...@@ -193,12 +195,13 @@ void ShadowController::Impl::OnWindowActivated(ActivationReason reason,
aura::Window* lost_active) { aura::Window* lost_active) {
if (gained_active) { if (gained_active) {
Shadow* shadow = GetShadowForWindow(gained_active); Shadow* shadow = GetShadowForWindow(gained_active);
if (shadow && !ShouldUseSmallShadowForWindow(gained_active)) if (shadow)
shadow->SetStyle(Shadow::STYLE_ACTIVE); shadow->SetStyle(GetShadowStyleForWindow(gained_active));
} }
if (lost_active) { if (lost_active) {
Shadow* shadow = GetShadowForWindow(lost_active); Shadow* shadow = GetShadowForWindow(lost_active);
if (shadow && !ShouldUseSmallShadowForWindow(lost_active)) { if (shadow &&
GetShadowStyleForWindow(lost_active) == Shadow::STYLE_INACTIVE) {
shadow->SetStyle(GetShadowStyleForWindowLosingActive(lost_active, shadow->SetStyle(GetShadowStyleForWindowLosingActive(lost_active,
gained_active)); gained_active));
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment