#include "third_party/blink/renderer/modules/presentation/presentation_request.h"
#include <memory>
#include "base/containers/to_vector.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/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_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/platform/bindings/exception_state.h"
#include "third_party/blink/renderer/platform/heap/garbage_collected.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");
}
}
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(
DynamicTo<LocalDOMWindow>(execution_context)
->GetFrame()
->IsInFencedFrameTree()
? "PresentationRequest is not supported in a fenced frame tree."
: "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,
StrCat({"'", 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(
StrCat({"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.empty()) {
exception_state.ThrowDOMException(
DOMExceptionCode::kNotSupportedError,
"An empty sequence of URLs is not supported.");
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) {
EventTarget::AddedEventListener(event_type, registered_listener);
if (event_type == event_type_names::kConnectionavailable) {
UseCounter::Count(
GetExecutionContext(),
WebFeature::kPresentationRequestConnectionAvailableEventListener);
}
}
bool PresentationRequest::HasPendingActivity() const {
if (!GetExecutionContext()) {
return false;
}
return HasEventListeners();
}
ScriptPromise<PresentationConnection> 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 EmptyPromise();
}
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 EmptyPromise();
}
PresentationController* controller = PresentationController::From(*window);
auto* resolver =
MakeGarbageCollected<ScriptPromiseResolver<PresentationConnection>>(
script_state, exception_state.GetContext());
controller->GetPresentationService()->StartPresentation(
urls_,
BindOnce(
&PresentationConnectionCallbacks::HandlePresentationResponse,
std::make_unique<PresentationConnectionCallbacks>(resolver, this)));
return resolver->Promise();
}
ScriptPromise<PresentationConnection> 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 EmptyPromise();
}
auto* resolver =
MakeGarbageCollected<ScriptPromiseResolver<PresentationConnection>>(
script_state, exception_state.GetContext());
ControllerPresentationConnection* existing_connection =
controller->FindExistingConnection(base::ToVector(urls_, ToWebURL), id);
if (existing_connection) {
controller->GetPresentationService()->ReconnectPresentation(
urls_, id,
BindOnce(&PresentationConnectionCallbacks::HandlePresentationResponse,
std::make_unique<PresentationConnectionCallbacks>(
resolver, existing_connection)));
} else {
controller->GetPresentationService()->ReconnectPresentation(
urls_, id,
BindOnce(
&PresentationConnectionCallbacks::HandlePresentationResponse,
std::make_unique<PresentationConnectionCallbacks>(resolver, this)));
}
return resolver->Promise();
}
ScriptPromise<PresentationAvailability> 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 EmptyPromise();
}
auto* resolver =
MakeGarbageCollected<ScriptPromiseResolver<PresentationAvailability>>(
script_state, exception_state.GetContext());
auto screen_availability =
controller->GetAvailabilityState()->GetScreenAvailability(urls_);
if (screen_availability == mojom::blink::ScreenAvailability::DISABLED) {
resolver->RejectWithDOMException(
DOMExceptionCode::kNotSupportedError,
PresentationAvailability::kNotSupportedErrorInfo);
return resolver->Promise();
}
if (!availability_) {
availability_ = PresentationAvailability::Take(
resolver->GetExecutionContext(), urls_,
screen_availability == mojom::blink::ScreenAvailability::AVAILABLE);
}
if (screen_availability != mojom::blink::ScreenAvailability::UNKNOWN) {
resolver->Resolve(availability_);
} else {
controller->GetAvailabilityState()->RequestAvailability(availability_);
availability_->AddResolver(resolver);
}
return resolver->Promise();
}
const Vector<KURL>& PresentationRequest::Urls() const {
return urls_;
}
void PresentationRequest::Trace(Visitor* visitor) const {
EventTarget::Trace(visitor);
ExecutionContextClient::Trace(visitor);
visitor->Trace(availability_);
}
PresentationRequest::PresentationRequest(ExecutionContext* execution_context,
const Vector<KURL>& urls)
: ActiveScriptWrappable<PresentationRequest>({}),
ExecutionContextClient(execution_context),
urls_(urls) {}
}