forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			842 lines
		
	
	
	
		
			25 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			842 lines
		
	
	
	
		
			25 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
 | |
| /* vim: set ts=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 "DocAccessibleParent.h"
 | |
| #include "mozilla/a11y/Platform.h"
 | |
| #include "mozilla/dom/TabParent.h"
 | |
| #include "xpcAccessibleDocument.h"
 | |
| #include "xpcAccEvents.h"
 | |
| #include "nsAccUtils.h"
 | |
| #include "nsCoreUtils.h"
 | |
| 
 | |
| #if defined(XP_WIN)
 | |
| #include "AccessibleWrap.h"
 | |
| #include "Compatibility.h"
 | |
| #include "mozilla/mscom/PassthruProxy.h"
 | |
| #include "mozilla/mscom/Ptr.h"
 | |
| #include "nsWinUtils.h"
 | |
| #include "RootAccessible.h"
 | |
| #endif
 | |
| 
 | |
| namespace mozilla {
 | |
| namespace a11y {
 | |
| uint64_t DocAccessibleParent::sMaxDocID = 0;
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| DocAccessibleParent::RecvShowEvent(const ShowEventData& aData,
 | |
|                                    const bool& aFromUser)
 | |
| {
 | |
|   if (mShutdown)
 | |
|     return IPC_OK();
 | |
| 
 | |
|   MOZ_ASSERT(CheckDocTree());
 | |
| 
 | |
|   if (aData.NewTree().IsEmpty()) {
 | |
|     return IPC_FAIL(this, "No children being added");
 | |
|   }
 | |
| 
 | |
|   ProxyAccessible* parent = GetAccessible(aData.ID());
 | |
| 
 | |
|   // XXX This should really never happen, but sometimes we fail to fire the
 | |
|   // required show events.
 | |
|   if (!parent) {
 | |
|     NS_ERROR("adding child to unknown accessible");
 | |
| #ifdef DEBUG
 | |
|     return IPC_FAIL(this, "unknown parent accessible");
 | |
| #else
 | |
|     return IPC_OK();
 | |
| #endif
 | |
|   }
 | |
| 
 | |
|   uint32_t newChildIdx = aData.Idx();
 | |
|   if (newChildIdx > parent->ChildrenCount()) {
 | |
|     NS_ERROR("invalid index to add child at");
 | |
| #ifdef DEBUG
 | |
|     return IPC_FAIL(this, "invalid index");
 | |
| #else
 | |
|     return IPC_OK();
 | |
| #endif
 | |
|   }
 | |
| 
 | |
|   uint32_t consumed = AddSubtree(parent, aData.NewTree(), 0, newChildIdx);
 | |
|   MOZ_ASSERT(consumed == aData.NewTree().Length());
 | |
| 
 | |
|   // XXX This shouldn't happen, but if we failed to add children then the below
 | |
|   // is pointless and can crash.
 | |
|   if (!consumed) {
 | |
|     return IPC_FAIL(this, "failed to add children");
 | |
|   }
 | |
| 
 | |
| #ifdef DEBUG
 | |
|   for (uint32_t i = 0; i < consumed; i++) {
 | |
|     uint64_t id = aData.NewTree()[i].ID();
 | |
|     MOZ_ASSERT(mAccessibles.GetEntry(id));
 | |
|   }
 | |
| #endif
 | |
| 
 | |
|   MOZ_ASSERT(CheckDocTree());
 | |
| 
 | |
|   // Just update, no events.
 | |
|   if (aData.EventSuppressed()) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   ProxyAccessible* target = parent->ChildAt(newChildIdx);
 | |
|   ProxyShowHideEvent(target, parent, true, aFromUser);
 | |
| 
 | |
|   if (!nsCoreUtils::AccEventObserversExist()) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   uint32_t type = nsIAccessibleEvent::EVENT_SHOW;
 | |
|   xpcAccessibleGeneric* xpcAcc = GetXPCAccessible(target);
 | |
|   xpcAccessibleDocument* doc = GetAccService()->GetXPCDocument(this);
 | |
|   nsINode* node = nullptr;
 | |
|   RefPtr<xpcAccEvent> event = new xpcAccEvent(type, xpcAcc, doc, node,
 | |
|                                               aFromUser);
 | |
|   nsCoreUtils::DispatchAccEvent(std::move(event));
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| uint32_t
 | |
| DocAccessibleParent::AddSubtree(ProxyAccessible* aParent,
 | |
|                                 const nsTArray<a11y::AccessibleData>& aNewTree,
 | |
|                                 uint32_t aIdx, uint32_t aIdxInParent)
 | |
| {
 | |
|   if (aNewTree.Length() <= aIdx) {
 | |
|     NS_ERROR("bad index in serialized tree!");
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   const AccessibleData& newChild = aNewTree[aIdx];
 | |
| 
 | |
|   if (mAccessibles.Contains(newChild.ID())) {
 | |
|     NS_ERROR("ID already in use");
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   ProxyAccessible* newProxy = new ProxyAccessible(
 | |
|     newChild.ID(), aParent, this, newChild.Role(), newChild.Interfaces());
 | |
| 
 | |
|   aParent->AddChildAt(aIdxInParent, newProxy);
 | |
|   mAccessibles.PutEntry(newChild.ID())->mProxy = newProxy;
 | |
|   ProxyCreated(newProxy, newChild.Interfaces());
 | |
| 
 | |
| #if defined(XP_WIN)
 | |
|   WrapperFor(newProxy)->SetID(newChild.MsaaID());
 | |
| #endif
 | |
| 
 | |
|   uint32_t accessibles = 1;
 | |
|   uint32_t kids = newChild.ChildrenCount();
 | |
|   for (uint32_t i = 0; i < kids; i++) {
 | |
|     uint32_t consumed = AddSubtree(newProxy, aNewTree, aIdx + accessibles, i);
 | |
|     if (!consumed)
 | |
|       return 0;
 | |
| 
 | |
|     accessibles += consumed;
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(newProxy->ChildrenCount() == kids);
 | |
| 
 | |
|   return accessibles;
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| DocAccessibleParent::RecvHideEvent(const uint64_t& aRootID,
 | |
|                                    const bool& aFromUser)
 | |
| {
 | |
|   if (mShutdown)
 | |
|     return IPC_OK();
 | |
| 
 | |
|   MOZ_ASSERT(CheckDocTree());
 | |
| 
 | |
|   // We shouldn't actually need this because mAccessibles shouldn't have an
 | |
|   // entry for the document itself, but it doesn't hurt to be explicit.
 | |
|   if (!aRootID) {
 | |
|     return IPC_FAIL(this, "Trying to hide entire document?");
 | |
|   }
 | |
| 
 | |
|   ProxyEntry* rootEntry = mAccessibles.GetEntry(aRootID);
 | |
|   if (!rootEntry) {
 | |
|     NS_ERROR("invalid root being removed!");
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   ProxyAccessible* root = rootEntry->mProxy;
 | |
|   if (!root) {
 | |
|     NS_ERROR("invalid root being removed!");
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   ProxyAccessible* parent = root->Parent();
 | |
|   ProxyShowHideEvent(root, parent, false, aFromUser);
 | |
| 
 | |
|   RefPtr<xpcAccHideEvent> event = nullptr;
 | |
|   if (nsCoreUtils::AccEventObserversExist()) {
 | |
|     uint32_t type = nsIAccessibleEvent::EVENT_HIDE;
 | |
|     xpcAccessibleGeneric* xpcAcc = GetXPCAccessible(root);
 | |
|     xpcAccessibleGeneric* xpcParent = GetXPCAccessible(parent);
 | |
|     ProxyAccessible* next = root->NextSibling();
 | |
|     xpcAccessibleGeneric* xpcNext = next ? GetXPCAccessible(next) : nullptr;
 | |
|     ProxyAccessible* prev = root->PrevSibling();
 | |
|     xpcAccessibleGeneric* xpcPrev = prev ? GetXPCAccessible(prev) : nullptr;
 | |
|     xpcAccessibleDocument* doc = GetAccService()->GetXPCDocument(this);
 | |
|     nsINode* node = nullptr;
 | |
|     event = new xpcAccHideEvent(type, xpcAcc, doc, node, aFromUser, xpcParent,
 | |
|                                 xpcNext, xpcPrev);
 | |
|   }
 | |
| 
 | |
|   parent->RemoveChild(root);
 | |
|   root->Shutdown();
 | |
| 
 | |
|   MOZ_ASSERT(CheckDocTree());
 | |
| 
 | |
|   if (event) {
 | |
|     nsCoreUtils::DispatchAccEvent(std::move(event));
 | |
|   }
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| DocAccessibleParent::RecvEvent(const uint64_t& aID, const uint32_t& aEventType)
 | |
| {
 | |
|   if (mShutdown) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   ProxyAccessible* proxy = GetAccessible(aID);
 | |
|   if (!proxy) {
 | |
|     NS_ERROR("no proxy for event!");
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   ProxyEvent(proxy, aEventType);
 | |
| 
 | |
|   if (!nsCoreUtils::AccEventObserversExist()) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   xpcAccessibleGeneric* xpcAcc = GetXPCAccessible(proxy);
 | |
|   xpcAccessibleDocument* doc = GetAccService()->GetXPCDocument(this);
 | |
|   nsINode* node = nullptr;
 | |
|   bool fromUser = true; // XXX fix me
 | |
|   RefPtr<xpcAccEvent> event = new xpcAccEvent(aEventType, xpcAcc, doc, node,
 | |
|                                               fromUser);
 | |
|   nsCoreUtils::DispatchAccEvent(std::move(event));
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| DocAccessibleParent::RecvStateChangeEvent(const uint64_t& aID,
 | |
|                                           const uint64_t& aState,
 | |
|                                           const bool& aEnabled)
 | |
| {
 | |
|   if (mShutdown) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   ProxyAccessible* target = GetAccessible(aID);
 | |
|   if (!target) {
 | |
|     NS_ERROR("we don't know about the target of a state change event!");
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   ProxyStateChangeEvent(target, aState, aEnabled);
 | |
| 
 | |
|   if (!nsCoreUtils::AccEventObserversExist()) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   xpcAccessibleGeneric* xpcAcc = GetXPCAccessible(target);
 | |
|   xpcAccessibleDocument* doc = GetAccService()->GetXPCDocument(this);
 | |
|   uint32_t type = nsIAccessibleEvent::EVENT_STATE_CHANGE;
 | |
|   bool extra;
 | |
|   uint32_t state = nsAccUtils::To32States(aState, &extra);
 | |
|   bool fromUser = true; // XXX fix this
 | |
|   nsINode* node = nullptr; // XXX can we do better?
 | |
|   RefPtr<xpcAccStateChangeEvent> event =
 | |
|     new xpcAccStateChangeEvent(type, xpcAcc, doc, node, fromUser, state, extra,
 | |
|                                aEnabled);
 | |
|   nsCoreUtils::DispatchAccEvent(std::move(event));
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| DocAccessibleParent::RecvCaretMoveEvent(const uint64_t& aID,
 | |
| #if defined(XP_WIN)
 | |
|                                         const LayoutDeviceIntRect& aCaretRect,
 | |
| #endif // defined (XP_WIN)
 | |
|                                         const int32_t& aOffset)
 | |
| {
 | |
|   if (mShutdown) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   ProxyAccessible* proxy = GetAccessible(aID);
 | |
|   if (!proxy) {
 | |
|     NS_ERROR("unknown caret move event target!");
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
| #if defined(XP_WIN)
 | |
|   ProxyCaretMoveEvent(proxy, aCaretRect);
 | |
| #else
 | |
|   ProxyCaretMoveEvent(proxy, aOffset);
 | |
| #endif
 | |
| 
 | |
|   if (!nsCoreUtils::AccEventObserversExist()) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   xpcAccessibleGeneric* xpcAcc = GetXPCAccessible(proxy);
 | |
|   xpcAccessibleDocument* doc = GetAccService()->GetXPCDocument(this);
 | |
|   nsINode* node = nullptr;
 | |
|   bool fromUser = true; // XXX fix me
 | |
|   uint32_t type = nsIAccessibleEvent::EVENT_TEXT_CARET_MOVED;
 | |
|   RefPtr<xpcAccCaretMoveEvent> event =
 | |
|     new xpcAccCaretMoveEvent(type, xpcAcc, doc, node, fromUser, aOffset);
 | |
|   nsCoreUtils::DispatchAccEvent(std::move(event));
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| DocAccessibleParent::RecvTextChangeEvent(const uint64_t& aID,
 | |
|                                          const nsString& aStr,
 | |
|                                          const int32_t& aStart,
 | |
|                                          const uint32_t& aLen,
 | |
|                                          const bool& aIsInsert,
 | |
|                                          const bool& aFromUser)
 | |
| {
 | |
|   if (mShutdown) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   ProxyAccessible* target = GetAccessible(aID);
 | |
|   if (!target) {
 | |
|     NS_ERROR("text change event target is unknown!");
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   ProxyTextChangeEvent(target, aStr, aStart, aLen, aIsInsert, aFromUser);
 | |
| 
 | |
|   if (!nsCoreUtils::AccEventObserversExist()) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   xpcAccessibleGeneric* xpcAcc = GetXPCAccessible(target);
 | |
|   xpcAccessibleDocument* doc = GetAccService()->GetXPCDocument(this);
 | |
|   uint32_t type = aIsInsert ? nsIAccessibleEvent::EVENT_TEXT_INSERTED :
 | |
|                               nsIAccessibleEvent::EVENT_TEXT_REMOVED;
 | |
|   nsINode* node = nullptr;
 | |
|   RefPtr<xpcAccTextChangeEvent> event =
 | |
|     new xpcAccTextChangeEvent(type, xpcAcc, doc, node, aFromUser, aStart, aLen,
 | |
|                               aIsInsert, aStr);
 | |
|   nsCoreUtils::DispatchAccEvent(std::move(event));
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| #if defined(XP_WIN)
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| DocAccessibleParent::RecvSyncTextChangeEvent(const uint64_t& aID,
 | |
|                                              const nsString& aStr,
 | |
|                                              const int32_t& aStart,
 | |
|                                              const uint32_t& aLen,
 | |
|                                              const bool& aIsInsert,
 | |
|                                              const bool& aFromUser)
 | |
| {
 | |
|   return RecvTextChangeEvent(aID, aStr, aStart, aLen, aIsInsert, aFromUser);
 | |
| }
 | |
| 
 | |
| #endif // defined(XP_WIN)
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| DocAccessibleParent::RecvSelectionEvent(const uint64_t& aID,
 | |
|                                         const uint64_t& aWidgetID,
 | |
|                                         const uint32_t& aType)
 | |
| {
 | |
|   if (mShutdown) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   ProxyAccessible* target = GetAccessible(aID);
 | |
|   ProxyAccessible* widget = GetAccessible(aWidgetID);
 | |
|   if (!target || !widget) {
 | |
|     NS_ERROR("invalid id in selection event");
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   ProxySelectionEvent(target, widget, aType);
 | |
|   if (!nsCoreUtils::AccEventObserversExist()) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
|   xpcAccessibleGeneric* xpcTarget = GetXPCAccessible(target);
 | |
|   xpcAccessibleDocument* xpcDoc = GetAccService()->GetXPCDocument(this);
 | |
|   RefPtr<xpcAccEvent> event = new xpcAccEvent(aType, xpcTarget, xpcDoc,
 | |
|                                               nullptr, false);
 | |
|   nsCoreUtils::DispatchAccEvent(std::move(event));
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| DocAccessibleParent::RecvVirtualCursorChangeEvent(const uint64_t& aID,
 | |
|                                                   const uint64_t& aOldPositionID,
 | |
|                                                   const int32_t& aOldStartOffset,
 | |
|                                                   const int32_t& aOldEndOffset,
 | |
|                                                   const uint64_t& aNewPositionID,
 | |
|                                                   const int32_t& aNewStartOffset,
 | |
|                                                   const int32_t& aNewEndOffset,
 | |
|                                                   const int16_t& aReason,
 | |
|                                                   const int16_t& aBoundaryType,
 | |
|                                                   const bool& aFromUser)
 | |
| {
 | |
|   ProxyAccessible* target = GetAccessible(aID);
 | |
|   ProxyAccessible* oldPosition = GetAccessible(aOldPositionID);
 | |
|   ProxyAccessible* newPosition = GetAccessible(aNewPositionID);
 | |
| 
 | |
|   if (!target) {
 | |
|     NS_ERROR("no proxy for event!");
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
| #if defined(ANDROID)
 | |
|   ProxyVirtualCursorChangeEvent(target,
 | |
|                                 oldPosition, aOldStartOffset, aOldEndOffset,
 | |
|                                 newPosition, aNewStartOffset, aNewEndOffset,
 | |
|                                 aReason, aBoundaryType, aFromUser);
 | |
| #endif
 | |
| 
 | |
|   if (!nsCoreUtils::AccEventObserversExist()) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   xpcAccessibleDocument* doc = GetAccService()->GetXPCDocument(this);
 | |
|   RefPtr<xpcAccVirtualCursorChangeEvent> event =
 | |
|     new xpcAccVirtualCursorChangeEvent(nsIAccessibleEvent::EVENT_VIRTUALCURSOR_CHANGED,
 | |
|                                        GetXPCAccessible(target), doc,
 | |
|                                        nullptr, aFromUser,
 | |
|                                        GetXPCAccessible(oldPosition),
 | |
|                                        aOldStartOffset, aOldEndOffset,
 | |
|                                        GetXPCAccessible(newPosition),
 | |
|                                        aNewStartOffset, aNewEndOffset,
 | |
|                                        aBoundaryType, aReason);
 | |
|   nsCoreUtils::DispatchAccEvent(std::move(event));
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| DocAccessibleParent::RecvScrollingEvent(const uint64_t& aID,
 | |
|                                         const uint64_t& aType,
 | |
|                                         const uint32_t& aScrollX,
 | |
|                                         const uint32_t& aScrollY,
 | |
|                                         const uint32_t& aMaxScrollX,
 | |
|                                         const uint32_t& aMaxScrollY)
 | |
| {
 | |
|   ProxyAccessible* target = GetAccessible(aID);
 | |
| 
 | |
|   if (!target) {
 | |
|     NS_ERROR("no proxy for event!");
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
| #if defined(ANDROID)
 | |
|   ProxyScrollingEvent(target, aScrollX, aScrollY, aMaxScrollX, aMaxScrollY);
 | |
| #endif
 | |
| 
 | |
|   if (!nsCoreUtils::AccEventObserversExist()) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   xpcAccessibleGeneric* xpcAcc = GetXPCAccessible(target);
 | |
|   xpcAccessibleDocument* doc = GetAccService()->GetXPCDocument(this);
 | |
|   nsINode* node = nullptr;
 | |
|   bool fromUser = true; // XXX: Determine if this was from user input.
 | |
|   RefPtr<xpcAccScrollingEvent> event =
 | |
|     new xpcAccScrollingEvent(aType, xpcAcc, doc, node, fromUser, aScrollX,
 | |
|                              aScrollY, aMaxScrollX, aMaxScrollY);
 | |
|   nsCoreUtils::DispatchAccEvent(std::move(event));
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| DocAccessibleParent::RecvRoleChangedEvent(const a11y::role& aRole)
 | |
| {
 | |
|   if (mShutdown) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   mRole = aRole;
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| DocAccessibleParent::RecvBindChildDoc(PDocAccessibleParent* aChildDoc, const uint64_t& aID)
 | |
| {
 | |
|   // One document should never directly be the child of another.
 | |
|   // We should always have at least an outer doc accessible in between.
 | |
|   MOZ_ASSERT(aID);
 | |
|   if (!aID)
 | |
|     return IPC_FAIL(this, "ID is 0!");
 | |
| 
 | |
|   if (mShutdown) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(CheckDocTree());
 | |
| 
 | |
|   auto childDoc = static_cast<DocAccessibleParent*>(aChildDoc);
 | |
|   childDoc->Unbind();
 | |
|   ipc::IPCResult result = AddChildDoc(childDoc, aID, false);
 | |
|   MOZ_ASSERT(result);
 | |
|   MOZ_ASSERT(CheckDocTree());
 | |
| #ifdef DEBUG
 | |
|   if (!result) {
 | |
|     return result;
 | |
|   }
 | |
| #else
 | |
|   result = IPC_OK();
 | |
| #endif
 | |
| 
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| ipc::IPCResult
 | |
| DocAccessibleParent::AddChildDoc(DocAccessibleParent* aChildDoc,
 | |
|                                  uint64_t aParentID, bool aCreating)
 | |
| {
 | |
|   // We do not use GetAccessible here because we want to be sure to not get the
 | |
|   // document it self.
 | |
|   ProxyEntry* e = mAccessibles.GetEntry(aParentID);
 | |
|   if (!e) {
 | |
|     return IPC_FAIL(this, "binding to nonexistant proxy!");
 | |
|   }
 | |
| 
 | |
|   ProxyAccessible* outerDoc = e->mProxy;
 | |
|   MOZ_ASSERT(outerDoc);
 | |
| 
 | |
|   // OuterDocAccessibles are expected to only have a document as a child.
 | |
|   // However for compatibility we tolerate replacing one document with another
 | |
|   // here.
 | |
|   if (outerDoc->ChildrenCount() > 1 ||
 | |
|       (outerDoc->ChildrenCount() == 1 && !outerDoc->ChildAt(0)->IsDoc())) {
 | |
|     return IPC_FAIL(this, "binding to proxy that can't be a outerDoc!");
 | |
|   }
 | |
| 
 | |
|   if (outerDoc->ChildrenCount() == 1) {
 | |
|     MOZ_ASSERT(outerDoc->ChildAt(0)->AsDoc());
 | |
|     outerDoc->ChildAt(0)->AsDoc()->Unbind();
 | |
|   }
 | |
| 
 | |
|   aChildDoc->SetParent(outerDoc);
 | |
|   outerDoc->SetChildDoc(aChildDoc);
 | |
|   mChildDocs.AppendElement(aChildDoc->mActorID);
 | |
|   aChildDoc->mParentDoc = mActorID;
 | |
| 
 | |
|   if (aCreating) {
 | |
|     ProxyCreated(aChildDoc, Interfaces::DOCUMENT | Interfaces::HYPERTEXT);
 | |
|   }
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| DocAccessibleParent::RecvShutdown()
 | |
| {
 | |
|   Destroy();
 | |
| 
 | |
|   auto mgr = static_cast<dom::TabParent*>(Manager());
 | |
|   if (!mgr->IsDestroyed()) {
 | |
|     if (!PDocAccessibleParent::Send__delete__(this)) {
 | |
|       return IPC_FAIL_NO_REASON(mgr);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| void
 | |
| DocAccessibleParent::Destroy()
 | |
| {
 | |
|   // If we are already shutdown that is because our containing tab parent is
 | |
|   // shutting down in which case we don't need to do anything.
 | |
|   if (mShutdown) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   mShutdown = true;
 | |
| 
 | |
|   MOZ_DIAGNOSTIC_ASSERT(LiveDocs().Contains(mActorID));
 | |
|   uint32_t childDocCount = mChildDocs.Length();
 | |
|   for (uint32_t i = 0; i < childDocCount; i++) {
 | |
|     for (uint32_t j = i + 1; j < childDocCount; j++) {
 | |
|       MOZ_DIAGNOSTIC_ASSERT(mChildDocs[i] != mChildDocs[j]);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // XXX This indirection through the hash map of live documents shouldn't be
 | |
|   // needed, but be paranoid for now.
 | |
|   int32_t actorID = mActorID;
 | |
|   for (uint32_t i = childDocCount - 1; i < childDocCount; i--) {
 | |
|     DocAccessibleParent* thisDoc = LiveDocs().Get(actorID);
 | |
|     MOZ_ASSERT(thisDoc);
 | |
|     if (!thisDoc) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     thisDoc->ChildDocAt(i)->Destroy();
 | |
|   }
 | |
| 
 | |
|   for (auto iter = mAccessibles.Iter(); !iter.Done(); iter.Next()) {
 | |
|     MOZ_ASSERT(iter.Get()->mProxy != this);
 | |
|     ProxyDestroyed(iter.Get()->mProxy);
 | |
|     iter.Remove();
 | |
|   }
 | |
| 
 | |
|   DocAccessibleParent* thisDoc = LiveDocs().Get(actorID);
 | |
|   MOZ_ASSERT(thisDoc);
 | |
|   if (!thisDoc) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // The code above should have already completely cleared these, but to be
 | |
|   // extra safe make sure they are cleared here.
 | |
|   thisDoc->mAccessibles.Clear();
 | |
|   thisDoc->mChildDocs.Clear();
 | |
| 
 | |
|   DocManager::NotifyOfRemoteDocShutdown(thisDoc);
 | |
|   thisDoc = LiveDocs().Get(actorID);
 | |
|   MOZ_ASSERT(thisDoc);
 | |
|   if (!thisDoc) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   ProxyDestroyed(thisDoc);
 | |
|   thisDoc = LiveDocs().Get(actorID);
 | |
|   MOZ_ASSERT(thisDoc);
 | |
|   if (!thisDoc) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (DocAccessibleParent* parentDoc = thisDoc->ParentDoc())
 | |
|     parentDoc->RemoveChildDoc(thisDoc);
 | |
|   else if (IsTopLevel())
 | |
|     GetAccService()->RemoteDocShutdown(this);
 | |
| }
 | |
| 
 | |
| DocAccessibleParent*
 | |
| DocAccessibleParent::ParentDoc() const
 | |
| {
 | |
|   if (mParentDoc == kNoParentDoc) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   return LiveDocs().Get(mParentDoc);
 | |
| }
 | |
| 
 | |
| bool
 | |
| DocAccessibleParent::CheckDocTree() const
 | |
| {
 | |
|   size_t childDocs = mChildDocs.Length();
 | |
|   for (size_t i = 0; i < childDocs; i++) {
 | |
|     const DocAccessibleParent* childDoc = ChildDocAt(i);
 | |
|     if (!childDoc || childDoc->ParentDoc() != this)
 | |
|       return false;
 | |
| 
 | |
|     if (!childDoc->CheckDocTree()) {
 | |
|       return false;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| xpcAccessibleGeneric*
 | |
| DocAccessibleParent::GetXPCAccessible(ProxyAccessible* aProxy)
 | |
| {
 | |
|   xpcAccessibleDocument* doc = GetAccService()->GetXPCDocument(this);
 | |
|   MOZ_ASSERT(doc);
 | |
| 
 | |
|   return doc->GetXPCAccessible(aProxy);
 | |
| }
 | |
| 
 | |
| #if defined(XP_WIN)
 | |
| void
 | |
| DocAccessibleParent::MaybeInitWindowEmulation()
 | |
| {
 | |
|   if (!nsWinUtils::IsWindowEmulationStarted()) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // XXX get the bounds from the tabParent instead of poking at accessibles
 | |
|   // which might not exist yet.
 | |
|   Accessible* outerDoc = OuterDocOfRemoteBrowser();
 | |
|   if (!outerDoc) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   RootAccessible* rootDocument = outerDoc->RootAccessible();
 | |
|   MOZ_ASSERT(rootDocument);
 | |
| 
 | |
|   bool isActive = true;
 | |
|   nsIntRect rect(CW_USEDEFAULT, CW_USEDEFAULT, 0, 0);
 | |
|   if (Compatibility::IsDolphin()) {
 | |
|     rect = Bounds();
 | |
|     nsIntRect rootRect = rootDocument->Bounds();
 | |
|     rect.MoveToX(rootRect.X() - rect.X());
 | |
|     rect.MoveToY(rect.Y() - rootRect.Y());
 | |
| 
 | |
|     auto tab = static_cast<dom::TabParent*>(Manager());
 | |
|     tab->GetDocShellIsActive(&isActive);
 | |
|   }
 | |
| 
 | |
|   nsWinUtils::NativeWindowCreateProc onCreate([this](HWND aHwnd) -> void {
 | |
|     IDispatchHolder hWndAccHolder;
 | |
| 
 | |
|     ::SetPropW(aHwnd, kPropNameDocAccParent, reinterpret_cast<HANDLE>(this));
 | |
| 
 | |
|     SetEmulatedWindowHandle(aHwnd);
 | |
| 
 | |
|     RefPtr<IAccessible> hwndAcc;
 | |
|     if (SUCCEEDED(::AccessibleObjectFromWindow(aHwnd, OBJID_WINDOW,
 | |
|                                                IID_IAccessible,
 | |
|                                                getter_AddRefs(hwndAcc)))) {
 | |
|       RefPtr<IDispatch> wrapped(mscom::PassthruProxy::Wrap<IDispatch>(WrapNotNull(hwndAcc)));
 | |
|       hWndAccHolder.Set(IDispatchHolder::COMPtrType(mscom::ToProxyUniquePtr(std::move(wrapped))));
 | |
|     }
 | |
| 
 | |
|     Unused << SendEmulatedWindow(reinterpret_cast<uintptr_t>(mEmulatedWindowHandle),
 | |
|                                  hWndAccHolder);
 | |
|   });
 | |
| 
 | |
|   HWND parentWnd = reinterpret_cast<HWND>(rootDocument->GetNativeWindow());
 | |
|   DebugOnly<HWND> hWnd = nsWinUtils::CreateNativeWindow(kClassNameTabContent,
 | |
|                                                         parentWnd,
 | |
|                                                         rect.X(), rect.Y(),
 | |
|                                                         rect.Width(), rect.Height(),
 | |
|                                                         isActive, &onCreate);
 | |
|   MOZ_ASSERT(hWnd);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @param aCOMProxy COM Proxy to the document in the content process.
 | |
|  */
 | |
