AccessibilityTableCell.cpp [plain text]
#include "config.h"
#include "AccessibilityTableCell.h"
#include "AXObjectCache.h"
#include "AccessibilityTable.h"
#include "AccessibilityTableRow.h"
#include "HTMLElement.h"
#include "HTMLNames.h"
#include "RenderObject.h"
#include "RenderTableCell.h"
namespace WebCore {
using namespace HTMLNames;
AccessibilityTableCell::AccessibilityTableCell(RenderObject* renderer)
: AccessibilityRenderObject(renderer)
{
}
AccessibilityTableCell::~AccessibilityTableCell()
{
}
PassRefPtr<AccessibilityTableCell> AccessibilityTableCell::create(RenderObject* renderer)
{
return adoptRef(new AccessibilityTableCell(renderer));
}
bool AccessibilityTableCell::computeAccessibilityIsIgnored() const
{
AccessibilityObjectInclusion decision = defaultObjectInclusion();
if (decision == IncludeObject)
return false;
if (decision == IgnoreObject)
return true;
if (!node())
return true;
if (!isTableCell())
return AccessibilityRenderObject::computeAccessibilityIsIgnored();
return false;
}
AccessibilityTable* AccessibilityTableCell::parentTable() const
{
if (!m_renderer || !m_renderer->isTableCell())
return 0;
if (!axObjectCache())
return 0;
AccessibilityObject* parentTable = axObjectCache()->get(toRenderTableCell(m_renderer)->table());
if (!parentTable || !parentTable->isTable())
return nullptr;
return toAccessibilityTable(parentTable);
}
bool AccessibilityTableCell::isTableCell() const
{
AccessibilityObject* parentTable = this->parentTable();
return parentTable && parentTable->isAccessibilityTable();
}
AccessibilityRole AccessibilityTableCell::determineAccessibilityRole()
{
AccessibilityRole defaultRole = AccessibilityRenderObject::determineAccessibilityRole();
if (!isTableCell())
return defaultRole;
return CellRole;
}
bool AccessibilityTableCell::isTableHeaderCell() const
{
return node() && node()->hasTagName(thTag);
}
bool AccessibilityTableCell::isTableCellInSameRowGroup(AccessibilityTableCell* otherTableCell)
{
Node* parentNode = node();
for ( ; parentNode; parentNode = parentNode->parentNode()) {
if (parentNode->hasTagName(theadTag) || parentNode->hasTagName(tbodyTag) || parentNode->hasTagName(tfootTag))
break;
}
Node* otherParentNode = otherTableCell->node();
for ( ; otherParentNode; otherParentNode = otherParentNode->parentNode()) {
if (otherParentNode->hasTagName(theadTag) || otherParentNode->hasTagName(tbodyTag) || otherParentNode->hasTagName(tfootTag))
break;
}
return otherParentNode == parentNode;
}
bool AccessibilityTableCell::isTableCellInSameColGroup(AccessibilityTableCell* tableCell)
{
std::pair<unsigned, unsigned> colRange;
columnIndexRange(colRange);
std::pair<unsigned, unsigned> otherColRange;
tableCell->columnIndexRange(otherColRange);
if (colRange.first <= (otherColRange.first + otherColRange.second))
return true;
return false;
}
String AccessibilityTableCell::expandedTextValue() const
{
return getAttribute(abbrAttr);
}
bool AccessibilityTableCell::supportsExpandedTextValue() const
{
return isTableHeaderCell() && hasAttribute(abbrAttr);
}
void AccessibilityTableCell::columnHeaders(AccessibilityChildrenVector& headers)
{
AccessibilityTable* parent = parentTable();
if (!parent)
return;
ariaElementsFromAttribute(headers, headersAttr);
if (!headers.isEmpty())
return;
std::pair<unsigned, unsigned> rowRange;
rowIndexRange(rowRange);
std::pair<unsigned, unsigned> colRange;
columnIndexRange(colRange);
for (unsigned row = 0; row < rowRange.first; row++) {
AccessibilityTableCell* tableCell = parent->cellForColumnAndRow(colRange.first, row);
if (!tableCell || tableCell == this || headers.contains(tableCell))
continue;
std::pair<unsigned, unsigned> childRowRange;
tableCell->rowIndexRange(childRowRange);
const AtomicString& scope = tableCell->getAttribute(scopeAttr);
if (scope == "col" || tableCell->isTableHeaderCell())
headers.append(tableCell);
else if (scope == "colgroup" && isTableCellInSameColGroup(tableCell))
headers.append(tableCell);
}
}
void AccessibilityTableCell::rowHeaders(AccessibilityChildrenVector& headers)
{
AccessibilityTable* parent = parentTable();
if (!parent)
return;
std::pair<unsigned, unsigned> rowRange;
rowIndexRange(rowRange);
std::pair<unsigned, unsigned> colRange;
columnIndexRange(colRange);
for (unsigned column = 0; column < colRange.first; column++) {
AccessibilityTableCell* tableCell = parent->cellForColumnAndRow(column, rowRange.first);
if (!tableCell || tableCell == this || headers.contains(tableCell))
continue;
const AtomicString& scope = tableCell->getAttribute(scopeAttr);
if (scope == "row")
headers.append(tableCell);
else if (scope == "rowgroup" && isTableCellInSameRowGroup(tableCell))
headers.append(tableCell);
}
}
void AccessibilityTableCell::rowIndexRange(std::pair<unsigned, unsigned>& rowRange)
{
if (!m_renderer || !m_renderer->isTableCell())
return;
RenderTableCell* renderCell = toRenderTableCell(m_renderer);
rowRange.first = renderCell->rowIndex();
rowRange.second = renderCell->rowSpan();
RenderTableSection* section = renderCell->section();
RenderTable* table = renderCell->table();
if (!table || !section)
return;
RenderTableSection* footerSection = table->footer();
unsigned rowOffset = 0;
for (RenderTableSection* tableSection = table->topSection(); tableSection; tableSection = table->sectionBelow(tableSection, SkipEmptySections)) {
if (tableSection == footerSection)
continue;
if (tableSection == section)
break;
rowOffset += tableSection->numRows();
}
rowRange.first += rowOffset;
}
void AccessibilityTableCell::columnIndexRange(std::pair<unsigned, unsigned>& columnRange)
{
if (!m_renderer || !m_renderer->isTableCell())
return;
const RenderTableCell& cell = *toRenderTableCell(m_renderer);
columnRange.first = cell.table()->colToEffCol(cell.col());
columnRange.second = cell.table()->colToEffCol(cell.col() + cell.colSpan()) - columnRange.first;
}
AccessibilityObject* AccessibilityTableCell::titleUIElement() const
{
if (isTableCell() || !m_renderer || !m_renderer->isTableCell())
return 0;
Node* node = m_renderer->node();
if (node && node->hasTagName(thTag))
return 0;
RenderTableCell* renderCell = toRenderTableCell(m_renderer);
int col = renderCell->col();
if (!col)
return 0;
int row = renderCell->rowIndex();
RenderTableSection* section = renderCell->section();
if (!section)
return 0;
RenderTableCell* headerCell = section->primaryCellAt(row, 0);
if (!headerCell || headerCell == renderCell)
return 0;
if (!headerCell->element() || !headerCell->element()->hasTagName(thTag))
return 0;
return axObjectCache()->getOrCreate(headerCell);
}
}