123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703 |
- /* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
- /* 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/. */
- #include "ARIAGridAccessible-inl.h"
- #include "Accessible-inl.h"
- #include "AccIterator.h"
- #include "nsAccUtils.h"
- #include "Role.h"
- #include "States.h"
- #include "nsIMutableArray.h"
- #include "nsIPersistentProperties2.h"
- #include "nsComponentManagerUtils.h"
- using namespace mozilla;
- using namespace mozilla::a11y;
- ////////////////////////////////////////////////////////////////////////////////
- // ARIAGridAccessible
- ////////////////////////////////////////////////////////////////////////////////
- ////////////////////////////////////////////////////////////////////////////////
- // Constructor
- ARIAGridAccessible::
- ARIAGridAccessible(nsIContent* aContent, DocAccessible* aDoc) :
- AccessibleWrap(aContent, aDoc)
- {
- }
- NS_IMPL_ISUPPORTS_INHERITED0(ARIAGridAccessible, Accessible)
- ////////////////////////////////////////////////////////////////////////////////
- // Table
- uint32_t
- ARIAGridAccessible::ColCount()
- {
- AccIterator rowIter(this, filters::GetRow);
- Accessible* row = rowIter.Next();
- if (!row)
- return 0;
- AccIterator cellIter(row, filters::GetCell);
- Accessible* cell = nullptr;
- uint32_t colCount = 0;
- while ((cell = cellIter.Next()))
- colCount++;
- return colCount;
- }
- uint32_t
- ARIAGridAccessible::RowCount()
- {
- uint32_t rowCount = 0;
- AccIterator rowIter(this, filters::GetRow);
- while (rowIter.Next())
- rowCount++;
- return rowCount;
- }
- Accessible*
- ARIAGridAccessible::CellAt(uint32_t aRowIndex, uint32_t aColumnIndex)
- {
- Accessible* row = GetRowAt(aRowIndex);
- if (!row)
- return nullptr;
- return GetCellInRowAt(row, aColumnIndex);
- }
- bool
- ARIAGridAccessible::IsColSelected(uint32_t aColIdx)
- {
- if (IsARIARole(nsGkAtoms::table))
- return false;
- AccIterator rowIter(this, filters::GetRow);
- Accessible* row = rowIter.Next();
- if (!row)
- return false;
- do {
- if (!nsAccUtils::IsARIASelected(row)) {
- Accessible* cell = GetCellInRowAt(row, aColIdx);
- if (!cell || !nsAccUtils::IsARIASelected(cell))
- return false;
- }
- } while ((row = rowIter.Next()));
- return true;
- }
- bool
- ARIAGridAccessible::IsRowSelected(uint32_t aRowIdx)
- {
- if (IsARIARole(nsGkAtoms::table))
- return false;
- Accessible* row = GetRowAt(aRowIdx);
- if(!row)
- return false;
- if (!nsAccUtils::IsARIASelected(row)) {
- AccIterator cellIter(row, filters::GetCell);
- Accessible* cell = nullptr;
- while ((cell = cellIter.Next())) {
- if (!nsAccUtils::IsARIASelected(cell))
- return false;
- }
- }
- return true;
- }
- bool
- ARIAGridAccessible::IsCellSelected(uint32_t aRowIdx, uint32_t aColIdx)
- {
- if (IsARIARole(nsGkAtoms::table))
- return false;
- Accessible* row = GetRowAt(aRowIdx);
- if(!row)
- return false;
- if (!nsAccUtils::IsARIASelected(row)) {
- Accessible* cell = GetCellInRowAt(row, aColIdx);
- if (!cell || !nsAccUtils::IsARIASelected(cell))
- return false;
- }
- return true;
- }
- uint32_t
- ARIAGridAccessible::SelectedCellCount()
- {
- if (IsARIARole(nsGkAtoms::table))
- return 0;
- uint32_t count = 0, colCount = ColCount();
- AccIterator rowIter(this, filters::GetRow);
- Accessible* row = nullptr;
- while ((row = rowIter.Next())) {
- if (nsAccUtils::IsARIASelected(row)) {
- count += colCount;
- continue;
- }
- AccIterator cellIter(row, filters::GetCell);
- Accessible* cell = nullptr;
- while ((cell = cellIter.Next())) {
- if (nsAccUtils::IsARIASelected(cell))
- count++;
- }
- }
- return count;
- }
- uint32_t
- ARIAGridAccessible::SelectedColCount()
- {
- if (IsARIARole(nsGkAtoms::table))
- return 0;
- uint32_t colCount = ColCount();
- if (!colCount)
- return 0;
- AccIterator rowIter(this, filters::GetRow);
- Accessible* row = rowIter.Next();
- if (!row)
- return 0;
- nsTArray<bool> isColSelArray(colCount);
- isColSelArray.AppendElements(colCount);
- memset(isColSelArray.Elements(), true, colCount * sizeof(bool));
- uint32_t selColCount = colCount;
- do {
- if (nsAccUtils::IsARIASelected(row))
- continue;
- AccIterator cellIter(row, filters::GetCell);
- Accessible* cell = nullptr;
- for (uint32_t colIdx = 0;
- (cell = cellIter.Next()) && colIdx < colCount; colIdx++)
- if (isColSelArray[colIdx] && !nsAccUtils::IsARIASelected(cell)) {
- isColSelArray[colIdx] = false;
- selColCount--;
- }
- } while ((row = rowIter.Next()));
- return selColCount;
- }
- uint32_t
- ARIAGridAccessible::SelectedRowCount()
- {
- if (IsARIARole(nsGkAtoms::table))
- return 0;
- uint32_t count = 0;
- AccIterator rowIter(this, filters::GetRow);
- Accessible* row = nullptr;
- while ((row = rowIter.Next())) {
- if (nsAccUtils::IsARIASelected(row)) {
- count++;
- continue;
- }
- AccIterator cellIter(row, filters::GetCell);
- Accessible* cell = cellIter.Next();
- if (!cell)
- continue;
- bool isRowSelected = true;
- do {
- if (!nsAccUtils::IsARIASelected(cell)) {
- isRowSelected = false;
- break;
- }
- } while ((cell = cellIter.Next()));
- if (isRowSelected)
- count++;
- }
- return count;
- }
- void
- ARIAGridAccessible::SelectedCells(nsTArray<Accessible*>* aCells)
- {
- if (IsARIARole(nsGkAtoms::table))
- return;
- AccIterator rowIter(this, filters::GetRow);
- Accessible* row = nullptr;
- while ((row = rowIter.Next())) {
- AccIterator cellIter(row, filters::GetCell);
- Accessible* cell = nullptr;
- if (nsAccUtils::IsARIASelected(row)) {
- while ((cell = cellIter.Next()))
- aCells->AppendElement(cell);
- continue;
- }
- while ((cell = cellIter.Next())) {
- if (nsAccUtils::IsARIASelected(cell))
- aCells->AppendElement(cell);
- }
- }
- }
- void
- ARIAGridAccessible::SelectedCellIndices(nsTArray<uint32_t>* aCells)
- {
- if (IsARIARole(nsGkAtoms::table))
- return;
- uint32_t colCount = ColCount();
- AccIterator rowIter(this, filters::GetRow);
- Accessible* row = nullptr;
- for (uint32_t rowIdx = 0; (row = rowIter.Next()); rowIdx++) {
- if (nsAccUtils::IsARIASelected(row)) {
- for (uint32_t colIdx = 0; colIdx < colCount; colIdx++)
- aCells->AppendElement(rowIdx * colCount + colIdx);
- continue;
- }
- AccIterator cellIter(row, filters::GetCell);
- Accessible* cell = nullptr;
- for (uint32_t colIdx = 0; (cell = cellIter.Next()); colIdx++) {
- if (nsAccUtils::IsARIASelected(cell))
- aCells->AppendElement(rowIdx * colCount + colIdx);
- }
- }
- }
- void
- ARIAGridAccessible::SelectedColIndices(nsTArray<uint32_t>* aCols)
- {
- if (IsARIARole(nsGkAtoms::table))
- return;
- uint32_t colCount = ColCount();
- if (!colCount)
- return;
- AccIterator rowIter(this, filters::GetRow);
- Accessible* row = rowIter.Next();
- if (!row)
- return;
- nsTArray<bool> isColSelArray(colCount);
- isColSelArray.AppendElements(colCount);
- memset(isColSelArray.Elements(), true, colCount * sizeof(bool));
- do {
- if (nsAccUtils::IsARIASelected(row))
- continue;
- AccIterator cellIter(row, filters::GetCell);
- Accessible* cell = nullptr;
- for (uint32_t colIdx = 0;
- (cell = cellIter.Next()) && colIdx < colCount; colIdx++)
- if (isColSelArray[colIdx] && !nsAccUtils::IsARIASelected(cell)) {
- isColSelArray[colIdx] = false;
- }
- } while ((row = rowIter.Next()));
- for (uint32_t colIdx = 0; colIdx < colCount; colIdx++)
- if (isColSelArray[colIdx])
- aCols->AppendElement(colIdx);
- }
- void
- ARIAGridAccessible::SelectedRowIndices(nsTArray<uint32_t>* aRows)
- {
- if (IsARIARole(nsGkAtoms::table))
- return;
- AccIterator rowIter(this, filters::GetRow);
- Accessible* row = nullptr;
- for (uint32_t rowIdx = 0; (row = rowIter.Next()); rowIdx++) {
- if (nsAccUtils::IsARIASelected(row)) {
- aRows->AppendElement(rowIdx);
- continue;
- }
- AccIterator cellIter(row, filters::GetCell);
- Accessible* cell = cellIter.Next();
- if (!cell)
- continue;
- bool isRowSelected = true;
- do {
- if (!nsAccUtils::IsARIASelected(cell)) {
- isRowSelected = false;
- break;
- }
- } while ((cell = cellIter.Next()));
- if (isRowSelected)
- aRows->AppendElement(rowIdx);
- }
- }
- void
- ARIAGridAccessible::SelectRow(uint32_t aRowIdx)
- {
- if (IsARIARole(nsGkAtoms::table))
- return;
- AccIterator rowIter(this, filters::GetRow);
- Accessible* row = nullptr;
- for (uint32_t rowIdx = 0; (row = rowIter.Next()); rowIdx++) {
- DebugOnly<nsresult> rv = SetARIASelected(row, rowIdx == aRowIdx);
- NS_ASSERTION(NS_SUCCEEDED(rv), "SetARIASelected() Shouldn't fail!");
- }
- }
- void
- ARIAGridAccessible::SelectCol(uint32_t aColIdx)
- {
- if (IsARIARole(nsGkAtoms::table))
- return;
- AccIterator rowIter(this, filters::GetRow);
- Accessible* row = nullptr;
- while ((row = rowIter.Next())) {
- // Unselect all cells in the row.
- DebugOnly<nsresult> rv = SetARIASelected(row, false);
- NS_ASSERTION(NS_SUCCEEDED(rv), "SetARIASelected() Shouldn't fail!");
- // Select cell at the column index.
- Accessible* cell = GetCellInRowAt(row, aColIdx);
- if (cell)
- SetARIASelected(cell, true);
- }
- }
- void
- ARIAGridAccessible::UnselectRow(uint32_t aRowIdx)
- {
- if (IsARIARole(nsGkAtoms::table))
- return;
- Accessible* row = GetRowAt(aRowIdx);
- if (row)
- SetARIASelected(row, false);
- }
- void
- ARIAGridAccessible::UnselectCol(uint32_t aColIdx)
- {
- if (IsARIARole(nsGkAtoms::table))
- return;
- AccIterator rowIter(this, filters::GetRow);
- Accessible* row = nullptr;
- while ((row = rowIter.Next())) {
- Accessible* cell = GetCellInRowAt(row, aColIdx);
- if (cell)
- SetARIASelected(cell, false);
- }
- }
- ////////////////////////////////////////////////////////////////////////////////
- // Protected
- Accessible*
- ARIAGridAccessible::GetRowAt(int32_t aRow)
- {
- int32_t rowIdx = aRow;
- AccIterator rowIter(this, filters::GetRow);
- Accessible* row = rowIter.Next();
- while (rowIdx != 0 && (row = rowIter.Next()))
- rowIdx--;
- return row;
- }
- Accessible*
- ARIAGridAccessible::GetCellInRowAt(Accessible* aRow, int32_t aColumn)
- {
- int32_t colIdx = aColumn;
- AccIterator cellIter(aRow, filters::GetCell);
- Accessible* cell = cellIter.Next();
- while (colIdx != 0 && (cell = cellIter.Next()))
- colIdx--;
- return cell;
- }
- nsresult
- ARIAGridAccessible::SetARIASelected(Accessible* aAccessible,
- bool aIsSelected, bool aNotify)
- {
- if (IsARIARole(nsGkAtoms::table))
- return NS_OK;
- nsIContent *content = aAccessible->GetContent();
- NS_ENSURE_STATE(content);
- nsresult rv = NS_OK;
- if (aIsSelected)
- rv = content->SetAttr(kNameSpaceID_None, nsGkAtoms::aria_selected,
- NS_LITERAL_STRING("true"), aNotify);
- else
- rv = content->SetAttr(kNameSpaceID_None, nsGkAtoms::aria_selected,
- NS_LITERAL_STRING("false"), aNotify);
- NS_ENSURE_SUCCESS(rv, rv);
- // No "smart" select/unselect for internal call.
- if (!aNotify)
- return NS_OK;
- // If row or cell accessible was selected then we're able to not bother about
- // selection of its cells or its row because our algorithm is row oriented,
- // i.e. we check selection on row firstly and then on cells.
- if (aIsSelected)
- return NS_OK;
- roles::Role role = aAccessible->Role();
- // If the given accessible is row that was unselected then remove
- // aria-selected from cell accessible.
- if (role == roles::ROW) {
- AccIterator cellIter(aAccessible, filters::GetCell);
- Accessible* cell = nullptr;
- while ((cell = cellIter.Next())) {
- rv = SetARIASelected(cell, false, false);
- NS_ENSURE_SUCCESS(rv, rv);
- }
- return NS_OK;
- }
- // If the given accessible is cell that was unselected and its row is selected
- // then remove aria-selected from row and put aria-selected on
- // siblings cells.
- if (role == roles::GRID_CELL || role == roles::ROWHEADER ||
- role == roles::COLUMNHEADER) {
- Accessible* row = aAccessible->Parent();
- if (row && row->Role() == roles::ROW &&
- nsAccUtils::IsARIASelected(row)) {
- rv = SetARIASelected(row, false, false);
- NS_ENSURE_SUCCESS(rv, rv);
- AccIterator cellIter(row, filters::GetCell);
- Accessible* cell = nullptr;
- while ((cell = cellIter.Next())) {
- if (cell != aAccessible) {
- rv = SetARIASelected(cell, true, false);
- NS_ENSURE_SUCCESS(rv, rv);
- }
- }
- }
- }
- return NS_OK;
- }
- ////////////////////////////////////////////////////////////////////////////////
- // ARIARowAccessible
- ////////////////////////////////////////////////////////////////////////////////
- ARIARowAccessible::
- ARIARowAccessible(nsIContent* aContent, DocAccessible* aDoc) :
- AccessibleWrap(aContent, aDoc)
- {
- mGenericTypes |= eTableRow;
- }
- NS_IMPL_ISUPPORTS_INHERITED0(ARIARowAccessible, Accessible)
- GroupPos
- ARIARowAccessible::GroupPosition()
- {
- int32_t count = 0, index = 0;
- Accessible* table = nsAccUtils::TableFor(this);
- if (table && nsCoreUtils::GetUIntAttr(table->GetContent(),
- nsGkAtoms::aria_rowcount, &count) &&
- nsCoreUtils::GetUIntAttr(mContent, nsGkAtoms::aria_rowindex, &index)) {
- return GroupPos(0, index, count);
- }
- return AccessibleWrap::GroupPosition();
- }
- ////////////////////////////////////////////////////////////////////////////////
- // ARIAGridCellAccessible
- ////////////////////////////////////////////////////////////////////////////////
- ////////////////////////////////////////////////////////////////////////////////
- // Constructor
- ARIAGridCellAccessible::
- ARIAGridCellAccessible(nsIContent* aContent, DocAccessible* aDoc) :
- HyperTextAccessibleWrap(aContent, aDoc)
- {
- mGenericTypes |= eTableCell;
- }
- NS_IMPL_ISUPPORTS_INHERITED0(ARIAGridCellAccessible, HyperTextAccessible)
- ////////////////////////////////////////////////////////////////////////////////
- // TableCell
- TableAccessible*
- ARIAGridCellAccessible::Table() const
- {
- Accessible* table = nsAccUtils::TableFor(Row());
- return table ? table->AsTable() : nullptr;
- }
- uint32_t
- ARIAGridCellAccessible::ColIdx() const
- {
- Accessible* row = Row();
- if (!row)
- return 0;
- int32_t indexInRow = IndexInParent();
- uint32_t colIdx = 0;
- for (int32_t idx = 0; idx < indexInRow; idx++) {
- Accessible* cell = row->GetChildAt(idx);
- roles::Role role = cell->Role();
- if (role == roles::CELL || role == roles::GRID_CELL ||
- role == roles::ROWHEADER || role == roles::COLUMNHEADER)
- colIdx++;
- }
- return colIdx;
- }
- uint32_t
- ARIAGridCellAccessible::RowIdx() const
- {
- return RowIndexFor(Row());
- }
- bool
- ARIAGridCellAccessible::Selected()
- {
- Accessible* row = Row();
- if (!row)
- return false;
- return nsAccUtils::IsARIASelected(row) || nsAccUtils::IsARIASelected(this);
- }
- ////////////////////////////////////////////////////////////////////////////////
- // Accessible
- void
- ARIAGridCellAccessible::ApplyARIAState(uint64_t* aState) const
- {
- HyperTextAccessibleWrap::ApplyARIAState(aState);
- // Return if the gridcell has aria-selected="true".
- if (*aState & states::SELECTED)
- return;
- // Check aria-selected="true" on the row.
- Accessible* row = Parent();
- if (!row || row->Role() != roles::ROW)
- return;
- nsIContent *rowContent = row->GetContent();
- if (nsAccUtils::HasDefinedARIAToken(rowContent,
- nsGkAtoms::aria_selected) &&
- !rowContent->AttrValueIs(kNameSpaceID_None,
- nsGkAtoms::aria_selected,
- nsGkAtoms::_false, eCaseMatters))
- *aState |= states::SELECTABLE | states::SELECTED;
- }
- already_AddRefed<nsIPersistentProperties>
- ARIAGridCellAccessible::NativeAttributes()
- {
- nsCOMPtr<nsIPersistentProperties> attributes =
- HyperTextAccessibleWrap::NativeAttributes();
- // Expose "table-cell-index" attribute.
- Accessible* thisRow = Row();
- if (!thisRow)
- return attributes.forget();
- int32_t colIdx = 0, colCount = 0;
- uint32_t childCount = thisRow->ChildCount();
- for (uint32_t childIdx = 0; childIdx < childCount; childIdx++) {
- Accessible* child = thisRow->GetChildAt(childIdx);
- if (child == this)
- colIdx = colCount;
- roles::Role role = child->Role();
- if (role == roles::CELL || role == roles::GRID_CELL ||
- role == roles::ROWHEADER || role == roles::COLUMNHEADER)
- colCount++;
- }
- int32_t rowIdx = RowIndexFor(thisRow);
- nsAutoString stringIdx;
- stringIdx.AppendInt(rowIdx * colCount + colIdx);
- nsAccUtils::SetAccAttr(attributes, nsGkAtoms::tableCellIndex, stringIdx);
- #ifdef DEBUG
- nsAutoString unused;
- attributes->SetStringProperty(NS_LITERAL_CSTRING("cppclass"),
- NS_LITERAL_STRING("ARIAGridCellAccessible"),
- unused);
- #endif
- return attributes.forget();
- }
- GroupPos
- ARIAGridCellAccessible::GroupPosition()
- {
- int32_t count = 0, index = 0;
- TableAccessible* table = Table();
- if (table && nsCoreUtils::GetUIntAttr(table->AsAccessible()->GetContent(),
- nsGkAtoms::aria_colcount, &count) &&
- nsCoreUtils::GetUIntAttr(mContent, nsGkAtoms::aria_colindex, &index)) {
- return GroupPos(0, index, count);
- }
- return GroupPos();
- }
|