mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 23:06:52 +00:00
cf2d7d1ae9
- Bug 1212114 - Stop using dom::Promise::MaybeRejectBrokenly() in various FileSystemTaskBase subclasses. r=baku (9f6ea5db27) - Bug 1253534 - Suspicious code with probably reversed parms in call to IsSingleLineTextControl, r=mounir (1a5ee1fd1e) - Bug 769117 - Mochitests for youtube flash -> html5 rewriting; r=bz r=hsivonen (6324471dd9) - Bug 1250148 - FormData should treat empty input type=file as empty string in FormData and as unnamed Blob in HTML submission, r=smaug (7043113247) - Bug 1173320 - patch 1/8 - Implement Directory object as string and not as BlobImpl, r=smaug (eca4bec6ea) - Bug 1173320 - patch 2/8 - Proper naming for the FileSystem path serialization, r=smaug (e1604ff2b1) - Bug 1173320 - patch 3/8 - Improve the Windows path management, r=smaug (f8da8541b5) - Bug 1252347 - Provide missing implementations of nsIBaseWindow::SetPositionDesktopPix. r=emk (97f9b92465) - Bug 1235066 - SVG elements should not display title attributes as tooltips. r=enndeakin (a77e809688) - Bug 1251809 - Add input[type=file] tooltip support for e10s. r=ehsan (9f72cf3cd2) - Bug 1173320 - patch 4/8 - Directory in FileList, r=smaug (c09d445043) - Bug 1246244 - Regression test. r=jaws,Margaret (8de084f9db) - Bug 1224105 - Allow windowless chrome docshells containing content docshells (r=smaug) (3343a2a966) - Bug 1230267 - Inconsistent display of SVG title as tooltip when multi-process is enabled. r=jst (b3fc298b7f) - Bug 1173320 - patch 5/8 - Cleanup manual string path management, r=smaug (8f6f0c4e44) - Bug 1250403 - Part 1. Define ARCH_CPU_ARM64 instead of ARCH_CPU_AARCH64. r=billm (d93a0b54da) - Bug 1250403 - Part 2. Import crbug #354405 for aarch64. r=billm (9dca949bcc) - Bug 1246501 - Add ppc specific atomic operations to ipc/chromium. r=Waldo (8054b612be) - Bug 1257305 - Avoid VS2015 about casting int to void*. r=dvander (6234acf6fc) - Bug 1207401 - Send B2G sandbox logging to both stderr and logcat. r=kang (ae404aa5ca) - Bug 1173320 - patch 6/8 - Make FileList clonable to workers if it doesn't contain Directories, r=smaug (ea6ba42f31) - Bug 1173320 - patch 7/8 - Tests for FileList and Directories, r=smaug (3f11503300) - Bug 1222522, part 1 - Make most dom/devicestorage/ tests work with e10s. r=dhylands (3cc7e339a5) - Bug 1173320 - patch 8/8 - Fix e10s tests for DeviceStorage API, r=smaug (949454bae7) - Bug 1222522, part 2 - Inline devicestorage_cleanup() in dom/devicestorage/ tests. r=dhylands (bcdee11385) - Bug 1258137 - OSFileSystem should not be kept alive by more than 1 Directory, r=smaug (225775f48d) - Bug 1255867. Remove some unnecessary AutoJSAPI uses. r=bholley (618cf018e8) - Bug 1237173 - Part2: Change type of duration to Maybe<StickyTimeDuration>. r=birtles (6b83473e05) - Bug 1238469 - Part 1: Refactor b2g emulator tests to remove the usage of custom mozharness configs; r=ahal (a128c8a8bb) - Bug 1238469 - Part 2: Refactor b2g mulet tests to remove the usage of custom mozharness configs; r=ahal (6c7ced3419) - Bug 1238469 - Part 3: Use b2g_emulator_unittest.py for b2g marionette tests; r=ahal (5fdb88b0e1) - Bug 1237173 - Part3: Throw TypeError if duration is NaN, negative value or not 'auto' string. r=birtles, r=smaug (d60b0318d2) - Bug 1136567 - Marionette test for selection carets' positions after changing orientation of device. r=Automatedtester (e7b25b1e4f) - Bug 1138839 - Part1 - Marionette test for selecting text inside an iframe. r=automatedtester (45db397a94) - Bug 1138839 - Part2 - Fix naming issue in marionette test for selection carets. r=automatedtester (c54850d9d3) - Bug 1198542: Update Marionette element IDs to be valid UUID. r=ato This allows WebDriver compatibility to create valid URI. (1b3de245de) - Bug 1204496: When searching by link text start from the startNode and not the rootNode; r=ato (0dfa3b8830) - Bug 1157725 - Rewrite test_mouse_action.py to be more robust. r=ato (40273abca2) - Bug 1141519: added test that puts marionette into a position that can cause hangs when in content scope; r=jgriffin (c99d8cf325) - Bug 1246407 - Rename parent directories for Marionette client and test harness; r=automatedtester (79d4e521fd) - Bug 1253989 Part 3 - Use @parameterized to rewrite selection mode tests. r=mtseng (baf189f1d5) - Bug 1253989 Part 4 - Rename AccessibleCaret test files. r=mtseng (d562edc8f4) - Bug 1251519 Part 3 - Add regression tests for caret dragging. r=mats (048dd5f103) - Bug 1253989 Part 5 - Remove touch caret and selection carets naming. r=mtseng (9929425b6d) - Bug 1253989 Part 6 - Refactor open_test_html(). r=mtseng (25a2424ca1) - Bug 1251519 Part 4 - Add tests for dragging caret to content boundary. r=mats (489b47269b) - Bug 1251519 Part 5 - Use union rect of child frames for clamping. r=mats (5520416749) - Bug 1216924 - Don't align to tiles (either real or virtual) if displayport suppression is enabled. r=BenWa (d470d188e4) - Bug 1254273 - Align the displayport to a max of 256 pixels even if the layer is larger. r=BenWa (0cf8cc0b6c) - Bug 1257938 part 3: Remove support for the "layout.css.sticky.enabled" pref (so we'll unconditionally support "position: sticky"). r=corey (fe12efd18f) - Bug 1257938 part 1: Adjust automated tests to assume position:sticky is unconditionally supported. r=corey (e749ac579e) - Bug 1257938 part 2: Remove separation between test_position_sticky.html & its helper-file, now that it doesn't need to tweak a pref. r=corey (631edab31a) - Bug 1257491 - Ensure that if the peek-messages code modifies the displayport, we schedule a repaint. r=BenWa (d6fb6ff96c) - Bug 1255006 - Ensure the displayport rect takes priority over a suppressed-margins displayport. r=kats (aa6cbc0250) - Bug 1259235 - Add IsScrollFrameWithSnapping to speed up event regions. r=mstange (2a744c311e) - Bug 1185140 - [css-grid][flexbox] Make grid/flex item blockification happen before creating table pseudos, per the latest specs. r=dholbert (dddb8b17ef) - Bug 1224424 - Replace mask-mode:auto keyword by mask-mode:match-source; r=dbaron (5a51b3b301) - Bug 1252039 - corrected MOZ_ASSERT expression in SeparatorRequiredBetweenTokens. r=dbaron (08790aa514) - Bug 1243734 - Part 1. Use MOZ_ENABLE_MASK_AS_SHORTHAND to define the type of mask property; r=dbaron (d8cd3a1c4a) - Bug 1243734 - Part 2. Set up gCSSProperties depends on mask-as-shorth and; r=dbaron (aa6b0259d8) - Bug 1243734 - Part 3. Set mask-mode reftest as failure before enable mask-as-shorthand; r=dbaron (6a326fbaf2) - Bug 1243734 - Part 5. Add MOZ_ENABLE_MASK_AS_SHORTHAND compile flag; r=ted r=dbaron (fcc1344ac8) - Bug 1142531: Check more bits in nsStyleContext::MoveTo assertion. r=heycam (8b62b139df) - Bug 1258147 - Pierce through display:contents style context ancestors when looking for CB context to compare our writing-mode to. r=jfkthame (956d8c25e5) - Bug 823483 patch 1 - Check for percentage max-width in addition to percentage width when deciding to ignore intrinsic min-width of replaced elements. r=dholbert (f88cb5f6a8) - Bug 823483 patch 3 - Limit effect of percentage width and max-width on intrinsic size to elements with replaced element sizing. r=dholbert (2573c3cfff) - Bug 823483 patch 4 - Make a percentage max-width override a fixed width for replaced element intrinsic size computation. r=dholbert (dda859f06c) - Bug 823483 patch 5 - Make (again) percentage width on text inputs make intrinsic minimum width be 0. r=dholbert (d46ada73ef) - Bug 1247929 patch 2 - Hard-code the Web-compatible set of form controls whose intrinsic minimum inline-size shrinks to 0 when inline-size (width) is specified as a percentage. r=dholbert (e4f0c80fcb) - Bug 1254968 - Add support for running JS builtins' constructors over Xray wrappers without unwrapping the newTarget. r=bholley,f=bz (56213ae395) - Bug 1249123 - Add telemetry for __defineGetter__/__defineSetter__ |this| values. data-review=bsmedberg r=till (52c5fd3488) - Bug 1232639 - Implement Object.{values,entries} in C++ to avoid native call overhead in tight loop. r=jorendorff (7262497283) - Bug 1254966 - Disambiguate JS Telemetry macro names. r=evilpie (781d0916c4) - Bug 1254384: Use generic shell switch syntax in js/src/jit-test tests. r=nbp (6f5975cc55) - Bug 1253016 - Remove legacy __defineGetter__/__defineSetter__ this behavior. r=till (cf1b7ad28c) - Bug 1253016 - Implement and test the new spec for legacy functions. r=till (1ff7762e3e)
692 lines
20 KiB
C++
692 lines
20 KiB
C++
/* 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/. */
|
|
|
|
#include "BrowserElementAudioChannel.h"
|
|
|
|
#include "mozilla/Preferences.h"
|
|
#include "mozilla/Services.h"
|
|
#include "mozilla/dom/BrowserElementAudioChannelBinding.h"
|
|
#include "mozilla/dom/DOMRequest.h"
|
|
#include "mozilla/dom/Element.h"
|
|
#include "mozilla/dom/TabParent.h"
|
|
#include "mozilla/dom/Promise.h"
|
|
#include "mozilla/dom/PromiseNativeHandler.h"
|
|
#include "mozilla/dom/ToJSValue.h"
|
|
#include "AudioChannelService.h"
|
|
#include "nsIAppsService.h"
|
|
#include "nsIBrowserElementAPI.h"
|
|
#include "nsIDocShell.h"
|
|
#include "nsIDOMDocument.h"
|
|
#include "nsIDOMDOMRequest.h"
|
|
#include "nsIObserverService.h"
|
|
#include "nsISupportsPrimitives.h"
|
|
#include "nsISystemMessagesInternal.h"
|
|
#include "nsITabParent.h"
|
|
#include "nsNetUtil.h"
|
|
#include "nsPIDOMWindow.h"
|
|
#include "nsServiceManagerUtils.h"
|
|
#include "nsContentUtils.h"
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
NS_IMPL_ADDREF_INHERITED(BrowserElementAudioChannel, DOMEventTargetHelper)
|
|
NS_IMPL_RELEASE_INHERITED(BrowserElementAudioChannel, DOMEventTargetHelper)
|
|
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION_INHERITED(BrowserElementAudioChannel)
|
|
NS_INTERFACE_MAP_ENTRY(nsISupportsWeakReference)
|
|
NS_INTERFACE_MAP_ENTRY(nsIObserver)
|
|
NS_INTERFACE_MAP_END_INHERITING(DOMEventTargetHelper)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_INHERITED(BrowserElementAudioChannel,
|
|
DOMEventTargetHelper,
|
|
mFrameLoader,
|
|
mFrameWindow,
|
|
mTabParent,
|
|
mBrowserElementAPI)
|
|
|
|
/* static */ already_AddRefed<BrowserElementAudioChannel>
|
|
BrowserElementAudioChannel::Create(nsPIDOMWindow* aWindow,
|
|
nsIFrameLoader* aFrameLoader,
|
|
nsIBrowserElementAPI* aAPI,
|
|
AudioChannel aAudioChannel,
|
|
const nsAString& aManifestURL,
|
|
ErrorResult& aRv)
|
|
{
|
|
RefPtr<BrowserElementAudioChannel> ac =
|
|
new BrowserElementAudioChannel(aWindow, aFrameLoader, aAPI,
|
|
aAudioChannel, aManifestURL);
|
|
|
|
aRv = ac->Initialize();
|
|
if (NS_WARN_IF(aRv.Failed())) {
|
|
return nullptr;
|
|
}
|
|
|
|
MOZ_LOG(AudioChannelService::GetAudioChannelLog(), LogLevel::Debug,
|
|
("BrowserElementAudioChannel, Create, channel = %p, type = %d\n",
|
|
ac.get(), aAudioChannel));
|
|
|
|
return ac.forget();
|
|
}
|
|
|
|
BrowserElementAudioChannel::BrowserElementAudioChannel(
|
|
nsPIDOMWindow* aWindow,
|
|
nsIFrameLoader* aFrameLoader,
|
|
nsIBrowserElementAPI* aAPI,
|
|
AudioChannel aAudioChannel,
|
|
const nsAString& aManifestURL)
|
|
: DOMEventTargetHelper(aWindow)
|
|
, mFrameLoader(aFrameLoader)
|
|
, mBrowserElementAPI(aAPI)
|
|
, mAudioChannel(aAudioChannel)
|
|
, mManifestURL(aManifestURL)
|
|
, mState(eStateUnknown)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
nsCOMPtr<nsIObserverService> obs = mozilla::services::GetObserverService();
|
|
if (obs) {
|
|
nsAutoString name;
|
|
AudioChannelService::GetAudioChannelString(aAudioChannel, name);
|
|
|
|
nsAutoCString topic;
|
|
topic.Assign("audiochannel-activity-");
|
|
topic.Append(NS_ConvertUTF16toUTF8(name));
|
|
|
|
obs->AddObserver(this, topic.get(), true);
|
|
}
|
|
}
|
|
|
|
BrowserElementAudioChannel::~BrowserElementAudioChannel()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
nsCOMPtr<nsIObserverService> obs = mozilla::services::GetObserverService();
|
|
if (obs) {
|
|
nsAutoString name;
|
|
AudioChannelService::GetAudioChannelString(mAudioChannel, name);
|
|
|
|
nsAutoCString topic;
|
|
topic.Assign("audiochannel-activity-");
|
|
topic.Append(NS_ConvertUTF16toUTF8(name));
|
|
|
|
obs->RemoveObserver(this, topic.get());
|
|
}
|
|
}
|
|
|
|
nsresult
|
|
BrowserElementAudioChannel::Initialize()
|
|
{
|
|
if (!mFrameLoader) {
|
|
nsCOMPtr<nsPIDOMWindow> window = GetOwner();
|
|
if (!window) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
mFrameWindow = window->GetScriptableTop();
|
|
mFrameWindow = mFrameWindow->GetOuterWindow();
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<nsIDocShell> docShell;
|
|
nsresult rv = mFrameLoader->GetDocShell(getter_AddRefs(docShell));
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
if (docShell) {
|
|
nsCOMPtr<nsPIDOMWindow> window = docShell->GetWindow();
|
|
if (!window) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
mFrameWindow = window->GetScriptableTop();
|
|
mFrameWindow = mFrameWindow->GetOuterWindow();
|
|
return NS_OK;
|
|
}
|
|
|
|
rv = mFrameLoader->GetTabParent(getter_AddRefs(mTabParent));
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
MOZ_ASSERT(mTabParent);
|
|
return NS_OK;
|
|
}
|
|
|
|
JSObject*
|
|
BrowserElementAudioChannel::WrapObject(JSContext *aCx,
|
|
JS::Handle<JSObject*> aGivenProto)
|
|
{
|
|
return BrowserElementAudioChannelBinding::Wrap(aCx, this, aGivenProto);
|
|
}
|
|
|
|
AudioChannel
|
|
BrowserElementAudioChannel::Name() const
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
return mAudioChannel;
|
|
}
|
|
|
|
namespace {
|
|
|
|
class BaseRunnable : public nsRunnable
|
|
{
|
|
protected:
|
|
nsCOMPtr<nsPIDOMWindow> mParentWindow;
|
|
nsCOMPtr<nsPIDOMWindow> mFrameWindow;
|
|
RefPtr<DOMRequest> mRequest;
|
|
AudioChannel mAudioChannel;
|
|
|
|
virtual void DoWork(AudioChannelService* aService,
|
|
JSContext* aCx) = 0;
|
|
|
|
public:
|
|
BaseRunnable(nsPIDOMWindow* aParentWindow, nsPIDOMWindow* aFrameWindow,
|
|
DOMRequest* aRequest, AudioChannel aAudioChannel)
|
|
: mParentWindow(aParentWindow)
|
|
, mFrameWindow(aFrameWindow)
|
|
, mRequest(aRequest)
|
|
, mAudioChannel(aAudioChannel)
|
|
{}
|
|
|
|
NS_IMETHODIMP Run() override
|
|
{
|
|
RefPtr<AudioChannelService> service = AudioChannelService::GetOrCreate();
|
|
if (!service) {
|
|
return NS_OK;
|
|
}
|
|
|
|
AutoJSAPI jsapi;
|
|
if (!jsapi.Init(mParentWindow)) {
|
|
mRequest->FireError(NS_ERROR_FAILURE);
|
|
return NS_OK;
|
|
}
|
|
|
|
DoWork(service, jsapi.cx());
|
|
return NS_OK;
|
|
}
|
|
};
|
|
|
|
class GetVolumeRunnable final : public BaseRunnable
|
|
{
|
|
public:
|
|
GetVolumeRunnable(nsPIDOMWindow* aParentWindow, nsPIDOMWindow* aFrameWindow,
|
|
DOMRequest* aRequest, AudioChannel aAudioChannel)
|
|
: BaseRunnable(aParentWindow, aFrameWindow, aRequest, aAudioChannel)
|
|
{}
|
|
|
|
protected:
|
|
virtual void DoWork(AudioChannelService* aService, JSContext* aCx) override
|
|
{
|
|
float volume = aService->GetAudioChannelVolume(mFrameWindow, mAudioChannel);
|
|
|
|
JS::Rooted<JS::Value> value(aCx);
|
|
if (!ToJSValue(aCx, volume, &value)) {
|
|
mRequest->FireError(NS_ERROR_FAILURE);
|
|
return;
|
|
}
|
|
|
|
mRequest->FireSuccess(value);
|
|
}
|
|
};
|
|
|
|
class GetMutedRunnable final : public BaseRunnable
|
|
{
|
|
public:
|
|
GetMutedRunnable(nsPIDOMWindow* aParentWindow, nsPIDOMWindow* aFrameWindow,
|
|
DOMRequest* aRequest, AudioChannel aAudioChannel)
|
|
: BaseRunnable(aParentWindow, aFrameWindow, aRequest, aAudioChannel)
|
|
{}
|
|
|
|
protected:
|
|
virtual void DoWork(AudioChannelService* aService, JSContext* aCx) override
|
|
{
|
|
bool muted = aService->GetAudioChannelMuted(mFrameWindow, mAudioChannel);
|
|
|
|
JS::Rooted<JS::Value> value(aCx);
|
|
if (!ToJSValue(aCx, muted, &value)) {
|
|
mRequest->FireError(NS_ERROR_FAILURE);
|
|
return;
|
|
}
|
|
|
|
mRequest->FireSuccess(value);
|
|
}
|
|
};
|
|
|
|
class IsActiveRunnable final : public BaseRunnable
|
|
{
|
|
bool mActive;
|
|
bool mValueKnown;
|
|
|
|
public:
|
|
IsActiveRunnable(nsPIDOMWindow* aParentWindow, nsPIDOMWindow* aFrameWindow,
|
|
DOMRequest* aRequest, AudioChannel aAudioChannel,
|
|
bool aActive)
|
|
: BaseRunnable(aParentWindow, aFrameWindow, aRequest, aAudioChannel)
|
|
, mActive(aActive)
|
|
, mValueKnown(true)
|
|
{}
|
|
|
|
IsActiveRunnable(nsPIDOMWindow* aParentWindow, nsPIDOMWindow* aFrameWindow,
|
|
DOMRequest* aRequest, AudioChannel aAudioChannel)
|
|
: BaseRunnable(aParentWindow, aFrameWindow, aRequest, aAudioChannel)
|
|
, mActive(true)
|
|
, mValueKnown(false)
|
|
{}
|
|
|
|
protected:
|
|
virtual void DoWork(AudioChannelService* aService, JSContext* aCx) override
|
|
{
|
|
if (!mValueKnown) {
|
|
mActive = aService->IsAudioChannelActive(mFrameWindow, mAudioChannel);
|
|
}
|
|
|
|
JS::Rooted<JS::Value> value(aCx);
|
|
if (!ToJSValue(aCx, mActive, &value)) {
|
|
mRequest->FireError(NS_ERROR_FAILURE);
|
|
return;
|
|
}
|
|
|
|
mRequest->FireSuccess(value);
|
|
}
|
|
};
|
|
|
|
class FireSuccessRunnable final : public BaseRunnable
|
|
{
|
|
public:
|
|
FireSuccessRunnable(nsPIDOMWindow* aParentWindow, nsPIDOMWindow* aFrameWindow,
|
|
DOMRequest* aRequest, AudioChannel aAudioChannel)
|
|
: BaseRunnable(aParentWindow, aFrameWindow, aRequest, aAudioChannel)
|
|
{}
|
|
|
|
protected:
|
|
virtual void DoWork(AudioChannelService* aService, JSContext* aCx) override
|
|
{
|
|
JS::Rooted<JS::Value> value(aCx);
|
|
mRequest->FireSuccess(value);
|
|
}
|
|
};
|
|
|
|
class RespondSuccessHandler final : public PromiseNativeHandler
|
|
{
|
|
public:
|
|
NS_DECL_ISUPPORTS
|
|
|
|
explicit RespondSuccessHandler(DOMRequest* aRequest)
|
|
: mDomRequest(aRequest)
|
|
{};
|
|
|
|
virtual void
|
|
ResolvedCallback(JSContext* aCx, JS::Handle<JS::Value> aValue) override;
|
|
|
|
virtual void
|
|
RejectedCallback(JSContext* aCx, JS::Handle<JS::Value> aValue) override;
|
|
|
|
private:
|
|
~RespondSuccessHandler() {};
|
|
|
|
RefPtr<DOMRequest> mDomRequest;
|
|
};
|
|
NS_IMPL_ISUPPORTS0(RespondSuccessHandler);
|
|
|
|
void
|
|
RespondSuccessHandler::ResolvedCallback(JSContext* aCx,
|
|
JS::Handle<JS::Value> aValue)
|
|
{
|
|
JS::Rooted<JS::Value> value(aCx);
|
|
mDomRequest->FireSuccess(value);
|
|
}
|
|
|
|
void
|
|
RespondSuccessHandler::RejectedCallback(JSContext* aCx,
|
|
JS::Handle<JS::Value> aValue)
|
|
{
|
|
mDomRequest->FireError(NS_ERROR_FAILURE);
|
|
}
|
|
|
|
} // anonymous namespace
|
|
|
|
already_AddRefed<dom::DOMRequest>
|
|
BrowserElementAudioChannel::GetVolume(ErrorResult& aRv)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (!mFrameWindow) {
|
|
nsCOMPtr<nsIDOMDOMRequest> request;
|
|
aRv = mBrowserElementAPI->GetAudioChannelVolume((uint32_t)mAudioChannel,
|
|
getter_AddRefs(request));
|
|
if (NS_WARN_IF(aRv.Failed())) {
|
|
return nullptr;
|
|
}
|
|
|
|
return request.forget().downcast<DOMRequest>();
|
|
}
|
|
|
|
RefPtr<DOMRequest> domRequest = new DOMRequest(GetOwner());
|
|
|
|
nsCOMPtr<nsIRunnable> runnable =
|
|
new GetVolumeRunnable(GetOwner(), mFrameWindow, domRequest, mAudioChannel);
|
|
NS_DispatchToMainThread(runnable);
|
|
|
|
return domRequest.forget();
|
|
}
|
|
|
|
already_AddRefed<dom::DOMRequest>
|
|
BrowserElementAudioChannel::SetVolume(float aVolume, ErrorResult& aRv)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (!mFrameWindow) {
|
|
nsCOMPtr<nsIDOMDOMRequest> request;
|
|
aRv = mBrowserElementAPI->SetAudioChannelVolume((uint32_t)mAudioChannel,
|
|
aVolume,
|
|
getter_AddRefs(request));
|
|
if (NS_WARN_IF(aRv.Failed())) {
|
|
return nullptr;
|
|
}
|
|
|
|
return request.forget().downcast<DOMRequest>();
|
|
}
|
|
|
|
RefPtr<AudioChannelService> service = AudioChannelService::GetOrCreate();
|
|
if (service) {
|
|
service->SetAudioChannelVolume(mFrameWindow, mAudioChannel, aVolume);
|
|
}
|
|
|
|
RefPtr<DOMRequest> domRequest = new DOMRequest(GetOwner());
|
|
nsCOMPtr<nsIRunnable> runnable = new FireSuccessRunnable(GetOwner(),
|
|
mFrameWindow,
|
|
domRequest,
|
|
mAudioChannel);
|
|
NS_DispatchToMainThread(runnable);
|
|
|
|
return domRequest.forget();
|
|
}
|
|
|
|
already_AddRefed<dom::DOMRequest>
|
|
BrowserElementAudioChannel::GetMuted(ErrorResult& aRv)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (!mFrameWindow) {
|
|
nsCOMPtr<nsIDOMDOMRequest> request;
|
|
aRv = mBrowserElementAPI->GetAudioChannelMuted((uint32_t)mAudioChannel,
|
|
getter_AddRefs(request));
|
|
if (NS_WARN_IF(aRv.Failed())) {
|
|
return nullptr;
|
|
}
|
|
|
|
return request.forget().downcast<DOMRequest>();
|
|
}
|
|
|
|
RefPtr<DOMRequest> domRequest = new DOMRequest(GetOwner());
|
|
|
|
nsCOMPtr<nsIRunnable> runnable =
|
|
new GetMutedRunnable(GetOwner(), mFrameWindow, domRequest, mAudioChannel);
|
|
NS_DispatchToMainThread(runnable);
|
|
|
|
return domRequest.forget();
|
|
}
|
|
|
|
already_AddRefed<dom::DOMRequest>
|
|
BrowserElementAudioChannel::SetMuted(bool aMuted, ErrorResult& aRv)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (!mFrameWindow) {
|
|
nsCOMPtr<nsIDOMDOMRequest> request;
|
|
aRv = mBrowserElementAPI->SetAudioChannelMuted((uint32_t)mAudioChannel,
|
|
aMuted,
|
|
getter_AddRefs(request));
|
|
if (NS_WARN_IF(aRv.Failed())) {
|
|
return nullptr;
|
|
}
|
|
|
|
return request.forget().downcast<DOMRequest>();
|
|
}
|
|
|
|
RefPtr<AudioChannelService> service = AudioChannelService::GetOrCreate();
|
|
if (service) {
|
|
service->SetAudioChannelMuted(mFrameWindow, mAudioChannel, aMuted);
|
|
}
|
|
|
|
RefPtr<DOMRequest> domRequest = new DOMRequest(GetOwner());
|
|
nsCOMPtr<nsIRunnable> runnable = new FireSuccessRunnable(GetOwner(),
|
|
mFrameWindow,
|
|
domRequest,
|
|
mAudioChannel);
|
|
NS_DispatchToMainThread(runnable);
|
|
|
|
return domRequest.forget();
|
|
}
|
|
|
|
already_AddRefed<dom::DOMRequest>
|
|
BrowserElementAudioChannel::IsActive(ErrorResult& aRv)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (mState != eStateUnknown) {
|
|
RefPtr<DOMRequest> domRequest = new DOMRequest(GetOwner());
|
|
|
|
nsCOMPtr<nsIRunnable> runnable =
|
|
new IsActiveRunnable(GetOwner(), mFrameWindow, domRequest, mAudioChannel,
|
|
mState == eStateActive);
|
|
NS_DispatchToMainThread(runnable);
|
|
|
|
return domRequest.forget();
|
|
}
|
|
|
|
if (!mFrameWindow) {
|
|
nsCOMPtr<nsIDOMDOMRequest> request;
|
|
aRv = mBrowserElementAPI->IsAudioChannelActive((uint32_t)mAudioChannel,
|
|
getter_AddRefs(request));
|
|
if (NS_WARN_IF(aRv.Failed())) {
|
|
return nullptr;
|
|
}
|
|
|
|
return request.forget().downcast<DOMRequest>();
|
|
}
|
|
|
|
RefPtr<DOMRequest> domRequest = new DOMRequest(GetOwner());
|
|
|
|
nsCOMPtr<nsIRunnable> runnable =
|
|
new IsActiveRunnable(GetOwner(), mFrameWindow, domRequest, mAudioChannel);
|
|
NS_DispatchToMainThread(runnable);
|
|
|
|
return domRequest.forget();
|
|
}
|
|
|
|
already_AddRefed<dom::DOMRequest>
|
|
BrowserElementAudioChannel::NotifyChannel(const nsAString& aEvent,
|
|
ErrorResult& aRv)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
MOZ_ASSERT(XRE_IsParentProcess());
|
|
|
|
if (!mFrameWindow) {
|
|
nsCOMPtr<nsIDOMDOMRequest> request;
|
|
aRv = mBrowserElementAPI->NotifyChannel(aEvent, mManifestURL,
|
|
(uint32_t)mAudioChannel,
|
|
getter_AddRefs(request));
|
|
if (NS_WARN_IF(aRv.Failed())) {
|
|
return nullptr;
|
|
}
|
|
|
|
return request.forget().downcast<DOMRequest>();
|
|
}
|
|
|
|
nsCOMPtr<nsISystemMessagesInternal> systemMessenger =
|
|
do_GetService("@mozilla.org/system-message-internal;1");
|
|
MOZ_ASSERT(systemMessenger);
|
|
|
|
JS::Rooted<JS::Value> value(nsContentUtils::RootingCxForThread());
|
|
value.setInt32((uint32_t)mAudioChannel);
|
|
|
|
nsCOMPtr<nsIURI> manifestURI;
|
|
nsresult rv = NS_NewURI(getter_AddRefs(manifestURI), mManifestURL);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return nullptr;
|
|
}
|
|
|
|
// Since the pageURI of the app has been registered to the system messager,
|
|
// when the app was installed. The system messager can only use the manifest
|
|
// to send the message to correct page.
|
|
nsCOMPtr<nsISupports> promise;
|
|
rv = systemMessenger->SendMessage(aEvent, value, nullptr, manifestURI,
|
|
JS::UndefinedHandleValue,
|
|
getter_AddRefs(promise));
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return nullptr;
|
|
}
|
|
|
|
RefPtr<Promise> promiseIns = static_cast<Promise*>(promise.get());
|
|
RefPtr<DOMRequest> request = new DOMRequest(GetOwner());
|
|
RefPtr<RespondSuccessHandler> handler = new RespondSuccessHandler(request);
|
|
promiseIns->AppendNativeHandler(handler);
|
|
|
|
return request.forget();
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
BrowserElementAudioChannel::Observe(nsISupports* aSubject, const char* aTopic,
|
|
const char16_t* aData)
|
|
{
|
|
nsAutoString name;
|
|
AudioChannelService::GetAudioChannelString(mAudioChannel, name);
|
|
|
|
nsAutoCString topic;
|
|
topic.Assign("audiochannel-activity-");
|
|
topic.Append(NS_ConvertUTF16toUTF8(name));
|
|
|
|
if (strcmp(topic.get(), aTopic)) {
|
|
return NS_OK;
|
|
}
|
|
|
|
// Message received from the child.
|
|
if (!mFrameWindow) {
|
|
if (mTabParent == aSubject) {
|
|
ProcessStateChanged(aData);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<nsISupportsPRUint64> wrapper = do_QueryInterface(aSubject);
|
|
if (!wrapper) {
|
|
bool isNested = false;
|
|
nsresult rv = IsFromNestedFrame(aSubject, isNested);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
if (isNested) {
|
|
ProcessStateChanged(aData);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
uint64_t windowID;
|
|
nsresult rv = wrapper->GetData(&windowID);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
if (windowID != mFrameWindow->WindowID()) {
|
|
return NS_OK;
|
|
}
|
|
|
|
ProcessStateChanged(aData);
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
BrowserElementAudioChannel::ProcessStateChanged(const char16_t* aData)
|
|
{
|
|
MOZ_LOG(AudioChannelService::GetAudioChannelLog(), LogLevel::Debug,
|
|
("BrowserElementAudioChannel, ProcessStateChanged, this = %p, "
|
|
"type = %d\n", this, mAudioChannel));
|
|
|
|
nsAutoString value(aData);
|
|
mState = value.EqualsASCII("active") ? eStateActive : eStateInactive;
|
|
DispatchTrustedEvent(NS_LITERAL_STRING("activestatechanged"));
|
|
}
|
|
|
|
bool
|
|
BrowserElementAudioChannel::IsSystemAppWindow(nsPIDOMWindow* aWindow) const
|
|
{
|
|
nsCOMPtr<nsIDocument> doc = aWindow->GetExtantDoc();
|
|
if (!doc) {
|
|
return false;
|
|
}
|
|
|
|
uint32_t appId;
|
|
nsCOMPtr<nsIPrincipal> principal = doc->NodePrincipal();
|
|
nsresult rv = principal->GetAppId(&appId);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return false;
|
|
}
|
|
|
|
if (appId == nsIScriptSecurityManager::NO_APP_ID ||
|
|
appId == nsIScriptSecurityManager::UNKNOWN_APP_ID) {
|
|
return false;
|
|
}
|
|
|
|
nsCOMPtr<nsIAppsService> appsService = do_GetService(APPS_SERVICE_CONTRACTID);
|
|
if (NS_WARN_IF(!appsService)) {
|
|
return false;
|
|
}
|
|
|
|
nsAdoptingString systemAppManifest =
|
|
mozilla::Preferences::GetString("b2g.system_manifest_url");
|
|
if (!systemAppManifest) {
|
|
return false;
|
|
}
|
|
|
|
uint32_t systemAppId;
|
|
appsService->GetAppLocalIdByManifestURL(systemAppManifest, &systemAppId);
|
|
|
|
if (systemAppId == appId) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
nsresult
|
|
BrowserElementAudioChannel::IsFromNestedFrame(nsISupports* aSubject,
|
|
bool& aIsNested) const
|
|
{
|
|
aIsNested = false;
|
|
nsCOMPtr<nsITabParent> iTabParent = do_QueryInterface(aSubject);
|
|
if (!iTabParent) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
RefPtr<TabParent> tabParent = TabParent::GetFrom(iTabParent);
|
|
if (!tabParent) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
Element* element = tabParent->GetOwnerElement();
|
|
if (!element) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
// Since the normal OOP processes are opened out from b2g process, the owner
|
|
// of their tabParent are the same - system app window. Therefore, in order
|
|
// to find the case of nested MozFrame, we need to exclude this situation.
|
|
nsCOMPtr<nsPIDOMWindow> window = element->OwnerDoc()->GetWindow();
|
|
if (window == mFrameWindow && !IsSystemAppWindow(window)) {
|
|
aIsNested = true;
|
|
return NS_OK;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
} // dom namespace
|
|
} // mozilla namespace
|