Index: components/view_manager/view_manager_client_apptest.cc |
diff --git a/components/view_manager/view_manager_client_apptest.cc b/components/view_manager/view_manager_client_apptest.cc |
deleted file mode 100644 |
index 7c5dec8a01597466935e11f17cf2b1d11d73bdbb..0000000000000000000000000000000000000000 |
--- a/components/view_manager/view_manager_client_apptest.cc |
+++ /dev/null |
@@ -1,825 +0,0 @@ |
-// Copyright 2014 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 "base/bind.h" |
-#include "base/logging.h" |
-#include "base/run_loop.h" |
-#include "components/view_manager/public/cpp/tests/view_manager_test_base.h" |
-#include "components/view_manager/public/cpp/util.h" |
-#include "components/view_manager/public/cpp/view_observer.h" |
-#include "components/view_manager/public/cpp/view_tree_connection.h" |
-#include "components/view_manager/public/cpp/view_tree_delegate.h" |
-#include "mojo/application/public/cpp/application_connection.h" |
-#include "mojo/application/public/cpp/application_impl.h" |
-#include "mojo/application/public/cpp/application_test_base.h" |
-#include "ui/mojo/geometry/geometry_util.h" |
- |
-namespace mojo { |
- |
-namespace { |
- |
-class BoundsChangeObserver : public ViewObserver { |
- public: |
- explicit BoundsChangeObserver(View* view) : view_(view) { |
- view_->AddObserver(this); |
- } |
- ~BoundsChangeObserver() override { view_->RemoveObserver(this); } |
- |
- private: |
- // Overridden from ViewObserver: |
- void OnViewBoundsChanged(View* view, |
- const Rect& old_bounds, |
- const Rect& new_bounds) override { |
- DCHECK_EQ(view, view_); |
- EXPECT_TRUE(ViewManagerTestBase::QuitRunLoop()); |
- } |
- |
- View* view_; |
- |
- MOJO_DISALLOW_COPY_AND_ASSIGN(BoundsChangeObserver); |
-}; |
- |
-// Wait until the bounds of the supplied view change; returns false on timeout. |
-bool WaitForBoundsToChange(View* view) { |
- BoundsChangeObserver observer(view); |
- return ViewManagerTestBase::DoRunLoopWithTimeout(); |
-} |
- |
-// Spins a run loop until the tree beginning at |root| has |tree_size| views |
-// (including |root|). |
-class TreeSizeMatchesObserver : public ViewObserver { |
- public: |
- TreeSizeMatchesObserver(View* tree, size_t tree_size) |
- : tree_(tree), tree_size_(tree_size) { |
- tree_->AddObserver(this); |
- } |
- ~TreeSizeMatchesObserver() override { tree_->RemoveObserver(this); } |
- |
- bool IsTreeCorrectSize() { return CountViews(tree_) == tree_size_; } |
- |
- private: |
- // Overridden from ViewObserver: |
- void OnTreeChanged(const TreeChangeParams& params) override { |
- if (IsTreeCorrectSize()) |
- EXPECT_TRUE(ViewManagerTestBase::QuitRunLoop()); |
- } |
- |
- size_t CountViews(const View* view) const { |
- size_t count = 1; |
- View::Children::const_iterator it = view->children().begin(); |
- for (; it != view->children().end(); ++it) |
- count += CountViews(*it); |
- return count; |
- } |
- |
- View* tree_; |
- size_t tree_size_; |
- |
- MOJO_DISALLOW_COPY_AND_ASSIGN(TreeSizeMatchesObserver); |
-}; |
- |
-// Wait until |view| has |tree_size| descendants; returns false on timeout. The |
-// count includes |view|. For example, if you want to wait for |view| to have |
-// a single child, use a |tree_size| of 2. |
-bool WaitForTreeSizeToMatch(View* view, size_t tree_size) { |
- TreeSizeMatchesObserver observer(view, tree_size); |
- return observer.IsTreeCorrectSize() || |
- ViewManagerTestBase::DoRunLoopWithTimeout(); |
-} |
- |
-class OrderChangeObserver : public ViewObserver { |
- public: |
- OrderChangeObserver(View* view) : view_(view) { view_->AddObserver(this); } |
- ~OrderChangeObserver() override { view_->RemoveObserver(this); } |
- |
- private: |
- // Overridden from ViewObserver: |
- void OnViewReordered(View* view, |
- View* relative_view, |
- OrderDirection direction) override { |
- DCHECK_EQ(view, view_); |
- EXPECT_TRUE(ViewManagerTestBase::QuitRunLoop()); |
- } |
- |
- View* view_; |
- |
- MOJO_DISALLOW_COPY_AND_ASSIGN(OrderChangeObserver); |
-}; |
- |
-// Wait until |view|'s tree size matches |tree_size|; returns false on timeout. |
-bool WaitForOrderChange(ViewTreeConnection* connection, View* view) { |
- OrderChangeObserver observer(view); |
- return ViewManagerTestBase::DoRunLoopWithTimeout(); |
-} |
- |
-// Tracks a view's destruction. Query is_valid() for current state. |
-class ViewTracker : public ViewObserver { |
- public: |
- explicit ViewTracker(View* view) : view_(view) { view_->AddObserver(this); } |
- ~ViewTracker() override { |
- if (view_) |
- view_->RemoveObserver(this); |
- } |
- |
- bool is_valid() const { return !!view_; } |
- |
- private: |
- // Overridden from ViewObserver: |
- void OnViewDestroyed(View* view) override { |
- DCHECK_EQ(view, view_); |
- view_ = nullptr; |
- } |
- |
- View* view_; |
- |
- MOJO_DISALLOW_COPY_AND_ASSIGN(ViewTracker); |
-}; |
- |
-} // namespace |
- |
-// ViewManager ----------------------------------------------------------------- |
- |
-struct EmbedResult { |
- EmbedResult(ViewTreeConnection* connection, ConnectionSpecificId id) |
- : connection(connection), connection_id(id) {} |
- EmbedResult() : connection(nullptr), connection_id(0) {} |
- |
- ViewTreeConnection* connection; |
- |
- // The id supplied to the callback from OnEmbed(). Depending upon the |
- // access policy this may or may not match the connection id of |
- // |connection|. |
- ConnectionSpecificId connection_id; |
-}; |
- |
-// These tests model synchronization of two peer connections to the view manager |
-// service, that are given access to some root view. |
- |
-class ViewManagerTest : public ViewManagerTestBase { |
- public: |
- ViewManagerTest() {} |
- |
- // Embeds another version of the test app @ view. This runs a run loop until |
- // a response is received, or a timeout. On success the new ViewManager is |
- // returned. |
- EmbedResult Embed(View* view) { |
- DCHECK(!embed_details_); |
- embed_details_.reset(new EmbedDetails); |
- view->Embed(ConnectToApplicationAndGetViewManagerClient(), |
- base::Bind(&ViewManagerTest::EmbedCallbackImpl, |
- base::Unretained(this))); |
- embed_details_->waiting = true; |
- if (!ViewManagerTestBase::DoRunLoopWithTimeout()) |
- return EmbedResult(); |
- const EmbedResult result(embed_details_->connection, |
- embed_details_->connection_id); |
- embed_details_.reset(); |
- return result; |
- } |
- |
- // Establishes a connection to this application and asks for a |
- // ViewTreeClient. |
- mojo::ViewTreeClientPtr ConnectToApplicationAndGetViewManagerClient() { |
- mojo::URLRequestPtr request(mojo::URLRequest::New()); |
- request->url = mojo::String::From(application_impl()->url()); |
- scoped_ptr<ApplicationConnection> connection = |
- application_impl()->ConnectToApplication(request.Pass()); |
- mojo::ViewTreeClientPtr client; |
- connection->ConnectToService(&client); |
- return client.Pass(); |
- } |
- |
- // ViewManagerTestBase: |
- void OnEmbed(View* root) override { |
- if (!embed_details_) { |
- ViewManagerTestBase::OnEmbed(root); |
- return; |
- } |
- |
- embed_details_->connection = root->connection(); |
- if (embed_details_->callback_run) |
- EXPECT_TRUE(ViewManagerTestBase::QuitRunLoop()); |
- } |
- |
- private: |
- // Used to track the state of a call to view->Embed(). |
- struct EmbedDetails { |
- EmbedDetails() |
- : callback_run(false), |
- result(false), |
- waiting(false), |
- connection_id(0), |
- connection(nullptr) {} |
- |
- // The callback supplied to Embed() was received. |
- bool callback_run; |
- |
- // The boolean supplied to the Embed() callback. |
- bool result; |
- |
- // Whether a MessageLoop is running. |
- bool waiting; |
- |
- // Connection id supplied to the Embed() callback. |
- ConnectionSpecificId connection_id; |
- |
- // The ViewTreeConnection that resulted from the Embed(). null if |result| |
- // is false. |
- ViewTreeConnection* connection; |
- }; |
- |
- void EmbedCallbackImpl(bool result, ConnectionSpecificId connection_id) { |
- embed_details_->callback_run = true; |
- embed_details_->result = result; |
- embed_details_->connection_id = connection_id; |
- if (embed_details_->waiting && (!result || embed_details_->connection)) |
- EXPECT_TRUE(ViewManagerTestBase::QuitRunLoop()); |
- } |
- |
- scoped_ptr<EmbedDetails> embed_details_; |
- |
- MOJO_DISALLOW_COPY_AND_ASSIGN(ViewManagerTest); |
-}; |
- |
-TEST_F(ViewManagerTest, RootView) { |
- ASSERT_NE(nullptr, window_manager()); |
- EXPECT_NE(nullptr, window_manager()->GetRoot()); |
-} |
- |
-TEST_F(ViewManagerTest, Embed) { |
- View* view = window_manager()->CreateView(); |
- ASSERT_NE(nullptr, view); |
- view->SetVisible(true); |
- window_manager()->GetRoot()->AddChild(view); |
- ViewTreeConnection* embedded = Embed(view).connection; |
- ASSERT_NE(nullptr, embedded); |
- |
- View* view_in_embedded = embedded->GetRoot(); |
- ASSERT_NE(nullptr, view_in_embedded); |
- EXPECT_EQ(view->id(), view_in_embedded->id()); |
- EXPECT_EQ(nullptr, view_in_embedded->parent()); |
- EXPECT_TRUE(view_in_embedded->children().empty()); |
-} |
- |
-// Window manager has two views, N1 and N11. Embeds A at N1. A should not see |
-// N11. |
-TEST_F(ViewManagerTest, EmbeddedDoesntSeeChild) { |
- View* view = window_manager()->CreateView(); |
- ASSERT_NE(nullptr, view); |
- view->SetVisible(true); |
- window_manager()->GetRoot()->AddChild(view); |
- View* nested = window_manager()->CreateView(); |
- ASSERT_NE(nullptr, nested); |
- nested->SetVisible(true); |
- view->AddChild(nested); |
- |
- ViewTreeConnection* embedded = Embed(view).connection; |
- ASSERT_NE(nullptr, embedded); |
- View* view_in_embedded = embedded->GetRoot(); |
- EXPECT_EQ(view->id(), view_in_embedded->id()); |
- EXPECT_EQ(nullptr, view_in_embedded->parent()); |
- EXPECT_TRUE(view_in_embedded->children().empty()); |
-} |
- |
-// TODO(beng): write a replacement test for the one that once existed here: |
-// This test validates the following scenario: |
-// - a view originating from one connection |
-// - a view originating from a second connection |
-// + the connection originating the view is destroyed |
-// -> the view should still exist (since the second connection is live) but |
-// should be disconnected from any views. |
-// http://crbug.com/396300 |
-// |
-// TODO(beng): The new test should validate the scenario as described above |
-// except that the second connection still has a valid tree. |
- |
-// Verifies that bounds changes applied to a view hierarchy in one connection |
-// are reflected to another. |
-TEST_F(ViewManagerTest, SetBounds) { |
- View* view = window_manager()->CreateView(); |
- view->SetVisible(true); |
- window_manager()->GetRoot()->AddChild(view); |
- ViewTreeConnection* embedded = Embed(view).connection; |
- ASSERT_NE(nullptr, embedded); |
- |
- View* view_in_embedded = embedded->GetViewById(view->id()); |
- EXPECT_EQ(view->bounds(), view_in_embedded->bounds()); |
- |
- Rect rect; |
- rect.width = rect.height = 100; |
- view->SetBounds(rect); |
- ASSERT_TRUE(WaitForBoundsToChange(view_in_embedded)); |
- EXPECT_EQ(view->bounds(), view_in_embedded->bounds()); |
-} |
- |
-// Verifies that bounds changes applied to a view owned by a different |
-// connection are refused. |
-TEST_F(ViewManagerTest, SetBoundsSecurity) { |
- View* view = window_manager()->CreateView(); |
- view->SetVisible(true); |
- window_manager()->GetRoot()->AddChild(view); |
- ViewTreeConnection* embedded = Embed(view).connection; |
- ASSERT_NE(nullptr, embedded); |
- |
- View* view_in_embedded = embedded->GetViewById(view->id()); |
- Rect rect; |
- rect.width = 800; |
- rect.height = 600; |
- view->SetBounds(rect); |
- ASSERT_TRUE(WaitForBoundsToChange(view_in_embedded)); |
- |
- rect.width = 1024; |
- rect.height = 768; |
- view_in_embedded->SetBounds(rect); |
- // Bounds change should have been rejected. |
- EXPECT_EQ(view->bounds(), view_in_embedded->bounds()); |
-} |
- |
-// Verifies that a view can only be destroyed by the connection that created it. |
-TEST_F(ViewManagerTest, DestroySecurity) { |
- View* view = window_manager()->CreateView(); |
- view->SetVisible(true); |
- window_manager()->GetRoot()->AddChild(view); |
- ViewTreeConnection* embedded = Embed(view).connection; |
- ASSERT_NE(nullptr, embedded); |
- |
- View* view_in_embedded = embedded->GetViewById(view->id()); |
- |
- ViewTracker tracker2(view_in_embedded); |
- view_in_embedded->Destroy(); |
- // View should not have been destroyed. |
- EXPECT_TRUE(tracker2.is_valid()); |
- |
- ViewTracker tracker1(view); |
- view->Destroy(); |
- EXPECT_FALSE(tracker1.is_valid()); |
-} |
- |
-TEST_F(ViewManagerTest, MultiRoots) { |
- View* view1 = window_manager()->CreateView(); |
- view1->SetVisible(true); |
- window_manager()->GetRoot()->AddChild(view1); |
- View* view2 = window_manager()->CreateView(); |
- view2->SetVisible(true); |
- window_manager()->GetRoot()->AddChild(view2); |
- ViewTreeConnection* embedded1 = Embed(view1).connection; |
- ASSERT_NE(nullptr, embedded1); |
- ViewTreeConnection* embedded2 = Embed(view2).connection; |
- ASSERT_NE(nullptr, embedded2); |
- EXPECT_NE(embedded1, embedded2); |
-} |
- |
-// TODO(alhaad): Currently, the RunLoop gets stuck waiting for order change. |
-// Debug and re-enable this. |
-TEST_F(ViewManagerTest, DISABLED_Reorder) { |
- View* view1 = window_manager()->CreateView(); |
- view1->SetVisible(true); |
- window_manager()->GetRoot()->AddChild(view1); |
- |
- ViewTreeConnection* embedded = Embed(view1).connection; |
- ASSERT_NE(nullptr, embedded); |
- |
- View* view11 = embedded->CreateView(); |
- view11->SetVisible(true); |
- embedded->GetRoot()->AddChild(view11); |
- View* view12 = embedded->CreateView(); |
- view12->SetVisible(true); |
- embedded->GetRoot()->AddChild(view12); |
- |
- View* root_in_embedded = embedded->GetRoot(); |
- |
- { |
- ASSERT_TRUE(WaitForTreeSizeToMatch(root_in_embedded, 3u)); |
- view11->MoveToFront(); |
- ASSERT_TRUE(WaitForOrderChange(embedded, root_in_embedded)); |
- |
- EXPECT_EQ(root_in_embedded->children().front(), |
- embedded->GetViewById(view12->id())); |
- EXPECT_EQ(root_in_embedded->children().back(), |
- embedded->GetViewById(view11->id())); |
- } |
- |
- { |
- view11->MoveToBack(); |
- ASSERT_TRUE(WaitForOrderChange(embedded, |
- embedded->GetViewById(view11->id()))); |
- |
- EXPECT_EQ(root_in_embedded->children().front(), |
- embedded->GetViewById(view11->id())); |
- EXPECT_EQ(root_in_embedded->children().back(), |
- embedded->GetViewById(view12->id())); |
- } |
-} |
- |
-namespace { |
- |
-class VisibilityChangeObserver : public ViewObserver { |
- public: |
- explicit VisibilityChangeObserver(View* view) : view_(view) { |
- view_->AddObserver(this); |
- } |
- ~VisibilityChangeObserver() override { view_->RemoveObserver(this); } |
- |
- private: |
- // Overridden from ViewObserver: |
- void OnViewVisibilityChanged(View* view) override { |
- EXPECT_EQ(view, view_); |
- EXPECT_TRUE(ViewManagerTestBase::QuitRunLoop()); |
- } |
- |
- View* view_; |
- |
- MOJO_DISALLOW_COPY_AND_ASSIGN(VisibilityChangeObserver); |
-}; |
- |
-} // namespace |
- |
-TEST_F(ViewManagerTest, Visible) { |
- View* view1 = window_manager()->CreateView(); |
- view1->SetVisible(true); |
- window_manager()->GetRoot()->AddChild(view1); |
- |
- // Embed another app and verify initial state. |
- ViewTreeConnection* embedded = Embed(view1).connection; |
- ASSERT_NE(nullptr, embedded); |
- ASSERT_NE(nullptr, embedded->GetRoot()); |
- View* embedded_root = embedded->GetRoot(); |
- EXPECT_TRUE(embedded_root->visible()); |
- EXPECT_TRUE(embedded_root->IsDrawn()); |
- |
- // Change the visible state from the first connection and verify its mirrored |
- // correctly to the embedded app. |
- { |
- VisibilityChangeObserver observer(embedded_root); |
- view1->SetVisible(false); |
- ASSERT_TRUE(ViewManagerTestBase::DoRunLoopWithTimeout()); |
- } |
- |
- EXPECT_FALSE(view1->visible()); |
- EXPECT_FALSE(view1->IsDrawn()); |
- |
- EXPECT_FALSE(embedded_root->visible()); |
- EXPECT_FALSE(embedded_root->IsDrawn()); |
- |
- // Make the node visible again. |
- { |
- VisibilityChangeObserver observer(embedded_root); |
- view1->SetVisible(true); |
- ASSERT_TRUE(ViewManagerTestBase::DoRunLoopWithTimeout()); |
- } |
- |
- EXPECT_TRUE(view1->visible()); |
- EXPECT_TRUE(view1->IsDrawn()); |
- |
- EXPECT_TRUE(embedded_root->visible()); |
- EXPECT_TRUE(embedded_root->IsDrawn()); |
-} |
- |
-namespace { |
- |
-class DrawnChangeObserver : public ViewObserver { |
- public: |
- explicit DrawnChangeObserver(View* view) : view_(view) { |
- view_->AddObserver(this); |
- } |
- ~DrawnChangeObserver() override { view_->RemoveObserver(this); } |
- |
- private: |
- // Overridden from ViewObserver: |
- void OnViewDrawnChanged(View* view) override { |
- EXPECT_EQ(view, view_); |
- EXPECT_TRUE(ViewManagerTestBase::QuitRunLoop()); |
- } |
- |
- View* view_; |
- |
- MOJO_DISALLOW_COPY_AND_ASSIGN(DrawnChangeObserver); |
-}; |
- |
-} // namespace |
- |
-TEST_F(ViewManagerTest, Drawn) { |
- View* view1 = window_manager()->CreateView(); |
- view1->SetVisible(true); |
- window_manager()->GetRoot()->AddChild(view1); |
- |
- // Embed another app and verify initial state. |
- ViewTreeConnection* embedded = Embed(view1).connection; |
- ASSERT_NE(nullptr, embedded); |
- ASSERT_NE(nullptr, embedded->GetRoot()); |
- View* embedded_root = embedded->GetRoot(); |
- EXPECT_TRUE(embedded_root->visible()); |
- EXPECT_TRUE(embedded_root->IsDrawn()); |
- |
- // Change the visibility of the root, this should propagate a drawn state |
- // change to |embedded|. |
- { |
- DrawnChangeObserver observer(embedded_root); |
- window_manager()->GetRoot()->SetVisible(false); |
- ASSERT_TRUE(DoRunLoopWithTimeout()); |
- } |
- |
- EXPECT_TRUE(view1->visible()); |
- EXPECT_FALSE(view1->IsDrawn()); |
- |
- EXPECT_TRUE(embedded_root->visible()); |
- EXPECT_FALSE(embedded_root->IsDrawn()); |
-} |
- |
-// TODO(beng): tests for view event dispatcher. |
-// - verify that we see events for all views. |
- |
-namespace { |
- |
-class FocusChangeObserver : public ViewObserver { |
- public: |
- explicit FocusChangeObserver(View* view) |
- : view_(view), last_gained_focus_(nullptr), last_lost_focus_(nullptr) { |
- view_->AddObserver(this); |
- } |
- ~FocusChangeObserver() override { view_->RemoveObserver(this); } |
- |
- View* last_gained_focus() { return last_gained_focus_; } |
- |
- View* last_lost_focus() { return last_lost_focus_; } |
- |
- private: |
- // Overridden from ViewObserver. |
- void OnViewFocusChanged(View* gained_focus, View* lost_focus) override { |
- EXPECT_TRUE(!gained_focus || gained_focus->HasFocus()); |
- EXPECT_FALSE(lost_focus && lost_focus->HasFocus()); |
- last_gained_focus_ = gained_focus; |
- last_lost_focus_ = lost_focus; |
- EXPECT_TRUE(ViewManagerTestBase::QuitRunLoop()); |
- } |
- |
- View* view_; |
- View* last_gained_focus_; |
- View* last_lost_focus_; |
- |
- MOJO_DISALLOW_COPY_AND_ASSIGN(FocusChangeObserver); |
-}; |
- |
-} // namespace |
- |
-TEST_F(ViewManagerTest, Focus) { |
- View* view1 = window_manager()->CreateView(); |
- view1->SetVisible(true); |
- window_manager()->GetRoot()->AddChild(view1); |
- |
- ViewTreeConnection* embedded = Embed(view1).connection; |
- ASSERT_NE(nullptr, embedded); |
- View* view11 = embedded->CreateView(); |
- view11->SetVisible(true); |
- embedded->GetRoot()->AddChild(view11); |
- |
- // TODO(alhaad): Figure out why switching focus between views from different |
- // connections is causing the tests to crash and add tests for that. |
- { |
- View* embedded_root = embedded->GetRoot(); |
- FocusChangeObserver observer(embedded_root); |
- embedded_root->SetFocus(); |
- ASSERT_TRUE(DoRunLoopWithTimeout()); |
- ASSERT_NE(nullptr, observer.last_gained_focus()); |
- EXPECT_EQ(embedded_root->id(), observer.last_gained_focus()->id()); |
- } |
- { |
- FocusChangeObserver observer(view11); |
- view11->SetFocus(); |
- ASSERT_TRUE(DoRunLoopWithTimeout()); |
- ASSERT_NE(nullptr, observer.last_gained_focus()); |
- ASSERT_NE(nullptr, observer.last_lost_focus()); |
- EXPECT_EQ(view11->id(), observer.last_gained_focus()->id()); |
- EXPECT_EQ(embedded->GetRoot()->id(), observer.last_lost_focus()->id()); |
- } |
- { |
- // Add an observer on the View that loses focus, and make sure the observer |
- // sees the right values. |
- FocusChangeObserver observer(view11); |
- embedded->GetRoot()->SetFocus(); |
- ASSERT_TRUE(DoRunLoopWithTimeout()); |
- ASSERT_NE(nullptr, observer.last_gained_focus()); |
- ASSERT_NE(nullptr, observer.last_lost_focus()); |
- EXPECT_EQ(view11->id(), observer.last_lost_focus()->id()); |
- EXPECT_EQ(embedded->GetRoot()->id(), observer.last_gained_focus()->id()); |
- } |
-} |
- |
-namespace { |
- |
-class DestroyedChangedObserver : public ViewObserver { |
- public: |
- DestroyedChangedObserver(ViewManagerTestBase* test, |
- View* view, |
- bool* got_destroy) |
- : test_(test), view_(view), got_destroy_(got_destroy) { |
- view_->AddObserver(this); |
- } |
- ~DestroyedChangedObserver() override { |
- if (view_) |
- view_->RemoveObserver(this); |
- } |
- |
- private: |
- // Overridden from ViewObserver: |
- void OnViewDestroyed(View* view) override { |
- EXPECT_EQ(view, view_); |
- view_->RemoveObserver(this); |
- *got_destroy_ = true; |
- view_ = nullptr; |
- |
- // We should always get OnViewDestroyed() before OnConnectionLost(). |
- EXPECT_FALSE(test_->view_tree_connection_destroyed()); |
- } |
- |
- ViewManagerTestBase* test_; |
- View* view_; |
- bool* got_destroy_; |
- |
- MOJO_DISALLOW_COPY_AND_ASSIGN(DestroyedChangedObserver); |
-}; |
- |
-} // namespace |
- |
-// Verifies deleting a ViewManager sends the right notifications. |
-TEST_F(ViewManagerTest, DeleteViewManager) { |
- View* view = window_manager()->CreateView(); |
- ASSERT_NE(nullptr, view); |
- view->SetVisible(true); |
- window_manager()->GetRoot()->AddChild(view); |
- ViewTreeConnection* connection = Embed(view).connection; |
- ASSERT_TRUE(connection); |
- bool got_destroy = false; |
- DestroyedChangedObserver observer(this, connection->GetRoot(), |
- &got_destroy); |
- delete connection; |
- EXPECT_TRUE(view_tree_connection_destroyed()); |
- EXPECT_TRUE(got_destroy); |
-} |
- |
-// Verifies two Embed()s in the same view trigger deletion of the first |
-// ViewManager. |
-TEST_F(ViewManagerTest, DisconnectTriggersDelete) { |
- View* view = window_manager()->CreateView(); |
- ASSERT_NE(nullptr, view); |
- view->SetVisible(true); |
- window_manager()->GetRoot()->AddChild(view); |
- ViewTreeConnection* connection = Embed(view).connection; |
- EXPECT_NE(connection, window_manager()); |
- View* embedded_view = connection->CreateView(); |
- // Embed again, this should trigger disconnect and deletion of connection. |
- bool got_destroy; |
- DestroyedChangedObserver observer(this, embedded_view, &got_destroy); |
- EXPECT_FALSE(view_tree_connection_destroyed()); |
- Embed(view); |
- EXPECT_TRUE(view_tree_connection_destroyed()); |
-} |
- |
-class ViewRemovedFromParentObserver : public ViewObserver { |
- public: |
- explicit ViewRemovedFromParentObserver(View* view) |
- : view_(view), was_removed_(false) { |
- view_->AddObserver(this); |
- } |
- ~ViewRemovedFromParentObserver() override { view_->RemoveObserver(this); } |
- |
- bool was_removed() const { return was_removed_; } |
- |
- private: |
- // Overridden from ViewObserver: |
- void OnTreeChanged(const TreeChangeParams& params) override { |
- if (params.target == view_ && !params.new_parent) |
- was_removed_ = true; |
- } |
- |
- View* view_; |
- bool was_removed_; |
- |
- MOJO_DISALLOW_COPY_AND_ASSIGN(ViewRemovedFromParentObserver); |
-}; |
- |
-TEST_F(ViewManagerTest, EmbedRemovesChildren) { |
- View* view1 = window_manager()->CreateView(); |
- View* view2 = window_manager()->CreateView(); |
- window_manager()->GetRoot()->AddChild(view1); |
- view1->AddChild(view2); |
- |
- ViewRemovedFromParentObserver observer(view2); |
- view1->Embed(ConnectToApplicationAndGetViewManagerClient()); |
- EXPECT_TRUE(observer.was_removed()); |
- EXPECT_EQ(nullptr, view2->parent()); |
- EXPECT_TRUE(view1->children().empty()); |
- |
- // Run the message loop so the Embed() call above completes. Without this |
- // we may end up reconnecting to the test and rerunning the test, which is |
- // problematic since the other services don't shut down. |
- ASSERT_TRUE(DoRunLoopWithTimeout()); |
-} |
- |
-namespace { |
- |
-class DestroyObserver : public ViewObserver { |
- public: |
- DestroyObserver(ViewManagerTestBase* test, |
- ViewTreeConnection* connection, |
- bool* got_destroy) |
- : test_(test), got_destroy_(got_destroy) { |
- connection->GetRoot()->AddObserver(this); |
- } |
- ~DestroyObserver() override {} |
- |
- private: |
- // Overridden from ViewObserver: |
- void OnViewDestroyed(View* view) override { |
- *got_destroy_ = true; |
- view->RemoveObserver(this); |
- |
- // We should always get OnViewDestroyed() before OnViewManagerDestroyed(). |
- EXPECT_FALSE(test_->view_tree_connection_destroyed()); |
- |
- EXPECT_TRUE(ViewManagerTestBase::QuitRunLoop()); |
- } |
- |
- ViewManagerTestBase* test_; |
- bool* got_destroy_; |
- |
- MOJO_DISALLOW_COPY_AND_ASSIGN(DestroyObserver); |
-}; |
- |
-} // namespace |
- |
-// Verifies deleting a View that is the root of another connection notifies |
-// observers in the right order (OnViewDestroyed() before |
-// OnViewManagerDestroyed()). |
-TEST_F(ViewManagerTest, ViewManagerDestroyedAfterRootObserver) { |
- View* embed_view = window_manager()->CreateView(); |
- window_manager()->GetRoot()->AddChild(embed_view); |
- |
- ViewTreeConnection* embedded_connection = Embed(embed_view).connection; |
- |
- bool got_destroy = false; |
- DestroyObserver observer(this, embedded_connection, &got_destroy); |
- // Delete the view |embedded_connection| is embedded in. This is async, |
- // but will eventually trigger deleting |embedded_connection|. |
- embed_view->Destroy(); |
- EXPECT_TRUE(DoRunLoopWithTimeout()); |
- EXPECT_TRUE(got_destroy); |
-} |
- |
-// Verifies an embed root sees views created beneath it from another |
-// connection. |
-TEST_F(ViewManagerTest, EmbedRootSeesHierarchyChanged) { |
- View* embed_view = window_manager()->CreateView(); |
- window_manager()->GetRoot()->AddChild(embed_view); |
- |
- embed_view->SetAccessPolicy(ViewTree::ACCESS_POLICY_EMBED_ROOT); |
- ViewTreeConnection* vm2 = Embed(embed_view).connection; |
- View* vm2_v1 = vm2->CreateView(); |
- vm2->GetRoot()->AddChild(vm2_v1); |
- |
- ViewTreeConnection* vm3 = Embed(vm2_v1).connection; |
- View* vm3_v1 = vm3->CreateView(); |
- vm3->GetRoot()->AddChild(vm3_v1); |
- |
- // As |vm2| is an embed root it should get notified about |vm3_v1|. |
- ASSERT_TRUE(WaitForTreeSizeToMatch(vm2_v1, 2)); |
-} |
- |
-TEST_F(ViewManagerTest, EmbedFromEmbedRoot) { |
- View* embed_view = window_manager()->CreateView(); |
- window_manager()->GetRoot()->AddChild(embed_view); |
- |
- // Give the connection embedded at |embed_view| embed root powers. |
- embed_view->SetAccessPolicy(ViewTree::ACCESS_POLICY_EMBED_ROOT); |
- const EmbedResult result1 = Embed(embed_view); |
- ViewTreeConnection* vm2 = result1.connection; |
- EXPECT_EQ(result1.connection_id, vm2->GetConnectionId()); |
- View* vm2_v1 = vm2->CreateView(); |
- vm2->GetRoot()->AddChild(vm2_v1); |
- |
- const EmbedResult result2 = Embed(vm2_v1); |
- ViewTreeConnection* vm3 = result2.connection; |
- EXPECT_EQ(result2.connection_id, vm3->GetConnectionId()); |
- View* vm3_v1 = vm3->CreateView(); |
- vm3->GetRoot()->AddChild(vm3_v1); |
- |
- // Embed from v3, the callback should not get the connection id as vm3 is not |
- // an embed root. |
- const EmbedResult result3 = Embed(vm3_v1); |
- ASSERT_TRUE(result3.connection); |
- EXPECT_EQ(0, result3.connection_id); |
- |
- // As |vm2| is an embed root it should get notified about |vm3_v1|. |
- ASSERT_TRUE(WaitForTreeSizeToMatch(vm2_v1, 2)); |
- |
- // Embed() from vm2 in vm3_v1. This is allowed as vm2 is an embed root, and |
- // further the callback should see the connection id. |
- ASSERT_EQ(1u, vm2_v1->children().size()); |
- View* vm3_v1_in_vm2 = vm2_v1->children()[0]; |
- const EmbedResult result4 = Embed(vm3_v1_in_vm2); |
- ASSERT_TRUE(result4.connection); |
- EXPECT_EQ(result4.connection_id, result4.connection->GetConnectionId()); |
-} |
- |
-} // namespace mojo |