| void
 | |
| DocAccessibleParent::SendParentCOMProxy()
 | |
| {
 | |
|   // Make sure that we're not racing with a tab shutdown
 | |
|   auto tab = static_cast<dom::TabParent*>(Manager());
 | |
|   MOZ_ASSERT(tab);
 | |
|   if (tab->IsDestroyed()) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Accessible* outerDoc = OuterDocOfRemoteBrowser();
 | |
|   if (!outerDoc) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   RefPtr<IAccessible> nativeAcc;
 | |
|   outerDoc->GetNativeInterface(getter_AddRefs(nativeAcc));
 | |
|   MOZ_ASSERT(nativeAcc);
 | |
| 
 | |
|   RefPtr<IDispatch> wrapped(mscom::PassthruProxy::Wrap<IDispatch>(WrapNotNull(nativeAcc)));
 | |
| 
 | |
|   IDispatchHolder::COMPtrType ptr(mscom::ToProxyUniquePtr(std::move(wrapped)));
 | |
|   IDispatchHolder holder(std::move(ptr));
 | |
|   if (!PDocAccessibleParent::SendParentCOMProxy(holder)) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
| #if defined(MOZ_CONTENT_SANDBOX)
 | |
|   mParentProxyStream = std::move(holder.GetPreservedStream());
 | |
| #endif // defined(MOZ_CONTENT_SANDBOX)
 | |
| }
 | |
