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) 1998
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
* Author: Eric Vaughan (evaughan@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 ***** */
|
|
|
|
|
|
|
|
// NOTE: alphabetically ordered
|
|
|
|
#include "nsAccessibleTreeWalker.h"
|
|
|
|
#include "nsAccessibilityAtoms.h"
|
|
|
|
#include "nsHTMLFormControlAccessible.h"
|
2007-04-06 08:44:46 -07:00
|
|
|
#include "nsIDOMDocument.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
#include "nsIDOMHTMLInputElement.h"
|
2007-05-07 11:51:36 -07:00
|
|
|
#include "nsIDOMNSHTMLElement.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
#include "nsIDOMNSEditableElement.h"
|
|
|
|
#include "nsIDOMNSHTMLButtonElement.h"
|
|
|
|
#include "nsIDOMHTMLFormElement.h"
|
|
|
|
#include "nsIDOMHTMLLegendElement.h"
|
|
|
|
#include "nsIDOMHTMLTextAreaElement.h"
|
|
|
|
#include "nsIFrame.h"
|
|
|
|
#include "nsINameSpaceManager.h"
|
|
|
|
#include "nsISelectionController.h"
|
|
|
|
#include "nsITextControlFrame.h"
|
|
|
|
|
|
|
|
// --- checkbox -----
|
|
|
|
|
|
|
|
nsHTMLCheckboxAccessible::nsHTMLCheckboxAccessible(nsIDOMNode* aNode, nsIWeakReference* aShell):
|
|
|
|
nsFormControlAccessible(aNode, aShell)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLCheckboxAccessible::GetRole(PRUint32 *_retval)
|
|
|
|
{
|
|
|
|
*_retval = nsIAccessibleRole::ROLE_CHECKBUTTON;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLCheckboxAccessible::GetNumActions(PRUint8 *_retval)
|
|
|
|
{
|
|
|
|
*_retval = 1;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLCheckboxAccessible::GetActionName(PRUint8 aIndex, nsAString& aName)
|
|
|
|
{
|
|
|
|
if (aIndex == eAction_Click) { // 0 is the magic value for default action
|
|
|
|
// check or uncheck
|
|
|
|
PRUint32 state;
|
2007-04-02 08:56:24 -07:00
|
|
|
GetState(&state, nsnull);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
if (state & nsIAccessibleStates::STATE_CHECKED)
|
|
|
|
aName.AssignLiteral("uncheck");
|
|
|
|
else
|
|
|
|
aName.AssignLiteral("check");
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLCheckboxAccessible::DoAction(PRUint8 index)
|
|
|
|
{
|
|
|
|
if (index == 0) { // 0 is the magic value for default action
|
|
|
|
return DoCommand();
|
|
|
|
}
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
2007-04-02 08:56:24 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsHTMLCheckboxAccessible::GetState(PRUint32 *aState, PRUint32 *aExtraState)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2007-04-02 08:56:24 -07:00
|
|
|
nsresult rv = nsFormControlAccessible::GetState(aState, aExtraState);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
2007-05-06 07:50:03 -07:00
|
|
|
*aState |= nsIAccessibleStates::STATE_CHECKABLE;
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
PRBool checked = PR_FALSE; // Radio buttons and check boxes can be checked
|
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMHTMLInputElement> htmlCheckboxElement(do_QueryInterface(mDOMNode));
|
2007-04-02 08:56:24 -07:00
|
|
|
if (htmlCheckboxElement)
|
2007-03-22 10:30:00 -07:00
|
|
|
htmlCheckboxElement->GetChecked(&checked);
|
|
|
|
|
2007-04-02 08:56:24 -07:00
|
|
|
if (checked)
|
|
|
|
*aState |= nsIAccessibleStates::STATE_CHECKED;
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
//------ Radio button -------
|
|
|
|
|
|
|
|
nsHTMLRadioButtonAccessible::nsHTMLRadioButtonAccessible(nsIDOMNode* aNode, nsIWeakReference* aShell):
|
|
|
|
nsRadioButtonAccessible(aNode, aShell)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2007-04-02 08:56:24 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsHTMLRadioButtonAccessible::GetState(PRUint32 *aState, PRUint32 *aExtraState)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2007-04-02 08:56:24 -07:00
|
|
|
nsresult rv = nsAccessibleWrap::GetState(aState, aExtraState);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
2007-05-06 07:50:03 -07:00
|
|
|
*aState |= nsIAccessibleStates::STATE_CHECKABLE;
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
PRBool checked = PR_FALSE; // Radio buttons and check boxes can be checked
|
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMHTMLInputElement> htmlRadioElement(do_QueryInterface(mDOMNode));
|
2007-04-02 08:56:24 -07:00
|
|
|
if (htmlRadioElement)
|
2007-03-22 10:30:00 -07:00
|
|
|
htmlRadioElement->GetChecked(&checked);
|
|
|
|
|
2007-04-02 08:56:24 -07:00
|
|
|
if (checked)
|
|
|
|
*aState |= nsIAccessibleStates::STATE_CHECKED;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2007-04-07 02:07:24 -07:00
|
|
|
nsresult
|
|
|
|
nsHTMLRadioButtonAccessible::GetAttributesInternal(nsIPersistentProperties *aAttributes)
|
2007-03-27 05:17:11 -07:00
|
|
|
{
|
|
|
|
NS_ENSURE_ARG_POINTER(aAttributes);
|
|
|
|
NS_ENSURE_TRUE(mDOMNode, NS_ERROR_FAILURE);
|
|
|
|
|
2007-04-07 02:07:24 -07:00
|
|
|
nsresult rv = nsRadioButtonAccessible::GetAttributesInternal(aAttributes);
|
2007-03-27 05:17:11 -07:00
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
|
|
|
nsAutoString nsURI;
|
|
|
|
mDOMNode->GetNamespaceURI(nsURI);
|
|
|
|
nsAutoString tagName;
|
|
|
|
mDOMNode->GetLocalName(tagName);
|
|
|
|
|
2007-04-03 01:27:43 -07:00
|
|
|
nsCOMPtr<nsIContent> content(do_QueryInterface(mDOMNode));
|
|
|
|
NS_ENSURE_STATE(content);
|
|
|
|
|
|
|
|
nsAutoString type;
|
|
|
|
content->GetAttr(kNameSpaceID_None, nsAccessibilityAtoms::type, type);
|
|
|
|
nsAutoString name;
|
|
|
|
content->GetAttr(kNameSpaceID_None, nsAccessibilityAtoms::name, name);
|
|
|
|
|
2007-04-06 08:44:46 -07:00
|
|
|
nsCOMPtr<nsIDOMNodeList> inputs;
|
|
|
|
|
2007-03-27 05:17:11 -07:00
|
|
|
nsCOMPtr<nsIDOMHTMLInputElement> radio(do_QueryInterface(mDOMNode));
|
|
|
|
nsCOMPtr<nsIDOMHTMLFormElement> form;
|
|
|
|
radio->GetForm(getter_AddRefs(form));
|
2007-04-06 08:44:46 -07:00
|
|
|
if (form) {
|
|
|
|
form->GetElementsByTagNameNS(nsURI, tagName, getter_AddRefs(inputs));
|
|
|
|
} else {
|
|
|
|
nsCOMPtr<nsIDOMDocument> document;
|
|
|
|
mDOMNode->GetOwnerDocument(getter_AddRefs(document));
|
|
|
|
if (document)
|
|
|
|
document->GetElementsByTagNameNS(nsURI, tagName, getter_AddRefs(inputs));
|
|
|
|
}
|
2007-03-27 05:17:11 -07:00
|
|
|
|
2007-04-03 01:27:43 -07:00
|
|
|
NS_ENSURE_TRUE(inputs, NS_OK);
|
2007-03-27 05:17:11 -07:00
|
|
|
|
2007-04-03 01:27:43 -07:00
|
|
|
PRUint32 inputsCount = 0;
|
|
|
|
inputs->GetLength(&inputsCount);
|
2007-03-27 05:17:11 -07:00
|
|
|
|
2007-04-03 01:27:43 -07:00
|
|
|
// Get posinset and setsize.
|
2007-03-27 05:17:11 -07:00
|
|
|
PRInt32 indexOf = 0;
|
2007-04-03 01:27:43 -07:00
|
|
|
PRInt32 count = 0;
|
|
|
|
|
|
|
|
for (PRUint32 index = 0; index < inputsCount; index++) {
|
|
|
|
nsCOMPtr<nsIDOMNode> itemNode;
|
|
|
|
inputs->Item(index, getter_AddRefs(itemNode));
|
|
|
|
|
|
|
|
nsCOMPtr<nsIContent> item(do_QueryInterface(itemNode));
|
|
|
|
if (item &&
|
|
|
|
item->AttrValueIs(kNameSpaceID_None, nsAccessibilityAtoms::type,
|
|
|
|
type, eCaseMatters) &&
|
|
|
|
item->AttrValueIs(kNameSpaceID_None, nsAccessibilityAtoms::name,
|
|
|
|
name, eCaseMatters)) {
|
|
|
|
|
|
|
|
count++;
|
|
|
|
|
|
|
|
if (itemNode == mDOMNode)
|
|
|
|
indexOf = count;
|
2007-03-27 05:17:11 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
nsAccessibilityUtils::
|
2007-04-07 02:07:24 -07:00
|
|
|
SetAccGroupAttrs(aAttributes, 0, indexOf, count);
|
2007-03-27 05:17:11 -07:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
// ----- Button -----
|
|
|
|
|
|
|
|
nsHTMLButtonAccessible::nsHTMLButtonAccessible(nsIDOMNode* aNode, nsIWeakReference* aShell):
|
2007-05-01 10:08:26 -07:00
|
|
|
nsHyperTextAccessibleWrap(aNode, aShell)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLButtonAccessible::GetNumActions(PRUint8 *_retval)
|
|
|
|
{
|
|
|
|
*_retval = 1;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLButtonAccessible::GetActionName(PRUint8 aIndex, nsAString& aName)
|
|
|
|
{
|
|
|
|
if (aIndex == eAction_Click) {
|
|
|
|
aName.AssignLiteral("press");
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLButtonAccessible::DoAction(PRUint8 index)
|
|
|
|
{
|
|
|
|
if (index == eAction_Click) {
|
|
|
|
return DoCommand();
|
|
|
|
}
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
2007-04-02 08:56:24 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsHTMLButtonAccessible::GetState(PRUint32 *aState, PRUint32 *aExtraState)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
nsCOMPtr<nsIDOMElement> element(do_QueryInterface(mDOMNode));
|
2007-04-02 08:56:24 -07:00
|
|
|
NS_ENSURE_TRUE(element, NS_ERROR_FAILURE);
|
|
|
|
|
2007-05-01 10:08:26 -07:00
|
|
|
nsresult rv = nsHyperTextAccessibleWrap::GetState(aState, aExtraState);
|
2007-04-02 08:56:24 -07:00
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
nsAutoString buttonType;
|
|
|
|
element->GetAttribute(NS_LITERAL_STRING("type"), buttonType);
|
|
|
|
if (buttonType.LowerCaseEqualsLiteral("submit"))
|
2007-04-02 08:56:24 -07:00
|
|
|
*aState |= nsIAccessibleStates::STATE_DEFAULT;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLButtonAccessible::GetRole(PRUint32 *_retval)
|
|
|
|
{
|
|
|
|
*_retval = nsIAccessibleRole::ROLE_PUSHBUTTON;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLButtonAccessible::GetName(nsAString& aName)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIContent> content(do_QueryInterface(mDOMNode));
|
|
|
|
if (!content) {
|
|
|
|
return NS_ERROR_FAILURE; // Node shut down
|
|
|
|
}
|
|
|
|
|
|
|
|
nsAutoString name;
|
|
|
|
if (!content->GetAttr(kNameSpaceID_None, nsAccessibilityAtoms::value,
|
|
|
|
name) &&
|
|
|
|
!content->GetAttr(kNameSpaceID_None, nsAccessibilityAtoms::alt,
|
|
|
|
name)) {
|
|
|
|
if (mRoleMapEntry) {
|
|
|
|
// Use HTML label or DHTML accessibility's labelledby attribute for name
|
|
|
|
GetHTMLName(name, PR_FALSE);
|
|
|
|
}
|
|
|
|
if (name.IsEmpty()) {
|
|
|
|
// Use the button's (default) label if nothing else works
|
|
|
|
nsIFrame* frame = GetFrame();
|
|
|
|
if (frame) {
|
|
|
|
nsIFormControlFrame* fcFrame;
|
|
|
|
frame->QueryInterface(NS_GET_IID(nsIFormControlFrame),
|
|
|
|
(void**) &fcFrame);
|
|
|
|
if (fcFrame)
|
|
|
|
fcFrame->GetFormProperty(nsAccessibilityAtoms::defaultLabel, name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (name.IsEmpty() &&
|
|
|
|
!content->GetAttr(kNameSpaceID_None, nsAccessibilityAtoms::title,
|
|
|
|
name) &&
|
|
|
|
!content->GetAttr(kNameSpaceID_None, nsAccessibilityAtoms::src,
|
|
|
|
name)) {
|
|
|
|
content->GetAttr(kNameSpaceID_None, nsAccessibilityAtoms::data, name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
name.CompressWhitespace();
|
|
|
|
aName = name;
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ----- HTML 4 Button: can contain arbitrary HTML content -----
|
|
|
|
|
|
|
|
nsHTML4ButtonAccessible::nsHTML4ButtonAccessible(nsIDOMNode* aNode, nsIWeakReference* aShell):
|
2007-05-01 10:08:26 -07:00
|
|
|
nsHyperTextAccessibleWrap(aNode, aShell)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTML4ButtonAccessible::GetNumActions(PRUint8 *_retval)
|
|
|
|
{
|
|
|
|
*_retval = 1;
|
|
|
|
return NS_OK;;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTML4ButtonAccessible::GetActionName(PRUint8 aIndex, nsAString& aName)
|
|
|
|
{
|
|
|
|
if (aIndex == eAction_Click) {
|
|
|
|
aName.AssignLiteral("press");
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTML4ButtonAccessible::DoAction(PRUint8 index)
|
|
|
|
{
|
|
|
|
if (index == 0) {
|
|
|
|
return DoCommand();
|
|
|
|
}
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTML4ButtonAccessible::GetRole(PRUint32 *_retval)
|
|
|
|
{
|
|
|
|
*_retval = nsIAccessibleRole::ROLE_PUSHBUTTON;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2007-04-02 08:56:24 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsHTML4ButtonAccessible::GetState(PRUint32 *aState, PRUint32 *aExtraState)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
nsCOMPtr<nsIDOMElement> element(do_QueryInterface(mDOMNode));
|
2007-04-02 08:56:24 -07:00
|
|
|
NS_ENSURE_TRUE(element, NS_ERROR_FAILURE); // Button accessible shut down
|
|
|
|
|
2007-05-01 10:08:26 -07:00
|
|
|
nsresult rv = nsHyperTextAccessibleWrap::GetState(aState, aExtraState);
|
2007-04-02 08:56:24 -07:00
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
|
|
|
*aState |= nsIAccessibleStates::STATE_FOCUSABLE;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
nsAutoString buttonType;
|
|
|
|
element->GetAttribute(NS_LITERAL_STRING("type"), buttonType);
|
|
|
|
if (buttonType.LowerCaseEqualsLiteral("submit"))
|
2007-04-02 08:56:24 -07:00
|
|
|
*aState |= nsIAccessibleStates::STATE_DEFAULT;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
// --- textfield -----
|
|
|
|
|
|
|
|
nsHTMLTextFieldAccessible::nsHTMLTextFieldAccessible(nsIDOMNode* aNode, nsIWeakReference* aShell):
|
2007-05-01 10:08:26 -07:00
|
|
|
nsHyperTextAccessibleWrap(aNode, aShell)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2007-05-01 10:08:26 -07:00
|
|
|
NS_IMPL_ISUPPORTS_INHERITED1(nsHTMLTextFieldAccessible, nsHyperTextAccessibleWrap,
|
2007-04-05 15:14:08 -07:00
|
|
|
nsIAccessibleText)
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLTextFieldAccessible::Init()
|
|
|
|
{
|
|
|
|
CheckForEditor();
|
2007-05-01 10:08:26 -07:00
|
|
|
return nsHyperTextAccessibleWrap::Init();
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLTextFieldAccessible::Shutdown()
|
|
|
|
{
|
|
|
|
if (mEditor) {
|
|
|
|
mEditor->RemoveEditActionListener(this);
|
|
|
|
mEditor = nsnull;
|
|
|
|
}
|
2007-05-01 10:08:26 -07:00
|
|
|
return nsHyperTextAccessibleWrap::Shutdown();
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLTextFieldAccessible::GetRole(PRUint32 *aRole)
|
|
|
|
{
|
|
|
|
*aRole = nsIAccessibleRole::ROLE_ENTRY;
|
|
|
|
nsCOMPtr<nsIContent> content(do_QueryInterface(mDOMNode));
|
|
|
|
if (content &&
|
|
|
|
content->AttrValueIs(kNameSpaceID_None, nsAccessibilityAtoms::type,
|
|
|
|
nsAccessibilityAtoms::password, eIgnoreCase)) {
|
|
|
|
*aRole = nsIAccessibleRole::ROLE_PASSWORD_TEXT;
|
|
|
|
}
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLTextFieldAccessible::GetValue(nsAString& _retval)
|
|
|
|
{
|
|
|
|
PRUint32 state;
|
2007-04-02 08:56:24 -07:00
|
|
|
GetState(&state, nsnull);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (state & nsIAccessibleStates::STATE_PROTECTED) // Don't return password text!
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMHTMLTextAreaElement> textArea(do_QueryInterface(mDOMNode));
|
|
|
|
if (textArea) {
|
|
|
|
return textArea->GetValue(_retval);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMHTMLInputElement> inputElement(do_QueryInterface(mDOMNode));
|
|
|
|
if (inputElement) {
|
|
|
|
return inputElement->GetValue(_retval);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
2007-04-02 08:56:24 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsHTMLTextFieldAccessible::GetState(PRUint32 *aState, PRUint32 *aExtraState)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2007-05-01 10:08:26 -07:00
|
|
|
nsresult rv = nsHyperTextAccessibleWrap::GetState(aState, aExtraState);
|
2007-04-02 08:56:24 -07:00
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// can be focusable, focused, protected. readonly, unavailable, selected
|
|
|
|
nsCOMPtr<nsIContent> content(do_QueryInterface(mDOMNode));
|
|
|
|
NS_ASSERTION(content, "Should not have gotten here if upcalled GetExtState() succeeded");
|
|
|
|
|
|
|
|
if (content->AttrValueIs(kNameSpaceID_None, nsAccessibilityAtoms::type,
|
|
|
|
nsAccessibilityAtoms::password, eIgnoreCase)) {
|
|
|
|
*aState |= nsIAccessibleStates::STATE_PROTECTED;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
nsCOMPtr<nsIAccessible> parent;
|
|
|
|
GetParent(getter_AddRefs(parent));
|
|
|
|
if (parent && Role(parent) == nsIAccessibleRole::ROLE_AUTOCOMPLETE) {
|
|
|
|
*aState |= nsIAccessibleStates::STATE_HASPOPUP;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (content->HasAttr(kNameSpaceID_None, nsAccessibilityAtoms::readonly)) {
|
|
|
|
*aState |= nsIAccessibleStates::STATE_READONLY;
|
|
|
|
}
|
|
|
|
|
2007-04-05 15:14:08 -07:00
|
|
|
if (!aExtraState || !(*aExtraState & nsIAccessibleStates::EXT_STATE_EDITABLE))
|
2007-04-02 08:56:24 -07:00
|
|
|
return NS_OK;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2007-04-05 15:14:08 -07:00
|
|
|
nsCOMPtr<nsIDOMHTMLInputElement> htmlInput(do_QueryInterface(mDOMNode));
|
2007-03-22 10:30:00 -07:00
|
|
|
// Is it an <input> or a <textarea> ?
|
2007-06-04 01:16:04 -07:00
|
|
|
if (htmlInput) {
|
|
|
|
*aExtraState |= nsIAccessibleStates::EXT_STATE_SINGLE_LINE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*aExtraState |= nsIAccessibleStates::EXT_STATE_MULTI_LINE;
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2007-04-05 15:14:08 -07:00
|
|
|
nsCOMPtr<nsIContent> bindingContent = content->GetBindingParent();
|
|
|
|
if (bindingContent &&
|
|
|
|
bindingContent->NodeInfo()->Equals(nsAccessibilityAtoms::textbox,
|
|
|
|
kNameSpaceID_XUL) &&
|
|
|
|
bindingContent->AttrValueIs(kNameSpaceID_None, nsAccessibilityAtoms::type,
|
|
|
|
nsAccessibilityAtoms::autocomplete,
|
|
|
|
eIgnoreCase)) {
|
|
|
|
// If parent is XUL textbox and value of @type attribute is "autocomplete",
|
|
|
|
// then this accessible supports autocompletion.
|
|
|
|
*aExtraState |= nsIAccessibleStates::EXT_STATE_SUPPORTS_AUTOCOMPLETION;
|
|
|
|
} else if (gIsFormFillEnabled && htmlInput &&
|
|
|
|
!(*aState & nsIAccessibleStates::STATE_PROTECTED)) {
|
|
|
|
// Check to see if autocompletion is allowed on this input. We don't expose
|
|
|
|
// it for password fields even though the entire password can be remembered
|
|
|
|
// for a page if the user asks it to be. However, the kind of autocomplete
|
|
|
|
// we're talking here is based on what the user types, where a popup of
|
|
|
|
// possible choices comes up.
|
|
|
|
nsAutoString autocomplete;
|
|
|
|
content->GetAttr(kNameSpaceID_None, nsAccessibilityAtoms::autocomplete,
|
|
|
|
autocomplete);
|
|
|
|
|
|
|
|
if (!autocomplete.LowerCaseEqualsLiteral("off")) {
|
|
|
|
nsCOMPtr<nsIDOMHTMLFormElement> form;
|
|
|
|
htmlInput->GetForm(getter_AddRefs(form));
|
|
|
|
nsCOMPtr<nsIContent> formContent(do_QueryInterface(form));
|
|
|
|
if (formContent) {
|
|
|
|
formContent->GetAttr(kNameSpaceID_None,
|
|
|
|
nsAccessibilityAtoms::autocomplete, autocomplete);
|
2007-04-05 14:52:03 -07:00
|
|
|
}
|
2007-04-05 15:14:08 -07:00
|
|
|
|
|
|
|
if (!formContent || !autocomplete.LowerCaseEqualsLiteral("off"))
|
|
|
|
*aExtraState |= nsIAccessibleStates::EXT_STATE_SUPPORTS_AUTOCOMPLETION;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLTextFieldAccessible::GetNumActions(PRUint8 *_retval)
|
|
|
|
{
|
|
|
|
*_retval = 1;
|
|
|
|
return NS_OK;;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLTextFieldAccessible::GetActionName(PRUint8 aIndex, nsAString& aName)
|
|
|
|
{
|
|
|
|
if (aIndex == eAction_Click) {
|
|
|
|
aName.AssignLiteral("activate");
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLTextFieldAccessible::DoAction(PRUint8 index)
|
|
|
|
{
|
|
|
|
if (index == 0) {
|
2007-05-07 11:51:36 -07:00
|
|
|
nsCOMPtr<nsIDOMNSHTMLElement> element(do_QueryInterface(mDOMNode));
|
|
|
|
if ( element ) {
|
|
|
|
return element->Focus();
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
void nsHTMLTextFieldAccessible::SetEditor(nsIEditor* aEditor)
|
|
|
|
{
|
|
|
|
mEditor = aEditor;
|
|
|
|
if (mEditor)
|
|
|
|
mEditor->AddEditActionListener(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void nsHTMLTextFieldAccessible::CheckForEditor()
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIDOMNSEditableElement> editableElt(do_QueryInterface(mDOMNode));
|
|
|
|
if (!editableElt) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsCOMPtr<nsIEditor> editor;
|
|
|
|
nsresult rv = editableElt->GetEditor(getter_AddRefs(editor));
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
SetEditor(editor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// --- groupbox -----
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The HTML for this is <fieldset> <legend>box-title</legend> form elements </fieldset>
|
|
|
|
*/
|
|
|
|
|
|
|
|
nsHTMLGroupboxAccessible::nsHTMLGroupboxAccessible(nsIDOMNode* aNode, nsIWeakReference* aShell):
|
|
|
|
nsAccessibleWrap(aNode, aShell)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLGroupboxAccessible::GetRole(PRUint32 *_retval)
|
|
|
|
{
|
|
|
|
*_retval = nsIAccessibleRole::ROLE_GROUPING;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2007-04-02 08:56:24 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsHTMLGroupboxAccessible::GetState(PRUint32 *aState, PRUint32 *aExtraState)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
// Groupbox doesn't support any states!
|
2007-04-02 08:56:24 -07:00
|
|
|
*aState = 0;
|
|
|
|
if (aExtraState)
|
|
|
|
*aExtraState = 0;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsHTMLGroupboxAccessible::GetName(nsAString& aName)
|
|
|
|
{
|
|
|
|
if (mRoleMapEntry) {
|
|
|
|
nsAccessible::GetName(aName);
|
|
|
|
if (!aName.IsEmpty()) {
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMElement> element(do_QueryInterface(mDOMNode));
|
|
|
|
if (element) {
|
|
|
|
nsCOMPtr<nsIDOMNodeList> legends;
|
|
|
|
nsAutoString nameSpaceURI;
|
|
|
|
element->GetNamespaceURI(nameSpaceURI);
|
|
|
|
element->GetElementsByTagNameNS(nameSpaceURI, NS_LITERAL_STRING("legend"),
|
|
|
|
getter_AddRefs(legends));
|
|
|
|
if (legends) {
|
|
|
|
nsCOMPtr<nsIDOMNode> legendNode;
|
|
|
|
legends->Item(0, getter_AddRefs(legendNode));
|
|
|
|
nsCOMPtr<nsIContent> legendContent(do_QueryInterface(legendNode));
|
|
|
|
if (legendContent) {
|
|
|
|
aName.Truncate(); // Default name is blank
|
|
|
|
return AppendFlatStringFromSubtree(legendContent, &aName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void nsHTMLGroupboxAccessible::CacheChildren()
|
|
|
|
{
|
|
|
|
if (!mWeakShell) {
|
|
|
|
// This node has been shut down
|
|
|
|
mAccChildCount = eChildCountUninitialized;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mAccChildCount == eChildCountUninitialized) {
|
|
|
|
PRBool allowsAnonChildren = PR_FALSE;
|
|
|
|
GetAllowsAnonChildAccessibles(&allowsAnonChildren);
|
|
|
|
nsAccessibleTreeWalker walker(mWeakShell, mDOMNode, allowsAnonChildren);
|
|
|
|
walker.mState.frame = GetFrame();
|
|
|
|
PRInt32 childCount = 0;
|
|
|
|
walker.GetFirstChild();
|
|
|
|
// Check for <legend> and skip it if it's there
|
|
|
|
if (walker.mState.accessible && walker.mState.domNode) {
|
|
|
|
nsCOMPtr<nsIDOMNode> mightBeLegendNode;
|
|
|
|
walker.mState.domNode->GetParentNode(getter_AddRefs(mightBeLegendNode));
|
|
|
|
nsCOMPtr<nsIDOMHTMLLegendElement> legend(do_QueryInterface(mightBeLegendNode));
|
|
|
|
if (legend) {
|
|
|
|
walker.GetNextSibling(); // Skip the legend
|
|
|
|
}
|
|
|
|
}
|
|
|
|
SetFirstChild(walker.mState.accessible);
|
|
|
|
nsCOMPtr<nsPIAccessible> privatePrevAccessible;
|
|
|
|
while (walker.mState.accessible) {
|
|
|
|
++ childCount;
|
|
|
|
privatePrevAccessible = do_QueryInterface(walker.mState.accessible);
|
|
|
|
privatePrevAccessible->SetParent(this);
|
|
|
|
walker.GetNextSibling();
|
|
|
|
privatePrevAccessible->SetNextSibling(walker.mState.accessible);
|
|
|
|
}
|
|
|
|
mAccChildCount = childCount;
|
|
|
|
}
|
|
|
|
}
|