Home | History | Annotate | Download | only in accessibility
      1 /*
      2  * Copyright (C) 2008 Apple Inc. All rights reserved.
      3  *
      4  * Redistribution and use in source and binary forms, with or without
      5  * modification, are permitted provided that the following conditions
      6  * are met:
      7  *
      8  * 1.  Redistributions of source code must retain the above copyright
      9  *     notice, this list of conditions and the following disclaimer.
     10  * 2.  Redistributions in binary form must reproduce the above copyright
     11  *     notice, this list of conditions and the following disclaimer in the
     12  *     documentation and/or other materials provided with the distribution.
     13  * 3.  Neither the name of Apple Computer, Inc. ("Apple") nor the names of
     14  *     its contributors may be used to endorse or promote products derived
     15  *     from this software without specific prior written permission.
     16  *
     17  * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
     18  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
     19  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
     20  * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
     21  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
     22  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
     23  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
     24  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
     25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
     26  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     27  */
     28 
     29 #include "config.h"
     30 #include "AccessibilityListBox.h"
     31 
     32 #include "AXObjectCache.h"
     33 #include "AccessibilityListBoxOption.h"
     34 #include "HTMLNames.h"
     35 #include "HTMLSelectElement.h"
     36 #include "HitTestResult.h"
     37 #include "RenderListBox.h"
     38 #include "RenderObject.h"
     39 
     40 using namespace std;
     41 
     42 namespace WebCore {
     43 
     44 using namespace HTMLNames;
     45 
     46 AccessibilityListBox::AccessibilityListBox(RenderObject* renderer)
     47     : AccessibilityRenderObject(renderer)
     48 {
     49 }
     50 
     51 AccessibilityListBox::~AccessibilityListBox()
     52 {
     53 }
     54 
     55 PassRefPtr<AccessibilityListBox> AccessibilityListBox::create(RenderObject* renderer)
     56 {
     57     return adoptRef(new AccessibilityListBox(renderer));
     58 }
     59 
     60 bool AccessibilityListBox::canSetSelectedChildrenAttribute() const
     61 {
     62     Node* selectNode = m_renderer->node();
     63     if (!selectNode)
     64         return false;
     65 
     66     return !static_cast<HTMLSelectElement*>(selectNode)->disabled();
     67 }
     68 
     69 void AccessibilityListBox::addChildren()
     70 {
     71     Node* selectNode = m_renderer->node();
     72     if (!selectNode)
     73         return;
     74 
     75     m_haveChildren = true;
     76 
     77     const Vector<Element*>& listItems = static_cast<HTMLSelectElement*>(selectNode)->listItems();
     78     unsigned length = listItems.size();
     79     for (unsigned i = 0; i < length; i++) {
     80         // The cast to HTMLElement below is safe because the only other possible listItem type
     81         // would be a WMLElement, but WML builds don't use accessibility features at all.
     82         AccessibilityObject* listOption = listBoxOptionAccessibilityObject(toHTMLElement(listItems[i]));
     83         if (listOption && !listOption->accessibilityIsIgnored())
     84             m_children.append(listOption);
     85     }
     86 }
     87 
     88 void AccessibilityListBox::setSelectedChildren(AccessibilityChildrenVector& children)
     89 {
     90     if (!canSetSelectedChildrenAttribute())
     91         return;
     92 
     93     Node* selectNode = m_renderer->node();
     94     if (!selectNode)
     95         return;
     96 
     97     // disable any selected options
     98     unsigned length = m_children.size();
     99     for (unsigned i = 0; i < length; i++) {
    100         AccessibilityListBoxOption* listBoxOption = static_cast<AccessibilityListBoxOption*>(m_children[i].get());
    101         if (listBoxOption->isSelected())
    102             listBoxOption->setSelected(false);
    103     }
    104 
    105     length = children.size();
    106     for (unsigned i = 0; i < length; i++) {
    107         AccessibilityObject* obj = children[i].get();
    108         if (obj->roleValue() != ListBoxOptionRole)
    109             continue;
    110 
    111         static_cast<AccessibilityListBoxOption*>(obj)->setSelected(true);
    112     }
    113 }
    114 
    115 void AccessibilityListBox::selectedChildren(AccessibilityChildrenVector& result)
    116 {
    117     ASSERT(result.isEmpty());
    118 
    119     if (!hasChildren())
    120         addChildren();
    121 
    122     unsigned length = m_children.size();
    123     for (unsigned i = 0; i < length; i++) {
    124         if (static_cast<AccessibilityListBoxOption*>(m_children[i].get())->isSelected())
    125             result.append(m_children[i]);
    126     }
    127 }
    128 
    129 void AccessibilityListBox::visibleChildren(AccessibilityChildrenVector& result)
    130 {
    131     ASSERT(result.isEmpty());
    132 
    133     if (!hasChildren())
    134         addChildren();
    135 
    136     unsigned length = m_children.size();
    137     for (unsigned i = 0; i < length; i++) {
    138         if (toRenderListBox(m_renderer)->listIndexIsVisible(i))
    139             result.append(m_children[i]);
    140     }
    141 }
    142 
    143 AccessibilityObject* AccessibilityListBox::listBoxOptionAccessibilityObject(HTMLElement* element) const
    144 {
    145     // skip hr elements
    146     if (!element || element->hasTagName(hrTag))
    147         return 0;
    148 
    149     AccessibilityObject* listBoxObject = m_renderer->document()->axObjectCache()->getOrCreate(ListBoxOptionRole);
    150     static_cast<AccessibilityListBoxOption*>(listBoxObject)->setHTMLElement(element);
    151 
    152     return listBoxObject;
    153 }
    154 
    155 bool AccessibilityListBox::accessibilityIsIgnored() const
    156 {
    157     AccessibilityObjectInclusion decision = accessibilityIsIgnoredBase();
    158     if (decision == IncludeObject)
    159         return false;
    160     if (decision == IgnoreObject)
    161         return true;
    162 
    163     return false;
    164 }
    165 
    166 AccessibilityObject* AccessibilityListBox::elementAccessibilityHitTest(const IntPoint& point) const
    167 {
    168     // the internal HTMLSelectElement methods for returning a listbox option at a point
    169     // ignore optgroup elements.
    170     if (!m_renderer)
    171         return 0;
    172 
    173     Node* node = m_renderer->node();
    174     if (!node)
    175         return 0;
    176 
    177     IntRect parentRect = boundingBoxRect();
    178 
    179     AccessibilityObject* listBoxOption = 0;
    180     unsigned length = m_children.size();
    181     for (unsigned i = 0; i < length; i++) {
    182         IntRect rect = toRenderListBox(m_renderer)->itemBoundingBoxRect(parentRect.x(), parentRect.y(), i);
    183         // The cast to HTMLElement below is safe because the only other possible listItem type
    184         // would be a WMLElement, but WML builds don't use accessibility features at all.
    185         if (rect.contains(point)) {
    186             listBoxOption = m_children[i].get();
    187             break;
    188         }
    189     }
    190 
    191     if (listBoxOption && !listBoxOption->accessibilityIsIgnored())
    192         return listBoxOption;
    193 
    194     return axObjectCache()->getOrCreate(m_renderer);
    195 }
    196 
    197 } // namespace WebCore
    198