| /* |
| * Copyright (C) 2010 Apple Inc. All Rights Reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY |
| * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
| * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR |
| * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
| * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
| * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
| * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY |
| * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #include "config.h" |
| #include "AccessibilityMenuList.h" |
| |
| #include "AXObjectCache.h" |
| #include "AccessibilityMenuListPopup.h" |
| #include "RenderMenuList.h" |
| |
| namespace WebCore { |
| |
| AccessibilityMenuList::AccessibilityMenuList(RenderMenuList* renderer) |
| : AccessibilityRenderObject(renderer) |
| { |
| } |
| |
| Ref<AccessibilityMenuList> AccessibilityMenuList::create(RenderMenuList* renderer) |
| { |
| return adoptRef(*new AccessibilityMenuList(renderer)); |
| } |
| |
| bool AccessibilityMenuList::press() |
| { |
| #if !PLATFORM(IOS_FAMILY) |
| auto element = this->element(); |
| AXObjectCache::AXNotification notification = AXObjectCache::AXPressDidFail; |
| if (element && !element->isDisabledFormControl() && is<RenderMenuList>(renderer())) { |
| RenderMenuList* menuList = downcast<RenderMenuList>(renderer()); |
| if (menuList->popupIsVisible()) |
| menuList->hidePopup(); |
| else |
| menuList->showPopup(); |
| notification = AXObjectCache::AXPressDidSucceed; |
| } |
| if (auto cache = axObjectCache()) |
| cache->postNotification(element, notification); |
| return true; |
| #endif |
| return false; |
| } |
| |
| void AccessibilityMenuList::addChildren() |
| { |
| if (!m_renderer) |
| return; |
| |
| AXObjectCache* cache = axObjectCache(); |
| if (!cache) |
| return; |
| |
| AccessibilityObject* list = cache->getOrCreate(AccessibilityRole::MenuListPopup); |
| if (!list) |
| return; |
| |
| downcast<AccessibilityMockObject>(*list).setParent(this); |
| if (list->accessibilityIsIgnored()) { |
| cache->remove(list->objectID()); |
| return; |
| } |
| |
| m_haveChildren = true; |
| m_children.append(list); |
| |
| list->addChildren(); |
| } |
| |
| void AccessibilityMenuList::childrenChanged() |
| { |
| if (m_children.isEmpty()) |
| return; |
| |
| ASSERT(m_children.size() == 1); |
| m_children[0]->childrenChanged(); |
| } |
| |
| bool AccessibilityMenuList::isCollapsed() const |
| { |
| #if !PLATFORM(IOS_FAMILY) |
| auto* renderer = this->renderer(); |
| if (is<RenderMenuList>(renderer)) |
| return !downcast<RenderMenuList>(*renderer).popupIsVisible(); |
| return true; |
| #else |
| return true; |
| #endif |
| } |
| |
| bool AccessibilityMenuList::canSetFocusAttribute() const |
| { |
| if (!node()) |
| return false; |
| |
| return !downcast<Element>(*node()).isDisabledFormControl(); |
| } |
| |
| void AccessibilityMenuList::didUpdateActiveOption(int optionIndex) |
| { |
| Ref<Document> document(m_renderer->document()); |
| AXObjectCache* cache = document->axObjectCache(); |
| |
| const auto& childObjects = children(); |
| if (!childObjects.isEmpty()) { |
| ASSERT(childObjects.size() == 1); |
| ASSERT(is<AccessibilityMenuListPopup>(*childObjects[0])); |
| |
| // We might be calling this method in situations where the renderers for list items |
| // associated to the menu list have not been created (e.g. they might be rendered |
| // in the UI process, as it's the case in the GTK+ port, which uses GtkMenuItem). |
| // So, we need to make sure that the accessibility popup object has some children |
| // before asking it to update its active option, or it will read invalid memory. |
| // You can reproduce the issue in the GTK+ port by removing this check and running |
| // accessibility/insert-selected-option-into-select-causes-crash.html (will crash). |
| int popupChildrenSize = static_cast<int>(childObjects[0]->children().size()); |
| if (is<AccessibilityMenuListPopup>(*childObjects[0]) && optionIndex >= 0 && optionIndex < popupChildrenSize) |
| downcast<AccessibilityMenuListPopup>(*childObjects[0]).didUpdateActiveOption(optionIndex); |
| } |
| |
| cache->postNotification(this, document.ptr(), AXObjectCache::AXMenuListValueChanged, TargetElement, PostSynchronously); |
| } |
| |
| } // namespace WebCore |