/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* ***** BEGIN LICENSE BLOCK ***** * Version: NPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Netscape 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/NPL/ * * 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: Aaron Leventhal (aaronl@netscape.com) * * * Alternatively, the contents of this file may be used under the terms of * either 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 NPL, 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 NPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #include "nsHTMLTableAccessible.h" #include "nsWeakReference.h" #include "nsReadableUtils.h" nsHTMLTableCellAccessible::nsHTMLTableCellAccessible(nsIDOMNode* aDomNode, nsIWeakReference* aShell): nsBlockAccessible(aDomNode, aShell) { } /* unsigned long getAccRole (); */ NS_IMETHODIMP nsHTMLTableCellAccessible::GetAccRole(PRUint32 *aResult) { *aResult = ROLE_CELL; return NS_OK; } NS_IMETHODIMP nsHTMLTableCellAccessible::GetAccState(PRUint32 *aResult) { nsAccessible::GetAccState(aResult); *aResult &= ~STATE_FOCUSABLE; // Inherit all states except focusable state since table cells cannot be focused return NS_OK; } nsHTMLTableCaptionAccessible::nsHTMLTableCaptionAccessible (nsIDOMNode* aDomNode, nsIWeakReference* aShell): nsAccessible(aDomNode, aShell) { } NS_IMETHODIMP nsHTMLTableCaptionAccessible::GetAccState(PRUint32 *aResult) { nsAccessible::GetAccState(aResult); *aResult &= ~STATE_FOCUSABLE; return NS_OK; } NS_IMETHODIMP nsHTMLTableCaptionAccessible::GetAccValue(nsAString& aResult) { aResult.Assign(NS_LITERAL_STRING("")); // Default name is blank nsCOMPtr captionContent(do_QueryInterface(mDOMNode)); AppendFlatStringFromSubtree(captionContent, &aResult); return NS_OK; } NS_IMPL_ISUPPORTS_INHERITED2(nsHTMLTableAccessible, nsBlockAccessible, nsIAccessible, nsIAccessibleTable) nsHTMLTableAccessible::nsHTMLTableAccessible(nsIDOMNode* aDomNode, nsIWeakReference* aShell): nsBlockAccessible(aDomNode, aShell) { } /* unsigned long getAccRole (); */ NS_IMETHODIMP nsHTMLTableAccessible::GetAccRole(PRUint32 *aResult) { *aResult = ROLE_TABLE; return NS_OK; } NS_IMETHODIMP nsHTMLTableAccessible::GetAccState(PRUint32 *aResult) { nsAccessible::GetAccState(aResult); *aResult &= ~STATE_FOCUSABLE; // Inherit all states except focusable state since tables cannot be focused return NS_OK; } NS_IMETHODIMP nsHTMLTableAccessible::GetAccName(nsAString& aResult) { aResult.Assign(NS_LITERAL_STRING("")); // Default name is blank nsCOMPtr element(do_QueryInterface(mDOMNode)); if (element) { nsCOMPtr captions; element->GetElementsByTagName(NS_LITERAL_STRING("caption"), getter_AddRefs(captions)); if (captions) { nsCOMPtr captionNode; captions->Item(0, getter_AddRefs(captionNode)); if (captionNode) { nsCOMPtr captionContent(do_QueryInterface(captionNode)); AppendFlatStringFromSubtree(captionContent, &aResult); } } } return NS_OK; } /* Implementation of nsIAccessibleTable */ NS_IMETHODIMP nsHTMLTableAccessible::GetCaption(nsIAccessible **aCaption) { nsresult rv = NS_OK; nsCOMPtr table(do_QueryInterface(mDOMNode)); NS_ENSURE_TRUE(table, NS_ERROR_FAILURE); nsCOMPtr caption; rv = table->GetCaption(getter_AddRefs(caption)); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr captionNode(do_QueryInterface(caption)); NS_ENSURE_TRUE(captionNode, NS_ERROR_FAILURE); nsCOMPtr accService(do_GetService("@mozilla.org/accessibilityService;1")); NS_ENSURE_TRUE(accService, NS_ERROR_FAILURE); return accService->CreateHTMLTableCaptionAccessible(captionNode, aCaption); } NS_IMETHODIMP nsHTMLTableAccessible::SetCaption(nsIAccessible *aCaption) { nsresult rv = NS_OK; nsCOMPtr table(do_QueryInterface(mDOMNode)); NS_ENSURE_TRUE(table, NS_ERROR_FAILURE); nsCOMPtr domNode; rv = aCaption->AccGetDOMNode(getter_AddRefs(domNode)); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr newDOMNode; rv = domNode->CloneNode(PR_TRUE, getter_AddRefs(newDOMNode)); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr captionElement(do_QueryInterface(newDOMNode)); NS_ENSURE_TRUE(captionElement, NS_ERROR_FAILURE); return table->SetCaption(captionElement); } NS_IMETHODIMP nsHTMLTableAccessible::GetSummary(nsAString &aSummary) { nsCOMPtr table(do_QueryInterface(mDOMNode)); NS_ENSURE_TRUE(table, NS_ERROR_FAILURE); return table->GetSummary(aSummary); } NS_IMETHODIMP nsHTMLTableAccessible::SetSummary(const nsAString &aSummary) { nsCOMPtr table(do_QueryInterface(mDOMNode)); NS_ENSURE_TRUE(table, NS_ERROR_FAILURE); return table->SetSummary(aSummary); } NS_IMETHODIMP nsHTMLTableAccessible::GetColumns(PRInt32 *aColumns) { nsITableLayout *tableLayout = nsnull; nsresult rv = GetTableLayout(&tableLayout); NS_ENSURE_SUCCESS(rv, rv); PRInt32 rows; return tableLayout->GetTableSize(rows, *aColumns); } NS_IMETHODIMP nsHTMLTableAccessible::GetColumnHeader(nsIAccessibleTable **aColumnHeader) { nsresult rv = NS_OK; nsCOMPtr table(do_QueryInterface(mDOMNode)); NS_ENSURE_TRUE(table, NS_ERROR_FAILURE); nsCOMPtr section; rv = table->GetTHead(getter_AddRefs(section)); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr accService(do_GetService("@mozilla.org/accessibilityService;1")); NS_ENSURE_TRUE(accService, NS_ERROR_FAILURE); nsCOMPtr accHead; rv = accService->CreateHTMLTableHeadAccessible(section, getter_AddRefs(accHead)); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr accTableHead(do_QueryInterface(accHead)); NS_ENSURE_TRUE(accTableHead, NS_ERROR_FAILURE); *aColumnHeader = accTableHead; NS_IF_ADDREF(*aColumnHeader); return rv; } NS_IMETHODIMP nsHTMLTableAccessible::GetRows(PRInt32 *aRows) { nsITableLayout *tableLayout = nsnull; nsresult rv = GetTableLayout(&tableLayout); NS_ENSURE_SUCCESS(rv, rv); PRInt32 columns; return tableLayout->GetTableSize(*aRows, columns); } NS_IMETHODIMP nsHTMLTableAccessible::GetRowHeader(nsIAccessibleTable **aRowHeader) { // Can not implement because there is no row header in html table return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsHTMLTableAccessible::GetSelectedColumns(PRUint32 *aNumColumns, PRInt32 **aColumns) { nsresult rv = NS_OK; PRInt32 columnCount; rv = GetColumns(&columnCount); NS_ENSURE_SUCCESS(rv, rv); PRBool *states = new PRBool[columnCount]; NS_ENSURE_TRUE(states, NS_ERROR_OUT_OF_MEMORY); *aNumColumns = 0; PRInt32 index; for (index = 0; index < columnCount; index++) { rv = IsColumnSelected(index, &states[index]); NS_ENSURE_SUCCESS(rv, rv); if (states[index]) { (*aNumColumns)++; } } PRInt32 *outArray = (PRInt32 *)nsMemory::Alloc((*aNumColumns) * sizeof(PRInt32)); if (!outArray) { delete []states; return NS_ERROR_OUT_OF_MEMORY; } PRInt32 curr = 0; for (index = 0; index < columnCount; index++) { if (states[index]) { outArray[curr++] = index; } } delete []states; *aColumns = outArray; return rv; } NS_IMETHODIMP nsHTMLTableAccessible::GetSelectedRows(PRUint32 *aNumRows, PRInt32 **aRows) { nsresult rv = NS_OK; PRInt32 rowCount; rv = GetRows(&rowCount); NS_ENSURE_SUCCESS(rv, rv); PRBool *states = new PRBool[rowCount]; NS_ENSURE_TRUE(states, NS_ERROR_OUT_OF_MEMORY); *aNumRows = 0; PRInt32 index; for (index = 0; index < rowCount; index++) { rv = IsRowSelected(index, &states[index]); NS_ENSURE_SUCCESS(rv, rv); if (states[index]) { (*aNumRows)++; } } PRInt32 *outArray = (PRInt32 *)nsMemory::Alloc((*aNumRows) * sizeof(PRInt32)); if (!outArray) { delete []states; return NS_ERROR_OUT_OF_MEMORY; } PRInt32 curr = 0; for (index = 0; index < rowCount; index++) { if (states[index]) { outArray[curr++] = index; } } delete []states; *aRows = outArray; return rv; } NS_IMETHODIMP nsHTMLTableAccessible::CellRefAt(PRInt32 aRow, PRInt32 aColumn, nsIAccessible **_retval) { nsresult rv = NS_OK; nsCOMPtr cellElement; rv = GetCellAt(aRow, aColumn, *getter_AddRefs(cellElement)); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr accService(do_GetService("@mozilla.org/accessibilityService;1")); NS_ENSURE_TRUE(accService, NS_ERROR_FAILURE); return accService->GetAccessibleFor(cellElement, _retval); } NS_IMETHODIMP nsHTMLTableAccessible::GetIndexAt(PRInt32 aRow, PRInt32 aColumn, PRInt32 *_retval) { NS_ENSURE_ARG_POINTER(_retval); nsresult rv = NS_OK; PRInt32 columns; rv = GetColumns(&columns); NS_ENSURE_SUCCESS(rv, rv); *_retval = aRow * columns + aColumn; return NS_OK; } NS_IMETHODIMP nsHTMLTableAccessible::GetColumnAtIndex(PRInt32 aIndex, PRInt32 *_retval) { NS_ENSURE_ARG_POINTER(_retval); nsresult rv = NS_OK; PRInt32 columns; rv = GetColumns(&columns); NS_ENSURE_SUCCESS(rv, rv); *_retval = aIndex % columns; return NS_OK; } NS_IMETHODIMP nsHTMLTableAccessible::GetRowAtIndex(PRInt32 aIndex, PRInt32 *_retval) { NS_ENSURE_ARG_POINTER(_retval); nsresult rv = NS_OK; PRInt32 columns; rv = GetColumns(&columns); NS_ENSURE_SUCCESS(rv, rv); *_retval = aIndex / columns; return NS_OK; } NS_IMETHODIMP nsHTMLTableAccessible::GetColumnExtentAt(PRInt32 aRow, PRInt32 aColumn, PRInt32 *_retval) { nsresult rv = NS_OK; nsCOMPtr domElement; rv = GetCellAt(aRow, aColumn, *getter_AddRefs(domElement)); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr cell(do_QueryInterface(domElement)); NS_ENSURE_TRUE(cell, NS_ERROR_FAILURE); return cell->GetColSpan(_retval); } NS_IMETHODIMP nsHTMLTableAccessible::GetRowExtentAt(PRInt32 aRow, PRInt32 aColumn, PRInt32 *_retval) { nsresult rv = NS_OK; nsCOMPtr domElement; rv = GetCellAt(aRow, aColumn, *getter_AddRefs(domElement)); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr cell(do_QueryInterface(domElement)); NS_ENSURE_TRUE(cell, NS_ERROR_FAILURE); return cell->GetRowSpan(_retval); } NS_IMETHODIMP nsHTMLTableAccessible::GetColumnDescription(PRInt32 aColumn, nsAString &_retval) { return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsHTMLTableAccessible::GetRowDescription(PRInt32 aRow, nsAString &_retval) { return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsHTMLTableAccessible::IsColumnSelected(PRInt32 aColumn, PRBool *_retval) { NS_ENSURE_ARG_POINTER(_retval); nsresult rv = NS_OK; PRInt32 rows; rv = GetRows(&rows); NS_ENSURE_SUCCESS(rv, rv); for (PRInt32 index = 0; index < rows; index++) { rv = IsCellSelected(index, aColumn, _retval); NS_ENSURE_SUCCESS(rv, rv); if (!*_retval) { break; } } return rv; } NS_IMETHODIMP nsHTMLTableAccessible::IsRowSelected(PRInt32 aRow, PRBool *_retval) { NS_ENSURE_ARG_POINTER(_retval); nsresult rv = NS_OK; PRInt32 columns; rv = GetColumns(&columns); NS_ENSURE_SUCCESS(rv, rv); for (PRInt32 index = 0; index < columns; index++) { rv = IsCellSelected(aRow, index, _retval); NS_ENSURE_SUCCESS(rv, rv); if (!*_retval) { break; } } return rv; } NS_IMETHODIMP nsHTMLTableAccessible::IsCellSelected(PRInt32 aRow, PRInt32 aColumn, PRBool *_retval) { nsITableLayout *tableLayout = nsnull; nsresult rv = GetTableLayout(&tableLayout); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr domElement; PRInt32 startRowIndex = 0, startColIndex = 0, rowSpan, colSpan, actualRowSpan, actualColSpan; return tableLayout->GetCellDataAt(aRow, aColumn, *getter_AddRefs(domElement), startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, *_retval); } nsresult nsHTMLTableAccessible::GetTableNode(nsIDOMNode **_retval) { nsresult rv = NS_OK; nsCOMPtr table(do_QueryInterface(mDOMNode)); if (table) { *_retval = table; NS_IF_ADDREF(*_retval); return rv; } nsCOMPtr section(do_QueryInterface(mDOMNode)); if (section) { nsCOMPtr parent; rv = section->GetParentNode(getter_AddRefs(parent)); NS_ENSURE_SUCCESS(rv, rv); *_retval = parent; NS_IF_ADDREF(*_retval); return rv; } return NS_ERROR_FAILURE; } nsresult nsHTMLTableAccessible::GetTableLayout(nsITableLayout **aLayoutObject) { nsresult rv = NS_OK; nsCOMPtr tableNode; rv = GetTableNode(getter_AddRefs(tableNode)); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr content(do_QueryInterface(tableNode)); NS_ENSURE_TRUE(content, NS_ERROR_FAILURE); nsCOMPtr document; rv = content->GetDocument(*getter_AddRefs(document)); NS_ENSURE_SUCCESS(rv, rv); nsCOMPtr presShell; rv = document->GetShellAt(0, getter_AddRefs(presShell)); NS_ENSURE_SUCCESS(rv, rv); nsISupports *layoutObject = nsnull; rv = presShell->GetLayoutObjectFor(content, &layoutObject); NS_ENSURE_SUCCESS(rv, rv); *aLayoutObject = nsnull; return layoutObject->QueryInterface(NS_GET_IID(nsITableLayout), (void **)aLayoutObject); } nsresult nsHTMLTableAccessible::GetCellAt(PRInt32 aRowIndex, PRInt32 aColIndex, nsIDOMElement* &aCell) { PRInt32 startRowIndex = 0, startColIndex = 0, rowSpan, colSpan, actualRowSpan, actualColSpan; PRBool isSelected; nsITableLayout *tableLayout = nsnull; nsresult rv = GetTableLayout(&tableLayout); NS_ENSURE_SUCCESS(rv, rv); return tableLayout->GetCellDataAt(aRowIndex, aColIndex, aCell, startRowIndex, startColIndex, rowSpan, colSpan, actualRowSpan, actualColSpan, isSelected); } //Class nsHTMLTableHeadAccessible nsHTMLTableHeadAccessible::nsHTMLTableHeadAccessible(nsIDOMNode *aDomNode, nsIWeakReference *aShell): nsHTMLTableAccessible(aDomNode, aShell) { } NS_IMETHODIMP nsHTMLTableHeadAccessible::GetAccRole(PRUint32 *aResult) { *aResult = ROLE_COLUMNHEADER; return NS_OK; } NS_IMETHODIMP nsHTMLTableHeadAccessible::GetCaption(nsIAccessible **aCaption) { return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsHTMLTableHeadAccessible::SetCaption(nsIAccessible *aCaption) { return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsHTMLTableHeadAccessible::GetSummary(nsAString &aSummary) { return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsHTMLTableHeadAccessible::SetSummary(const nsAString &aSummary) { return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsHTMLTableHeadAccessible::GetColumnHeader(nsIAccessibleTable **aColumnHeader) { return NS_ERROR_NOT_IMPLEMENTED; } NS_IMETHODIMP nsHTMLTableHeadAccessible::GetRows(PRInt32 *aRows) { nsresult rv = NS_OK; nsCOMPtr head(do_QueryInterface(mDOMNode)); NS_ENSURE_TRUE(head, NS_ERROR_FAILURE); nsCOMPtr rows; rv = head->GetRows(getter_AddRefs(rows)); NS_ENSURE_SUCCESS(rv, rv); return rows->GetLength((PRUint32 *)aRows); } /* End of Implementation of nsIAccessibleTable */