| OLD | NEW |
| (Empty) |
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #include "components/mus/connection_manager.h" | |
| 6 | |
| 7 #include "base/logging.h" | |
| 8 #include "base/stl_util.h" | |
| 9 #include "cc/output/compositor_frame.h" | |
| 10 #include "cc/quads/shared_quad_state.h" | |
| 11 #include "cc/quads/surface_draw_quad.h" | |
| 12 #include "components/mus/client_connection.h" | |
| 13 #include "components/mus/connection_manager_delegate.h" | |
| 14 #include "components/mus/server_view.h" | |
| 15 #include "components/mus/view_coordinate_conversions.h" | |
| 16 #include "components/mus/view_tree_host_connection.h" | |
| 17 #include "components/mus/view_tree_impl.h" | |
| 18 #include "mojo/application/public/cpp/application_connection.h" | |
| 19 #include "mojo/converters/geometry/geometry_type_converters.h" | |
| 20 #include "mojo/converters/input_events/input_events_type_converters.h" | |
| 21 #include "mojo/converters/surfaces/surfaces_type_converters.h" | |
| 22 #include "ui/gfx/geometry/size_conversions.h" | |
| 23 | |
| 24 namespace mus { | |
| 25 | |
| 26 ConnectionManager::ScopedChange::ScopedChange( | |
| 27 ViewTreeImpl* connection, | |
| 28 ConnectionManager* connection_manager, | |
| 29 bool is_delete_view) | |
| 30 : connection_manager_(connection_manager), | |
| 31 connection_id_(connection->id()), | |
| 32 is_delete_view_(is_delete_view) { | |
| 33 connection_manager_->PrepareForChange(this); | |
| 34 } | |
| 35 | |
| 36 ConnectionManager::ScopedChange::~ScopedChange() { | |
| 37 connection_manager_->FinishChange(); | |
| 38 } | |
| 39 | |
| 40 ConnectionManager::ConnectionManager( | |
| 41 ConnectionManagerDelegate* delegate, | |
| 42 const scoped_refptr<SurfacesState>& surfaces_state) | |
| 43 : delegate_(delegate), | |
| 44 surfaces_state_(surfaces_state), | |
| 45 next_connection_id_(1), | |
| 46 next_host_id_(0), | |
| 47 current_change_(nullptr), | |
| 48 in_destructor_(false) {} | |
| 49 | |
| 50 ConnectionManager::~ConnectionManager() { | |
| 51 in_destructor_ = true; | |
| 52 | |
| 53 // Copy the HostConnectionMap because it will be mutated as the connections | |
| 54 // are closed. | |
| 55 HostConnectionMap host_connection_map(host_connection_map_); | |
| 56 for (auto& pair : host_connection_map) | |
| 57 pair.second->CloseConnection(); | |
| 58 | |
| 59 STLDeleteValues(&connection_map_); | |
| 60 // All the connections should have been destroyed. | |
| 61 DCHECK(host_connection_map_.empty()); | |
| 62 DCHECK(connection_map_.empty()); | |
| 63 } | |
| 64 | |
| 65 void ConnectionManager::AddHost(ViewTreeHostConnection* host_connection) { | |
| 66 DCHECK_EQ(0u, host_connection_map_.count(host_connection->view_tree_host())); | |
| 67 host_connection_map_[host_connection->view_tree_host()] = host_connection; | |
| 68 } | |
| 69 | |
| 70 ServerView* ConnectionManager::CreateServerView(const ViewId& id) { | |
| 71 ServerView* view = new ServerView(this, id); | |
| 72 view->AddObserver(this); | |
| 73 return view; | |
| 74 } | |
| 75 | |
| 76 ConnectionSpecificId ConnectionManager::GetAndAdvanceNextConnectionId() { | |
| 77 const ConnectionSpecificId id = next_connection_id_++; | |
| 78 DCHECK_LT(id, next_connection_id_); | |
| 79 return id; | |
| 80 } | |
| 81 | |
| 82 uint16_t ConnectionManager::GetAndAdvanceNextHostId() { | |
| 83 const uint16_t id = next_host_id_++; | |
| 84 DCHECK_LT(id, next_host_id_); | |
| 85 return id; | |
| 86 } | |
| 87 | |
| 88 void ConnectionManager::OnConnectionError(ClientConnection* connection) { | |
| 89 // This will be null if the root has been destroyed. | |
| 90 const ViewId* view_id = connection->service()->root(); | |
| 91 ServerView* view = | |
| 92 view_id ? GetView(*connection->service()->root()) : nullptr; | |
| 93 // If the ViewTree root is a viewport root, then we'll wait until | |
| 94 // the root connection goes away to cleanup. | |
| 95 if (view && (GetRootView(view) == view)) | |
| 96 return; | |
| 97 | |
| 98 scoped_ptr<ClientConnection> connection_owner(connection); | |
| 99 | |
| 100 connection_map_.erase(connection->service()->id()); | |
| 101 | |
| 102 // Notify remaining connections so that they can cleanup. | |
| 103 for (auto& pair : connection_map_) { | |
| 104 pair.second->service()->OnWillDestroyViewTreeImpl(connection->service()); | |
| 105 } | |
| 106 } | |
| 107 | |
| 108 void ConnectionManager::OnHostConnectionClosed( | |
| 109 ViewTreeHostConnection* connection) { | |
| 110 auto it = host_connection_map_.find(connection->view_tree_host()); | |
| 111 DCHECK(it != host_connection_map_.end()); | |
| 112 | |
| 113 // Get the ClientConnection by ViewTreeImpl ID. | |
| 114 ConnectionMap::iterator service_connection_it = | |
| 115 connection_map_.find(it->first->GetViewTree()->id()); | |
| 116 DCHECK(service_connection_it != connection_map_.end()); | |
| 117 | |
| 118 // Tear down the associated ViewTree connection. | |
| 119 // TODO(fsamuel): I don't think this is quite right, we should tear down all | |
| 120 // connections within the root's viewport. We should probably employ an | |
| 121 // observer pattern to do this. Each ViewTreeImpl should track its | |
| 122 // parent's lifetime. | |
| 123 host_connection_map_.erase(it); | |
| 124 OnConnectionError(service_connection_it->second); | |
| 125 | |
| 126 // If we have no more roots left, let the app know so it can terminate. | |
| 127 if (!host_connection_map_.size()) | |
| 128 delegate_->OnNoMoreRootConnections(); | |
| 129 } | |
| 130 | |
| 131 void ConnectionManager::EmbedAtView(ConnectionSpecificId creator_id, | |
| 132 const ViewId& view_id, | |
| 133 uint32_t policy_bitmask, | |
| 134 mojo::URLRequestPtr request) { | |
| 135 mojo::ViewTreePtr service_ptr; | |
| 136 ClientConnection* client_connection = | |
| 137 delegate_->CreateClientConnectionForEmbedAtView( | |
| 138 this, GetProxy(&service_ptr), creator_id, request.Pass(), view_id, | |
| 139 policy_bitmask); | |
| 140 AddConnection(client_connection); | |
| 141 client_connection->service()->Init(client_connection->client(), | |
| 142 service_ptr.Pass()); | |
| 143 OnConnectionMessagedClient(client_connection->service()->id()); | |
| 144 } | |
| 145 | |
| 146 ViewTreeImpl* ConnectionManager::EmbedAtView(ConnectionSpecificId creator_id, | |
| 147 const ViewId& view_id, | |
| 148 uint32_t policy_bitmask, | |
| 149 mojo::ViewTreeClientPtr client) { | |
| 150 mojo::ViewTreePtr service_ptr; | |
| 151 ClientConnection* client_connection = | |
| 152 delegate_->CreateClientConnectionForEmbedAtView( | |
| 153 this, GetProxy(&service_ptr), creator_id, view_id, policy_bitmask, | |
| 154 client.Pass()); | |
| 155 AddConnection(client_connection); | |
| 156 client_connection->service()->Init(client_connection->client(), | |
| 157 service_ptr.Pass()); | |
| 158 OnConnectionMessagedClient(client_connection->service()->id()); | |
| 159 | |
| 160 return client_connection->service(); | |
| 161 } | |
| 162 | |
| 163 ViewTreeImpl* ConnectionManager::GetConnection( | |
| 164 ConnectionSpecificId connection_id) { | |
| 165 ConnectionMap::iterator i = connection_map_.find(connection_id); | |
| 166 return i == connection_map_.end() ? nullptr : i->second->service(); | |
| 167 } | |
| 168 | |
| 169 ServerView* ConnectionManager::GetView(const ViewId& id) { | |
| 170 for (auto& pair : host_connection_map_) { | |
| 171 if (pair.first->root_view()->id() == id) | |
| 172 return pair.first->root_view(); | |
| 173 } | |
| 174 ViewTreeImpl* service = GetConnection(id.connection_id); | |
| 175 return service ? service->GetView(id) : nullptr; | |
| 176 } | |
| 177 | |
| 178 bool ConnectionManager::IsViewAttachedToRoot(const ServerView* view) const { | |
| 179 for (auto& pair : host_connection_map_) { | |
| 180 if (pair.first->IsViewAttachedToRoot(view)) | |
| 181 return true; | |
| 182 } | |
| 183 return false; | |
| 184 } | |
| 185 | |
| 186 void ConnectionManager::SchedulePaint(const ServerView* view, | |
| 187 const gfx::Rect& bounds) { | |
| 188 for (auto& pair : host_connection_map_) { | |
| 189 if (pair.first->SchedulePaintIfInViewport(view, bounds)) | |
| 190 return; | |
| 191 } | |
| 192 } | |
| 193 | |
| 194 void ConnectionManager::OnConnectionMessagedClient(ConnectionSpecificId id) { | |
| 195 if (current_change_) | |
| 196 current_change_->MarkConnectionAsMessaged(id); | |
| 197 } | |
| 198 | |
| 199 bool ConnectionManager::DidConnectionMessageClient( | |
| 200 ConnectionSpecificId id) const { | |
| 201 return current_change_ && current_change_->DidMessageConnection(id); | |
| 202 } | |
| 203 | |
| 204 mojo::ViewportMetricsPtr ConnectionManager::GetViewportMetricsForView( | |
| 205 const ServerView* view) { | |
| 206 ViewTreeHostImpl* host = GetViewTreeHostByView(view); | |
| 207 if (host) | |
| 208 return host->GetViewportMetrics().Clone(); | |
| 209 | |
| 210 if (!host_connection_map_.empty()) | |
| 211 return host_connection_map_.begin()->first->GetViewportMetrics().Clone(); | |
| 212 | |
| 213 mojo::ViewportMetricsPtr metrics = mojo::ViewportMetrics::New(); | |
| 214 metrics->size_in_pixels = mojo::Size::New(); | |
| 215 return metrics.Pass(); | |
| 216 } | |
| 217 | |
| 218 const ViewTreeImpl* ConnectionManager::GetConnectionWithRoot( | |
| 219 const ViewId& id) const { | |
| 220 for (auto& pair : connection_map_) { | |
| 221 if (pair.second->service()->IsRoot(id)) | |
| 222 return pair.second->service(); | |
| 223 } | |
| 224 return nullptr; | |
| 225 } | |
| 226 | |
| 227 ViewTreeHostImpl* ConnectionManager::GetViewTreeHostByView( | |
| 228 const ServerView* view) { | |
| 229 return const_cast<ViewTreeHostImpl*>( | |
| 230 static_cast<const ConnectionManager*>(this)->GetViewTreeHostByView(view)); | |
| 231 } | |
| 232 | |
| 233 const ViewTreeHostImpl* ConnectionManager::GetViewTreeHostByView( | |
| 234 const ServerView* view) const { | |
| 235 while (view && view->parent()) | |
| 236 view = view->parent(); | |
| 237 for (auto& pair : host_connection_map_) { | |
| 238 if (view == pair.first->root_view()) | |
| 239 return pair.first; | |
| 240 } | |
| 241 return nullptr; | |
| 242 } | |
| 243 | |
| 244 ViewTreeImpl* ConnectionManager::GetEmbedRoot(ViewTreeImpl* service) { | |
| 245 while (service) { | |
| 246 const ViewId* root_id = service->root(); | |
| 247 if (!root_id || root_id->connection_id == service->id()) | |
| 248 return nullptr; | |
| 249 | |
| 250 ViewTreeImpl* parent_service = GetConnection(root_id->connection_id); | |
| 251 service = parent_service; | |
| 252 if (service && service->is_embed_root()) | |
| 253 return service; | |
| 254 } | |
| 255 return nullptr; | |
| 256 } | |
| 257 | |
| 258 void ConnectionManager::ProcessViewBoundsChanged(const ServerView* view, | |
| 259 const gfx::Rect& old_bounds, | |
| 260 const gfx::Rect& new_bounds) { | |
| 261 for (auto& pair : connection_map_) { | |
| 262 pair.second->service()->ProcessViewBoundsChanged( | |
| 263 view, old_bounds, new_bounds, IsChangeSource(pair.first)); | |
| 264 } | |
| 265 } | |
| 266 | |
| 267 void ConnectionManager::ProcessWillChangeViewHierarchy( | |
| 268 const ServerView* view, | |
| 269 const ServerView* new_parent, | |
| 270 const ServerView* old_parent) { | |
| 271 for (auto& pair : connection_map_) { | |
| 272 pair.second->service()->ProcessWillChangeViewHierarchy( | |
| 273 view, new_parent, old_parent, IsChangeSource(pair.first)); | |
| 274 } | |
| 275 } | |
| 276 | |
| 277 void ConnectionManager::ProcessViewHierarchyChanged( | |
| 278 const ServerView* view, | |
| 279 const ServerView* new_parent, | |
| 280 const ServerView* old_parent) { | |
| 281 for (auto& pair : connection_map_) { | |
| 282 pair.second->service()->ProcessViewHierarchyChanged( | |
| 283 view, new_parent, old_parent, IsChangeSource(pair.first)); | |
| 284 } | |
| 285 } | |
| 286 | |
| 287 void ConnectionManager::ProcessViewReorder( | |
| 288 const ServerView* view, | |
| 289 const ServerView* relative_view, | |
| 290 const mojo::OrderDirection direction) { | |
| 291 for (auto& pair : connection_map_) { | |
| 292 pair.second->service()->ProcessViewReorder(view, relative_view, direction, | |
| 293 IsChangeSource(pair.first)); | |
| 294 } | |
| 295 } | |
| 296 | |
| 297 void ConnectionManager::ProcessViewDeleted(const ViewId& view) { | |
| 298 for (auto& pair : connection_map_) { | |
| 299 pair.second->service()->ProcessViewDeleted(view, | |
| 300 IsChangeSource(pair.first)); | |
| 301 } | |
| 302 } | |
| 303 | |
| 304 void ConnectionManager::ProcessViewportMetricsChanged( | |
| 305 const mojo::ViewportMetrics& old_metrics, | |
| 306 const mojo::ViewportMetrics& new_metrics) { | |
| 307 for (auto& pair : connection_map_) { | |
| 308 pair.second->service()->ProcessViewportMetricsChanged( | |
| 309 old_metrics, new_metrics, IsChangeSource(pair.first)); | |
| 310 } | |
| 311 } | |
| 312 | |
| 313 void ConnectionManager::PrepareForChange(ScopedChange* change) { | |
| 314 // Should only ever have one change in flight. | |
| 315 CHECK(!current_change_); | |
| 316 current_change_ = change; | |
| 317 } | |
| 318 | |
| 319 void ConnectionManager::FinishChange() { | |
| 320 // PrepareForChange/FinishChange should be balanced. | |
| 321 CHECK(current_change_); | |
| 322 current_change_ = NULL; | |
| 323 } | |
| 324 | |
| 325 void ConnectionManager::AddConnection(ClientConnection* connection) { | |
| 326 DCHECK_EQ(0u, connection_map_.count(connection->service()->id())); | |
| 327 connection_map_[connection->service()->id()] = connection; | |
| 328 } | |
| 329 | |
| 330 scoped_ptr<cc::CompositorFrame> | |
| 331 ConnectionManager::UpdateViewTreeFromCompositorFrame( | |
| 332 const mojo::CompositorFramePtr& input) { | |
| 333 return ConvertToCompositorFrame(input, this); | |
| 334 } | |
| 335 | |
| 336 SurfacesState* ConnectionManager::GetSurfacesState() { | |
| 337 return surfaces_state_.get(); | |
| 338 } | |
| 339 | |
| 340 void ConnectionManager::OnScheduleViewPaint(const ServerView* view) { | |
| 341 if (!in_destructor_) | |
| 342 SchedulePaint(view, gfx::Rect(view->bounds().size())); | |
| 343 } | |
| 344 | |
| 345 const ServerView* ConnectionManager::GetRootView(const ServerView* view) const { | |
| 346 const ViewTreeHostImpl* host = GetViewTreeHostByView(view); | |
| 347 return host ? host->root_view() : nullptr; | |
| 348 } | |
| 349 | |
| 350 void ConnectionManager::OnViewDestroyed(ServerView* view) { | |
| 351 if (!in_destructor_) | |
| 352 ProcessViewDeleted(view->id()); | |
| 353 } | |
| 354 | |
| 355 void ConnectionManager::OnWillChangeViewHierarchy(ServerView* view, | |
| 356 ServerView* new_parent, | |
| 357 ServerView* old_parent) { | |
| 358 if (in_destructor_) | |
| 359 return; | |
| 360 | |
| 361 ProcessWillChangeViewHierarchy(view, new_parent, old_parent); | |
| 362 } | |
| 363 | |
| 364 void ConnectionManager::OnViewHierarchyChanged(ServerView* view, | |
| 365 ServerView* new_parent, | |
| 366 ServerView* old_parent) { | |
| 367 if (in_destructor_) | |
| 368 return; | |
| 369 | |
| 370 ProcessViewHierarchyChanged(view, new_parent, old_parent); | |
| 371 | |
| 372 // TODO(beng): optimize. | |
| 373 if (old_parent) | |
| 374 SchedulePaint(old_parent, gfx::Rect(old_parent->bounds().size())); | |
| 375 if (new_parent) | |
| 376 SchedulePaint(new_parent, gfx::Rect(new_parent->bounds().size())); | |
| 377 } | |
| 378 | |
| 379 void ConnectionManager::OnViewBoundsChanged(ServerView* view, | |
| 380 const gfx::Rect& old_bounds, | |
| 381 const gfx::Rect& new_bounds) { | |
| 382 if (in_destructor_) | |
| 383 return; | |
| 384 | |
| 385 ProcessViewBoundsChanged(view, old_bounds, new_bounds); | |
| 386 if (!view->parent()) | |
| 387 return; | |
| 388 | |
| 389 // TODO(sky): optimize this. | |
| 390 SchedulePaint(view->parent(), old_bounds); | |
| 391 SchedulePaint(view->parent(), new_bounds); | |
| 392 } | |
| 393 | |
| 394 void ConnectionManager::OnViewReordered(ServerView* view, | |
| 395 ServerView* relative, | |
| 396 mojo::OrderDirection direction) { | |
| 397 if (!in_destructor_) | |
| 398 SchedulePaint(view, gfx::Rect(view->bounds().size())); | |
| 399 } | |
| 400 | |
| 401 void ConnectionManager::OnWillChangeViewVisibility(ServerView* view) { | |
| 402 if (in_destructor_) | |
| 403 return; | |
| 404 | |
| 405 // Need to repaint if the view was drawn (which means it's in the process of | |
| 406 // hiding) or the view is transitioning to drawn. | |
| 407 if (view->parent() && | |
| 408 (view->IsDrawn() || (!view->visible() && view->parent()->IsDrawn()))) { | |
| 409 SchedulePaint(view->parent(), view->bounds()); | |
| 410 } | |
| 411 | |
| 412 for (auto& pair : connection_map_) { | |
| 413 pair.second->service()->ProcessWillChangeViewVisibility( | |
| 414 view, IsChangeSource(pair.first)); | |
| 415 } | |
| 416 } | |
| 417 | |
| 418 void ConnectionManager::OnViewSharedPropertyChanged( | |
| 419 ServerView* view, | |
| 420 const std::string& name, | |
| 421 const std::vector<uint8_t>* new_data) { | |
| 422 for (auto& pair : connection_map_) { | |
| 423 pair.second->service()->ProcessViewPropertyChanged( | |
| 424 view, name, new_data, IsChangeSource(pair.first)); | |
| 425 } | |
| 426 } | |
| 427 | |
| 428 void ConnectionManager::OnViewTextInputStateChanged( | |
| 429 ServerView* view, | |
| 430 const ui::TextInputState& state) { | |
| 431 ViewTreeHostImpl* host = GetViewTreeHostByView(view); | |
| 432 host->UpdateTextInputState(view, state); | |
| 433 } | |
| 434 | |
| 435 bool ConnectionManager::ConvertSurfaceDrawQuad( | |
| 436 const mojo::QuadPtr& input, | |
| 437 const mojo::CompositorFrameMetadataPtr& metadata, | |
| 438 cc::SharedQuadState* sqs, | |
| 439 cc::RenderPass* render_pass) { | |
| 440 unsigned int id = static_cast<unsigned int>( | |
| 441 input->surface_quad_state->surface.To<cc::SurfaceId>().id); | |
| 442 // TODO(fsamuel): Security checks: | |
| 443 // 1. We need to make sure the embedder can only position views it's allowed | |
| 444 // to access. | |
| 445 // 2. We need to make sure that the embedder cannot place views in areas | |
| 446 // outside of its own bounds. | |
| 447 ServerView* view = GetView(ViewIdFromTransportId(id)); | |
| 448 // If a CompositorFrame message arrives late, say during a navigation, then | |
| 449 // it may contain view IDs that no longer exist. | |
| 450 if (!view || view->surface_id().is_null()) | |
| 451 return false; | |
| 452 | |
| 453 // TODO(fsamuel): This shouldn't be in the ConnectionManager. Let's find a | |
| 454 // better home for it. http://crbug.com/533029. | |
| 455 cc::SurfaceDrawQuad* surface_quad = | |
| 456 render_pass->CreateAndAppendDrawQuad<cc::SurfaceDrawQuad>(); | |
| 457 surface_quad->SetAll( | |
| 458 sqs, | |
| 459 input->rect.To<gfx::Rect>(), | |
| 460 input->opaque_rect.To<gfx::Rect>(), | |
| 461 input->visible_rect.To<gfx::Rect>(), | |
| 462 input->needs_blending, | |
| 463 view->surface_id()); | |
| 464 return true; | |
| 465 } | |
| 466 | |
| 467 } // namespace mus | |
| OLD | NEW |