// Copyright (c) 2012 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 "ui/aura/window.h" #include "base/basictypes.h" #include "base/compiler_specific.h" #include "base/stringprintf.h" #include "testing/gtest/include/gtest/gtest.h" #include "ui/aura/client/stacking_client.h" #include "ui/aura/client/visibility_client.h" #include "ui/aura/event.h" #include "ui/aura/focus_manager.h" #include "ui/aura/layout_manager.h" #include "ui/aura/root_window.h" #include "ui/aura/root_window_observer.h" #include "ui/aura/test/aura_test_base.h" #include "ui/aura/test/event_generator.h" #include "ui/aura/test/test_window_delegate.h" #include "ui/aura/test/test_windows.h" #include "ui/aura/window_delegate.h" #include "ui/aura/window_observer.h" #include "ui/aura/window_property.h" #include "ui/base/hit_test.h" #include "ui/base/keycodes/keyboard_codes.h" #include "ui/compositor/layer.h" #include "ui/compositor/scoped_layer_animation_settings.h" #include "ui/gfx/canvas.h" #include "ui/gfx/screen.h" DECLARE_WINDOW_PROPERTY_TYPE(const char*) DECLARE_WINDOW_PROPERTY_TYPE(int) namespace aura { namespace test { typedef AuraTestBase WindowTest; namespace { // Used for verifying destruction methods are invoked. class DestroyTrackingDelegateImpl : public TestWindowDelegate { public: DestroyTrackingDelegateImpl() : destroying_count_(0), destroyed_count_(0), in_destroying_(false) {} void clear_destroying_count() { destroying_count_ = 0; } int destroying_count() const { return destroying_count_; } void clear_destroyed_count() { destroyed_count_ = 0; } int destroyed_count() const { return destroyed_count_; } bool in_destroying() const { return in_destroying_; } virtual void OnWindowDestroying() OVERRIDE { EXPECT_FALSE(in_destroying_); in_destroying_ = true; destroying_count_++; } virtual void OnWindowDestroyed() OVERRIDE { EXPECT_TRUE(in_destroying_); in_destroying_ = false; destroyed_count_++; } private: int destroying_count_; int destroyed_count_; bool in_destroying_; DISALLOW_COPY_AND_ASSIGN(DestroyTrackingDelegateImpl); }; // Used to verify that when OnWindowDestroying is invoked the parent is also // is in the process of being destroyed. class ChildWindowDelegateImpl : public DestroyTrackingDelegateImpl { public: explicit ChildWindowDelegateImpl( DestroyTrackingDelegateImpl* parent_delegate) : parent_delegate_(parent_delegate) { } virtual void OnWindowDestroying() OVERRIDE { EXPECT_TRUE(parent_delegate_->in_destroying()); DestroyTrackingDelegateImpl::OnWindowDestroying(); } private: DestroyTrackingDelegateImpl* parent_delegate_; DISALLOW_COPY_AND_ASSIGN(ChildWindowDelegateImpl); }; // Used to verify that a Window is removed from its parent when // OnWindowDestroyed is called. class DestroyOrphanDelegate : public TestWindowDelegate { public: DestroyOrphanDelegate() : window_(NULL) { } void set_window(Window* window) { window_ = window; } virtual void OnWindowDestroyed() OVERRIDE { EXPECT_FALSE(window_->parent()); } private: Window* window_; DISALLOW_COPY_AND_ASSIGN(DestroyOrphanDelegate); }; // Used in verifying mouse capture. class CaptureWindowDelegateImpl : public TestWindowDelegate { public: CaptureWindowDelegateImpl() { ResetCounts(); } void ResetCounts() { capture_changed_event_count_ = 0; capture_lost_count_ = 0; mouse_event_count_ = 0; touch_event_count_ = 0; gesture_event_count_ = 0; } int capture_changed_event_count() const { return capture_changed_event_count_; } int capture_lost_count() const { return capture_lost_count_; } int mouse_event_count() const { return mouse_event_count_; } int touch_event_count() const { return touch_event_count_; } int gesture_event_count() const { return gesture_event_count_; } virtual bool OnMouseEvent(MouseEvent* event) OVERRIDE { if (event->type() == ui::ET_MOUSE_CAPTURE_CHANGED) capture_changed_event_count_++; mouse_event_count_++; return false; } virtual ui::TouchStatus OnTouchEvent(TouchEvent* event) OVERRIDE { touch_event_count_++; return ui::TOUCH_STATUS_UNKNOWN; } virtual ui::GestureStatus OnGestureEvent(GestureEvent* event) OVERRIDE { gesture_event_count_++; return ui::GESTURE_STATUS_UNKNOWN; } virtual void OnCaptureLost() OVERRIDE { capture_lost_count_++; } private: int capture_changed_event_count_; int capture_lost_count_; int mouse_event_count_; int touch_event_count_; int gesture_event_count_; DISALLOW_COPY_AND_ASSIGN(CaptureWindowDelegateImpl); }; // Keeps track of the location of the gesture. class GestureTrackPositionDelegate : public TestWindowDelegate { public: GestureTrackPositionDelegate() {} virtual ui::GestureStatus OnGestureEvent(GestureEvent* event) OVERRIDE { position_ = event->location(); return ui::GESTURE_STATUS_CONSUMED; } const gfx::Point& position() const { return position_; } private: gfx::Point position_; DISALLOW_COPY_AND_ASSIGN(GestureTrackPositionDelegate); }; // Keeps track of mouse events. class MouseTrackingDelegate : public TestWindowDelegate { public: MouseTrackingDelegate() : mouse_enter_count_(0), mouse_move_count_(0), mouse_leave_count_(0) { } virtual bool OnMouseEvent(MouseEvent* event) OVERRIDE { switch (event->type()) { case ui::ET_MOUSE_MOVED: mouse_move_count_++; break; case ui::ET_MOUSE_ENTERED: mouse_enter_count_++; break; case ui::ET_MOUSE_EXITED: mouse_leave_count_++; break; default: break; } return false; } std::string GetMouseCountsAndReset() { std::string result = StringPrintf("%d %d %d", mouse_enter_count_, mouse_move_count_, mouse_leave_count_); mouse_enter_count_ = 0; mouse_move_count_ = 0; mouse_leave_count_ = 0; return result; } private: int mouse_enter_count_; int mouse_move_count_; int mouse_leave_count_; DISALLOW_COPY_AND_ASSIGN(MouseTrackingDelegate); }; base::TimeDelta getTime() { return base::Time::NowFromSystemTime() - base::Time(); } } // namespace TEST_F(WindowTest, GetChildById) { scoped_ptr w1(CreateTestWindowWithId(1, NULL)); scoped_ptr w11(CreateTestWindowWithId(11, w1.get())); scoped_ptr w111(CreateTestWindowWithId(111, w11.get())); scoped_ptr w12(CreateTestWindowWithId(12, w1.get())); EXPECT_EQ(NULL, w1->GetChildById(57)); EXPECT_EQ(w12.get(), w1->GetChildById(12)); EXPECT_EQ(w111.get(), w1->GetChildById(111)); } // Make sure that Window::Contains correctly handles children, grandchildren, // and not containing NULL or parents. TEST_F(WindowTest, Contains) { Window parent(NULL); parent.Init(ui::LAYER_NOT_DRAWN); Window child1(NULL); child1.Init(ui::LAYER_NOT_DRAWN); Window child2(NULL); child2.Init(ui::LAYER_NOT_DRAWN); child1.SetParent(&parent); child2.SetParent(&child1); EXPECT_TRUE(parent.Contains(&parent)); EXPECT_TRUE(parent.Contains(&child1)); EXPECT_TRUE(parent.Contains(&child2)); EXPECT_FALSE(parent.Contains(NULL)); EXPECT_FALSE(child1.Contains(&parent)); EXPECT_FALSE(child2.Contains(&child1)); } TEST_F(WindowTest, ConvertPointToWindow) { // Window::ConvertPointToWindow is mostly identical to // Layer::ConvertPointToLayer, except NULL values for |source| are permitted, // in which case the function just returns. scoped_ptr w1(CreateTestWindowWithId(1, NULL)); gfx::Point reference_point(100, 100); gfx::Point test_point = reference_point; Window::ConvertPointToWindow(NULL, w1.get(), &test_point); EXPECT_EQ(reference_point, test_point); } TEST_F(WindowTest, HitTest) { Window w1(new ColorTestWindowDelegate(SK_ColorWHITE)); w1.set_id(1); w1.Init(ui::LAYER_TEXTURED); w1.SetBounds(gfx::Rect(10, 20, 50, 60)); w1.Show(); w1.SetParent(NULL); // Points are in the Window's coordinates. EXPECT_TRUE(w1.HitTest(gfx::Point(1, 1))); EXPECT_FALSE(w1.HitTest(gfx::Point(-1, -1))); // We can expand the bounds slightly to track events outside our border. w1.set_hit_test_bounds_override_outer(gfx::Insets(-1, -1, -1, -1)); EXPECT_TRUE(w1.HitTest(gfx::Point(-1, -1))); EXPECT_FALSE(w1.HitTest(gfx::Point(-2, -2))); // TODO(beng): clip Window to parent. } TEST_F(WindowTest, GetEventHandlerForPoint) { scoped_ptr w1( CreateTestWindow(SK_ColorWHITE, 1, gfx::Rect(10, 10, 500, 500), NULL)); scoped_ptr w11( CreateTestWindow(SK_ColorGREEN, 11, gfx::Rect(5, 5, 100, 100), w1.get())); scoped_ptr w111( CreateTestWindow(SK_ColorCYAN, 111, gfx::Rect(5, 5, 75, 75), w11.get())); scoped_ptr w1111( CreateTestWindow(SK_ColorRED, 1111, gfx::Rect(5, 5, 50, 50), w111.get())); scoped_ptr w12( CreateTestWindow(SK_ColorMAGENTA, 12, gfx::Rect(10, 420, 25, 25), w1.get())); scoped_ptr w121( CreateTestWindow(SK_ColorYELLOW, 121, gfx::Rect(5, 5, 5, 5), w12.get())); scoped_ptr w13( CreateTestWindow(SK_ColorGRAY, 13, gfx::Rect(5, 470, 50, 50), w1.get())); Window* root = root_window(); w1->parent()->SetBounds(gfx::Rect(500, 500)); EXPECT_EQ(NULL, root->GetEventHandlerForPoint(gfx::Point(5, 5))); EXPECT_EQ(w1.get(), root->GetEventHandlerForPoint(gfx::Point(11, 11))); EXPECT_EQ(w11.get(), root->GetEventHandlerForPoint(gfx::Point(16, 16))); EXPECT_EQ(w111.get(), root->GetEventHandlerForPoint(gfx::Point(21, 21))); EXPECT_EQ(w1111.get(), root->GetEventHandlerForPoint(gfx::Point(26, 26))); EXPECT_EQ(w12.get(), root->GetEventHandlerForPoint(gfx::Point(21, 431))); EXPECT_EQ(w121.get(), root->GetEventHandlerForPoint(gfx::Point(26, 436))); EXPECT_EQ(w13.get(), root->GetEventHandlerForPoint(gfx::Point(26, 481))); } TEST_F(WindowTest, GetEventHandlerForPointWithOverride) { // If our child is flush to our top-left corner he gets events just inside the // window edges. scoped_ptr parent( CreateTestWindow(SK_ColorWHITE, 1, gfx::Rect(10, 20, 400, 500), NULL)); scoped_ptr child( CreateTestWindow(SK_ColorRED, 2, gfx::Rect(0, 0, 60, 70), parent.get())); EXPECT_EQ(child.get(), parent->GetEventHandlerForPoint(gfx::Point(0, 0))); EXPECT_EQ(child.get(), parent->GetEventHandlerForPoint(gfx::Point(1, 1))); // We can override the hit test bounds of the parent to make the parent grab // events along that edge. parent->set_hit_test_bounds_override_inner(gfx::Insets(1, 1, 1, 1)); EXPECT_EQ(parent.get(), parent->GetEventHandlerForPoint(gfx::Point(0, 0))); EXPECT_EQ(child.get(), parent->GetEventHandlerForPoint(gfx::Point(1, 1))); } TEST_F(WindowTest, GetTopWindowContainingPoint) { Window* root = root_window(); root->SetBounds(gfx::Rect(0, 0, 300, 300)); scoped_ptr w1( CreateTestWindow(SK_ColorWHITE, 1, gfx::Rect(10, 10, 100, 100), NULL)); scoped_ptr w11( CreateTestWindow(SK_ColorGREEN, 11, gfx::Rect(0, 0, 120, 120), w1.get())); scoped_ptr w2( CreateTestWindow(SK_ColorRED, 2, gfx::Rect(5, 5, 55, 55), NULL)); scoped_ptr w3( CreateTestWindowWithDelegate( NULL, 3, gfx::Rect(200, 200, 100, 100), NULL)); scoped_ptr w31( CreateTestWindow(SK_ColorCYAN, 31, gfx::Rect(0, 0, 50, 50), w3.get())); scoped_ptr w311( CreateTestWindow(SK_ColorBLUE, 311, gfx::Rect(0, 0, 10, 10), w31.get())); EXPECT_EQ(NULL, root->GetTopWindowContainingPoint(gfx::Point(0, 0))); EXPECT_EQ(w2.get(), root->GetTopWindowContainingPoint(gfx::Point(5, 5))); EXPECT_EQ(w2.get(), root->GetTopWindowContainingPoint(gfx::Point(10, 10))); EXPECT_EQ(w2.get(), root->GetTopWindowContainingPoint(gfx::Point(59, 59))); EXPECT_EQ(w1.get(), root->GetTopWindowContainingPoint(gfx::Point(60, 60))); EXPECT_EQ(w1.get(), root->GetTopWindowContainingPoint(gfx::Point(109, 109))); EXPECT_EQ(NULL, root->GetTopWindowContainingPoint(gfx::Point(110, 110))); EXPECT_EQ(w31.get(), root->GetTopWindowContainingPoint(gfx::Point(200, 200))); EXPECT_EQ(w31.get(), root->GetTopWindowContainingPoint(gfx::Point(220, 220))); EXPECT_EQ(NULL, root->GetTopWindowContainingPoint(gfx::Point(260, 260))); } TEST_F(WindowTest, GetToplevelWindow) { const gfx::Rect kBounds(0, 0, 10, 10); TestWindowDelegate delegate; scoped_ptr w1(CreateTestWindowWithId(1, root_window())); scoped_ptr w11( CreateTestWindowWithDelegate(&delegate, 11, kBounds, w1.get())); scoped_ptr w111(CreateTestWindowWithId(111, w11.get())); scoped_ptr w1111( CreateTestWindowWithDelegate(&delegate, 1111, kBounds, w111.get())); EXPECT_TRUE(root_window()->GetToplevelWindow() == NULL); EXPECT_TRUE(w1->GetToplevelWindow() == NULL); EXPECT_EQ(w11.get(), w11->GetToplevelWindow()); EXPECT_EQ(w11.get(), w111->GetToplevelWindow()); EXPECT_EQ(w11.get(), w1111->GetToplevelWindow()); } // Various destruction assertions. TEST_F(WindowTest, DestroyTest) { DestroyTrackingDelegateImpl parent_delegate; ChildWindowDelegateImpl child_delegate(&parent_delegate); { scoped_ptr parent( CreateTestWindowWithDelegate(&parent_delegate, 0, gfx::Rect(), NULL)); CreateTestWindowWithDelegate(&child_delegate, 0, gfx::Rect(), parent.get()); } // Both the parent and child should have been destroyed. EXPECT_EQ(1, parent_delegate.destroying_count()); EXPECT_EQ(1, parent_delegate.destroyed_count()); EXPECT_EQ(1, child_delegate.destroying_count()); EXPECT_EQ(1, child_delegate.destroyed_count()); } // Tests that a window is orphaned before OnWindowDestroyed is called. TEST_F(WindowTest, OrphanedBeforeOnDestroyed) { TestWindowDelegate parent_delegate; DestroyOrphanDelegate child_delegate; { scoped_ptr parent( CreateTestWindowWithDelegate(&parent_delegate, 0, gfx::Rect(), NULL)); scoped_ptr child(CreateTestWindowWithDelegate(&child_delegate, 0, gfx::Rect(), parent.get())); child_delegate.set_window(child.get()); } } // Make sure StackChildAtTop moves both the window and layer to the front. TEST_F(WindowTest, StackChildAtTop) { Window parent(NULL); parent.Init(ui::LAYER_NOT_DRAWN); Window child1(NULL); child1.Init(ui::LAYER_NOT_DRAWN); Window child2(NULL); child2.Init(ui::LAYER_NOT_DRAWN); child1.SetParent(&parent); child2.SetParent(&parent); ASSERT_EQ(2u, parent.children().size()); EXPECT_EQ(&child1, parent.children()[0]); EXPECT_EQ(&child2, parent.children()[1]); ASSERT_EQ(2u, parent.layer()->children().size()); EXPECT_EQ(child1.layer(), parent.layer()->children()[0]); EXPECT_EQ(child2.layer(), parent.layer()->children()[1]); parent.StackChildAtTop(&child1); ASSERT_EQ(2u, parent.children().size()); EXPECT_EQ(&child1, parent.children()[1]); EXPECT_EQ(&child2, parent.children()[0]); ASSERT_EQ(2u, parent.layer()->children().size()); EXPECT_EQ(child1.layer(), parent.layer()->children()[1]); EXPECT_EQ(child2.layer(), parent.layer()->children()[0]); } // Make sure StackChildBelow works. TEST_F(WindowTest, StackChildBelow) { Window parent(NULL); parent.Init(ui::LAYER_NOT_DRAWN); Window child1(NULL); child1.Init(ui::LAYER_NOT_DRAWN); child1.set_id(1); Window child2(NULL); child2.Init(ui::LAYER_NOT_DRAWN); child2.set_id(2); Window child3(NULL); child3.Init(ui::LAYER_NOT_DRAWN); child3.set_id(3); child1.SetParent(&parent); child2.SetParent(&parent); child3.SetParent(&parent); EXPECT_EQ("1 2 3", ChildWindowIDsAsString(&parent)); parent.StackChildBelow(&child1, &child2); EXPECT_EQ("1 2 3", ChildWindowIDsAsString(&parent)); parent.StackChildBelow(&child2, &child1); EXPECT_EQ("2 1 3", ChildWindowIDsAsString(&parent)); parent.StackChildBelow(&child3, &child2); EXPECT_EQ("3 2 1", ChildWindowIDsAsString(&parent)); parent.StackChildBelow(&child3, &child1); EXPECT_EQ("2 3 1", ChildWindowIDsAsString(&parent)); } // Various assertions for StackChildAbove. TEST_F(WindowTest, StackChildAbove) { Window parent(NULL); parent.Init(ui::LAYER_NOT_DRAWN); Window child1(NULL); child1.Init(ui::LAYER_NOT_DRAWN); Window child2(NULL); child2.Init(ui::LAYER_NOT_DRAWN); Window child3(NULL); child3.Init(ui::LAYER_NOT_DRAWN); child1.SetParent(&parent); child2.SetParent(&parent); // Move 1 in front of 2. parent.StackChildAbove(&child1, &child2); ASSERT_EQ(2u, parent.children().size()); EXPECT_EQ(&child2, parent.children()[0]); EXPECT_EQ(&child1, parent.children()[1]); ASSERT_EQ(2u, parent.layer()->children().size()); EXPECT_EQ(child2.layer(), parent.layer()->children()[0]); EXPECT_EQ(child1.layer(), parent.layer()->children()[1]); // Add 3, resulting in order [2, 1, 3], then move 2 in front of 1, resulting // in [1, 2, 3]. child3.SetParent(&parent); parent.StackChildAbove(&child2, &child1); ASSERT_EQ(3u, parent.children().size()); EXPECT_EQ(&child1, parent.children()[0]); EXPECT_EQ(&child2, parent.children()[1]); EXPECT_EQ(&child3, parent.children()[2]); ASSERT_EQ(3u, parent.layer()->children().size()); EXPECT_EQ(child1.layer(), parent.layer()->children()[0]); EXPECT_EQ(child2.layer(), parent.layer()->children()[1]); EXPECT_EQ(child3.layer(), parent.layer()->children()[2]); // Move 1 in front of 3, resulting in [2, 3, 1]. parent.StackChildAbove(&child1, &child3); ASSERT_EQ(3u, parent.children().size()); EXPECT_EQ(&child2, parent.children()[0]); EXPECT_EQ(&child3, parent.children()[1]); EXPECT_EQ(&child1, parent.children()[2]); ASSERT_EQ(3u, parent.layer()->children().size()); EXPECT_EQ(child2.layer(), parent.layer()->children()[0]); EXPECT_EQ(child3.layer(), parent.layer()->children()[1]); EXPECT_EQ(child1.layer(), parent.layer()->children()[2]); // Moving 1 in front of 2 should lower it, resulting in [2, 1, 3]. parent.StackChildAbove(&child1, &child2); ASSERT_EQ(3u, parent.children().size()); EXPECT_EQ(&child2, parent.children()[0]); EXPECT_EQ(&child1, parent.children()[1]); EXPECT_EQ(&child3, parent.children()[2]); ASSERT_EQ(3u, parent.layer()->children().size()); EXPECT_EQ(child2.layer(), parent.layer()->children()[0]); EXPECT_EQ(child1.layer(), parent.layer()->children()[1]); EXPECT_EQ(child3.layer(), parent.layer()->children()[2]); } // Various capture assertions. TEST_F(WindowTest, CaptureTests) { CaptureWindowDelegateImpl delegate; scoped_ptr window(CreateTestWindowWithDelegate( &delegate, 0, gfx::Rect(0, 0, 20, 20), NULL)); EXPECT_FALSE(window->HasCapture()); delegate.ResetCounts(); // Do a capture. window->SetCapture(); EXPECT_TRUE(window->HasCapture()); EXPECT_EQ(0, delegate.capture_lost_count()); EXPECT_EQ(0, delegate.capture_changed_event_count()); EventGenerator generator(root_window(), gfx::Point(50, 50)); generator.PressLeftButton(); EXPECT_EQ(1, delegate.mouse_event_count()); generator.ReleaseLeftButton(); EXPECT_EQ(2, delegate.mouse_event_count()); delegate.ResetCounts(); TouchEvent touchev(ui::ET_TOUCH_PRESSED, gfx::Point(50, 50), 0, getTime()); root_window()->DispatchTouchEvent(&touchev); EXPECT_EQ(1, delegate.touch_event_count()); delegate.ResetCounts(); window->ReleaseCapture(); EXPECT_FALSE(window->HasCapture()); EXPECT_EQ(1, delegate.capture_lost_count()); EXPECT_EQ(1, delegate.capture_changed_event_count()); EXPECT_EQ(1, delegate.mouse_event_count()); EXPECT_EQ(1, delegate.touch_event_count()); generator.PressLeftButton(); EXPECT_EQ(1, delegate.mouse_event_count()); TouchEvent touchev2(ui::ET_TOUCH_PRESSED, gfx::Point(250, 250), 1, getTime()); root_window()->DispatchTouchEvent(&touchev2); EXPECT_EQ(1, delegate.touch_event_count()); // Removing the capture window from parent should reset the capture window // in the root window. window->SetCapture(); EXPECT_EQ(window.get(), root_window()->capture_window()); window->parent()->RemoveChild(window.get()); EXPECT_FALSE(window->HasCapture()); EXPECT_EQ(NULL, root_window()->capture_window()); } TEST_F(WindowTest, TouchCaptureCancelsOtherTouches) { CaptureWindowDelegateImpl delegate1; scoped_ptr w1(CreateTestWindowWithDelegate( &delegate1, 0, gfx::Rect(0, 0, 20, 20), NULL)); CaptureWindowDelegateImpl delegate2; scoped_ptr w2(CreateTestWindowWithDelegate( &delegate2, 0, gfx::Rect(20, 20, 20, 20), NULL)); // Press on w1. TouchEvent press(ui::ET_TOUCH_PRESSED, gfx::Point(10, 10), 0, getTime()); root_window()->DispatchTouchEvent(&press); EXPECT_EQ(1, delegate1.gesture_event_count()); delegate1.ResetCounts(); w2->SetCapture(); // The touch was cancelled when the other window // attained a touch lock. EXPECT_EQ(1, delegate1.touch_event_count()); EXPECT_EQ(0, delegate2.touch_event_count()); delegate1.ResetCounts(); delegate2.ResetCounts(); TouchEvent move(ui::ET_TOUCH_MOVED, gfx::Point(10, 10), 0, getTime()); root_window()->DispatchTouchEvent(&move); // This touch id is now ignored, no scroll fired. EXPECT_EQ(0, delegate1.gesture_event_count()); EXPECT_EQ(0, delegate2.gesture_event_count()); TouchEvent release(ui::ET_TOUCH_RELEASED, gfx::Point(10, 10), 0, getTime()); root_window()->DispatchTouchEvent(&release); EXPECT_EQ(0, delegate1.gesture_event_count()); EXPECT_EQ(0, delegate2.gesture_event_count()); // A new press is captured by w2. TouchEvent press2(ui::ET_TOUCH_PRESSED, gfx::Point(10, 10), 0, getTime()); root_window()->DispatchTouchEvent(&press2); EXPECT_EQ(0, delegate1.gesture_event_count()); EXPECT_EQ(1, delegate2.gesture_event_count()); } TEST_F(WindowTest, TouchCaptureDoesntCancelCapturedTouches) { CaptureWindowDelegateImpl delegate; scoped_ptr window(CreateTestWindowWithDelegate( &delegate, 0, gfx::Rect(0, 0, 20, 20), NULL)); TouchEvent press(ui::ET_TOUCH_PRESSED, gfx::Point(10, 10), 0, getTime()); root_window()->DispatchTouchEvent(&press); EXPECT_EQ(1, delegate.gesture_event_count()); delegate.ResetCounts(); window->SetCapture(); EXPECT_EQ(0, delegate.gesture_event_count()); delegate.ResetCounts(); // The move event should still create a gesture, as this touch was // on the window which was captured. TouchEvent release(ui::ET_TOUCH_RELEASED, gfx::Point(10, 10), 0, getTime() + base::TimeDelta::FromMilliseconds(50)); root_window()->DispatchTouchEvent(&release); EXPECT_EQ(2, delegate.gesture_event_count()); } // Changes capture while capture is already ongoing. TEST_F(WindowTest, ChangeCaptureWhileMouseDown) { CaptureWindowDelegateImpl delegate; scoped_ptr window(CreateTestWindowWithDelegate( &delegate, 0, gfx::Rect(0, 0, 20, 20), NULL)); CaptureWindowDelegateImpl delegate2; scoped_ptr w2(CreateTestWindowWithDelegate( &delegate2, 0, gfx::Rect(20, 20, 20, 20), NULL)); // Execute the scheduled draws so that mouse events are not // aggregated. RunAllPendingInMessageLoop(); EXPECT_FALSE(window->HasCapture()); // Do a capture. delegate.ResetCounts(); window->SetCapture(); EXPECT_TRUE(window->HasCapture()); EXPECT_EQ(0, delegate.capture_lost_count()); EXPECT_EQ(0, delegate.capture_changed_event_count()); EventGenerator generator(root_window(), gfx::Point(50, 50)); generator.PressLeftButton(); EXPECT_EQ(0, delegate.capture_lost_count()); EXPECT_EQ(0, delegate.capture_changed_event_count()); EXPECT_EQ(1, delegate.mouse_event_count()); // Set capture to |w2|, should implicitly unset capture for |window|. delegate.ResetCounts(); delegate2.ResetCounts(); w2->SetCapture(); generator.MoveMouseTo(gfx::Point(40, 40), 2); EXPECT_EQ(1, delegate.capture_lost_count()); EXPECT_EQ(1, delegate.capture_changed_event_count()); EXPECT_EQ(1, delegate.mouse_event_count()); EXPECT_EQ(2, delegate2.mouse_event_count()); } // Verifies capture is reset when a window is destroyed. TEST_F(WindowTest, ReleaseCaptureOnDestroy) { CaptureWindowDelegateImpl delegate; scoped_ptr window(CreateTestWindowWithDelegate( &delegate, 0, gfx::Rect(0, 0, 20, 20), NULL)); EXPECT_FALSE(window->HasCapture()); // Do a capture. window->SetCapture(); EXPECT_TRUE(window->HasCapture()); // Destroy the window. window.reset(); // Make sure the root window doesn't reference the window anymore. EXPECT_EQ(NULL, root_window()->mouse_pressed_handler()); EXPECT_EQ(NULL, root_window()->capture_window()); } TEST_F(WindowTest, GetBoundsInRootWindow) { scoped_ptr viewport(CreateTestWindowWithBounds( gfx::Rect(0, 0, 300, 300), NULL)); scoped_ptr child(CreateTestWindowWithBounds( gfx::Rect(0, 0, 100, 100), viewport.get())); // Sanity check. EXPECT_EQ("0,0 100x100", child->GetBoundsInRootWindow().ToString()); // The |child| window's screen bounds should move along with the |viewport|. viewport->SetBounds(gfx::Rect(-100, -100, 300, 300)); EXPECT_EQ("-100,-100 100x100", child->GetBoundsInRootWindow().ToString()); // The |child| window is moved to the 0,0 in screen coordinates. // |GetBoundsInRootWindow()| should return 0,0. child->SetBounds(gfx::Rect(100, 100, 100, 100)); EXPECT_EQ("0,0 100x100", child->GetBoundsInRootWindow().ToString()); } class MouseEnterExitWindowDelegate : public TestWindowDelegate { public: MouseEnterExitWindowDelegate() : entered_(false), exited_(false) {} virtual bool OnMouseEvent(MouseEvent* event) OVERRIDE { switch (event->type()) { case ui::ET_MOUSE_ENTERED: entered_ = true; break; case ui::ET_MOUSE_EXITED: exited_ = true; break; default: break; } return false; } bool entered() const { return entered_; } bool exited() const { return exited_; } private: bool entered_; bool exited_; DISALLOW_COPY_AND_ASSIGN(MouseEnterExitWindowDelegate); }; // Verifies that the WindowDelegate receives MouseExit and MouseEnter events for // mouse transitions from window to window. TEST_F(WindowTest, MouseEnterExit) { MouseEnterExitWindowDelegate d1; scoped_ptr w1( CreateTestWindowWithDelegate(&d1, 1, gfx::Rect(10, 10, 50, 50), NULL)); MouseEnterExitWindowDelegate d2; scoped_ptr w2( CreateTestWindowWithDelegate(&d2, 2, gfx::Rect(70, 70, 50, 50), NULL)); test::EventGenerator generator(root_window()); generator.MoveMouseToCenterOf(w1.get()); EXPECT_TRUE(d1.entered()); EXPECT_FALSE(d1.exited()); EXPECT_FALSE(d2.entered()); EXPECT_FALSE(d2.exited()); generator.MoveMouseToCenterOf(w2.get()); EXPECT_TRUE(d1.entered()); EXPECT_TRUE(d1.exited()); EXPECT_TRUE(d2.entered()); EXPECT_FALSE(d2.exited()); } // Creates a window with a delegate (w111) that can handle events at a lower // z-index than a window without a delegate (w12). w12 is sized to fill the // entire bounds of the container. This test verifies that // GetEventHandlerForPoint() skips w12 even though its bounds contain the event, // because it has no children that can handle the event and it has no delegate // allowing it to handle the event itself. TEST_F(WindowTest, GetEventHandlerForPoint_NoDelegate) { TestWindowDelegate d111; scoped_ptr w1(CreateTestWindowWithDelegate(NULL, 1, gfx::Rect(0, 0, 500, 500), NULL)); scoped_ptr w11(CreateTestWindowWithDelegate(NULL, 11, gfx::Rect(0, 0, 500, 500), w1.get())); scoped_ptr w111(CreateTestWindowWithDelegate(&d111, 111, gfx::Rect(50, 50, 450, 450), w11.get())); scoped_ptr w12(CreateTestWindowWithDelegate(NULL, 12, gfx::Rect(0, 0, 500, 500), w1.get())); gfx::Point target_point = w111->bounds().CenterPoint(); EXPECT_EQ(w111.get(), w1->GetEventHandlerForPoint(target_point)); } class VisibilityWindowDelegate : public TestWindowDelegate { public: VisibilityWindowDelegate() : shown_(0), hidden_(0) { } int shown() const { return shown_; } int hidden() const { return hidden_; } void Clear() { shown_ = 0; hidden_ = 0; } virtual void OnWindowVisibilityChanged(bool visible) OVERRIDE { if (visible) shown_++; else hidden_++; } private: int shown_; int hidden_; DISALLOW_COPY_AND_ASSIGN(VisibilityWindowDelegate); }; // Verifies show/hide propagate correctly to children and the layer. TEST_F(WindowTest, Visibility) { VisibilityWindowDelegate d; VisibilityWindowDelegate d2; scoped_ptr w1(CreateTestWindowWithDelegate(&d, 1, gfx::Rect(), NULL)); scoped_ptr w2( CreateTestWindowWithDelegate(&d2, 2, gfx::Rect(), w1.get())); scoped_ptr w3(CreateTestWindowWithId(3, w2.get())); // Create shows all the windows. EXPECT_TRUE(w1->IsVisible()); EXPECT_TRUE(w2->IsVisible()); EXPECT_TRUE(w3->IsVisible()); EXPECT_EQ(1, d.shown()); d.Clear(); w1->Hide(); EXPECT_FALSE(w1->IsVisible()); EXPECT_FALSE(w2->IsVisible()); EXPECT_FALSE(w3->IsVisible()); EXPECT_EQ(1, d.hidden()); EXPECT_EQ(0, d.shown()); w2->Show(); EXPECT_FALSE(w1->IsVisible()); EXPECT_FALSE(w2->IsVisible()); EXPECT_FALSE(w3->IsVisible()); w3->Hide(); EXPECT_FALSE(w1->IsVisible()); EXPECT_FALSE(w2->IsVisible()); EXPECT_FALSE(w3->IsVisible()); d.Clear(); w1->Show(); EXPECT_TRUE(w1->IsVisible()); EXPECT_TRUE(w2->IsVisible()); EXPECT_FALSE(w3->IsVisible()); EXPECT_EQ(0, d.hidden()); EXPECT_EQ(1, d.shown()); w3->Show(); EXPECT_TRUE(w1->IsVisible()); EXPECT_TRUE(w2->IsVisible()); EXPECT_TRUE(w3->IsVisible()); // Verify that if an ancestor isn't visible and we change the visibility of a // child window that OnChildWindowVisibilityChanged() is still invoked. w1->Hide(); d2.Clear(); w2->Hide(); EXPECT_EQ(1, d2.hidden()); EXPECT_EQ(0, d2.shown()); d2.Clear(); w2->Show(); EXPECT_EQ(0, d2.hidden()); EXPECT_EQ(1, d2.shown()); } TEST_F(WindowTest, IgnoreEventsTest) { TestWindowDelegate d11; TestWindowDelegate d12; TestWindowDelegate d111; TestWindowDelegate d121; scoped_ptr w1(CreateTestWindowWithDelegate(NULL, 1, gfx::Rect(0, 0, 500, 500), NULL)); scoped_ptr w11(CreateTestWindowWithDelegate(&d11, 11, gfx::Rect(0, 0, 500, 500), w1.get())); scoped_ptr w111(CreateTestWindowWithDelegate(&d111, 111, gfx::Rect(50, 50, 450, 450), w11.get())); scoped_ptr w12(CreateTestWindowWithDelegate(&d12, 12, gfx::Rect(0, 0, 500, 500), w1.get())); scoped_ptr w121(CreateTestWindowWithDelegate(&d121, 121, gfx::Rect(150, 150, 50, 50), w12.get())); EXPECT_EQ(w12.get(), w1->GetEventHandlerForPoint(gfx::Point(10, 10))); w12->set_ignore_events(true); EXPECT_EQ(w11.get(), w1->GetEventHandlerForPoint(gfx::Point(10, 10))); w12->set_ignore_events(false); EXPECT_EQ(w121.get(), w1->GetEventHandlerForPoint(gfx::Point(160, 160))); w121->set_ignore_events(true); EXPECT_EQ(w12.get(), w1->GetEventHandlerForPoint(gfx::Point(160, 160))); w12->set_ignore_events(true); EXPECT_EQ(w111.get(), w1->GetEventHandlerForPoint(gfx::Point(160, 160))); w111->set_ignore_events(true); EXPECT_EQ(w11.get(), w1->GetEventHandlerForPoint(gfx::Point(160, 160))); } // Tests transformation on the root window. TEST_F(WindowTest, Transform) { gfx::Size size = root_window()->GetHostSize(); EXPECT_EQ(gfx::Rect(size), gfx::Screen::GetMonitorNearestPoint(gfx::Point()).bounds()); // Rotate it clock-wise 90 degrees. ui::Transform transform; transform.SetRotate(90.0f); transform.ConcatTranslate(size.height(), 0); root_window()->SetTransform(transform); // The size should be the transformed size. gfx::Size transformed_size(size.height(), size.width()); EXPECT_EQ(transformed_size.ToString(), root_window()->bounds().size().ToString()); EXPECT_EQ( gfx::Rect(transformed_size).ToString(), gfx::Screen::GetMonitorNearestPoint(gfx::Point()).bounds().ToString()); // Host size shouldn't change. EXPECT_EQ(size.ToString(), root_window()->GetHostSize().ToString()); } TEST_F(WindowTest, TransformGesture) { gfx::Size size = root_window()->GetHostSize(); scoped_ptr delegate( new GestureTrackPositionDelegate); scoped_ptr window(CreateTestWindowWithDelegate(delegate.get(), -1234, gfx::Rect(0, 0, 20, 20), NULL)); // Rotate the root-window clock-wise 90 degrees. ui::Transform transform; transform.SetRotate(90.0f); transform.ConcatTranslate(size.height(), 0); root_window()->SetTransform(transform); TouchEvent press(ui::ET_TOUCH_PRESSED, gfx::Point(size.height() - 10, 10), 0, getTime()); root_window()->DispatchTouchEvent(&press); EXPECT_EQ(gfx::Point(10, 10).ToString(), delegate->position().ToString()); } // Various assertions for transient children. TEST_F(WindowTest, TransientChildren) { scoped_ptr parent(CreateTestWindowWithId(0, NULL)); scoped_ptr w1(CreateTestWindowWithId(1, parent.get())); scoped_ptr w3(CreateTestWindowWithId(3, parent.get())); Window* w2 = CreateTestWindowWithId(2, parent.get()); w1->AddTransientChild(w2); // w2 is now owned by w1. // Stack w1 at the top (end), this should force w2 to be last (on top of w1). parent->StackChildAtTop(w1.get()); ASSERT_EQ(3u, parent->children().size()); EXPECT_EQ(w2, parent->children().back()); // Destroy w1, which should also destroy w3 (since it's a transient child). w1.reset(); w2 = NULL; ASSERT_EQ(1u, parent->children().size()); EXPECT_EQ(w3.get(), parent->children()[0]); w1.reset(CreateTestWindowWithId(4, parent.get())); w2 = CreateTestWindowWithId(5, w3.get()); w1->AddTransientChild(w2); parent->StackChildAtTop(w3.get()); // Stack w1 at the top (end), this shouldn't affect w2 since it has a // different parent. parent->StackChildAtTop(w1.get()); ASSERT_EQ(2u, parent->children().size()); EXPECT_EQ(w3.get(), parent->children()[0]); EXPECT_EQ(w1.get(), parent->children()[1]); // Hiding parent should hide transient children. EXPECT_TRUE(w2->IsVisible()); w1->Hide(); EXPECT_FALSE(w2->IsVisible()); } // Tests that when a focused window is closed, its parent inherits the focus. TEST_F(WindowTest, FocusedWindowTest) { scoped_ptr parent(CreateTestWindowWithId(0, NULL)); scoped_ptr child(CreateTestWindowWithId(1, parent.get())); parent->Show(); child->Focus(); EXPECT_TRUE(child->HasFocus()); EXPECT_FALSE(parent->HasFocus()); child.reset(); EXPECT_TRUE(parent->HasFocus()); } namespace { DEFINE_WINDOW_PROPERTY_KEY(int, kIntKey, -2); DEFINE_WINDOW_PROPERTY_KEY(const char*, kStringKey, "squeamish"); } TEST_F(WindowTest, Property) { scoped_ptr w(CreateTestWindowWithId(0, NULL)); static const char native_prop_key[] = "fnord"; // Non-existent properties should return the default values. EXPECT_EQ(-2, w->GetProperty(kIntKey)); EXPECT_EQ(std::string("squeamish"), w->GetProperty(kStringKey)); EXPECT_EQ(NULL, w->GetNativeWindowProperty(native_prop_key)); // A set property value should be returned again (even if it's the default // value). w->SetProperty(kIntKey, INT_MAX); EXPECT_EQ(INT_MAX, w->GetProperty(kIntKey)); w->SetProperty(kIntKey, -2); EXPECT_EQ(-2, w->GetProperty(kIntKey)); w->SetProperty(kIntKey, INT_MIN); EXPECT_EQ(INT_MIN, w->GetProperty(kIntKey)); w->SetProperty(kStringKey, static_cast(NULL)); EXPECT_EQ(NULL, w->GetProperty(kStringKey)); w->SetProperty(kStringKey, "squeamish"); EXPECT_EQ(std::string("squeamish"), w->GetProperty(kStringKey)); w->SetProperty(kStringKey, "ossifrage"); EXPECT_EQ(std::string("ossifrage"), w->GetProperty(kStringKey)); w->SetNativeWindowProperty(native_prop_key, &*w); EXPECT_EQ(&*w, w->GetNativeWindowProperty(native_prop_key)); w->SetNativeWindowProperty(native_prop_key, NULL); EXPECT_EQ(NULL, w->GetNativeWindowProperty(native_prop_key)); // ClearProperty should restore the default value. w->ClearProperty(kIntKey); EXPECT_EQ(-2, w->GetProperty(kIntKey)); w->ClearProperty(kStringKey); EXPECT_EQ(std::string("squeamish"), w->GetProperty(kStringKey)); } namespace { class TestProperty { public: TestProperty() {} virtual ~TestProperty() { last_deleted_ = this; } static TestProperty* last_deleted() { return last_deleted_; } private: static TestProperty* last_deleted_; DISALLOW_COPY_AND_ASSIGN(TestProperty); }; TestProperty* TestProperty::last_deleted_ = NULL; DEFINE_OWNED_WINDOW_PROPERTY_KEY(TestProperty, kOwnedKey, NULL); } // namespace TEST_F(WindowTest, OwnedProperty) { scoped_ptr w(CreateTestWindowWithId(0, NULL)); EXPECT_EQ(NULL, w->GetProperty(kOwnedKey)); TestProperty* p1 = new TestProperty(); w->SetProperty(kOwnedKey, p1); EXPECT_EQ(p1, w->GetProperty(kOwnedKey)); EXPECT_EQ(NULL, TestProperty::last_deleted()); TestProperty* p2 = new TestProperty(); w->SetProperty(kOwnedKey, p2); EXPECT_EQ(p2, w->GetProperty(kOwnedKey)); EXPECT_EQ(p1, TestProperty::last_deleted()); w->ClearProperty(kOwnedKey); EXPECT_EQ(NULL, w->GetProperty(kOwnedKey)); EXPECT_EQ(p2, TestProperty::last_deleted()); TestProperty* p3 = new TestProperty(); w->SetProperty(kOwnedKey, p3); EXPECT_EQ(p3, w->GetProperty(kOwnedKey)); EXPECT_EQ(p2, TestProperty::last_deleted()); w.reset(); EXPECT_EQ(p3, TestProperty::last_deleted()); } TEST_F(WindowTest, SetBoundsInternalShouldCheckTargetBounds) { // We cannot short-circuit animations in this test. ui::LayerAnimator::set_disable_animations_for_test(false); scoped_ptr w1( CreateTestWindowWithBounds(gfx::Rect(0, 0, 100, 100), NULL)); EXPECT_FALSE(!w1->layer()); w1->layer()->GetAnimator()->set_disable_timer_for_test(true); ui::AnimationContainerElement* element = w1->layer()->GetAnimator(); EXPECT_EQ("0,0 100x100", w1->bounds().ToString()); EXPECT_EQ("0,0 100x100", w1->layer()->GetTargetBounds().ToString()); // Animate to a different position. { ui::ScopedLayerAnimationSettings settings(w1->layer()->GetAnimator()); w1->SetBounds(gfx::Rect(100, 100, 100, 100)); } EXPECT_EQ("0,0 100x100", w1->bounds().ToString()); EXPECT_EQ("100,100 100x100", w1->layer()->GetTargetBounds().ToString()); // Animate back to the first position. The animation hasn't started yet, so // the current bounds are still (0, 0, 100, 100), but the target bounds are // (100, 100, 100, 100). If we step the animator ahead, we should find that // we're at (0, 0, 100, 100). That is, the second animation should be applied. { ui::ScopedLayerAnimationSettings settings(w1->layer()->GetAnimator()); w1->SetBounds(gfx::Rect(0, 0, 100, 100)); } EXPECT_EQ("0,0 100x100", w1->bounds().ToString()); EXPECT_EQ("0,0 100x100", w1->layer()->GetTargetBounds().ToString()); // Confirm that the target bounds are reached. base::TimeTicks start_time = w1->layer()->GetAnimator()->last_step_time(); element->Step(start_time + base::TimeDelta::FromMilliseconds(1000)); EXPECT_EQ("0,0 100x100", w1->bounds().ToString()); } typedef std::pair PropertyChangeInfo; class WindowObserverTest : public WindowTest, public WindowObserver { public: struct VisibilityInfo { bool window_visible; bool visible_param; }; WindowObserverTest() : added_count_(0), removed_count_(0), destroyed_count_(0), old_property_value_(-3) { } virtual ~WindowObserverTest() {} const VisibilityInfo* GetVisibilityInfo() const { return visibility_info_.get(); } void ResetVisibilityInfo() { visibility_info_.reset(); } // Returns a description of the WindowObserver methods that have been invoked. std::string WindowObserverCountStateAndClear() { std::string result( base::StringPrintf("added=%d removed=%d", added_count_, removed_count_)); added_count_ = removed_count_ = 0; return result; } int DestroyedCountAndClear() { int result = destroyed_count_; destroyed_count_ = 0; return result; } // Return a tuple of the arguments passed in OnPropertyChanged callback. PropertyChangeInfo PropertyChangeInfoAndClear() { PropertyChangeInfo result(property_key_, old_property_value_); property_key_ = NULL; old_property_value_ = -3; return result; } private: virtual void OnWindowAdded(Window* new_window) OVERRIDE { added_count_++; } virtual void OnWillRemoveWindow(Window* window) OVERRIDE { removed_count_++; } virtual void OnWindowVisibilityChanged(Window* window, bool visible) OVERRIDE { visibility_info_.reset(new VisibilityInfo); visibility_info_->window_visible = window->IsVisible(); visibility_info_->visible_param = visible; } virtual void OnWindowDestroyed(Window* window) OVERRIDE { EXPECT_FALSE(window->parent()); destroyed_count_++; } virtual void OnWindowPropertyChanged(Window* window, const void* key, intptr_t old) OVERRIDE { property_key_ = key; old_property_value_ = old; } int added_count_; int removed_count_; int destroyed_count_; scoped_ptr visibility_info_; const void* property_key_; intptr_t old_property_value_; DISALLOW_COPY_AND_ASSIGN(WindowObserverTest); }; // Various assertions for WindowObserver. TEST_F(WindowObserverTest, WindowObserver) { scoped_ptr w1(CreateTestWindowWithId(1, NULL)); w1->AddObserver(this); // Create a new window as a child of w1, our observer should be notified. scoped_ptr w2(CreateTestWindowWithId(2, w1.get())); EXPECT_EQ("added=1 removed=0", WindowObserverCountStateAndClear()); // Delete w2, which should result in the remove notification. w2.reset(); EXPECT_EQ("added=0 removed=1", WindowObserverCountStateAndClear()); // Create a window that isn't parented to w1, we shouldn't get any // notification. scoped_ptr w3(CreateTestWindowWithId(3, NULL)); EXPECT_EQ("added=0 removed=0", WindowObserverCountStateAndClear()); // Similarly destroying w3 shouldn't notify us either. w3.reset(); EXPECT_EQ("added=0 removed=0", WindowObserverCountStateAndClear()); w1->RemoveObserver(this); } // Test if OnWindowVisibilityChagned is invoked with expected // parameters. TEST_F(WindowObserverTest, WindowVisibility) { scoped_ptr w1(CreateTestWindowWithId(1, NULL)); scoped_ptr w2(CreateTestWindowWithId(1, w1.get())); w2->AddObserver(this); // Hide should make the window invisible and the passed visible // parameter is false. w2->Hide(); EXPECT_FALSE(!GetVisibilityInfo()); EXPECT_FALSE(!GetVisibilityInfo()); if (!GetVisibilityInfo()) return; EXPECT_FALSE(GetVisibilityInfo()->window_visible); EXPECT_FALSE(GetVisibilityInfo()->visible_param); // If parent isn't visible, showing window won't make the window visible, but // passed visible value must be true. w1->Hide(); ResetVisibilityInfo(); EXPECT_TRUE(!GetVisibilityInfo()); w2->Show(); EXPECT_FALSE(!GetVisibilityInfo()); if (!GetVisibilityInfo()) return; EXPECT_FALSE(GetVisibilityInfo()->window_visible); EXPECT_TRUE(GetVisibilityInfo()->visible_param); // If parent is visible, showing window will make the window // visible and the passed visible value is true. w1->Show(); w2->Hide(); ResetVisibilityInfo(); w2->Show(); EXPECT_FALSE(!GetVisibilityInfo()); if (!GetVisibilityInfo()) return; EXPECT_TRUE(GetVisibilityInfo()->window_visible); EXPECT_TRUE(GetVisibilityInfo()->visible_param); } // Test if OnWindowDestroyed is invoked as expected. TEST_F(WindowObserverTest, WindowDestroyed) { // Delete a window should fire a destroyed notification. scoped_ptr w1(CreateTestWindowWithId(1, NULL)); w1->AddObserver(this); w1.reset(); EXPECT_EQ(1, DestroyedCountAndClear()); // Observe on child and delete parent window should fire a notification. scoped_ptr parent(CreateTestWindowWithId(1, NULL)); Window* child = CreateTestWindowWithId(1, parent.get()); // owned by parent child->AddObserver(this); parent.reset(); EXPECT_EQ(1, DestroyedCountAndClear()); } TEST_F(WindowObserverTest, PropertyChanged) { // Setting property should fire a property change notification. scoped_ptr w1(CreateTestWindowWithId(1, NULL)); w1->AddObserver(this); static const WindowProperty prop = {-2}; static const char native_prop_key[] = "fnord"; w1->SetProperty(&prop, 1); EXPECT_EQ(PropertyChangeInfo(&prop, -2), PropertyChangeInfoAndClear()); w1->SetProperty(&prop, -2); EXPECT_EQ(PropertyChangeInfo(&prop, 1), PropertyChangeInfoAndClear()); w1->SetProperty(&prop, 3); EXPECT_EQ(PropertyChangeInfo(&prop, -2), PropertyChangeInfoAndClear()); w1->ClearProperty(&prop); EXPECT_EQ(PropertyChangeInfo(&prop, 3), PropertyChangeInfoAndClear()); w1->SetNativeWindowProperty(native_prop_key, &*w1); EXPECT_EQ(PropertyChangeInfo(native_prop_key, 0), PropertyChangeInfoAndClear()); w1->SetNativeWindowProperty(native_prop_key, NULL); EXPECT_EQ(PropertyChangeInfo(native_prop_key, reinterpret_cast(&*w1)), PropertyChangeInfoAndClear()); // Sanity check to see if |PropertyChangeInfoAndClear| really clears. EXPECT_EQ(PropertyChangeInfo( reinterpret_cast(NULL), -3), PropertyChangeInfoAndClear()); } TEST_F(WindowTest, AcquireLayer) { scoped_ptr window1(CreateTestWindowWithId(1, NULL)); scoped_ptr window2(CreateTestWindowWithId(2, NULL)); ui::Layer* parent = window1->parent()->layer(); EXPECT_EQ(2U, parent->children().size()); Window::TestApi window1_test_api(window1.get()); Window::TestApi window2_test_api(window2.get()); EXPECT_TRUE(window1_test_api.OwnsLayer()); EXPECT_TRUE(window2_test_api.OwnsLayer()); // After acquisition, window1 should not own its layer, but it should still // be available to the window. scoped_ptr window1_layer(window1->AcquireLayer()); EXPECT_FALSE(window1_test_api.OwnsLayer()); EXPECT_TRUE(window1_layer.get() == window1->layer()); // Upon destruction, window1's layer should still be valid, and in the layer // hierarchy, but window2's should be gone, and no longer in the hierarchy. window1.reset(); window2.reset(); // This should be set by the window's destructor. EXPECT_TRUE(window1_layer->delegate() == NULL); EXPECT_EQ(1U, parent->children().size()); } TEST_F(WindowTest, StackWindowsWhoseLayersHaveNoDelegate) { scoped_ptr window1(CreateTestWindowWithId(1, NULL)); scoped_ptr window2(CreateTestWindowWithId(2, NULL)); // This brings window1 (and its layer) to the front. root_window()->StackChildAbove(window1.get(), window2.get()); EXPECT_EQ(root_window()->children().front(), window2.get()); EXPECT_EQ(root_window()->children().back(), window1.get()); EXPECT_EQ(root_window()->layer()->children().front(), window2->layer()); EXPECT_EQ(root_window()->layer()->children().back(), window1->layer()); // Since window1 does not have a delegate, window2 should not move in // front of it, nor should its layer. window1->layer()->set_delegate(NULL); root_window()->StackChildAbove(window2.get(), window1.get()); EXPECT_EQ(root_window()->children().front(), window2.get()); EXPECT_EQ(root_window()->children().back(), window1.get()); EXPECT_EQ(root_window()->layer()->children().front(), window2->layer()); EXPECT_EQ(root_window()->layer()->children().back(), window1->layer()); } TEST_F(WindowTest, StackTransientsWhoseLayersHaveNoDelegate) { RootWindow* root = root_window(); // Create a window with several transients, then a couple windows on top. scoped_ptr window1(CreateTestWindowWithId(1, NULL)); scoped_ptr window11(CreateTransientChild(11, window1.get())); scoped_ptr window12(CreateTransientChild(12, window1.get())); scoped_ptr window13(CreateTransientChild(13, window1.get())); scoped_ptr window2(CreateTestWindowWithId(2, NULL)); scoped_ptr window3(CreateTestWindowWithId(3, NULL)); EXPECT_EQ("1 11 12 13 2 3", ChildWindowIDsAsString(root)); // Remove the delegates of a couple of transients, as if they are closing // and animating out. window11->layer()->set_delegate(NULL); window13->layer()->set_delegate(NULL); // Move window1 to the front. All transients should move with it, and their // order should be preserved. root->StackChildAtTop(window1.get()); EXPECT_EQ("2 3 1 11 12 13", ChildWindowIDsAsString(root)); } class TestVisibilityClient : public client::VisibilityClient { public: explicit TestVisibilityClient(RootWindow* root_window) : ignore_visibility_changes_(false) { client::SetVisibilityClient(root_window, this); } virtual ~TestVisibilityClient() { } void set_ignore_visibility_changes(bool ignore_visibility_changes) { ignore_visibility_changes_ = ignore_visibility_changes; } // Overridden from client::VisibilityClient: virtual void UpdateLayerVisibility(aura::Window* window, bool visible) OVERRIDE { if (!ignore_visibility_changes_) window->layer()->SetVisible(visible); } private: bool ignore_visibility_changes_; DISALLOW_COPY_AND_ASSIGN(TestVisibilityClient); }; TEST_F(WindowTest, VisibilityClientIsVisible) { TestVisibilityClient client(root_window()); scoped_ptr window(CreateTestWindowWithId(1, NULL)); EXPECT_TRUE(window->IsVisible()); EXPECT_TRUE(window->layer()->visible()); window->Hide(); EXPECT_FALSE(window->IsVisible()); EXPECT_FALSE(window->layer()->visible()); window->Show(); client.set_ignore_visibility_changes(true); window->Hide(); EXPECT_FALSE(window->IsVisible()); EXPECT_TRUE(window->layer()->visible()); } #if !defined(OS_WIN) // Temporarily disabled for windows. See crbug.com/112222. // Tests mouse events on window change. TEST_F(WindowTest, MouseEventsOnWindowChange) { gfx::Size size = root_window()->GetHostSize(); EventGenerator generator(root_window()); generator.MoveMouseTo(50, 50); MouseTrackingDelegate d1; scoped_ptr w1(CreateTestWindowWithDelegate(&d1, 1, gfx::Rect(0, 0, 100, 100), root_window())); RunAllPendingInMessageLoop(); // The format of result is "Enter/Mouse/Leave". EXPECT_EQ("1 1 0", d1.GetMouseCountsAndReset()); // Adding new window. MouseTrackingDelegate d11; scoped_ptr w11(CreateTestWindowWithDelegate( &d11, 1, gfx::Rect(0, 0, 100, 100), w1.get())); RunAllPendingInMessageLoop(); EXPECT_EQ("0 0 1", d1.GetMouseCountsAndReset()); EXPECT_EQ("1 1 0", d11.GetMouseCountsAndReset()); // Move bounds. w11->SetBounds(gfx::Rect(0, 0, 10, 10)); RunAllPendingInMessageLoop(); EXPECT_EQ("1 1 0", d1.GetMouseCountsAndReset()); EXPECT_EQ("0 0 1", d11.GetMouseCountsAndReset()); w11->SetBounds(gfx::Rect(0, 0, 60, 60)); RunAllPendingInMessageLoop(); EXPECT_EQ("0 0 1", d1.GetMouseCountsAndReset()); EXPECT_EQ("1 1 0", d11.GetMouseCountsAndReset()); // Detach, then re-attach. w1->RemoveChild(w11.get()); RunAllPendingInMessageLoop(); EXPECT_EQ("1 1 0", d1.GetMouseCountsAndReset()); // Window is detached, so no event is set. EXPECT_EQ("0 0 0", d11.GetMouseCountsAndReset()); w1->AddChild(w11.get()); RunAllPendingInMessageLoop(); EXPECT_EQ("0 0 1", d1.GetMouseCountsAndReset()); // Window is detached, so no event is set. EXPECT_EQ("1 1 0", d11.GetMouseCountsAndReset()); // Visibility Change w11->Hide(); RunAllPendingInMessageLoop(); EXPECT_EQ("1 1 0", d1.GetMouseCountsAndReset()); EXPECT_EQ("0 0 0", d11.GetMouseCountsAndReset()); w11->Show(); RunAllPendingInMessageLoop(); EXPECT_EQ("0 0 1", d1.GetMouseCountsAndReset()); EXPECT_EQ("1 1 0", d11.GetMouseCountsAndReset()); // Transform: move d11 by 100 100. ui::Transform transform; transform.ConcatTranslate(100, 100); w11->SetTransform(transform); RunAllPendingInMessageLoop(); EXPECT_EQ("1 1 0", d1.GetMouseCountsAndReset()); EXPECT_EQ("0 0 1", d11.GetMouseCountsAndReset()); w11->SetTransform(ui::Transform()); RunAllPendingInMessageLoop(); EXPECT_EQ("0 0 1", d1.GetMouseCountsAndReset()); EXPECT_EQ("1 1 0", d11.GetMouseCountsAndReset()); // Closing a window. w11.reset(); RunAllPendingInMessageLoop(); EXPECT_EQ("1 1 0", d1.GetMouseCountsAndReset()); } #endif class StackingMadrigalLayoutManager : public LayoutManager { public: explicit StackingMadrigalLayoutManager(RootWindow* root_window) : root_window_(root_window) { root_window_->SetLayoutManager(this); } virtual ~StackingMadrigalLayoutManager() { } private: // Overridden from LayoutManager: virtual void OnWindowResized() OVERRIDE {} virtual void OnWindowAddedToLayout(Window* child) OVERRIDE {} virtual void OnWillRemoveWindowFromLayout(Window* child) OVERRIDE {} virtual void OnWindowRemovedFromLayout(Window* child) OVERRIDE {} virtual void OnChildWindowVisibilityChanged(Window* child, bool visible) OVERRIDE { Window::Windows::const_iterator it = root_window_->children().begin(); Window* last_window = NULL; for (; it != root_window_->children().end(); ++it) { if (*it == child && last_window) { if (!visible) root_window_->StackChildAbove(last_window, *it); else root_window_->StackChildAbove(*it, last_window); break; } last_window = *it; } } virtual void SetChildBounds(Window* child, const gfx::Rect& requested_bounds) OVERRIDE { SetChildBoundsDirect(child, requested_bounds); } RootWindow* root_window_; DISALLOW_COPY_AND_ASSIGN(StackingMadrigalLayoutManager); }; class StackingMadrigalVisibilityClient : public client::VisibilityClient { public: explicit StackingMadrigalVisibilityClient(RootWindow* root_window) : ignored_window_(NULL) { client::SetVisibilityClient(root_window, this); } virtual ~StackingMadrigalVisibilityClient() { } void set_ignored_window(Window* ignored_window) { ignored_window_ = ignored_window; } private: // Overridden from client::VisibilityClient: virtual void UpdateLayerVisibility(Window* window, bool visible) OVERRIDE { if (!visible) { if (window == ignored_window_) window->layer()->set_delegate(NULL); else window->layer()->SetVisible(visible); } else { window->layer()->SetVisible(visible); } } Window* ignored_window_; DISALLOW_COPY_AND_ASSIGN(StackingMadrigalVisibilityClient); }; // This test attempts to reconstruct a circumstance that can happen when the // aura client attempts to manipulate the visibility and delegate of a layer // independent of window visibility. // A use case is where the client attempts to keep a window visible onscreen // even after code has called Hide() on the window. The use case for this would // be that window hides are animated (e.g. the window fades out). To prevent // spurious updating the client code may also clear window's layer's delegate, // so that the window cannot attempt to paint or update it further. The window // uses the presence of a NULL layer delegate as a signal in stacking to note // that the window is being manipulated by such a use case and its stacking // should not be adjusted. // One issue that can arise when a window opens two transient children, and the // first is hidden. Subsequent attempts to activate the transient parent can // result in the transient parent being stacked above the second transient // child. A fix is made to Window::StackAbove to prevent this, and this test // verifies this fix. TEST_F(WindowTest, StackingMadrigal) { new StackingMadrigalLayoutManager(root_window()); StackingMadrigalVisibilityClient visibility_client(root_window()); scoped_ptr window1(CreateTestWindowWithId(1, NULL)); scoped_ptr window11(CreateTransientChild(11, window1.get())); visibility_client.set_ignored_window(window11.get()); window11->Show(); window11->Hide(); // As a transient, window11 should still be stacked above window1, even when // hidden. EXPECT_TRUE(WindowIsAbove(window11.get(), window1.get())); EXPECT_TRUE(LayerIsAbove(window11.get(), window1.get())); // A new transient should still be above window1. It will appear behind // window11 because we don't stack windows on top of targets with NULL // delegates. scoped_ptr window12(CreateTransientChild(12, window1.get())); window12->Show(); EXPECT_TRUE(WindowIsAbove(window12.get(), window1.get())); EXPECT_TRUE(LayerIsAbove(window12.get(), window1.get())); // In earlier versions of the StackChildAbove() method, attempting to stack // window1 above window12 at this point would actually restack the layers // resulting in window12's layer being below window1's layer (though the // windows themselves would still be correctly stacked, so events would pass // through.) root_window()->StackChildAbove(window1.get(), window12.get()); // Both window12 and its layer should be stacked above window1. EXPECT_TRUE(WindowIsAbove(window12.get(), window1.get())); EXPECT_TRUE(LayerIsAbove(window12.get(), window1.get())); } // Test for an issue where attempting to stack a primary window on top of a // transient with a NULL layer delegate causes that primary window to be moved, // but the layer order not changed to match. http://crbug.com/112562 TEST_F(WindowTest, StackOverClosingTransient) { scoped_ptr window1(CreateTestWindowWithId(1, NULL)); scoped_ptr transient1(CreateTransientChild(11, window1.get())); scoped_ptr window2(CreateTestWindowWithId(2, NULL)); scoped_ptr transient2(CreateTransientChild(21, window2.get())); // Both windows and layers are stacked in creation order. RootWindow* root = root_window(); ASSERT_EQ(4u, root->children().size()); EXPECT_EQ(root->children()[0], window1.get()); EXPECT_EQ(root->children()[1], transient1.get()); EXPECT_EQ(root->children()[2], window2.get()); EXPECT_EQ(root->children()[3], transient2.get()); ASSERT_EQ(4u, root->layer()->children().size()); EXPECT_EQ(root->layer()->children()[0], window1->layer()); EXPECT_EQ(root->layer()->children()[1], transient1->layer()); EXPECT_EQ(root->layer()->children()[2], window2->layer()); EXPECT_EQ(root->layer()->children()[3], transient2->layer()); // This brings window1 and its transient to the front. // root_window()->StackChildAbove(window1.get(), window2.get()); root->StackChildAtTop(window1.get()); EXPECT_EQ(root->children()[0], window2.get()); EXPECT_EQ(root->children()[1], transient2.get()); EXPECT_EQ(root->children()[2], window1.get()); EXPECT_EQ(root->children()[3], transient1.get()); EXPECT_EQ(root->layer()->children()[0], window2->layer()); EXPECT_EQ(root->layer()->children()[1], transient2->layer()); EXPECT_EQ(root->layer()->children()[2], window1->layer()); EXPECT_EQ(root->layer()->children()[3], transient1->layer()); // Pretend we're closing the top-most transient, then bring window2 to the // front. This mimics activating a browser window while the status bubble // is fading out. The transient should stay topmost. transient1->layer()->set_delegate(NULL); root->StackChildAtTop(window2.get()); EXPECT_EQ(root->children()[0], window1.get()); EXPECT_EQ(root->children()[1], window2.get()); EXPECT_EQ(root->children()[2], transient2.get()); EXPECT_EQ(root->children()[3], transient1.get()); EXPECT_EQ(root->layer()->children()[0], window1->layer()); EXPECT_EQ(root->layer()->children()[1], window2->layer()); EXPECT_EQ(root->layer()->children()[2], transient2->layer()); EXPECT_EQ(root->layer()->children()[3], transient1->layer()); // Close the transient. Remaining windows are stable. transient1.reset(); ASSERT_EQ(3u, root->children().size()); EXPECT_EQ(root->children()[0], window1.get()); EXPECT_EQ(root->children()[1], window2.get()); EXPECT_EQ(root->children()[2], transient2.get()); ASSERT_EQ(3u, root->layer()->children().size()); EXPECT_EQ(root->layer()->children()[0], window1->layer()); EXPECT_EQ(root->layer()->children()[1], window2->layer()); EXPECT_EQ(root->layer()->children()[2], transient2->layer()); // Open another window on top. scoped_ptr window3(CreateTestWindowWithId(3, NULL)); ASSERT_EQ(4u, root->children().size()); EXPECT_EQ(root->children()[0], window1.get()); EXPECT_EQ(root->children()[1], window2.get()); EXPECT_EQ(root->children()[2], transient2.get()); EXPECT_EQ(root->children()[3], window3.get()); ASSERT_EQ(4u, root->layer()->children().size()); EXPECT_EQ(root->layer()->children()[0], window1->layer()); EXPECT_EQ(root->layer()->children()[1], window2->layer()); EXPECT_EQ(root->layer()->children()[2], transient2->layer()); EXPECT_EQ(root->layer()->children()[3], window3->layer()); // Pretend we're closing the topmost non-transient window, then bring // window2 to the top. It should not move. window3->layer()->set_delegate(NULL); root->StackChildAtTop(window2.get()); ASSERT_EQ(4u, root->children().size()); EXPECT_EQ(root->children()[0], window1.get()); EXPECT_EQ(root->children()[1], window2.get()); EXPECT_EQ(root->children()[2], transient2.get()); EXPECT_EQ(root->children()[3], window3.get()); ASSERT_EQ(4u, root->layer()->children().size()); EXPECT_EQ(root->layer()->children()[0], window1->layer()); EXPECT_EQ(root->layer()->children()[1], window2->layer()); EXPECT_EQ(root->layer()->children()[2], transient2->layer()); EXPECT_EQ(root->layer()->children()[3], window3->layer()); // Bring window1 to the top. It should move ahead of window2, but not // ahead of window3 (with NULL delegate). root->StackChildAtTop(window1.get()); ASSERT_EQ(4u, root->children().size()); EXPECT_EQ(root->children()[0], window2.get()); EXPECT_EQ(root->children()[1], transient2.get()); EXPECT_EQ(root->children()[2], window1.get()); EXPECT_EQ(root->children()[3], window3.get()); ASSERT_EQ(4u, root->layer()->children().size()); EXPECT_EQ(root->layer()->children()[0], window2->layer()); EXPECT_EQ(root->layer()->children()[1], transient2->layer()); EXPECT_EQ(root->layer()->children()[2], window1->layer()); EXPECT_EQ(root->layer()->children()[3], window3->layer()); } class RootWindowAttachmentObserver : public WindowObserver { public: RootWindowAttachmentObserver() : added_count_(0), removed_count_(0) {} virtual ~RootWindowAttachmentObserver() {} int added_count() const { return added_count_; } int removed_count() const { return removed_count_; } void Clear() { added_count_ = 0; removed_count_ = 0; } // Overridden from WindowObserver: virtual void OnWindowAddedToRootWindow(Window* window) OVERRIDE { ++added_count_; } virtual void OnWindowRemovingFromRootWindow(Window* window) OVERRIDE { ++removed_count_; } private: int added_count_; int removed_count_; DISALLOW_COPY_AND_ASSIGN(RootWindowAttachmentObserver); }; TEST_F(WindowTest, RootWindowAttachment) { RootWindowAttachmentObserver observer; // Test a direct add/remove from the RootWindow. scoped_ptr w1(new Window(NULL)); w1->Init(ui::LAYER_NOT_DRAWN); w1->AddObserver(&observer); w1->SetParent(NULL); EXPECT_EQ(1, observer.added_count()); EXPECT_EQ(0, observer.removed_count()); w1.reset(); EXPECT_EQ(1, observer.added_count()); EXPECT_EQ(1, observer.removed_count()); observer.Clear(); // Test an indirect add/remove from the RootWindow. w1.reset(new Window(NULL)); w1->Init(ui::LAYER_NOT_DRAWN); Window* w11 = new Window(NULL); w11->Init(ui::LAYER_NOT_DRAWN); w11->AddObserver(&observer); w11->SetParent(w1.get()); EXPECT_EQ(0, observer.added_count()); EXPECT_EQ(0, observer.removed_count()); w1->SetParent(NULL); EXPECT_EQ(1, observer.added_count()); EXPECT_EQ(0, observer.removed_count()); w1.reset(); // Deletes w11. w11 = NULL; EXPECT_EQ(1, observer.added_count()); EXPECT_EQ(1, observer.removed_count()); observer.Clear(); // Test an indirect add/remove with nested observers. w1.reset(new Window(NULL)); w1->Init(ui::LAYER_NOT_DRAWN); w11 = new Window(NULL); w11->Init(ui::LAYER_NOT_DRAWN); w11->AddObserver(&observer); w11->SetParent(w1.get()); Window* w111 = new Window(NULL); w111->Init(ui::LAYER_NOT_DRAWN); w111->AddObserver(&observer); w111->SetParent(w11); EXPECT_EQ(0, observer.added_count()); EXPECT_EQ(0, observer.removed_count()); w1->SetParent(NULL); EXPECT_EQ(2, observer.added_count()); EXPECT_EQ(0, observer.removed_count()); w1.reset(); // Deletes w11 and w111. w11 = NULL; w111 = NULL; EXPECT_EQ(2, observer.added_count()); EXPECT_EQ(2, observer.removed_count()); } TEST_F(WindowTest, OwnedByParentFalse) { // By default, a window is owned by its parent. If this is set to false, the // window will not be destroyed when its parent is. scoped_ptr w1(new Window(NULL)); w1->Init(ui::LAYER_NOT_DRAWN); scoped_ptr w2(new Window(NULL)); w2->set_owned_by_parent(false); w2->Init(ui::LAYER_NOT_DRAWN); w2->SetParent(w1.get()); w1.reset(); // We should be able to deref w2 still, but its parent should now be NULL. EXPECT_EQ(NULL, w2->parent()); } namespace { // Used By DeleteWindowFromOnWindowDestroyed. Destroys a Window from // OnWindowDestroyed(). class OwningWindowDelegate : public TestWindowDelegate { public: OwningWindowDelegate() {} void SetOwnedWindow(Window* window) { owned_window_.reset(window); } virtual void OnWindowDestroyed() OVERRIDE { owned_window_.reset(NULL); } private: scoped_ptr owned_window_; DISALLOW_COPY_AND_ASSIGN(OwningWindowDelegate); }; } // namespace // Creates a window with two child windows. When the first child window is // destroyed (WindowDelegate::OnWindowDestroyed) it deletes the second child. // This synthesizes BrowserView and the status bubble. Both are children of the // same parent and destroying BrowserView triggers it destroying the status // bubble. TEST_F(WindowTest, DeleteWindowFromOnWindowDestroyed) { scoped_ptr parent(new Window(NULL)); parent->Init(ui::LAYER_NOT_DRAWN); OwningWindowDelegate delegate; Window* c1 = new Window(&delegate); c1->Init(ui::LAYER_NOT_DRAWN); c1->SetParent(parent.get()); Window* c2 = new Window(NULL); c2->Init(ui::LAYER_NOT_DRAWN); c2->SetParent(parent.get()); delegate.SetOwnedWindow(c2); parent.reset(); } } // namespace test } // namespace aura