blob: 99fd685692776660cf1362ae31fccf4095ee1537 [file] [log] [blame]
// 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_request.h"
#include <memory>
#include "third_party/blink/renderer/bindings/core/v8/callback_promise_adapter.h"
#include "third_party/blink/renderer/bindings/core/v8/script_promise.h"
#include "third_party/blink/renderer/bindings/core/v8/script_promise_resolver.h"
#include "third_party/blink/renderer/core/dom/dom_exception.h"
#include "third_party/blink/renderer/core/execution_context/execution_context.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/core/frame/settings.h"
#include "third_party/blink/renderer/core/loader/mixed_content_checker.h"
#include "third_party/blink/renderer/modules/event_target_modules.h"
#include "third_party/blink/renderer/modules/presentation/presentation_availability.h"
#include "third_party/blink/renderer/modules/presentation/presentation_availability_callbacks.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/modules/presentation/presentation_connection_callbacks.h"
#include "third_party/blink/renderer/modules/presentation/presentation_controller.h"
#include "third_party/blink/renderer/modules/presentation/presentation_error.h"
#include "third_party/blink/renderer/platform/bindings/exception_state.h"
#include "third_party/blink/renderer/platform/heap/heap.h"
#include "third_party/blink/renderer/platform/instrumentation/use_counter.h"
namespace blink {
namespace {
bool IsKnownProtocolForPresentationUrl(const KURL& url) {
return url.ProtocolIsInHTTPFamily() || url.ProtocolIs("cast") ||
url.ProtocolIs("cast-dial");
}
} // anonymous namespace
// static
PresentationRequest* PresentationRequest::Create(
ExecutionContext* execution_context,
const String& url,
ExceptionState& exception_state) {
Vector<String> urls(1);
urls[0] = url;
return Create(execution_context, urls, exception_state);
}
PresentationRequest* PresentationRequest::Create(
ExecutionContext* execution_context,
const Vector<String>& urls,
ExceptionState& exception_state) {
if (execution_context->IsSandboxed(
network::mojom::blink::WebSandboxFlags::kPresentationController)) {
exception_state.ThrowSecurityError(
"The document is sandboxed and lacks the 'allow-presentation' flag.");
return nullptr;
}
Vector<KURL> parsed_urls;
for (const auto& url : urls) {
const KURL& parsed_url = KURL(execution_context->Url(), url);
if (!parsed_url.IsValid()) {
exception_state.ThrowDOMException(
DOMExceptionCode::kSyntaxError,
"'" + url + "' can't be resolved to a valid URL.");
return nullptr;
}
if (parsed_url.ProtocolIsInHTTPFamily() &&
MixedContentChecker::IsMixedContent(
execution_context->GetSecurityOrigin(), parsed_url)) {
exception_state.ThrowSecurityError(
"Presentation of an insecure document [" + url +
"] is prohibited from a secure context.");
return nullptr;
}
if (IsKnownProtocolForPresentationUrl(parsed_url))
parsed_urls.push_back(parsed_url);
}
if (parsed_urls.IsEmpty()) {
exception_state.ThrowDOMException(DOMExceptionCode::kNotSupportedError,
"Do not support empty sequence of URLs.");
return nullptr;
}
return MakeGarbageCollected<PresentationRequest>(execution_context,
parsed_urls);
}
const AtomicString& PresentationRequest::InterfaceName() const {
return event_target_names::kPresentationRequest;
}
ExecutionContext* PresentationRequest::GetExecutionContext() const {
return ExecutionContextClient::GetExecutionContext();
}
void PresentationRequest::AddedEventListener(
const AtomicString& event_type,
RegisteredEventListener& registered_listener) {
EventTargetWithInlineData::AddedEventListener(event_type,
registered_listener);
if (event_type == event_type_names::kConnectionavailable) {
UseCounter::Count(
GetExecutionContext(),
WebFeature::kPresentationRequestConnectionAvailableEventListener);
}
}
bool PresentationRequest::HasPendingActivity() const {
// Prevents garbage collecting of this object when not hold by another
// object but still has listeners registered.
if (!GetExecutionContext())
return false;
if (HasEventListeners())
return true;
return availability_property_ &&
availability_property_->GetState() ==
PresentationAvailabilityProperty::kPending;
}
ScriptPromise PresentationRequest::start(ScriptState* script_state,
ExceptionState& exception_state) {
if (!script_state->ContextIsValid()) {
exception_state.ThrowDOMException(
DOMExceptionCode::kInvalidStateError,
"The PresentationRequest is no longer associated to a frame.");
return ScriptPromise();
}
LocalDOMWindow* window = LocalDOMWindow::From(script_state);
if (window->GetFrame()->GetSettings()->GetPresentationRequiresUserGesture() &&
!LocalFrame::HasTransientUserActivation(window->GetFrame())) {
exception_state.ThrowDOMException(
DOMExceptionCode::kInvalidAccessError,
"PresentationRequest::start() requires user gesture.");
return ScriptPromise();
}
PresentationController* controller = PresentationController::From(*window);
auto* resolver = MakeGarbageCollected<ScriptPromiseResolver>(script_state);
controller->GetPresentationService()->StartPresentation(
urls_,
WTF::Bind(
&PresentationConnectionCallbacks::HandlePresentationResponse,
std::make_unique<PresentationConnectionCallbacks>(resolver, this)));
return resolver->Promise();
}
ScriptPromise PresentationRequest::reconnect(ScriptState* script_state,
const String& id,
ExceptionState& exception_state) {
PresentationController* controller =
PresentationController::FromContext(GetExecutionContext());
if (!controller) {
exception_state.ThrowDOMException(
DOMExceptionCode::kInvalidStateError,
"The PresentationRequest is no longer associated to a frame.");
return ScriptPromise();
}
auto* resolver = MakeGarbageCollected<ScriptPromiseResolver>(script_state);
ControllerPresentationConnection* existing_connection =
controller->FindExistingConnection(urls_, id);
if (existing_connection) {
controller->GetPresentationService()->ReconnectPresentation(
urls_, id,
WTF::Bind(&PresentationConnectionCallbacks::HandlePresentationResponse,
std::make_unique<PresentationConnectionCallbacks>(
resolver, existing_connection)));
} else {
controller->GetPresentationService()->ReconnectPresentation(
urls_, id,
WTF::Bind(
&PresentationConnectionCallbacks::HandlePresentationResponse,
std::make_unique<PresentationConnectionCallbacks>(resolver, this)));
}
return resolver->Promise();
}
ScriptPromise PresentationRequest::getAvailability(
ScriptState* script_state,
ExceptionState& exception_state) {
PresentationController* controller =
PresentationController::FromContext(GetExecutionContext());
if (!controller) {
exception_state.ThrowDOMException(
DOMExceptionCode::kInvalidStateError,
"The PresentationRequest is no longer associated to a frame.");
return ScriptPromise();
}
if (!availability_property_) {
availability_property_ =
MakeGarbageCollected<PresentationAvailabilityProperty>(
ExecutionContext::From(script_state));
controller->GetAvailabilityState()->RequestAvailability(
urls_, MakeGarbageCollected<PresentationAvailabilityCallbacks>(
availability_property_, urls_));
}
return availability_property_->Promise(script_state->World());
}
const Vector<KURL>& PresentationRequest::Urls() const {
return urls_;
}
void PresentationRequest::Trace(Visitor* visitor) const {
visitor->Trace(availability_property_);
EventTargetWithInlineData::Trace(visitor);
ExecutionContextClient::Trace(visitor);
}
PresentationRequest::PresentationRequest(ExecutionContext* execution_context,
const Vector<KURL>& urls)
: ExecutionContextClient(execution_context), urls_(urls) {}
} // namespace blink