forked from mirrors/gecko-dev
This implements Jamie's suggested fixes for a screenreader issue when the skeleton UI is enabled. Most of the work here is just pulling out pieces from the files we needed to include in mozglue so that any references to, say, nsString or other pieces from libxul either no longer exist or are only included when building libxul. In a few cases this meant creating whole files to house single functions, which isn't so pretty, but it was the best I could come up with to get the job done. Differential Revision: https://phabricator.services.mozilla.com/D117663
223 lines
6.1 KiB
C++
223 lines
6.1 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
/* 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 "mozilla/mscom/ActivationContext.h"
|
|
|
|
#include "mozilla/Assertions.h"
|
|
#include "mozilla/DebugOnly.h"
|
|
#include "mozilla/mscom/Utils.h"
|
|
|
|
namespace mozilla {
|
|
namespace mscom {
|
|
|
|
ActivationContext::ActivationContext(ActCtxResource aResource)
|
|
: ActivationContext(aResource.mModule, aResource.mId) {}
|
|
|
|
ActivationContext::ActivationContext(HMODULE aLoadFromModule, WORD aResourceId)
|
|
: mActCtx(INVALID_HANDLE_VALUE) {
|
|
ACTCTXW actCtx = {sizeof(actCtx)};
|
|
actCtx.dwFlags = ACTCTX_FLAG_RESOURCE_NAME_VALID | ACTCTX_FLAG_HMODULE_VALID;
|
|
actCtx.lpResourceName = MAKEINTRESOURCEW(aResourceId);
|
|
actCtx.hModule = aLoadFromModule;
|
|
|
|
Init(actCtx);
|
|
}
|
|
|
|
void ActivationContext::Init(ACTCTXW& aActCtx) {
|
|
MOZ_ASSERT(mActCtx == INVALID_HANDLE_VALUE);
|
|
mActCtx = ::CreateActCtxW(&aActCtx);
|
|
MOZ_ASSERT(mActCtx != INVALID_HANDLE_VALUE);
|
|
}
|
|
|
|
void ActivationContext::AddRef() {
|
|
if (mActCtx == INVALID_HANDLE_VALUE) {
|
|
return;
|
|
}
|
|
::AddRefActCtx(mActCtx);
|
|
}
|
|
|
|
ActivationContext::ActivationContext(ActivationContext&& aOther)
|
|
: mActCtx(aOther.mActCtx) {
|
|
aOther.mActCtx = INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
ActivationContext& ActivationContext::operator=(ActivationContext&& aOther) {
|
|
Release();
|
|
|
|
mActCtx = aOther.mActCtx;
|
|
aOther.mActCtx = INVALID_HANDLE_VALUE;
|
|
return *this;
|
|
}
|
|
|
|
ActivationContext::ActivationContext(const ActivationContext& aOther)
|
|
: mActCtx(aOther.mActCtx) {
|
|
AddRef();
|
|
}
|
|
|
|
ActivationContext& ActivationContext::operator=(
|
|
const ActivationContext& aOther) {
|
|
Release();
|
|
mActCtx = aOther.mActCtx;
|
|
AddRef();
|
|
return *this;
|
|
}
|
|
|
|
void ActivationContext::Release() {
|
|
if (mActCtx == INVALID_HANDLE_VALUE) {
|
|
return;
|
|
}
|
|
::ReleaseActCtx(mActCtx);
|
|
mActCtx = INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
ActivationContext::~ActivationContext() { Release(); }
|
|
|
|
#if defined(MOZILLA_INTERNAL_API)
|
|
|
|
/* static */ Result<uintptr_t, HRESULT> ActivationContext::GetCurrent() {
|
|
HANDLE actCtx;
|
|
if (!::GetCurrentActCtx(&actCtx)) {
|
|
return Result<uintptr_t, HRESULT>(HRESULT_FROM_WIN32(::GetLastError()));
|
|
}
|
|
|
|
return reinterpret_cast<uintptr_t>(actCtx);
|
|
}
|
|
|
|
/* static */
|
|
HRESULT ActivationContext::GetCurrentManifestPath(nsAString& aOutManifestPath) {
|
|
aOutManifestPath.Truncate();
|
|
|
|
SIZE_T bytesNeeded;
|
|
BOOL ok = ::QueryActCtxW(QUERY_ACTCTX_FLAG_USE_ACTIVE_ACTCTX, nullptr,
|
|
nullptr, ActivationContextDetailedInformation,
|
|
nullptr, 0, &bytesNeeded);
|
|
if (!ok) {
|
|
DWORD err = ::GetLastError();
|
|
if (err != ERROR_INSUFFICIENT_BUFFER) {
|
|
return HRESULT_FROM_WIN32(err);
|
|
}
|
|
}
|
|
|
|
auto ctxBuf = MakeUnique<BYTE[]>(bytesNeeded);
|
|
|
|
ok = ::QueryActCtxW(QUERY_ACTCTX_FLAG_USE_ACTIVE_ACTCTX, nullptr, nullptr,
|
|
ActivationContextDetailedInformation, ctxBuf.get(),
|
|
bytesNeeded, nullptr);
|
|
if (!ok) {
|
|
return HRESULT_FROM_WIN32(::GetLastError());
|
|
}
|
|
|
|
auto ctxInfo =
|
|
reinterpret_cast<ACTIVATION_CONTEXT_DETAILED_INFORMATION*>(ctxBuf.get());
|
|
|
|
// assemblyIndex is 1-based, and we want the last index, so we can just copy
|
|
// ctxInfo->ulAssemblyCount directly.
|
|
DWORD assemblyIndex = ctxInfo->ulAssemblyCount;
|
|
ok = ::QueryActCtxW(
|
|
QUERY_ACTCTX_FLAG_USE_ACTIVE_ACTCTX, nullptr, &assemblyIndex,
|
|
AssemblyDetailedInformationInActivationContext, nullptr, 0, &bytesNeeded);
|
|
if (!ok) {
|
|
DWORD err = ::GetLastError();
|
|
if (err != ERROR_INSUFFICIENT_BUFFER) {
|
|
return HRESULT_FROM_WIN32(err);
|
|
}
|
|
}
|
|
|
|
auto assemblyBuf = MakeUnique<BYTE[]>(bytesNeeded);
|
|
|
|
ok = ::QueryActCtxW(QUERY_ACTCTX_FLAG_USE_ACTIVE_ACTCTX, nullptr,
|
|
&assemblyIndex,
|
|
AssemblyDetailedInformationInActivationContext,
|
|
assemblyBuf.get(), bytesNeeded, &bytesNeeded);
|
|
if (!ok) {
|
|
return HRESULT_FROM_WIN32(::GetLastError());
|
|
}
|
|
|
|
auto assemblyInfo =
|
|
reinterpret_cast<ACTIVATION_CONTEXT_ASSEMBLY_DETAILED_INFORMATION*>(
|
|
assemblyBuf.get());
|
|
aOutManifestPath = nsDependentString(
|
|
assemblyInfo->lpAssemblyManifestPath,
|
|
(assemblyInfo->ulManifestPathLength + 1) / sizeof(wchar_t));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
#endif // defined(MOZILLA_INTERNAL_API)
|
|
|
|
ActivationContextRegion::ActivationContextRegion() : mActCookie(0) {}
|
|
|
|
ActivationContextRegion::ActivationContextRegion(
|
|
const ActivationContext& aActCtx)
|
|
: mActCtx(aActCtx), mActCookie(0) {
|
|
Activate();
|
|
}
|
|
|
|
ActivationContextRegion& ActivationContextRegion::operator=(
|
|
const ActivationContext& aActCtx) {
|
|
Deactivate();
|
|
mActCtx = aActCtx;
|
|
Activate();
|
|
return *this;
|
|
}
|
|
|
|
ActivationContextRegion::ActivationContextRegion(ActivationContext&& aActCtx)
|
|
: mActCtx(std::move(aActCtx)), mActCookie(0) {
|
|
Activate();
|
|
}
|
|
|
|
ActivationContextRegion& ActivationContextRegion::operator=(
|
|
ActivationContext&& aActCtx) {
|
|
Deactivate();
|
|
mActCtx = std::move(aActCtx);
|
|
Activate();
|
|
return *this;
|
|
}
|
|
|
|
ActivationContextRegion::ActivationContextRegion(ActivationContextRegion&& aRgn)
|
|
: mActCtx(std::move(aRgn.mActCtx)), mActCookie(aRgn.mActCookie) {
|
|
aRgn.mActCookie = 0;
|
|
}
|
|
|
|
ActivationContextRegion& ActivationContextRegion::operator=(
|
|
ActivationContextRegion&& aRgn) {
|
|
Deactivate();
|
|
mActCtx = std::move(aRgn.mActCtx);
|
|
mActCookie = aRgn.mActCookie;
|
|
aRgn.mActCookie = 0;
|
|
return *this;
|
|
}
|
|
|
|
void ActivationContextRegion::Activate() {
|
|
if (mActCtx.mActCtx == INVALID_HANDLE_VALUE) {
|
|
return;
|
|
}
|
|
|
|
#ifdef MOZ_DIAGNOSTIC_ASSERT_ENABLED
|
|
BOOL activated =
|
|
#endif
|
|
::ActivateActCtx(mActCtx.mActCtx, &mActCookie);
|
|
MOZ_DIAGNOSTIC_ASSERT(activated);
|
|
}
|
|
|
|
bool ActivationContextRegion::Deactivate() {
|
|
if (!mActCookie) {
|
|
return true;
|
|
}
|
|
|
|
BOOL deactivated = ::DeactivateActCtx(0, mActCookie);
|
|
MOZ_DIAGNOSTIC_ASSERT(deactivated);
|
|
if (deactivated) {
|
|
mActCookie = 0;
|
|
}
|
|
|
|
return !!deactivated;
|
|
}
|
|
|
|
ActivationContextRegion::~ActivationContextRegion() { Deactivate(); }
|
|
|
|
} // namespace mscom
|
|
} // namespace mozilla
|