2007-03-22 10:30:00 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
2012-05-21 04:12:37 -07:00
|
|
|
/* 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/. */
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2012-04-05 07:27:37 -07:00
|
|
|
#include "OuterDocAccessible.h"
|
2010-04-26 23:52:03 -07:00
|
|
|
|
|
|
|
#include "nsAccUtils.h"
|
2012-05-27 02:01:40 -07:00
|
|
|
#include "DocAccessible.h"
|
2012-01-11 19:07:35 -08:00
|
|
|
#include "Role.h"
|
|
|
|
#include "States.h"
|
2010-04-26 23:52:03 -07:00
|
|
|
|
2012-05-23 02:21:40 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
#include "Logging.h"
|
|
|
|
#endif
|
|
|
|
|
2012-04-05 07:27:37 -07:00
|
|
|
using namespace mozilla;
|
2011-07-27 05:43:01 -07:00
|
|
|
using namespace mozilla::a11y;
|
|
|
|
|
2010-06-08 09:39:58 -07:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
2012-04-05 07:27:37 -07:00
|
|
|
// OuterDocAccessible
|
2010-06-08 09:39:58 -07:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2012-04-05 07:27:37 -07:00
|
|
|
OuterDocAccessible::
|
2012-05-27 02:01:40 -07:00
|
|
|
OuterDocAccessible(nsIContent* aContent, DocAccessible* aDoc) :
|
2012-02-07 14:38:54 -08:00
|
|
|
nsAccessibleWrap(aContent, aDoc)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2012-04-05 07:27:37 -07:00
|
|
|
OuterDocAccessible::~OuterDocAccessible()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2010-06-08 09:39:58 -07:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// nsISupports
|
|
|
|
|
2012-04-05 07:27:37 -07:00
|
|
|
NS_IMPL_ISUPPORTS_INHERITED0(OuterDocAccessible,
|
2010-06-08 09:39:58 -07:00
|
|
|
nsAccessible)
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// nsAccessible public (DON'T add methods here)
|
|
|
|
|
2012-01-11 19:07:35 -08:00
|
|
|
role
|
2012-04-05 07:27:37 -07:00
|
|
|
OuterDocAccessible::NativeRole()
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2012-01-11 19:07:35 -08:00
|
|
|
return roles::INTERNAL_FRAME;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2011-03-28 21:44:20 -07:00
|
|
|
nsAccessible*
|
2012-04-05 07:27:37 -07:00
|
|
|
OuterDocAccessible::ChildAtPoint(PRInt32 aX, PRInt32 aY,
|
|
|
|
EWhichChildAtPoint aWhichChild)
|
2007-09-18 14:44:43 -07:00
|
|
|
{
|
2009-05-11 03:57:28 -07:00
|
|
|
PRInt32 docX = 0, docY = 0, docWidth = 0, docHeight = 0;
|
|
|
|
nsresult rv = GetBounds(&docX, &docY, &docWidth, &docHeight);
|
2011-03-28 21:44:20 -07:00
|
|
|
NS_ENSURE_SUCCESS(rv, nsnull);
|
2007-09-18 14:44:43 -07:00
|
|
|
|
2009-05-11 03:57:28 -07:00
|
|
|
if (aX < docX || aX >= docX + docWidth || aY < docY || aY >= docY + docHeight)
|
2011-03-28 21:44:20 -07:00
|
|
|
return nsnull;
|
2007-09-18 14:44:43 -07:00
|
|
|
|
2009-05-11 03:57:28 -07:00
|
|
|
// Always return the inner doc as direct child accessible unless bounds
|
|
|
|
// outside of it.
|
2011-03-28 21:44:20 -07:00
|
|
|
nsAccessible* child = GetChildAt(0);
|
|
|
|
NS_ENSURE_TRUE(child, nsnull);
|
2009-05-11 03:57:28 -07:00
|
|
|
|
2011-04-17 01:48:02 -07:00
|
|
|
if (aWhichChild == eDeepestChild)
|
2011-06-13 15:20:54 -07:00
|
|
|
return child->ChildAtPoint(aX, aY, eDeepestChild);
|
2011-03-28 21:44:20 -07:00
|
|
|
return child;
|
2008-09-17 06:11:39 -07:00
|
|
|
}
|
|
|
|
|
2008-03-14 13:49:38 -07:00
|
|
|
nsresult
|
2012-04-05 07:27:37 -07:00
|
|
|
OuterDocAccessible::GetAttributesInternal(nsIPersistentProperties* aAttributes)
|
2008-03-14 13:49:38 -07:00
|
|
|
{
|
|
|
|
nsAutoString tag;
|
|
|
|
aAttributes->GetStringProperty(NS_LITERAL_CSTRING("tag"), tag);
|
|
|
|
if (!tag.IsEmpty()) {
|
|
|
|
// We're overriding the ARIA attributes on an sub document, but we don't want to
|
|
|
|
// override the other attributes
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
return nsAccessible::GetAttributesInternal(aAttributes);
|
|
|
|
}
|
2008-06-24 06:03:06 -07:00
|
|
|
|
2010-06-08 09:39:58 -07:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// nsIAccessible
|
|
|
|
|
2011-06-05 12:35:43 -07:00
|
|
|
PRUint8
|
2012-04-05 07:27:37 -07:00
|
|
|
OuterDocAccessible::ActionCount()
|
2008-06-24 06:03:06 -07:00
|
|
|
{
|
2010-06-08 09:39:58 -07:00
|
|
|
// Internal frame, which is the doc's parent, should not have a click action.
|
2011-06-05 12:35:43 -07:00
|
|
|
return 0;
|
2008-06-24 06:03:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2012-04-05 07:27:37 -07:00
|
|
|
OuterDocAccessible::GetActionName(PRUint8 aIndex, nsAString& aName)
|
2008-06-24 06:03:06 -07:00
|
|
|
{
|
|
|
|
aName.Truncate();
|
|
|
|
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2012-04-05 07:27:37 -07:00
|
|
|
OuterDocAccessible::GetActionDescription(PRUint8 aIndex,
|
|
|
|
nsAString& aDescription)
|
2008-06-24 06:03:06 -07:00
|
|
|
{
|
|
|
|
aDescription.Truncate();
|
|
|
|
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2012-04-05 07:27:37 -07:00
|
|
|
OuterDocAccessible::DoAction(PRUint8 aIndex)
|
2008-06-24 06:03:06 -07:00
|
|
|
{
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
2010-06-08 09:39:58 -07:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// nsAccessNode public
|
|
|
|
|
2010-06-11 21:04:35 -07:00
|
|
|
void
|
2012-04-05 07:27:37 -07:00
|
|
|
OuterDocAccessible::Shutdown()
|
2010-06-08 09:39:58 -07:00
|
|
|
{
|
2010-06-17 19:44:09 -07:00
|
|
|
// XXX: sometimes outerdoc accessible is shutdown because of layout style
|
|
|
|
// change however the presshell of underlying document isn't destroyed and
|
|
|
|
// the document doesn't get pagehide events. Shutdown underlying document if
|
|
|
|
// any to avoid hanging document accessible.
|
2012-05-23 02:21:40 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
if (logging::IsEnabled(logging::eDocDestroy)) {
|
|
|
|
logging::Msg("A11y outerdoc shutdown");
|
|
|
|
logging::Address("outerdoc", this);
|
|
|
|
}
|
|
|
|
#endif
|
2010-06-17 19:44:09 -07:00
|
|
|
|
2012-04-05 07:27:37 -07:00
|
|
|
nsAccessible* childAcc = mChildren.SafeElementAt(0, nsnull);
|
2010-06-14 16:19:10 -07:00
|
|
|
if (childAcc) {
|
2012-05-23 02:21:40 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
if (logging::IsEnabled(logging::eDocDestroy)) {
|
|
|
|
logging::DocDestroy("outerdoc's child document shutdown",
|
|
|
|
childAcc->GetDocumentNode());
|
|
|
|
}
|
|
|
|
#endif
|
2010-10-28 02:34:26 -07:00
|
|
|
childAcc->Shutdown();
|
2010-06-14 16:19:10 -07:00
|
|
|
}
|
|
|
|
|
2010-06-11 21:04:35 -07:00
|
|
|
nsAccessibleWrap::Shutdown();
|
2010-06-08 09:39:58 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// nsAccessible public
|
|
|
|
|
|
|
|
void
|
2012-04-05 07:27:37 -07:00
|
|
|
OuterDocAccessible::InvalidateChildren()
|
2010-06-08 09:39:58 -07:00
|
|
|
{
|
2010-06-14 16:19:10 -07:00
|
|
|
// Do not invalidate children because nsAccDocManager is responsible for
|
|
|
|
// document accessible lifetime when DOM document is created or destroyed. If
|
|
|
|
// DOM document isn't destroyed but its presshell is destroyed (for example,
|
|
|
|
// when DOM node of outerdoc accessible is hidden), then outerdoc accessible
|
|
|
|
// notifies nsAccDocManager about this. If presshell is created for existing
|
|
|
|
// DOM document (for example when DOM node of outerdoc accessible is shown)
|
|
|
|
// then allow nsAccDocManager to handle this case since the document
|
|
|
|
// accessible is created and appended as a child when it's requested.
|
2010-06-08 09:39:58 -07:00
|
|
|
|
2011-01-27 21:15:04 -08:00
|
|
|
SetChildrenFlag(eChildrenUninitialized);
|
2010-06-08 09:39:58 -07:00
|
|
|
}
|
|
|
|
|
2011-09-28 23:19:26 -07:00
|
|
|
bool
|
2012-04-05 07:27:37 -07:00
|
|
|
OuterDocAccessible::AppendChild(nsAccessible* aAccessible)
|
2010-06-08 09:39:58 -07:00
|
|
|
{
|
2010-11-09 02:22:59 -08:00
|
|
|
// We keep showing the old document for a bit after creating the new one,
|
|
|
|
// and while building the new DOM and frame tree. That's done on purpose
|
|
|
|
// to avoid weird flashes of default background color.
|
|
|
|
// The old viewer will be destroyed after the new one is created.
|
|
|
|
// For a11y, it should be safe to shut down the old document now.
|
|
|
|
if (mChildren.Length())
|
|
|
|
mChildren[0]->Shutdown();
|
2010-06-08 09:39:58 -07:00
|
|
|
|
2011-12-22 14:31:06 -08:00
|
|
|
if (!nsAccessibleWrap::AppendChild(aAccessible))
|
2011-10-17 07:59:28 -07:00
|
|
|
return false;
|
2010-06-08 09:39:58 -07:00
|
|
|
|
2012-05-23 02:21:40 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
if (logging::IsEnabled(logging::eDocCreate)) {
|
|
|
|
logging::DocCreate("append document to outerdoc",
|
|
|
|
aAccessible->GetDocumentNode());
|
|
|
|
logging::Address("outerdoc", this);
|
|
|
|
}
|
|
|
|
#endif
|
2010-06-17 19:44:09 -07:00
|
|
|
|
2011-10-17 07:59:28 -07:00
|
|
|
return true;
|
2010-06-08 09:39:58 -07:00
|
|
|
}
|
|
|
|
|
2011-09-28 23:19:26 -07:00
|
|
|
bool
|
2012-04-05 07:27:37 -07:00
|
|
|
OuterDocAccessible::RemoveChild(nsAccessible* aAccessible)
|
2010-06-08 09:39:58 -07:00
|
|
|
{
|
2012-04-05 07:27:37 -07:00
|
|
|
nsAccessible* child = mChildren.SafeElementAt(0, nsnull);
|
2010-06-08 09:39:58 -07:00
|
|
|
if (child != aAccessible) {
|
|
|
|
NS_ERROR("Wrong child to remove!");
|
2011-10-17 07:59:28 -07:00
|
|
|
return false;
|
2010-06-08 09:39:58 -07:00
|
|
|
}
|
|
|
|
|
2012-05-23 02:21:40 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
if (logging::IsEnabled(logging::eDocDestroy)) {
|
|
|
|
logging::DocDestroy("remove document from outerdoc", child->GetDocumentNode(),
|
|
|
|
child->AsDoc());
|
|
|
|
logging::Address("outerdoc", this);
|
|
|
|
}
|
|
|
|
#endif
|
2010-06-17 19:44:09 -07:00
|
|
|
|
2011-12-22 14:31:06 -08:00
|
|
|
bool wasRemoved = nsAccessibleWrap::RemoveChild(child);
|
2010-06-17 19:44:09 -07:00
|
|
|
|
2010-06-08 09:39:58 -07:00
|
|
|
NS_ASSERTION(!mChildren.Length(),
|
|
|
|
"This child document of outerdoc accessible wasn't removed!");
|
|
|
|
|
2010-07-01 18:22:41 -07:00
|
|
|
return wasRemoved;
|
2010-06-08 09:39:58 -07:00
|
|
|
}
|
|
|
|
|
2010-06-17 19:44:09 -07:00
|
|
|
|
2010-06-08 09:39:58 -07:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// nsAccessible protected
|
|
|
|
|
|
|
|
void
|
2012-04-05 07:27:37 -07:00
|
|
|
OuterDocAccessible::CacheChildren()
|
2010-06-08 09:39:58 -07:00
|
|
|
{
|
|
|
|
// Request document accessible for the content document to make sure it's
|
2011-02-01 08:08:43 -08:00
|
|
|
// created. It will appended to outerdoc accessible children asynchronously.
|
|
|
|
nsIDocument* outerDoc = mContent->GetCurrentDoc();
|
|
|
|
if (outerDoc) {
|
|
|
|
nsIDocument* innerDoc = outerDoc->GetSubDocumentFor(mContent);
|
|
|
|
if (innerDoc)
|
|
|
|
GetAccService()->GetDocAccessible(innerDoc);
|
|
|
|
}
|
2010-06-08 09:39:58 -07:00
|
|
|
}
|