Chromium Code Reviews| Index: ash/magnifier/magnification_controller.cc |
| diff --git a/ash/magnifier/magnification_controller.cc b/ash/magnifier/magnification_controller.cc |
| index 487aa7a3221d9c4fd82fd95b4b09b715ee89f904..98a239b109658def4d7cf14b1f404ea8e04084a8 100644 |
| --- a/ash/magnifier/magnification_controller.cc |
| +++ b/ash/magnifier/magnification_controller.cc |
| @@ -4,6 +4,7 @@ |
| #include "ash/magnifier/magnification_controller.h" |
| +#include "ash/display/display_controller.h" |
| #include "ash/shell.h" |
| #include "ash/shell_delegate.h" |
| #include "ash/system/tray/system_tray_delegate.h" |
| @@ -46,7 +47,8 @@ namespace ash { |
| class MagnificationControllerImpl : virtual public MagnificationController, |
| public ui::EventHandler, |
| - public ui::ImplicitAnimationObserver { |
| + public ui::ImplicitAnimationObserver, |
| + public aura::WindowObserver { |
| public: |
| MagnificationControllerImpl(); |
| virtual ~MagnificationControllerImpl(); |
| @@ -70,6 +72,9 @@ class MagnificationControllerImpl : virtual public MagnificationController, |
| // ui::ImplicitAnimationObserver overrides: |
| virtual void OnImplicitAnimationsCompleted() OVERRIDE; |
| + // aura::WindowObserver overrides: |
| + virtual void OnWindowDestroying(aura::Window* root_window) OVERRIDE; |
| + |
| // Redraws the magnification window with the given origin position and the |
| // given scale. Returns true if the window is changed; otherwise, false. |
| // These methods should be called internally just after the scale and/or |
| @@ -122,6 +127,7 @@ class MagnificationControllerImpl : virtual public MagnificationController, |
| virtual void OnScrollEvent(ui::ScrollEvent* event) OVERRIDE; |
| virtual void OnTouchEvent(ui::TouchEvent* event) OVERRIDE; |
| + // Target root window. This must not be NULL. |
| aura::RootWindow* root_window_; |
| // True if the magnified window is currently animating a change. Otherwise, |
| @@ -158,16 +164,22 @@ MagnificationControllerImpl::MagnificationControllerImpl() |
| scale_(kNonMagnifiedScale) { |
| Shell::GetInstance()->AddPreTargetHandler(this); |
| - if (root_window_) |
| - point_of_interest_ = root_window_->bounds().CenterPoint(); |
| + CHECK(root_window_); |
|
oshima
2013/02/27 19:07:04
CHECK/DCHECK is useful if you want to fail early,
|
| + root_window_->AddObserver(this); |
| + point_of_interest_ = root_window_->bounds().CenterPoint(); |
| } |
| MagnificationControllerImpl::~MagnificationControllerImpl() { |
| + DCHECK(root_window_); |
|
oshima
2013/02/27 19:07:04
ditto
|
| + root_window_->RemoveObserver(this); |
| + |
| Shell::GetInstance()->RemovePreTargetHandler(this); |
| } |
| void MagnificationControllerImpl::RedrawKeepingMousePosition( |
| float scale, bool animate) { |
| + DCHECK(root_window_); |
|
oshima
2013/02/27 19:07:04
ditto
|
| + |
| gfx::Point mouse_in_root = point_of_interest_; |
| // mouse_in_root is invalid value when the cursor is hidden. |
| @@ -187,6 +199,8 @@ void MagnificationControllerImpl::RedrawKeepingMousePosition( |
| bool MagnificationControllerImpl::Redraw(const gfx::PointF& position, |
| float scale, |
| bool animate) { |
| + DCHECK(root_window_); |
|
oshima
2013/02/27 19:07:04
ditto
|
| + |
| const gfx::PointF position_in_dip = |
| ui::ConvertPointToDIP(root_window_->layer(), position); |
| return RedrawDIP(position_in_dip, scale, animate); |
| @@ -195,6 +209,8 @@ bool MagnificationControllerImpl::Redraw(const gfx::PointF& position, |
| bool MagnificationControllerImpl::RedrawDIP(const gfx::PointF& position_in_dip, |
| float scale, |
| bool animate) { |
| + DCHECK(root_window_); |
| + |
| float x = position_in_dip.x(); |
| float y = position_in_dip.y(); |
| @@ -253,6 +269,8 @@ void MagnificationControllerImpl::EnsureRectIsVisibleWithScale( |
| const gfx::Rect& target_rect, |
| float scale, |
| bool animate) { |
| + DCHECK(root_window_); |
|
oshima
2013/02/27 19:07:04
ditto
|
| + |
| const gfx::Rect target_rect_in_dip = |
| ui::ConvertRectToDIP(root_window_->layer(), target_rect); |
| EnsureRectIsVisibleDIP(target_rect_in_dip, scale, animate); |
| @@ -299,6 +317,8 @@ void MagnificationControllerImpl::EnsurePointIsVisibleWithScale( |
| } |
| void MagnificationControllerImpl::OnMouseMove(const gfx::Point& location) { |
| + DCHECK(root_window_); |
| + |
| gfx::Point mouse(location); |
| int x = origin_.x(); |
| @@ -353,6 +373,8 @@ void MagnificationControllerImpl::OnMouseMove(const gfx::Point& location) { |
| void MagnificationControllerImpl::AfterAnimationMoveCursorTo( |
| const gfx::Point& location) { |
| + DCHECK(root_window_); |
| + |
| aura::client::CursorClient* cursor_client = |
| aura::client::GetCursorClient(root_window_); |
| if (cursor_client) { |
| @@ -416,16 +438,42 @@ void MagnificationControllerImpl::OnImplicitAnimationsCompleted() { |
| is_on_animation_ = false; |
| } |
| +void MagnificationControllerImpl::OnWindowDestroying( |
| + aura::Window* root_window) { |
| + if (root_window == root_window_) { |
| + // There must be at least one root window because this controller is |
| + // destroyed before the root windows get destroyed. |
| + DCHECK(root_window); |
| + |
| + aura::RootWindow* active_root_window = Shell::GetActiveRootWindow(); |
| + CHECK(active_root_window); |
|
oshima
2013/02/27 17:21:43
what happens when the last root window gets detroy
|
| + |
| + // The destroyed root window must not be active. |
| + CHECK_NE(active_root_window, root_window); |
| + |
| + SwitchTargetRootWindow(active_root_window); |
| + point_of_interest_ = active_root_window->bounds().CenterPoint(); |
| + } |
| +} |
| + |
| void MagnificationControllerImpl::SwitchTargetRootWindow( |
| aura::RootWindow* new_root_window) { |
| + DCHECK(root_window_); |
| + DCHECK(new_root_window); |
| + |
| if (new_root_window == root_window_) |
| return; |
| + // Stores the previous scale. |
| float scale = GetScale(); |
| + // Unmagnify the previous root window. |
| + root_window_->RemoveObserver(this); |
| RedrawKeepingMousePosition(1.0f, true); |
| + |
| root_window_ = new_root_window; |
| RedrawKeepingMousePosition(scale, true); |
| + root_window_->AddObserver(this); |
| } |
| //////////////////////////////////////////////////////////////////////////////// |
| @@ -511,12 +559,15 @@ void MagnificationControllerImpl::OnMouseEvent(ui::MouseEvent* event) { |
| gfx::Rect root_bounds = current_root->bounds(); |
| if (root_bounds.Contains(event->root_location())) { |
| - if (current_root != root_window_) |
| - SwitchTargetRootWindow(current_root); |
| - |
| + // This must be before |SwitchTargetRootWindow()|. |
| point_of_interest_ = event->root_location(); |
| - if (IsMagnified() && event->type() == ui::ET_MOUSE_MOVED) |
| + if (current_root != root_window_) { |
| + DCHECK(current_root); |
| + SwitchTargetRootWindow(current_root); |
| + } |
| + |
| + if (IsMagnified() && root_window_ && event->type() == ui::ET_MOUSE_MOVED) |
|
oshima
2013/02/27 19:07:04
you don't need to check root_window_ anymore, do y
|
| OnMouseMove(event->root_location()); |
| } |
| } |
| @@ -543,7 +594,7 @@ void MagnificationControllerImpl::OnScrollEvent(ui::ScrollEvent* event) { |
| void MagnificationControllerImpl::OnTouchEvent(ui::TouchEvent* event) { |
| aura::Window* target = static_cast<aura::Window*>(event->target()); |
| aura::RootWindow* current_root = target->GetRootWindow(); |
| - if (current_root == root_window_) { |
| + if (current_root && current_root == root_window_) { |
|
oshima
2013/02/27 19:07:04
when can |current_root| be NULL?
|
| gfx::Rect root_bounds = current_root->bounds(); |
| if (root_bounds.Contains(event->root_location())) |
| point_of_interest_ = event->root_location(); |