2015-03-25 14:47:56 +03:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
|
2015-03-26 06:16:21 +03:00
|
|
|
#include "ipc/PresentationIPCService.h"
|
2015-03-25 14:47:56 +03:00
|
|
|
#include "mozilla/Services.h"
|
2015-03-30 10:46:11 +03:00
|
|
|
#include "mozIApplication.h"
|
|
|
|
#include "nsIAppsService.h"
|
2015-03-25 14:47:56 +03:00
|
|
|
#include "nsIObserverService.h"
|
2015-03-30 09:27:27 +03:00
|
|
|
#include "nsIPresentationControlChannel.h"
|
|
|
|
#include "nsIPresentationDeviceManager.h"
|
|
|
|
#include "nsIPresentationDevicePrompt.h"
|
2015-03-25 14:47:56 +03:00
|
|
|
#include "nsIPresentationListener.h"
|
2015-03-30 10:48:11 +03:00
|
|
|
#include "nsIPresentationRequestUIGlue.h"
|
2015-03-30 10:46:11 +03:00
|
|
|
#include "nsIPresentationSessionRequest.h"
|
|
|
|
#include "nsNetUtil.h"
|
2015-03-25 14:47:56 +03:00
|
|
|
#include "nsServiceManagerUtils.h"
|
|
|
|
#include "nsThreadUtils.h"
|
2015-03-26 06:16:21 +03:00
|
|
|
#include "nsXULAppAPI.h"
|
2015-03-25 14:47:56 +03:00
|
|
|
#include "PresentationService.h"
|
|
|
|
|
|
|
|
using namespace mozilla;
|
|
|
|
using namespace mozilla::dom;
|
|
|
|
|
2015-03-30 09:27:27 +03:00
|
|
|
namespace mozilla {
|
|
|
|
namespace dom {
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Implementation of PresentationDeviceRequest
|
|
|
|
*/
|
|
|
|
|
|
|
|
class PresentationDeviceRequest final : public nsIPresentationDeviceRequest
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
NS_DECL_ISUPPORTS
|
|
|
|
NS_DECL_NSIPRESENTATIONDEVICEREQUEST
|
|
|
|
|
|
|
|
PresentationDeviceRequest(const nsAString& aRequestUrl,
|
|
|
|
const nsAString& aId,
|
|
|
|
const nsAString& aOrigin);
|
|
|
|
|
|
|
|
private:
|
|
|
|
virtual ~PresentationDeviceRequest();
|
|
|
|
|
|
|
|
nsString mRequestUrl;
|
|
|
|
nsString mId;
|
|
|
|
nsString mOrigin;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace dom
|
|
|
|
} // namespace mozilla
|
|
|
|
|
|
|
|
NS_IMPL_ISUPPORTS(PresentationDeviceRequest, nsIPresentationDeviceRequest)
|
|
|
|
|
|
|
|
PresentationDeviceRequest::PresentationDeviceRequest(const nsAString& aRequestUrl,
|
|
|
|
const nsAString& aId,
|
|
|
|
const nsAString& aOrigin)
|
|
|
|
: mRequestUrl(aRequestUrl)
|
|
|
|
, mId(aId)
|
|
|
|
, mOrigin(aOrigin)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(!mRequestUrl.IsEmpty());
|
|
|
|
MOZ_ASSERT(!mId.IsEmpty());
|
|
|
|
MOZ_ASSERT(!mOrigin.IsEmpty());
|
|
|
|
}
|
|
|
|
|
|
|
|
PresentationDeviceRequest::~PresentationDeviceRequest()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
PresentationDeviceRequest::GetOrigin(nsAString& aOrigin)
|
|
|
|
{
|
|
|
|
aOrigin = mOrigin;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
PresentationDeviceRequest::GetRequestURL(nsAString& aRequestUrl)
|
|
|
|
{
|
|
|
|
aRequestUrl = mRequestUrl;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
PresentationDeviceRequest::Select(nsIPresentationDevice* aDevice)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
MOZ_ASSERT(aDevice);
|
|
|
|
|
|
|
|
nsCOMPtr<nsIPresentationService> service =
|
|
|
|
do_GetService(PRESENTATION_SERVICE_CONTRACTID);
|
|
|
|
if (NS_WARN_IF(!service)) {
|
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update device in the session info.
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<PresentationSessionInfo> info =
|
2015-03-30 09:27:27 +03:00
|
|
|
static_cast<PresentationService*>(service.get())->GetSessionInfo(mId);
|
|
|
|
if (NS_WARN_IF(!info)) {
|
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
}
|
|
|
|
info->SetDevice(aDevice);
|
|
|
|
|
|
|
|
// Establish a control channel. If we failed to do so, the callback is called
|
|
|
|
// with an error message.
|
|
|
|
nsCOMPtr<nsIPresentationControlChannel> ctrlChannel;
|
|
|
|
nsresult rv = aDevice->EstablishControlChannel(mRequestUrl, mId, getter_AddRefs(ctrlChannel));
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
2015-09-09 11:38:26 +03:00
|
|
|
return info->ReplyError(NS_ERROR_DOM_OPERATION_ERR);
|
2015-03-30 09:27:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize the session info with the control channel.
|
|
|
|
rv = info->Init(ctrlChannel);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
2015-09-09 11:38:26 +03:00
|
|
|
return info->ReplyError(NS_ERROR_DOM_OPERATION_ERR);
|
2015-03-30 09:27:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
PresentationDeviceRequest::Cancel()
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIPresentationService> service =
|
|
|
|
do_GetService(PRESENTATION_SERVICE_CONTRACTID);
|
|
|
|
if (NS_WARN_IF(!service)) {
|
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
}
|
|
|
|
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<PresentationSessionInfo> info =
|
2015-03-30 09:27:27 +03:00
|
|
|
static_cast<PresentationService*>(service.get())->GetSessionInfo(mId);
|
|
|
|
if (NS_WARN_IF(!info)) {
|
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
}
|
|
|
|
|
2015-09-09 11:38:26 +03:00
|
|
|
return info->ReplyError(NS_ERROR_DOM_ABORT_ERR);
|
2015-03-30 09:27:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Implementation of PresentationService
|
|
|
|
*/
|
|
|
|
|
2015-03-25 14:47:56 +03:00
|
|
|
NS_IMPL_ISUPPORTS(PresentationService, nsIPresentationService, nsIObserver)
|
|
|
|
|
|
|
|
PresentationService::PresentationService()
|
2015-03-30 09:27:27 +03:00
|
|
|
: mIsAvailable(false)
|
2015-03-25 14:47:56 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
PresentationService::~PresentationService()
|
|
|
|
{
|
2015-03-30 09:27:27 +03:00
|
|
|
HandleShutdown();
|
2015-03-25 14:47:56 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
PresentationService::Init()
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
|
|
|
|
nsCOMPtr<nsIObserverService> obs = services::GetObserverService();
|
|
|
|
if (NS_WARN_IF(!obs)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-03-30 09:27:27 +03:00
|
|
|
nsresult rv = obs->AddObserver(this, NS_XPCOM_SHUTDOWN_OBSERVER_ID, false);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
rv = obs->AddObserver(this, PRESENTATION_DEVICE_CHANGE_TOPIC, false);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
|
|
return false;
|
|
|
|
}
|
2015-03-30 10:46:11 +03:00
|
|
|
rv = obs->AddObserver(this, PRESENTATION_SESSION_REQUEST_TOPIC, false);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
|
|
return false;
|
|
|
|
}
|
2015-03-25 14:47:56 +03:00
|
|
|
|
2015-03-30 09:27:27 +03:00
|
|
|
nsCOMPtr<nsIPresentationDeviceManager> deviceManager =
|
|
|
|
do_GetService(PRESENTATION_DEVICE_MANAGER_CONTRACTID);
|
|
|
|
if (NS_WARN_IF(!deviceManager)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
rv = deviceManager->GetDeviceAvailable(&mIsAvailable);
|
|
|
|
return !NS_WARN_IF(NS_FAILED(rv));
|
2015-03-25 14:47:56 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
PresentationService::Observe(nsISupports* aSubject,
|
|
|
|
const char* aTopic,
|
|
|
|
const char16_t* aData)
|
|
|
|
{
|
2015-03-30 09:27:27 +03:00
|
|
|
if (!strcmp(aTopic, NS_XPCOM_SHUTDOWN_OBSERVER_ID)) {
|
|
|
|
HandleShutdown();
|
|
|
|
return NS_OK;
|
|
|
|
} else if (!strcmp(aTopic, PRESENTATION_DEVICE_CHANGE_TOPIC)) {
|
|
|
|
return HandleDeviceChange();
|
2015-03-30 10:46:11 +03:00
|
|
|
} else if (!strcmp(aTopic, PRESENTATION_SESSION_REQUEST_TOPIC)) {
|
|
|
|
nsCOMPtr<nsIPresentationSessionRequest> request(do_QueryInterface(aSubject));
|
|
|
|
if (NS_WARN_IF(!request)) {
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return HandleSessionRequest(request);
|
2015-03-30 09:27:27 +03:00
|
|
|
} else if (!strcmp(aTopic, "profile-after-change")) {
|
|
|
|
// It's expected since we add and entry to |kLayoutCategories| in
|
|
|
|
// |nsLayoutModule.cpp| to launch this service earlier.
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
MOZ_ASSERT(false, "Unexpected topic for PresentationService");
|
|
|
|
return NS_ERROR_UNEXPECTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
PresentationService::HandleShutdown()
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
|
2015-09-10 11:29:08 +03:00
|
|
|
mAvailabilityListeners.Clear();
|
|
|
|
mRespondingListeners.Clear();
|
2015-03-30 09:27:27 +03:00
|
|
|
mSessionInfo.Clear();
|
2015-08-31 08:24:35 +03:00
|
|
|
mRespondingSessionIds.Clear();
|
2015-03-30 09:27:27 +03:00
|
|
|
|
|
|
|
nsCOMPtr<nsIObserverService> obs = services::GetObserverService();
|
|
|
|
if (obs) {
|
|
|
|
obs->RemoveObserver(this, NS_XPCOM_SHUTDOWN_OBSERVER_ID);
|
|
|
|
obs->RemoveObserver(this, PRESENTATION_DEVICE_CHANGE_TOPIC);
|
2015-03-30 10:46:11 +03:00
|
|
|
obs->RemoveObserver(this, PRESENTATION_SESSION_REQUEST_TOPIC);
|
2015-03-30 09:27:27 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
PresentationService::HandleDeviceChange()
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIPresentationDeviceManager> deviceManager =
|
|
|
|
do_GetService(PRESENTATION_DEVICE_MANAGER_CONTRACTID);
|
|
|
|
if (NS_WARN_IF(!deviceManager)) {
|
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isAvailable;
|
|
|
|
nsresult rv = deviceManager->GetDeviceAvailable(&isAvailable);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isAvailable != mIsAvailable) {
|
|
|
|
mIsAvailable = isAvailable;
|
|
|
|
NotifyAvailableChange(mIsAvailable);
|
|
|
|
}
|
2015-03-25 14:47:56 +03:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2015-03-30 10:46:11 +03:00
|
|
|
nsresult
|
|
|
|
PresentationService::HandleSessionRequest(nsIPresentationSessionRequest* aRequest)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIPresentationControlChannel> ctrlChannel;
|
|
|
|
nsresult rv = aRequest->GetControlChannel(getter_AddRefs(ctrlChannel));
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv) || !ctrlChannel)) {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsAutoString url;
|
|
|
|
rv = aRequest->GetUrl(url);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
2015-09-09 11:38:26 +03:00
|
|
|
ctrlChannel->Close(rv);
|
2015-03-30 10:46:11 +03:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsAutoString sessionId;
|
|
|
|
rv = aRequest->GetPresentationId(sessionId);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
2015-09-09 11:38:26 +03:00
|
|
|
ctrlChannel->Close(rv);
|
2015-03-30 10:46:11 +03:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsCOMPtr<nsIPresentationDevice> device;
|
|
|
|
rv = aRequest->GetDevice(getter_AddRefs(device));
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
2015-09-09 11:38:26 +03:00
|
|
|
ctrlChannel->Close(rv);
|
2015-03-30 10:46:11 +03:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef MOZ_WIDGET_GONK
|
|
|
|
// Verify the existence of the app if necessary.
|
|
|
|
nsCOMPtr<nsIURI> uri;
|
|
|
|
rv = NS_NewURI(getter_AddRefs(uri), url);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
|
|
ctrlChannel->Close(NS_ERROR_DOM_BAD_URI);
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isApp;
|
|
|
|
rv = uri->SchemeIs("app", &isApp);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
2015-09-09 11:38:26 +03:00
|
|
|
ctrlChannel->Close(rv);
|
2015-03-30 10:46:11 +03:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (NS_WARN_IF(isApp && !IsAppInstalled(uri))) {
|
|
|
|
ctrlChannel->Close(NS_ERROR_DOM_NOT_FOUND_ERR);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Create or reuse session info.
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<PresentationSessionInfo> info = GetSessionInfo(sessionId);
|
2015-03-30 10:46:11 +03:00
|
|
|
if (NS_WARN_IF(info)) {
|
2015-09-09 11:38:26 +03:00
|
|
|
// TODO Bug 1195605. Update here after session join/resume becomes supported.
|
|
|
|
ctrlChannel->Close(NS_ERROR_DOM_OPERATION_ERR);
|
2015-03-30 10:46:11 +03:00
|
|
|
return NS_ERROR_DOM_ABORT_ERR;
|
|
|
|
}
|
|
|
|
|
2015-09-09 12:41:55 +03:00
|
|
|
info = new PresentationPresentingInfo(url, sessionId, device);
|
2015-03-30 10:46:11 +03:00
|
|
|
rv = info->Init(ctrlChannel);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
2015-09-09 11:38:26 +03:00
|
|
|
ctrlChannel->Close(rv);
|
2015-03-30 10:46:11 +03:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
mSessionInfo.Put(sessionId, info);
|
|
|
|
|
|
|
|
// Notify the receiver to launch.
|
2015-03-30 10:48:11 +03:00
|
|
|
nsCOMPtr<nsIPresentationRequestUIGlue> glue =
|
|
|
|
do_CreateInstance(PRESENTATION_REQUEST_UI_GLUE_CONTRACTID);
|
|
|
|
if (NS_WARN_IF(!glue)) {
|
2015-09-09 11:38:26 +03:00
|
|
|
ctrlChannel->Close(NS_ERROR_DOM_OPERATION_ERR);
|
|
|
|
return info->ReplyError(NS_ERROR_DOM_OPERATION_ERR);
|
2015-03-30 10:46:11 +03:00
|
|
|
}
|
2015-03-30 10:48:11 +03:00
|
|
|
nsCOMPtr<nsISupports> promise;
|
|
|
|
rv = glue->SendRequest(url, sessionId, getter_AddRefs(promise));
|
2015-03-30 10:46:11 +03:00
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
2015-09-09 11:38:26 +03:00
|
|
|
ctrlChannel->Close(rv);
|
|
|
|
return info->ReplyError(NS_ERROR_DOM_OPERATION_ERR);
|
2015-03-30 10:46:11 +03:00
|
|
|
}
|
2015-03-30 10:48:11 +03:00
|
|
|
nsCOMPtr<Promise> realPromise = do_QueryInterface(promise);
|
2015-09-09 12:41:55 +03:00
|
|
|
static_cast<PresentationPresentingInfo*>(info.get())->SetPromise(realPromise);
|
2015-03-30 10:46:11 +03:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2015-03-25 14:47:56 +03:00
|
|
|
void
|
|
|
|
PresentationService::NotifyAvailableChange(bool aIsAvailable)
|
|
|
|
{
|
2015-09-10 11:29:08 +03:00
|
|
|
nsTObserverArray<nsCOMPtr<nsIPresentationAvailabilityListener>>::ForwardIterator iter(mAvailabilityListeners);
|
2015-03-25 14:47:56 +03:00
|
|
|
while (iter.HasMore()) {
|
2015-09-10 11:29:08 +03:00
|
|
|
nsCOMPtr<nsIPresentationAvailabilityListener> listener = iter.GetNext();
|
2015-03-25 14:47:56 +03:00
|
|
|
NS_WARN_IF(NS_FAILED(listener->NotifyAvailableChange(aIsAvailable)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-30 10:46:11 +03:00
|
|
|
bool
|
|
|
|
PresentationService::IsAppInstalled(nsIURI* aUri)
|
|
|
|
{
|
|
|
|
nsAutoCString prePath;
|
|
|
|
nsresult rv = aUri->GetPrePath(prePath);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsAutoString manifestUrl;
|
|
|
|
AppendUTF8toUTF16(prePath, manifestUrl);
|
|
|
|
manifestUrl.AppendLiteral("/manifest.webapp");
|
|
|
|
|
|
|
|
nsCOMPtr<nsIAppsService> appsService = do_GetService(APPS_SERVICE_CONTRACTID);
|
|
|
|
if (NS_WARN_IF(!appsService)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsCOMPtr<mozIApplication> app;
|
|
|
|
appsService->GetAppByManifestURL(manifestUrl, getter_AddRefs(app));
|
|
|
|
if (NS_WARN_IF(!app)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-03-25 14:47:56 +03:00
|
|
|
NS_IMETHODIMP
|
|
|
|
PresentationService::StartSession(const nsAString& aUrl,
|
|
|
|
const nsAString& aSessionId,
|
|
|
|
const nsAString& aOrigin,
|
2016-01-13 14:21:25 +03:00
|
|
|
const nsAString& aDeviceId,
|
2015-03-25 14:47:56 +03:00
|
|
|
nsIPresentationServiceCallback* aCallback)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
MOZ_ASSERT(aCallback);
|
|
|
|
MOZ_ASSERT(!aSessionId.IsEmpty());
|
|
|
|
|
2015-03-30 09:27:27 +03:00
|
|
|
// Create session info and set the callback. The callback is called when the
|
|
|
|
// request is finished.
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<PresentationSessionInfo> info =
|
2015-09-09 12:41:55 +03:00
|
|
|
new PresentationControllingInfo(aUrl, aSessionId, aCallback);
|
2015-03-30 09:27:27 +03:00
|
|
|
mSessionInfo.Put(aSessionId, info);
|
|
|
|
|
|
|
|
nsCOMPtr<nsIPresentationDeviceRequest> request =
|
|
|
|
new PresentationDeviceRequest(aUrl, aSessionId, aOrigin);
|
2016-01-13 14:21:25 +03:00
|
|
|
|
|
|
|
if (aDeviceId.IsVoid()) {
|
|
|
|
// Pop up a prompt and ask user to select a device.
|
|
|
|
nsCOMPtr<nsIPresentationDevicePrompt> prompt =
|
|
|
|
do_GetService(PRESENTATION_DEVICE_PROMPT_CONTRACTID);
|
|
|
|
if (NS_WARN_IF(!prompt)) {
|
|
|
|
return info->ReplyError(NS_ERROR_DOM_OPERATION_ERR);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult rv = prompt->PromptDeviceSelection(request);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
|
|
return info->ReplyError(NS_ERROR_DOM_OPERATION_ERR);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find the designated device from available device list.
|
|
|
|
nsCOMPtr<nsIPresentationDeviceManager> deviceManager =
|
|
|
|
do_GetService(PRESENTATION_DEVICE_MANAGER_CONTRACTID);
|
|
|
|
if (NS_WARN_IF(!deviceManager)) {
|
|
|
|
return info->ReplyError(NS_ERROR_DOM_OPERATION_ERR);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsCOMPtr<nsIArray> devices;
|
|
|
|
nsresult rv = deviceManager->GetAvailableDevices(getter_AddRefs(devices));
|
2015-03-30 09:27:27 +03:00
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
2015-09-09 11:38:26 +03:00
|
|
|
return info->ReplyError(NS_ERROR_DOM_OPERATION_ERR);
|
2015-03-30 09:27:27 +03:00
|
|
|
}
|
2015-03-25 14:47:56 +03:00
|
|
|
|
2016-01-13 14:21:25 +03:00
|
|
|
nsCOMPtr<nsISimpleEnumerator> enumerator;
|
|
|
|
rv = devices->Enumerate(getter_AddRefs(enumerator));
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
|
|
return info->ReplyError(NS_ERROR_DOM_OPERATION_ERR);
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_ConvertUTF16toUTF8 utf8DeviceId(aDeviceId);
|
|
|
|
bool hasMore;
|
|
|
|
while(NS_SUCCEEDED(enumerator->HasMoreElements(&hasMore)) && hasMore){
|
|
|
|
nsCOMPtr<nsISupports> isupports;
|
|
|
|
rv = enumerator->GetNext(getter_AddRefs(isupports));
|
|
|
|
|
|
|
|
nsCOMPtr<nsIPresentationDevice> device(do_QueryInterface(isupports));
|
|
|
|
MOZ_ASSERT(device);
|
|
|
|
|
|
|
|
nsAutoCString id;
|
|
|
|
if (NS_SUCCEEDED(device->GetId(id)) && id.Equals(utf8DeviceId)) {
|
|
|
|
request->Select(device);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reject if designated device is not available.
|
|
|
|
return info->ReplyError(NS_ERROR_DOM_NOT_FOUND_ERR);
|
2015-03-25 14:47:56 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
PresentationService::SendSessionMessage(const nsAString& aSessionId,
|
|
|
|
nsIInputStream* aStream)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
MOZ_ASSERT(aStream);
|
|
|
|
MOZ_ASSERT(!aSessionId.IsEmpty());
|
|
|
|
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<PresentationSessionInfo> info = GetSessionInfo(aSessionId);
|
2015-04-22 11:01:38 +03:00
|
|
|
if (NS_WARN_IF(!info)) {
|
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
}
|
2015-03-25 14:47:56 +03:00
|
|
|
|
2015-04-22 11:01:38 +03:00
|
|
|
return info->Send(aStream);
|
2015-03-25 14:47:56 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2015-10-08 13:11:10 +03:00
|
|
|
PresentationService::CloseSession(const nsAString& aSessionId)
|
2015-03-25 14:47:56 +03:00
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
MOZ_ASSERT(!aSessionId.IsEmpty());
|
|
|
|
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<PresentationSessionInfo> info = GetSessionInfo(aSessionId);
|
2015-04-22 11:01:38 +03:00
|
|
|
if (NS_WARN_IF(!info)) {
|
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
}
|
2015-03-25 14:47:56 +03:00
|
|
|
|
2015-10-08 13:11:10 +03:00
|
|
|
return info->Close(NS_OK, nsIPresentationSessionListener::STATE_CLOSED);
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
PresentationService::TerminateSession(const nsAString& aSessionId)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
MOZ_ASSERT(!aSessionId.IsEmpty());
|
|
|
|
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<PresentationSessionInfo> info = GetSessionInfo(aSessionId);
|
2015-10-08 13:11:10 +03:00
|
|
|
if (NS_WARN_IF(!info)) {
|
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return info->Close(NS_OK, nsIPresentationSessionListener::STATE_TERMINATED);
|
2015-03-25 14:47:56 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2015-09-10 11:29:08 +03:00
|
|
|
PresentationService::RegisterAvailabilityListener(nsIPresentationAvailabilityListener* aListener)
|
2015-03-25 14:47:56 +03:00
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
|
2015-09-10 11:29:08 +03:00
|
|
|
if (NS_WARN_IF(mAvailabilityListeners.Contains(aListener))) {
|
2015-03-25 14:47:56 +03:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2015-09-10 11:29:08 +03:00
|
|
|
mAvailabilityListeners.AppendElement(aListener);
|
2015-03-25 14:47:56 +03:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2015-09-10 11:29:08 +03:00
|
|
|
PresentationService::UnregisterAvailabilityListener(nsIPresentationAvailabilityListener* aListener)
|
2015-03-25 14:47:56 +03:00
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
|
2015-09-10 11:29:08 +03:00
|
|
|
mAvailabilityListeners.RemoveElement(aListener);
|
2015-03-25 14:47:56 +03:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
PresentationService::RegisterSessionListener(const nsAString& aSessionId,
|
|
|
|
nsIPresentationSessionListener* aListener)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
MOZ_ASSERT(aListener);
|
|
|
|
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<PresentationSessionInfo> info = GetSessionInfo(aSessionId);
|
2015-03-25 14:47:56 +03:00
|
|
|
if (NS_WARN_IF(!info)) {
|
2015-03-30 10:46:11 +03:00
|
|
|
// Notify the listener of TERMINATED since no correspondent session info is
|
|
|
|
// available possibly due to establishment failure. This would be useful at
|
|
|
|
// the receiver side, since a presentation session is created at beginning
|
|
|
|
// and here is the place to realize the underlying establishment fails.
|
|
|
|
nsresult rv = aListener->NotifyStateChange(aSessionId,
|
|
|
|
nsIPresentationSessionListener::STATE_TERMINATED);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
|
|
return rv;
|
|
|
|
}
|
2015-03-25 14:47:56 +03:00
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
}
|
|
|
|
|
2015-03-30 09:27:27 +03:00
|
|
|
return info->SetListener(aListener);
|
2015-03-25 14:47:56 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
PresentationService::UnregisterSessionListener(const nsAString& aSessionId)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<PresentationSessionInfo> info = GetSessionInfo(aSessionId);
|
2015-03-25 14:47:56 +03:00
|
|
|
if (info) {
|
2015-10-08 13:11:10 +03:00
|
|
|
NS_WARN_IF(NS_FAILED(info->Close(NS_OK, nsIPresentationSessionListener::STATE_TERMINATED)));
|
2015-08-31 08:24:35 +03:00
|
|
|
UntrackSessionInfo(aSessionId);
|
2015-03-30 09:27:27 +03:00
|
|
|
return info->SetListener(nullptr);
|
2015-03-25 14:47:56 +03:00
|
|
|
}
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2015-09-10 11:29:08 +03:00
|
|
|
NS_IMETHODIMP
|
|
|
|
PresentationService::RegisterRespondingListener(uint64_t aWindowId,
|
|
|
|
nsIPresentationRespondingListener* aListener)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
MOZ_ASSERT(aListener);
|
|
|
|
|
|
|
|
nsCOMPtr<nsIPresentationRespondingListener> listener;
|
|
|
|
if (mRespondingListeners.Get(aWindowId, getter_AddRefs(listener))) {
|
|
|
|
return (listener == aListener) ? NS_OK : NS_ERROR_DOM_INVALID_STATE_ERR;
|
|
|
|
}
|
|
|
|
|
|
|
|
mRespondingListeners.Put(aWindowId, aListener);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
PresentationService::UnregisterRespondingListener(uint64_t aWindowId)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
|
|
|
|
mRespondingListeners.Remove(aWindowId);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2015-03-25 14:47:56 +03:00
|
|
|
NS_IMETHODIMP
|
2015-08-31 08:24:35 +03:00
|
|
|
PresentationService::GetExistentSessionIdAtLaunch(uint64_t aWindowId,
|
|
|
|
nsAString& aSessionId)
|
2015-03-25 14:47:56 +03:00
|
|
|
{
|
2015-08-31 08:24:35 +03:00
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
|
|
|
|
nsString* sessionId = mRespondingSessionIds.Get(aWindowId);
|
|
|
|
if (sessionId) {
|
|
|
|
aSessionId.Assign(*sessionId);
|
|
|
|
} else {
|
|
|
|
aSessionId.Truncate();
|
|
|
|
}
|
2015-03-25 14:47:56 +03:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2015-08-31 08:24:35 +03:00
|
|
|
PresentationService::NotifyReceiverReady(const nsAString& aSessionId,
|
|
|
|
uint64_t aWindowId)
|
2015-03-25 14:47:56 +03:00
|
|
|
{
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<PresentationSessionInfo> info = GetSessionInfo(aSessionId);
|
2015-03-30 10:46:11 +03:00
|
|
|
if (NS_WARN_IF(!info)) {
|
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
}
|
2015-03-25 14:47:56 +03:00
|
|
|
|
2015-08-31 08:24:35 +03:00
|
|
|
// Only track the responding info when an actual window ID, which would never
|
|
|
|
// be 0, is provided (for an in-process receiver page).
|
|
|
|
if (aWindowId != 0) {
|
|
|
|
mRespondingSessionIds.Put(aWindowId, new nsAutoString(aSessionId));
|
|
|
|
mRespondingWindowIds.Put(aSessionId, aWindowId);
|
|
|
|
}
|
|
|
|
|
2015-09-09 12:41:55 +03:00
|
|
|
return static_cast<PresentationPresentingInfo*>(info.get())->NotifyResponderReady();
|
2015-03-25 14:47:56 +03:00
|
|
|
}
|
|
|
|
|
2015-08-31 08:24:35 +03:00
|
|
|
NS_IMETHODIMP
|
|
|
|
PresentationService::UntrackSessionInfo(const nsAString& aSessionId)
|
|
|
|
{
|
|
|
|
// Remove the session info.
|
|
|
|
mSessionInfo.Remove(aSessionId);
|
|
|
|
|
|
|
|
// Remove the in-process responding info if there's still any.
|
|
|
|
uint64_t windowId = 0;
|
|
|
|
if(mRespondingWindowIds.Get(aSessionId, &windowId)) {
|
|
|
|
mRespondingWindowIds.Remove(aSessionId);
|
|
|
|
mRespondingSessionIds.Remove(windowId);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
PresentationService::IsSessionAccessible(const nsAString& aSessionId,
|
|
|
|
base::ProcessId aProcessId)
|
|
|
|
{
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<PresentationSessionInfo> info = GetSessionInfo(aSessionId);
|
2015-08-31 08:24:35 +03:00
|
|
|
if (NS_WARN_IF(!info)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return info->IsAccessible(aProcessId);
|
|
|
|
}
|
|
|
|
|
2015-03-25 14:47:56 +03:00
|
|
|
already_AddRefed<nsIPresentationService>
|
|
|
|
NS_CreatePresentationService()
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
|
2015-03-26 06:16:21 +03:00
|
|
|
nsCOMPtr<nsIPresentationService> service;
|
|
|
|
if (XRE_GetProcessType() == GeckoProcessType_Content) {
|
|
|
|
service = new mozilla::dom::PresentationIPCService();
|
|
|
|
} else {
|
|
|
|
service = new PresentationService();
|
|
|
|
if (NS_WARN_IF(!static_cast<PresentationService*>(service.get())->Init())) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return service.forget();
|
2015-03-25 14:47:56 +03:00
|
|
|
}
|