| 
 | |
| void
 | |
| DocAccessibleParent::SetEmulatedWindowHandle(HWND aWindowHandle)
 | |
| {
 | |
|   if (!aWindowHandle && mEmulatedWindowHandle && IsTopLevel()) {
 | |
|     ::DestroyWindow(mEmulatedWindowHandle);
 | |
|   }
 | |
|   mEmulatedWindowHandle = aWindowHandle;
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| DocAccessibleParent::RecvGetWindowedPluginIAccessible(
 | |
|       const WindowsHandle& aHwnd, IAccessibleHolder* aPluginCOMProxy)
 | |
| {
 | |
| #if defined(MOZ_CONTENT_SANDBOX)
 | |
|   // We don't actually want the accessible object for aHwnd, but rather the
 | |
|   // one that belongs to its child (see HTMLWin32ObjectAccessible).
 | |
|   HWND childWnd = ::GetWindow(reinterpret_cast<HWND>(aHwnd), GW_CHILD);
 | |
|   if (!childWnd) {
 | |
|     // We're seeing this in the wild - the plugin is windowed but we no longer
 | |
|     // have a window.
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   IAccessible* rawAccPlugin = nullptr;
 | |
|   HRESULT hr = ::AccessibleObjectFromWindow(childWnd, OBJID_WINDOW,
 | |
|                                             IID_IAccessible,
 | |
|                                             (void**)&rawAccPlugin);
 | |
|   if (FAILED(hr)) {
 | |
|     // This might happen if the plugin doesn't handle WM_GETOBJECT properly.
 | |
|     // We should not consider that a failure.
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   aPluginCOMProxy->Set(IAccessibleHolder::COMPtrType(rawAccPlugin));
 | |
| 
 | |
|   return IPC_OK();
 | |
| #else
 | |
|   return IPC_FAIL(this, "Message unsupported in this build configuration");
 | |
| #endif
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult
 | |
| DocAccessibleParent::RecvFocusEvent(const uint64_t& aID,
 | |
|                                     const LayoutDeviceIntRect& aCaretRect)
 | |
| {
 | |
|   if (mShutdown) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   ProxyAccessible* proxy = GetAccessible(aID);
 | |
|   if (!proxy) {
 | |
|     NS_ERROR("no proxy for event!");
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   ProxyFocusEvent(proxy, aCaretRect);
 | |
| 
 | |
|   if (!nsCoreUtils::AccEventObserversExist()) {
 | |
|     return IPC_OK();
 | |
|   }
 | |
| 
 | |
|   xpcAccessibleGeneric* xpcAcc = GetXPCAccessible(proxy);
 | |
|   xpcAccessibleDocument* doc = GetAccService()->GetXPCDocument(this);
 | |
|   nsINode* node = nullptr;
 | |
|   bool fromUser = true; // XXX fix me
 | |
|   RefPtr<xpcAccEvent> event = new xpcAccEvent(nsIAccessibleEvent::EVENT_FOCUS,
 | |
|                                               xpcAcc, doc, node, fromUser);
 | |
|   nsCoreUtils::DispatchAccEvent(std::move(event));
 | |
| 
 | |
|   return IPC_OK();
 | |
| }
 | |
| 
 | |
| #endif // defined(XP_WIN)
 | |
| 
 | |
| } // a11y
 | |
| } // mozilla
 | 
