| // Copyright 2015 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 "third_party/blink/renderer/modules/presentation/presentation_controller.h" |
| |
| #include <memory> |
| #include "third_party/blink/public/common/browser_interface_broker_proxy.h" |
| #include "third_party/blink/public/platform/web_string.h" |
| #include "third_party/blink/public/platform/web_vector.h" |
| #include "third_party/blink/renderer/core/frame/deprecation.h" |
| #include "third_party/blink/renderer/core/frame/local_dom_window.h" |
| #include "third_party/blink/renderer/core/frame/local_frame.h" |
| #include "third_party/blink/renderer/modules/presentation/presentation_availability_callbacks.h" |
| #include "third_party/blink/renderer/modules/presentation/presentation_availability_observer.h" |
| #include "third_party/blink/renderer/modules/presentation/presentation_availability_state.h" |
| #include "third_party/blink/renderer/modules/presentation/presentation_connection.h" |
| #include "third_party/blink/renderer/platform/instrumentation/use_counter.h" |
| |
| namespace blink { |
| |
| PresentationController::PresentationController(LocalDOMWindow& window) |
| : Supplement<LocalDOMWindow>(window), |
| presentation_controller_receiver_(this, &window) {} |
| |
| PresentationController::~PresentationController() = default; |
| |
| // static |
| const char PresentationController::kSupplementName[] = "PresentationController"; |
| |
| // static |
| PresentationController* PresentationController::From(LocalDOMWindow& window) { |
| PresentationController* controller = |
| Supplement<LocalDOMWindow>::From<PresentationController>(window); |
| if (!controller) { |
| controller = MakeGarbageCollected<PresentationController>(window); |
| Supplement<LocalDOMWindow>::ProvideTo(window, controller); |
| } |
| return controller; |
| } |
| |
| // static |
| PresentationController* PresentationController::FromContext( |
| ExecutionContext* execution_context) { |
| if (!execution_context || execution_context->IsContextDestroyed()) |
| return nullptr; |
| return From(*To<LocalDOMWindow>(execution_context)); |
| } |
| |
| void PresentationController::Trace(Visitor* visitor) const { |
| visitor->Trace(presentation_controller_receiver_); |
| visitor->Trace(presentation_); |
| visitor->Trace(connections_); |
| visitor->Trace(availability_state_); |
| Supplement<LocalDOMWindow>::Trace(visitor); |
| } |
| |
| void PresentationController::SetPresentation(Presentation* presentation) { |
| presentation_ = presentation; |
| } |
| |
| void PresentationController::RegisterConnection( |
| ControllerPresentationConnection* connection) { |
| connections_.insert(connection); |
| } |
| |
| PresentationAvailabilityState* PresentationController::GetAvailabilityState() { |
| if (!availability_state_) { |
| availability_state_ = MakeGarbageCollected<PresentationAvailabilityState>( |
| GetPresentationService().get()); |
| } |
| |
| return availability_state_; |
| } |
| |
| void PresentationController::AddAvailabilityObserver( |
| PresentationAvailabilityObserver* observer) { |
| GetAvailabilityState()->AddObserver(observer); |
| } |
| |
| void PresentationController::RemoveAvailabilityObserver( |
| PresentationAvailabilityObserver* observer) { |
| GetAvailabilityState()->RemoveObserver(observer); |
| } |
| |
| void PresentationController::OnScreenAvailabilityUpdated( |
| const KURL& url, |
| mojom::blink::ScreenAvailability availability) { |
| GetAvailabilityState()->UpdateAvailability(url, availability); |
| } |
| |
| void PresentationController::OnConnectionStateChanged( |
| mojom::blink::PresentationInfoPtr presentation_info, |
| mojom::blink::PresentationConnectionState state) { |
| PresentationConnection* connection = FindConnection(*presentation_info); |
| if (!connection) |
| return; |
| |
| connection->DidChangeState(state); |
| } |
| |
| void PresentationController::OnConnectionClosed( |
| mojom::blink::PresentationInfoPtr presentation_info, |
| mojom::blink::PresentationConnectionCloseReason reason, |
| const String& message) { |
| PresentationConnection* connection = FindConnection(*presentation_info); |
| if (!connection) |
| return; |
| |
| connection->DidClose(reason, message); |
| } |
| |
| void PresentationController::OnDefaultPresentationStarted( |
| mojom::blink::PresentationConnectionResultPtr result) { |
| DCHECK(result); |
| DCHECK(result->presentation_info); |
| DCHECK(result->connection_remote && result->connection_receiver); |
| if (!presentation_ || !presentation_->defaultRequest()) |
| return; |
| |
| auto* connection = ControllerPresentationConnection::Take( |
| this, *result->presentation_info, presentation_->defaultRequest()); |
| // TODO(btolsch): Convert this and similar calls to just use InterfacePtrInfo |
| // instead of constructing an InterfacePtr every time we have |
| // InterfacePtrInfo. |
| connection->Init(std::move(result->connection_remote), |
| std::move(result->connection_receiver)); |
| } |
| |
| ControllerPresentationConnection* |
| PresentationController::FindExistingConnection( |
| const blink::WebVector<blink::WebURL>& presentation_urls, |
| const blink::WebString& presentation_id) { |
| for (const auto& connection : connections_) { |
| for (const auto& presentation_url : presentation_urls) { |
| if (connection->GetState() != |
| mojom::blink::PresentationConnectionState::TERMINATED && |
| connection->Matches(presentation_id, presentation_url)) { |
| return connection.Get(); |
| } |
| } |
| } |
| return nullptr; |
| } |
| |
| mojo::Remote<mojom::blink::PresentationService>& |
| PresentationController::GetPresentationService() { |
| if (!presentation_service_remote_ && GetSupplementable()) { |
| scoped_refptr<base::SingleThreadTaskRunner> task_runner = |
| GetSupplementable()->GetTaskRunner(TaskType::kPresentation); |
| GetSupplementable()->GetBrowserInterfaceBroker().GetInterface( |
| presentation_service_remote_.BindNewPipeAndPassReceiver(task_runner)); |
| presentation_service_remote_->SetController( |
| presentation_controller_receiver_.BindNewPipeAndPassRemote( |
| task_runner)); |
| } |
| return presentation_service_remote_; |
| } |
| |
| ControllerPresentationConnection* PresentationController::FindConnection( |
| const mojom::blink::PresentationInfo& presentation_info) const { |
| for (const auto& connection : connections_) { |
| if (connection->Matches(presentation_info.id, presentation_info.url)) |
| return connection.Get(); |
| } |
| |
| return nullptr; |
| } |
| |
| } // namespace blink |