2007-03-22 10:30:00 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
|
|
* the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
|
|
|
* The Original Code is mozilla.org code.
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is
|
|
|
|
* Netscape Communications Corporation.
|
|
|
|
* Portions created by the Initial Developer are Copyright (C) 2003
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
* Original Author: Aaron Leventhal (aaronl@netscape.com)
|
|
|
|
*
|
|
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
|
|
* either of the GNU General Public License Version 2 or later (the "GPL"),
|
|
|
|
* or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
|
|
* the provisions above, a recipient may use your version of this file under
|
|
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
|
|
*
|
|
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
|
|
|
|
#include "nsOuterDocAccessible.h"
|
2010-04-26 23:52:03 -07:00
|
|
|
|
|
|
|
#include "nsAccUtils.h"
|
2010-06-08 09:39:58 -07:00
|
|
|
#include "nsDocAccessible.h"
|
2010-04-26 23:52:03 -07:00
|
|
|
|
2010-06-08 09:39:58 -07:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// nsOuterDocAccessible
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-06-11 01:23:18 -07:00
|
|
|
nsOuterDocAccessible::
|
|
|
|
nsOuterDocAccessible(nsIContent *aContent, nsIWeakReference *aShell) :
|
|
|
|
nsAccessibleWrap(aContent, aShell)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2010-06-08 09:39:58 -07:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// nsISupports
|
|
|
|
|
|
|
|
NS_IMPL_ISUPPORTS_INHERITED0(nsOuterDocAccessible,
|
|
|
|
nsAccessible)
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// nsAccessible public (DON'T add methods here)
|
|
|
|
|
2010-09-04 19:14:01 -07:00
|
|
|
PRUint32
|
|
|
|
nsOuterDocAccessible::NativeRole()
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-09-04 19:14:01 -07:00
|
|
|
return nsIAccessibleRole::ROLE_INTERNAL_FRAME;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2008-11-03 19:37:46 -08:00
|
|
|
nsresult
|
|
|
|
nsOuterDocAccessible::GetStateInternal(PRUint32 *aState, PRUint32 *aExtraState)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2008-11-03 19:37:46 -08:00
|
|
|
nsresult rv = nsAccessible::GetStateInternal(aState, aExtraState);
|
2008-11-26 20:04:05 -08:00
|
|
|
NS_ENSURE_A11Y_SUCCESS(rv, rv);
|
2008-11-03 19:37:46 -08:00
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
*aState &= ~nsIAccessibleStates::STATE_FOCUSABLE;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2009-05-11 03:57:28 -07:00
|
|
|
nsresult
|
2007-09-18 14:44:43 -07:00
|
|
|
nsOuterDocAccessible::GetChildAtPoint(PRInt32 aX, PRInt32 aY,
|
2009-05-11 03:57:28 -07:00
|
|
|
PRBool aDeepestChild,
|
|
|
|
nsIAccessible **aChild)
|
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);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
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)
|
|
|
|
return NS_OK;
|
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.
|
2008-09-17 06:11:39 -07:00
|
|
|
nsCOMPtr<nsIAccessible> childAcc;
|
2009-05-11 03:57:28 -07:00
|
|
|
rv = GetFirstChild(getter_AddRefs(childAcc));
|
2008-09-17 06:11:39 -07:00
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
|
|
|
if (!childAcc)
|
|
|
|
return NS_OK;
|
|
|
|
|
2009-05-11 03:57:28 -07:00
|
|
|
if (aDeepestChild)
|
|
|
|
return childAcc->GetDeepestChildAtPoint(aX, aY, aChild);
|
|
|
|
|
|
|
|
NS_ADDREF(*aChild = childAcc);
|
|
|
|
return NS_OK;
|
2008-09-17 06:11:39 -07:00
|
|
|
}
|
|
|
|
|
2008-03-14 13:49:38 -07:00
|
|
|
nsresult
|
|
|
|
nsOuterDocAccessible::GetAttributesInternal(nsIPersistentProperties *aAttributes)
|
|
|
|
{
|
|
|
|
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
|
|
|
|
|
2008-06-24 06:03:06 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsOuterDocAccessible::GetNumActions(PRUint8 *aNumActions)
|
|
|
|
{
|
|
|
|
NS_ENSURE_ARG_POINTER(aNumActions);
|
|
|
|
*aNumActions = 0;
|
|
|
|
|
2010-06-08 09:39:58 -07:00
|
|
|
// Internal frame, which is the doc's parent, should not have a click action.
|
2008-06-24 06:03:06 -07:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsOuterDocAccessible::GetActionName(PRUint8 aIndex, nsAString& aName)
|
|
|
|
{
|
|
|
|
aName.Truncate();
|
|
|
|
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsOuterDocAccessible::GetActionDescription(PRUint8 aIndex, nsAString& aDescription)
|
|
|
|
{
|
|
|
|
aDescription.Truncate();
|
|
|
|
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsOuterDocAccessible::DoAction(PRUint8 aIndex)
|
|
|
|
{
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
2010-06-08 09:39:58 -07:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// nsAccessNode public
|
|
|
|
|
2010-06-11 21:04:35 -07:00
|
|
|
void
|
2010-06-08 09:39:58 -07:00
|
|
|
nsOuterDocAccessible::Shutdown()
|
|
|
|
{
|
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.
|
|
|
|
NS_LOG_ACCDOCDESTROY_MSG("A11y outerdoc shutdown")
|
|
|
|
NS_LOG_ACCDOCDESTROY_ACCADDRESS("outerdoc", this)
|
|
|
|
|
2010-06-14 16:19:10 -07:00
|
|
|
nsAccessible *childAcc = mChildren.SafeElementAt(0, nsnull);
|
|
|
|
if (childAcc) {
|
2010-06-17 19:44:09 -07:00
|
|
|
NS_LOG_ACCDOCDESTROY("outerdoc's child document shutdown",
|
|
|
|
childAcc->GetDocumentNode())
|
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
|
|
|
|
nsOuterDocAccessible::InvalidateChildren()
|
|
|
|
{
|
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
|
|
|
|
2010-08-15 04:28:49 -07:00
|
|
|
mChildrenFlags = eChildrenUninitialized;
|
2010-06-08 09:39:58 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsOuterDocAccessible::AppendChild(nsAccessible *aAccessible)
|
|
|
|
{
|
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
|
|
|
|
2010-07-01 18:22:41 -07:00
|
|
|
if (!nsAccessible::AppendChild(aAccessible))
|
2010-06-08 09:39:58 -07:00
|
|
|
return PR_FALSE;
|
|
|
|
|
2010-06-17 19:44:09 -07:00
|
|
|
NS_LOG_ACCDOCCREATE("append document to outerdoc",
|
|
|
|
aAccessible->GetDocumentNode())
|
|
|
|
NS_LOG_ACCDOCCREATE_ACCADDRESS("outerdoc", this)
|
|
|
|
|
2010-06-08 09:39:58 -07:00
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsOuterDocAccessible::RemoveChild(nsAccessible *aAccessible)
|
|
|
|
{
|
|
|
|
nsAccessible *child = mChildren.SafeElementAt(0, nsnull);
|
|
|
|
if (child != aAccessible) {
|
|
|
|
NS_ERROR("Wrong child to remove!");
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
2010-06-17 19:44:09 -07:00
|
|
|
NS_LOG_ACCDOCDESTROY("remove document from outerdoc",
|
|
|
|
child->GetDocumentNode())
|
|
|
|
NS_LOG_ACCDOCDESTROY_ACCADDRESS("outerdoc", this)
|
|
|
|
|
2010-07-01 18:22:41 -07:00
|
|
|
PRBool wasRemoved = nsAccessible::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
|
|
|
|
nsOuterDocAccessible::CacheChildren()
|
|
|
|
{
|
|
|
|
// Request document accessible for the content document to make sure it's
|
|
|
|
// created because once it's created it appends itself as a child.
|
2010-06-11 01:23:18 -07:00
|
|
|
nsIDocument *outerDoc = mContent->GetCurrentDoc();
|
2010-06-08 09:39:58 -07:00
|
|
|
if (!outerDoc)
|
|
|
|
return;
|
|
|
|
|
2010-06-11 01:23:18 -07:00
|
|
|
nsIDocument *innerDoc = outerDoc->GetSubDocumentFor(mContent);
|
|
|
|
if (!innerDoc)
|
2010-06-08 09:39:58 -07:00
|
|
|
return;
|
|
|
|
|
|
|
|
nsDocAccessible *docAcc = GetAccService()->GetDocAccessible(innerDoc);
|
|
|
|
NS_ASSERTION(docAcc && docAcc->GetParent() == this,
|
|
|
|
"Document accessible isn't a child of outerdoc accessible!");
|
|
|
|
}
|