blob: 97b637d301c721b96de5f3993d14f8d38660b221 [file] [log] [blame]
// Copyright 2013 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/chromeos/ui/focus_ring_layer.h"
#include "ash/system/tray/actionable_view.h"
#include "ash/system/tray/tray_background_view.h"
#include "ash/system/tray/tray_popup_header_button.h"
#include "base/bind.h"
#include "ui/aura/window.h"
#include "ui/compositor/layer.h"
#include "ui/gfx/canvas.h"
#include "ui/views/controls/button/label_button.h"
#include "ui/views/view.h"
#include "ui/views/widget/widget.h"
namespace chromeos {
namespace {
const int kShadowRadius = 10;
const int kShadowAlpha = 90;
const SkColor kShadowColor = SkColorSetRGB(77, 144, 254);
} // namespace
FocusRingLayer::FocusRingLayer()
: window_(NULL),
root_window_(NULL) {
}
FocusRingLayer::~FocusRingLayer() {}
void FocusRingLayer::Update() {
if (!window_)
return;
aura::Window* root_window = window_->GetRootWindow();
if (!layer_ || root_window != root_window_) {
root_window_ = root_window;
ui::Layer* root_layer = root_window->layer();
layer_.reset(new ui::Layer(ui::LAYER_TEXTURED));
layer_->set_name("FocusRing");
layer_->set_delegate(this);
layer_->SetFillsBoundsOpaquely(false);
root_layer->Add(layer_.get());
}
// Keep moving it to the top in case new layers have been added
// since we created this layer.
layer_->parent()->StackAtTop(layer_.get());
// Translate native window coordinates to root window coordinates.
gfx::Point origin = focus_ring_.origin();
aura::Window::ConvertPointToTarget(window_, root_window_, &origin);
gfx::Rect layer_bounds = focus_ring_;
layer_bounds.set_origin(origin);
int inset = -(kShadowRadius + 2);
layer_bounds.Inset(inset, inset, inset, inset);
layer_->SetBounds(layer_bounds);
}
void FocusRingLayer::SetForView(views::View* view) {
if (!view) {
if (layer_ && !focus_ring_.IsEmpty())
layer_->SchedulePaint(focus_ring_);
focus_ring_ = gfx::Rect();
return;
}
DCHECK(view->GetWidget());
window_ = view->GetWidget()->GetNativeWindow();
gfx::Rect view_bounds = view->GetContentsBounds();
// Workarounds that attempts to pick a better bounds.
if (view->GetClassName() == views::LabelButton::kViewClassName) {
view_bounds = view->GetLocalBounds();
view_bounds.Inset(2, 2, 2, 2);
}
// Workarounds for system tray items that have customized focus borders. The
// insets here must be consistent with the ones used by those classes.
if (view->GetClassName() == ash::ActionableView::kViewClassName) {
view_bounds = view->GetLocalBounds();
view_bounds.Inset(1, 1, 3, 3);
} else if (view->GetClassName() == ash::TrayBackgroundView::kViewClassName) {
view_bounds.Inset(1, 1, 3, 3);
} else if (view->GetClassName() ==
ash::TrayPopupHeaderButton::kViewClassName) {
view_bounds = view->GetLocalBounds();
view_bounds.Inset(2, 1, 2, 2);
}
focus_ring_ = view->ConvertRectToWidget(view_bounds);
Update();
}
void FocusRingLayer::OnPaintLayer(gfx::Canvas* canvas) {
if (focus_ring_.IsEmpty())
return;
// Convert the focus ring from native-window-relative coordinates to
// layer-relative coordinates.
gfx::Point origin = focus_ring_.origin();
aura::Window::ConvertPointToTarget(window_, root_window_, &origin);
origin -= layer_->bounds().OffsetFromOrigin();
gfx::Rect bounds = focus_ring_;
bounds.set_origin(origin);
SkPaint paint;
paint.setColor(kShadowColor);
paint.setFlags(SkPaint::kAntiAlias_Flag);
paint.setStyle(SkPaint::kStroke_Style);
paint.setStrokeWidth(2);
int r = kShadowRadius;
for (int i = 0; i < r; i++) {
// Fade out alpha quadratically.
paint.setAlpha((kShadowAlpha * (r - i) * (r - i)) / (r * r));
gfx::Rect outsetRect = bounds;
outsetRect.Inset(-i, -i, -i, -i);
canvas->DrawRect(outsetRect, paint);
}
}
void FocusRingLayer::OnDeviceScaleFactorChanged(float device_scale_factor) {
Update();
}
base::Closure FocusRingLayer::PrepareForLayerBoundsChange() {
return base::Bind(&base::DoNothing);
}
} // namespace chromeos