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
|
|
|
|
2011-04-09 16:38:06 -07:00
|
|
|
#include "States.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
|
|
|
|
2011-07-27 05:43:01 -07:00
|
|
|
using namespace mozilla::a11y;
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2011-03-28 21:44:20 -07:00
|
|
|
nsAccessible*
|
2011-06-13 15:20:54 -07:00
|
|
|
nsOuterDocAccessible::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
|
|
|
|
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
|
|
|
|
|
2011-06-05 12:35:43 -07:00
|
|
|
PRUint8
|
|
|
|
nsOuterDocAccessible::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
|
|
|
|
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
|
|
|
|
2011-01-27 21:15:04 -08:00
|
|
|
SetChildrenFlag(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;
|
|
|
|
}
|
|
|
|
|
2011-05-21 00:00:14 -07:00
|
|
|
NS_LOG_ACCDOCDESTROY_FOR("remove document from outerdoc",
|
|
|
|
child->GetDocumentNode(), child)
|
2010-06-17 19:44:09 -07:00
|
|
|
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
|
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
|
|
|
}
|