summaryrefslogtreecommitdiffstats
path: root/accessible/windows/msaa/MsaaRootAccessible.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'accessible/windows/msaa/MsaaRootAccessible.cpp')
-rw-r--r--accessible/windows/msaa/MsaaRootAccessible.cpp113
1 files changed, 113 insertions, 0 deletions
diff --git a/accessible/windows/msaa/MsaaRootAccessible.cpp b/accessible/windows/msaa/MsaaRootAccessible.cpp
new file mode 100644
index 0000000000..0698f892ef
--- /dev/null
+++ b/accessible/windows/msaa/MsaaRootAccessible.cpp
@@ -0,0 +1,113 @@
+/* -*- 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/. */
+
+#include "mozilla/a11y/DocAccessibleParent.h"
+#include "mozilla/dom/BrowserParent.h"
+#include "mozilla/WindowsVersion.h"
+#include "MsaaRootAccessible.h"
+#include "Relation.h"
+#include "RootAccessible.h"
+
+using namespace mozilla;
+using namespace mozilla::a11y;
+
+RootAccessible* MsaaRootAccessible::RootAcc() {
+ return static_cast<RootAccessible*>(LocalAcc());
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// Aggregated IUnknown
+HRESULT
+MsaaRootAccessible::InternalQueryInterface(REFIID aIid, void** aOutInterface) {
+ if (!aOutInterface) {
+ return E_INVALIDARG;
+ }
+
+ // InternalQueryInterface should always return its internal unknown
+ // when queried for IID_IUnknown...
+ if (aIid == IID_IUnknown) {
+ RefPtr<IUnknown> punk(&mInternalUnknown);
+ punk.forget(aOutInterface);
+ return S_OK;
+ }
+
+ // ...Otherwise we pass through to the base COM implementation of
+ // QueryInterface which is provided by MsaaDocAccessible.
+ return MsaaDocAccessible::QueryInterface(aIid, aOutInterface);
+}
+
+ULONG
+MsaaRootAccessible::InternalAddRef() { return MsaaDocAccessible::AddRef(); }
+
+ULONG
+MsaaRootAccessible::InternalRelease() { return MsaaDocAccessible::Release(); }
+
+already_AddRefed<IUnknown> MsaaRootAccessible::Aggregate(IUnknown* aOuter) {
+ MOZ_ASSERT(mOuter &&
+ (mOuter == &mInternalUnknown || mOuter == aOuter || !aOuter));
+ if (!aOuter) {
+ // If there is no aOuter then we should always set mOuter to
+ // mInternalUnknown. This is standard COM aggregation stuff.
+ mOuter = &mInternalUnknown;
+ return nullptr;
+ }
+
+ mOuter = aOuter;
+ return GetInternalUnknown();
+}
+
+already_AddRefed<IUnknown> MsaaRootAccessible::GetInternalUnknown() {
+ RefPtr<IUnknown> result(&mInternalUnknown);
+ return result.forget();
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// MsaaRootAccessible
+STDMETHODIMP
+MsaaRootAccessible::accNavigate(
+ /* [in] */ long navDir,
+ /* [optional][in] */ VARIANT varStart,
+ /* [retval][out] */ VARIANT __RPC_FAR* pvarEndUpAt) {
+ // Special handling for NAVRELATION_EMBEDS.
+ // When we only have a single process, this can be handled the same way as
+ // any other relation.
+ // However, for multi process, the normal relation mechanism doesn't work
+ // because it can't handle remote objects.
+ if (navDir != NAVRELATION_EMBEDS || varStart.vt != VT_I4 ||
+ varStart.lVal != CHILDID_SELF) {
+ // We only handle EMBEDS on the root here.
+ // Forward to the base implementation.
+ return MsaaDocAccessible::accNavigate(navDir, varStart, pvarEndUpAt);
+ }
+
+ if (!pvarEndUpAt) {
+ return E_INVALIDARG;
+ }
+ RootAccessible* rootAcc = RootAcc();
+ if (!rootAcc) {
+ return CO_E_OBJNOTCONNECTED;
+ }
+
+ Accessible* target = nullptr;
+ // Get the document in the active tab.
+ RemoteAccessible* docProxy = rootAcc->GetPrimaryRemoteTopLevelContentDoc();
+ if (docProxy) {
+ target = docProxy;
+ } else {
+ // The base implementation could handle this, but we may as well
+ // just handle it here.
+ Relation rel = rootAcc->RelationByType(RelationType::EMBEDS);
+ target = rel.Next();
+ }
+
+ if (!target) {
+ return E_FAIL;
+ }
+
+ VariantInit(pvarEndUpAt);
+ pvarEndUpAt->pdispVal = NativeAccessible(target);
+ pvarEndUpAt->vt = VT_DISPATCH;
+ return S_OK;
+}