Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "ash/shelf/shelf_tooltip_manager.h" | 5 #include "ash/shelf/shelf_tooltip_manager.h" |
| 6 | 6 |
| 7 #include "ash/common/shelf/wm_shelf.h" | |
| 7 #include "ash/common/shell_window_ids.h" | 8 #include "ash/common/shell_window_ids.h" |
| 8 #include "ash/shelf/shelf.h" | 9 #include "ash/common/wm_lookup.h" |
| 9 #include "ash/shelf/shelf_layout_manager.h" | 10 #include "ash/common/wm_root_window_controller.h" |
| 11 #include "ash/common/wm_shell.h" | |
| 12 #include "ash/common/wm_window.h" | |
| 10 #include "ash/shelf/shelf_view.h" | 13 #include "ash/shelf/shelf_view.h" |
| 11 #include "ash/shell.h" | |
| 12 #include "ash/wm/window_animations.h" | |
| 13 #include "base/bind.h" | 14 #include "base/bind.h" |
| 14 #include "base/strings/string16.h" | 15 #include "base/strings/string16.h" |
| 15 #include "base/threading/thread_task_runner_handle.h" | 16 #include "base/threading/thread_task_runner_handle.h" |
| 16 #include "base/time/time.h" | 17 #include "base/time/time.h" |
| 17 #include "ui/aura/window.h" | 18 #include "ui/aura/window.h" |
|
msw
2016/07/22 22:41:39
Can/should I avoid this? Use WmWindow and add supp
| |
| 18 #include "ui/events/event.h" | 19 #include "ui/events/event.h" |
| 19 #include "ui/events/event_constants.h" | 20 #include "ui/events/event_constants.h" |
| 20 #include "ui/gfx/geometry/insets.h" | 21 #include "ui/gfx/geometry/insets.h" |
| 21 #include "ui/views/bubble/bubble_dialog_delegate.h" | 22 #include "ui/views/bubble/bubble_dialog_delegate.h" |
| 22 #include "ui/views/controls/label.h" | 23 #include "ui/views/controls/label.h" |
| 23 #include "ui/views/layout/fill_layout.h" | 24 #include "ui/views/layout/fill_layout.h" |
| 24 #include "ui/views/widget/widget.h" | 25 #include "ui/views/widget/widget.h" |
| 25 #include "ui/wm/core/window_animations.h" | |
| 26 | 26 |
| 27 namespace ash { | 27 namespace ash { |
| 28 namespace { | 28 namespace { |
| 29 | 29 |
| 30 const int kTooltipTopBottomMargin = 3; | 30 const int kTooltipTopBottomMargin = 3; |
| 31 const int kTooltipLeftRightMargin = 10; | 31 const int kTooltipLeftRightMargin = 10; |
| 32 const int kTooltipAppearanceDelay = 1000; // msec | 32 const int kTooltipAppearanceDelay = 1000; // msec |
| 33 const int kTooltipMinHeight = 29 - 2 * kTooltipTopBottomMargin; | 33 const int kTooltipMinHeight = 29 - 2 * kTooltipTopBottomMargin; |
| 34 const SkColor kTooltipTextColor = SkColorSetRGB(0x22, 0x22, 0x22); | 34 const SkColor kTooltipTextColor = SkColorSetRGB(0x22, 0x22, 0x22); |
| 35 | 35 |
| (...skipping 10 matching lines...) Expand all Loading... | |
| 46 } // namespace | 46 } // namespace |
| 47 | 47 |
| 48 // The implementation of tooltip of the launcher. | 48 // The implementation of tooltip of the launcher. |
| 49 class ShelfTooltipManager::ShelfTooltipBubble | 49 class ShelfTooltipManager::ShelfTooltipBubble |
| 50 : public views::BubbleDialogDelegateView { | 50 : public views::BubbleDialogDelegateView { |
| 51 public: | 51 public: |
| 52 ShelfTooltipBubble(views::View* anchor, | 52 ShelfTooltipBubble(views::View* anchor, |
| 53 views::BubbleBorder::Arrow arrow, | 53 views::BubbleBorder::Arrow arrow, |
| 54 const base::string16& text) | 54 const base::string16& text) |
| 55 : views::BubbleDialogDelegateView(anchor, arrow) { | 55 : views::BubbleDialogDelegateView(anchor, arrow) { |
| 56 gfx::Insets insets = | 56 gfx::Insets insets(kArrowOffsetTopBottom, kArrowOffsetLeftRight); |
| 57 gfx::Insets(kArrowOffsetTopBottom, kArrowOffsetLeftRight); | |
| 58 // Adjust the anchor location for asymmetrical borders of shelf item. | 57 // Adjust the anchor location for asymmetrical borders of shelf item. |
| 59 if (anchor->border()) | 58 if (anchor->border()) |
| 60 insets += anchor->border()->GetInsets(); | 59 insets += anchor->border()->GetInsets(); |
| 61 | 60 |
| 62 set_anchor_view_insets(insets); | 61 set_anchor_view_insets(insets); |
| 63 set_close_on_deactivate(false); | 62 set_close_on_deactivate(false); |
| 64 set_can_activate(false); | 63 set_can_activate(false); |
| 65 set_accept_events(false); | 64 set_accept_events(false); |
| 66 set_margins(gfx::Insets(kTooltipTopBottomMargin, kTooltipLeftRightMargin)); | 65 set_margins(gfx::Insets(kTooltipTopBottomMargin, kTooltipLeftRightMargin)); |
| 67 set_shadow(views::BubbleBorder::SMALL_SHADOW); | 66 set_shadow(views::BubbleBorder::SMALL_SHADOW); |
| 68 SetLayoutManager(new views::FillLayout()); | 67 SetLayoutManager(new views::FillLayout()); |
| 69 // The anchor may not have the widget in tests. | |
| 70 if (anchor->GetWidget() && anchor->GetWidget()->GetNativeWindow()) { | |
| 71 set_parent_window(ash::Shell::GetContainer( | |
| 72 anchor->GetWidget()->GetNativeWindow()->GetRootWindow(), | |
| 73 ash::kShellWindowId_SettingBubbleContainer)); | |
| 74 } | |
| 75 views::Label* label = new views::Label(text); | 68 views::Label* label = new views::Label(text); |
| 76 label->SetHorizontalAlignment(gfx::ALIGN_LEFT); | 69 label->SetHorizontalAlignment(gfx::ALIGN_LEFT); |
| 77 label->SetEnabledColor(kTooltipTextColor); | 70 label->SetEnabledColor(kTooltipTextColor); |
| 78 AddChildView(label); | 71 AddChildView(label); |
| 79 views::BubbleDialogDelegateView::CreateBubble(this); | 72 views::BubbleDialogDelegateView::CreateBubble(this); |
| 80 } | 73 } |
| 81 | 74 |
| 82 private: | 75 private: |
| 83 // BubbleDialogDelegateView overrides: | 76 // BubbleDialogDelegateView overrides: |
| 84 gfx::Size GetPreferredSize() const override { | 77 gfx::Size GetPreferredSize() const override { |
| 85 const gfx::Size size = BubbleDialogDelegateView::GetPreferredSize(); | 78 const gfx::Size size = BubbleDialogDelegateView::GetPreferredSize(); |
| 86 return gfx::Size(std::min(size.width(), kTooltipMaxWidth), | 79 return gfx::Size(std::min(size.width(), kTooltipMaxWidth), |
| 87 std::max(size.height(), kTooltipMinHeight)); | 80 std::max(size.height(), kTooltipMinHeight)); |
| 88 } | 81 } |
| 89 | 82 |
| 83 void OnBeforeBubbleWidgetInit(views::Widget::InitParams* params, | |
| 84 views::Widget* bubble_widget) const override { | |
| 85 // Place the bubble in the same display as the anchor. | |
| 86 WmLookup::Get() | |
| 87 ->GetWindowForWidget(anchor_widget()) | |
| 88 ->GetRootWindowController() | |
| 89 ->ConfigureWidgetInitParamsForContainer( | |
| 90 bubble_widget, kShellWindowId_SettingBubbleContainer, params); | |
| 91 } | |
| 92 | |
| 90 int GetDialogButtons() const override { return ui::DIALOG_BUTTON_NONE; } | 93 int GetDialogButtons() const override { return ui::DIALOG_BUTTON_NONE; } |
| 91 | 94 |
| 92 DISALLOW_COPY_AND_ASSIGN(ShelfTooltipBubble); | 95 DISALLOW_COPY_AND_ASSIGN(ShelfTooltipBubble); |
| 93 }; | 96 }; |
| 94 | 97 |
| 95 ShelfTooltipManager::ShelfTooltipManager(ShelfView* shelf_view) | 98 ShelfTooltipManager::ShelfTooltipManager(ShelfView* shelf_view) |
| 96 : timer_delay_(kTooltipAppearanceDelay), | 99 : timer_delay_(kTooltipAppearanceDelay), |
| 97 shelf_view_(shelf_view), | 100 shelf_view_(shelf_view), |
| 98 root_window_(nullptr), | |
| 99 shelf_layout_manager_(nullptr), | |
| 100 bubble_(nullptr), | 101 bubble_(nullptr), |
| 101 weak_factory_(this) {} | 102 weak_factory_(this) { |
| 103 shelf_view_->wm_shelf()->AddObserver(this); | |
| 104 WmShell::Get()->AddPointerDownWatcher(this); | |
| 105 } | |
| 102 | 106 |
| 103 ShelfTooltipManager::~ShelfTooltipManager() { | 107 ShelfTooltipManager::~ShelfTooltipManager() { |
| 104 WillDeleteShelfLayoutManager(); | 108 WmShell::Get()->RemovePointerDownWatcher(this); |
| 105 | 109 shelf_view_->wm_shelf()->RemoveObserver(this); |
| 106 Shell::GetInstance()->RemovePointerDownWatcher(this); | 110 if (shelf_view_->GetWidget() && shelf_view_->GetWidget()->GetNativeWindow()) |
| 107 | 111 shelf_view_->GetWidget()->GetNativeWindow()->PrependPreTargetHandler(this); |
| 108 if (root_window_) { | |
| 109 root_window_->RemoveObserver(this); | |
| 110 root_window_->RemovePreTargetHandler(this); | |
| 111 root_window_ = nullptr; | |
| 112 } | |
| 113 } | 112 } |
| 114 | 113 |
| 115 void ShelfTooltipManager::Init() { | 114 void ShelfTooltipManager::Init() { |
| 116 shelf_layout_manager_ = shelf_view_->shelf()->shelf_layout_manager(); | 115 shelf_view_->GetWidget()->GetNativeWindow()->PrependPreTargetHandler(this); |
| 117 shelf_layout_manager_->AddObserver(this); | |
| 118 | |
| 119 root_window_ = shelf_view_->GetWidget()->GetNativeWindow()->GetRootWindow(); | |
| 120 root_window_->AddPreTargetHandler(this); | |
| 121 root_window_->AddObserver(this); | |
| 122 | |
| 123 Shell::GetInstance()->AddPointerDownWatcher(this); | |
| 124 } | 116 } |
| 125 | 117 |
| 126 void ShelfTooltipManager::Close() { | 118 void ShelfTooltipManager::Close() { |
| 127 timer_.Stop(); | 119 timer_.Stop(); |
| 128 if (bubble_) | 120 if (bubble_) |
| 129 bubble_->GetWidget()->Close(); | 121 bubble_->GetWidget()->Close(); |
| 130 bubble_ = nullptr; | 122 bubble_ = nullptr; |
| 131 } | 123 } |
| 132 | 124 |
| 133 bool ShelfTooltipManager::IsVisible() const { | 125 bool ShelfTooltipManager::IsVisible() const { |
| 134 return bubble_ && bubble_->GetWidget()->IsVisible(); | 126 return bubble_ && bubble_->GetWidget()->IsVisible(); |
| 135 } | 127 } |
| 136 | 128 |
| 137 views::View* ShelfTooltipManager::GetCurrentAnchorView() const { | 129 views::View* ShelfTooltipManager::GetCurrentAnchorView() const { |
| 138 return bubble_ ? bubble_->GetAnchorView() : nullptr; | 130 return bubble_ ? bubble_->GetAnchorView() : nullptr; |
| 139 } | 131 } |
| 140 | 132 |
| 141 void ShelfTooltipManager::ShowTooltip(views::View* view) { | 133 void ShelfTooltipManager::ShowTooltip(views::View* view) { |
| 142 timer_.Stop(); | 134 timer_.Stop(); |
| 143 if (bubble_) { | 135 if (bubble_) { |
| 144 // Cancel the hiding animation to hide the old bubble immediately. | 136 // Cancel the hiding animation to hide the old bubble immediately. |
| 145 gfx::NativeView native_view = bubble_->GetWidget()->GetNativeView(); | 137 WmLookup::Get() |
| 146 ::wm::SetWindowVisibilityAnimationTransition(native_view, | 138 ->GetWindowForWidget(bubble_->GetWidget()) |
| 147 ::wm::ANIMATE_NONE); | 139 ->SetVisibilityAnimationTransition(::wm::ANIMATE_NONE); |
| 148 Close(); | 140 Close(); |
| 149 } | 141 } |
| 150 | 142 |
| 151 if (!ShouldShowTooltipForView(view)) | 143 if (!ShouldShowTooltipForView(view)) |
| 152 return; | 144 return; |
| 153 | 145 |
| 154 Shelf* shelf = shelf_view_->shelf(); | 146 views::BubbleBorder::Arrow arrow = views::BubbleBorder::Arrow::NONE; |
| 155 views::BubbleBorder::Arrow arrow = shelf->SelectValueForShelfAlignment( | 147 switch (shelf_view_->wm_shelf()->GetAlignment()) { |
| 156 views::BubbleBorder::BOTTOM_CENTER, views::BubbleBorder::LEFT_CENTER, | 148 case SHELF_ALIGNMENT_BOTTOM: |
| 157 views::BubbleBorder::RIGHT_CENTER); | 149 case SHELF_ALIGNMENT_BOTTOM_LOCKED: |
| 150 arrow = views::BubbleBorder::BOTTOM_CENTER; | |
| 151 break; | |
| 152 case SHELF_ALIGNMENT_LEFT: | |
| 153 arrow = views::BubbleBorder::LEFT_CENTER; | |
| 154 break; | |
| 155 case SHELF_ALIGNMENT_RIGHT: | |
| 156 arrow = views::BubbleBorder::RIGHT_CENTER; | |
| 157 break; | |
| 158 } | |
| 158 | 159 |
| 159 base::string16 text = shelf_view_->GetTitleForView(view); | 160 base::string16 text = shelf_view_->GetTitleForView(view); |
| 160 bubble_ = new ShelfTooltipBubble(view, arrow, text); | 161 bubble_ = new ShelfTooltipBubble(view, arrow, text); |
| 161 gfx::NativeView native_view = bubble_->GetWidget()->GetNativeView(); | 162 WmWindow* window = WmLookup::Get()->GetWindowForWidget(bubble_->GetWidget()); |
| 162 ::wm::SetWindowVisibilityAnimationType( | 163 window->SetVisibilityAnimationType( |
| 163 native_view, ::wm::WINDOW_VISIBILITY_ANIMATION_TYPE_VERTICAL); | 164 ::wm::WINDOW_VISIBILITY_ANIMATION_TYPE_VERTICAL); |
| 164 ::wm::SetWindowVisibilityAnimationTransition(native_view, ::wm::ANIMATE_HIDE); | 165 window->SetVisibilityAnimationTransition(::wm::ANIMATE_HIDE); |
| 165 bubble_->GetWidget()->Show(); | 166 bubble_->GetWidget()->Show(); |
| 166 } | 167 } |
| 167 | 168 |
| 168 void ShelfTooltipManager::ShowTooltipWithDelay(views::View* view) { | 169 void ShelfTooltipManager::ShowTooltipWithDelay(views::View* view) { |
| 169 if (ShouldShowTooltipForView(view)) { | 170 if (ShouldShowTooltipForView(view)) { |
| 170 timer_.Start(FROM_HERE, base::TimeDelta::FromMilliseconds(timer_delay_), | 171 timer_.Start(FROM_HERE, base::TimeDelta::FromMilliseconds(timer_delay_), |
| 171 base::Bind(&ShelfTooltipManager::ShowTooltip, | 172 base::Bind(&ShelfTooltipManager::ShowTooltip, |
| 172 weak_factory_.GetWeakPtr(), view)); | 173 weak_factory_.GetWeakPtr(), view)); |
| 173 } | 174 } |
| 174 } | 175 } |
| 175 | 176 |
| 176 void ShelfTooltipManager::OnMousePressed(const ui::MouseEvent& event, | 177 void ShelfTooltipManager::OnMousePressed(const ui::MouseEvent& event, |
| 177 const gfx::Point& location_in_screen, | 178 const gfx::Point& location_in_screen, |
| 178 views::Widget* target) { | 179 views::Widget* target) { |
| 179 // Close on any mouse press events inside or outside the tooltip. | 180 // Close on any mouse press events inside or outside the tooltip. |
| 180 Close(); | 181 Close(); |
| 181 } | 182 } |
| 182 | 183 |
| 183 void ShelfTooltipManager::OnTouchPressed(const ui::TouchEvent& event, | 184 void ShelfTooltipManager::OnTouchPressed(const ui::TouchEvent& event, |
| 184 const gfx::Point& location_in_screen, | 185 const gfx::Point& location_in_screen, |
| 185 views::Widget* target) { | 186 views::Widget* target) { |
| 186 // Close on any touch press events inside or outside the tooltip. | 187 // Close on any touch press events inside or outside the tooltip. |
| 187 Close(); | 188 Close(); |
| 188 } | 189 } |
| 189 | 190 |
| 190 void ShelfTooltipManager::OnEvent(ui::Event* event) { | 191 void ShelfTooltipManager::OnMouseEvent(ui::MouseEvent* event) { |
| 191 // Mouse and touch press events are handled via views::PointerDownWatcher. | 192 DCHECK_EQ(shelf_view_->GetWidget()->GetNativeWindow(), event->target()); |
| 192 if (event->type() == ui::ET_MOUSE_PRESSED || | 193 if (event->type() == ui::ET_MOUSE_EXITED) { |
| 193 event->type() == ui::ET_TOUCH_PRESSED) { | 194 Close(); |
| 194 return; | 195 return; |
| 195 } | 196 } |
| 196 | 197 |
| 197 if (event->type() == ui::ET_MOUSE_EXITED || !event->IsMouseEvent() || | 198 if (event->type() != ui::ET_MOUSE_MOVED) |
| 198 event->target() != shelf_view_->GetWidget()->GetNativeWindow()) { | |
| 199 if (!event->IsKeyEvent()) | |
| 200 Close(); | |
| 201 return; | 199 return; |
| 202 } | |
| 203 | 200 |
| 204 gfx::Point point = static_cast<ui::LocatedEvent*>(event)->location(); | 201 gfx::Point point = event->AsLocatedEvent()->location(); |
| 205 aura::Window::ConvertPointToTarget( | |
| 206 static_cast<aura::Window*>(event->target()), | |
| 207 shelf_view_->GetWidget()->GetNativeWindow(), &point); | |
| 208 views::View::ConvertPointFromWidget(shelf_view_, &point); | 202 views::View::ConvertPointFromWidget(shelf_view_, &point); |
| 209 views::View* view = shelf_view_->GetTooltipHandlerForPoint(point); | 203 views::View* view = shelf_view_->GetTooltipHandlerForPoint(point); |
| 210 const bool should_show = ShouldShowTooltipForView(view); | 204 const bool should_show = ShouldShowTooltipForView(view); |
| 211 | 205 |
| 212 timer_.Stop(); | 206 timer_.Stop(); |
| 213 if (IsVisible() && should_show && bubble_->GetAnchorView() != view) | 207 if (IsVisible() && should_show && bubble_->GetAnchorView() != view) |
| 214 ShowTooltip(view); | 208 ShowTooltip(view); |
| 215 else if (!IsVisible() && should_show && event->type() == ui::ET_MOUSE_MOVED) | 209 else if (!IsVisible() && should_show) |
| 216 ShowTooltipWithDelay(view); | 210 ShowTooltipWithDelay(view); |
| 217 else if (IsVisible() && shelf_view_->ShouldHideTooltip(point)) | 211 else if (IsVisible() && shelf_view_->ShouldHideTooltip(point)) |
| 218 Close(); | 212 Close(); |
| 219 } | 213 } |
| 220 | 214 |
| 221 void ShelfTooltipManager::OnWindowDestroying(aura::Window* window) { | |
| 222 if (window == root_window_) { | |
| 223 root_window_->RemoveObserver(this); | |
| 224 root_window_->RemovePreTargetHandler(this); | |
| 225 root_window_ = nullptr; | |
| 226 } | |
| 227 } | |
| 228 | |
| 229 void ShelfTooltipManager::WillDeleteShelfLayoutManager() { | |
| 230 if (shelf_layout_manager_) | |
| 231 shelf_layout_manager_->RemoveObserver(this); | |
| 232 shelf_layout_manager_ = nullptr; | |
| 233 shelf_view_ = nullptr; | |
| 234 } | |
| 235 | |
| 236 void ShelfTooltipManager::WillChangeVisibilityState( | 215 void ShelfTooltipManager::WillChangeVisibilityState( |
| 237 ShelfVisibilityState new_state) { | 216 ShelfVisibilityState new_state) { |
| 238 if (new_state == SHELF_HIDDEN) | 217 if (new_state == SHELF_HIDDEN) |
| 239 Close(); | 218 Close(); |
| 240 } | 219 } |
| 241 | 220 |
| 242 void ShelfTooltipManager::OnAutoHideStateChanged(ShelfAutoHideState new_state) { | 221 void ShelfTooltipManager::OnAutoHideStateChanged(ShelfAutoHideState new_state) { |
| 243 if (new_state == SHELF_AUTO_HIDE_HIDDEN) { | 222 if (new_state == SHELF_AUTO_HIDE_HIDDEN) { |
| 244 timer_.Stop(); | 223 timer_.Stop(); |
| 245 // AutoHide state change happens during an event filter, so immediate close | 224 // AutoHide state change happens during an event filter, so immediate close |
| 246 // may cause a crash in the HandleMouseEvent() after the filter. So we just | 225 // may cause a crash in the HandleMouseEvent() after the filter. So we just |
| 247 // schedule the Close here. | 226 // schedule the Close here. |
| 248 base::ThreadTaskRunnerHandle::Get()->PostTask( | 227 base::ThreadTaskRunnerHandle::Get()->PostTask( |
| 249 FROM_HERE, | 228 FROM_HERE, |
| 250 base::Bind(&ShelfTooltipManager::Close, weak_factory_.GetWeakPtr())); | 229 base::Bind(&ShelfTooltipManager::Close, weak_factory_.GetWeakPtr())); |
| 251 } | 230 } |
| 252 } | 231 } |
| 253 | 232 |
| 254 bool ShelfTooltipManager::ShouldShowTooltipForView(views::View* view) { | 233 bool ShelfTooltipManager::ShouldShowTooltipForView(views::View* view) { |
| 255 return shelf_view_ && shelf_view_->ShouldShowTooltipForView(view) && | 234 return shelf_view_ && shelf_view_->ShouldShowTooltipForView(view) && |
| 256 shelf_layout_manager_ && shelf_layout_manager_->IsVisible(); | 235 shelf_view_->wm_shelf()->GetVisibilityState() == SHELF_VISIBLE; |
| 257 } | 236 } |
| 258 | 237 |
| 259 } // namespace ash | 238 } // namespace ash |
| OLD | NEW |