| /* |
| * Copyright (C) 2003, 2006, 2007, 2008, 2009, 2010, 2011 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 COMPUTER, 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 COMPUTER, 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. |
| */ |
| |
| #ifndef AXObjectCache_h |
| #define AXObjectCache_h |
| |
| #include "AccessibilityObject.h" |
| #include "Timer.h" |
| #include <limits.h> |
| #include <wtf/Forward.h> |
| #include <wtf/HashMap.h> |
| #include <wtf/HashSet.h> |
| #include <wtf/RefPtr.h> |
| |
| namespace WebCore { |
| |
| class Document; |
| class HTMLAreaElement; |
| class Node; |
| class Page; |
| class RenderObject; |
| class ScrollView; |
| class VisiblePosition; |
| class Widget; |
| |
| struct TextMarkerData { |
| AXID axID; |
| Node* node; |
| int offset; |
| EAffinity affinity; |
| }; |
| |
| class AXComputedObjectAttributeCache { |
| public: |
| AccessibilityObjectInclusion getIgnored(AXID) const; |
| void setIgnored(AXID, AccessibilityObjectInclusion); |
| |
| private: |
| struct CachedAXObjectAttributes { |
| CachedAXObjectAttributes() : ignored(DefaultBehavior) { } |
| |
| AccessibilityObjectInclusion ignored; |
| }; |
| |
| HashMap<AXID, CachedAXObjectAttributes> m_idMapping; |
| }; |
| |
| enum PostTarget { TargetElement, TargetObservableParent }; |
| |
| enum PostType { PostSynchronously, PostAsynchronously }; |
| |
| class AXObjectCache { |
| WTF_MAKE_NONCOPYABLE(AXObjectCache); WTF_MAKE_FAST_ALLOCATED; |
| public: |
| explicit AXObjectCache(Document&); |
| ~AXObjectCache(); |
| |
| static AccessibilityObject* focusedUIElementForPage(const Page*); |
| |
| // Returns the root object for the entire document. |
| AccessibilityObject* rootObject(); |
| // Returns the root object for a specific frame. |
| AccessibilityObject* rootObjectForFrame(Frame*); |
| |
| // For AX objects with elements that back them. |
| AccessibilityObject* getOrCreate(RenderObject*); |
| AccessibilityObject* getOrCreate(Widget*); |
| AccessibilityObject* getOrCreate(Node*); |
| |
| // used for objects without backing elements |
| AccessibilityObject* getOrCreate(AccessibilityRole); |
| |
| // will only return the AccessibilityObject if it already exists |
| AccessibilityObject* get(RenderObject*); |
| AccessibilityObject* get(Widget*); |
| AccessibilityObject* get(Node*); |
| |
| void remove(RenderObject*); |
| void remove(Node*); |
| void remove(Widget*); |
| void remove(AXID); |
| |
| void detachWrapper(AccessibilityObject*, AccessibilityDetachmentType); |
| void attachWrapper(AccessibilityObject*); |
| void childrenChanged(Node*); |
| void childrenChanged(RenderObject*, RenderObject* newChild = nullptr); |
| void childrenChanged(AccessibilityObject*); |
| void checkedStateChanged(Node*); |
| void selectedChildrenChanged(Node*); |
| void selectedChildrenChanged(RenderObject*); |
| // Called by a node when text or a text equivalent (e.g. alt) attribute is changed. |
| void textChanged(Node*); |
| void textChanged(RenderObject*); |
| // Called when a node has just been attached, so we can make sure we have the right subclass of AccessibilityObject. |
| void updateCacheAfterNodeIsAttached(Node*); |
| |
| void handleActiveDescendantChanged(Node*); |
| void handleAriaRoleChanged(Node*); |
| void handleFocusedUIElementChanged(Node* oldFocusedNode, Node* newFocusedNode); |
| void handleScrolledToAnchor(const Node* anchorNode); |
| void handleAriaExpandedChange(Node*); |
| void handleScrollbarUpdate(ScrollView*); |
| |
| void handleAttributeChanged(const QualifiedName& attrName, Element*); |
| void recomputeIsIgnored(RenderObject* renderer); |
| |
| #if HAVE(ACCESSIBILITY) |
| static void enableAccessibility(); |
| static void disableAccessibility(); |
| |
| // Enhanced user interface accessibility can be toggled by the assistive technology. |
| static void setEnhancedUserInterfaceAccessibility(bool flag) { gAccessibilityEnhancedUserInterfaceEnabled = flag; } |
| |
| static bool accessibilityEnabled() { return gAccessibilityEnabled; } |
| static bool accessibilityEnhancedUserInterfaceEnabled() { return gAccessibilityEnhancedUserInterfaceEnabled; } |
| #else |
| static void enableAccessibility() { } |
| static void disableAccessibility() { } |
| static void setEnhancedUserInterfaceAccessibility(bool) { } |
| static bool accessibilityEnabled() { return false; } |
| static bool accessibilityEnhancedUserInterfaceEnabled() { return false; } |
| #endif |
| |
| void removeAXID(AccessibilityObject*); |
| bool isIDinUse(AXID id) const { return m_idsInUse.contains(id); } |
| |
| Element* rootAXEditableElement(Node*); |
| const Element* rootAXEditableElement(const Node*); |
| bool nodeIsTextControl(const Node*); |
| |
| AXID platformGenerateAXID() const; |
| AccessibilityObject* objectFromAXID(AXID id) const { return m_objects.get(id); } |
| |
| // Text marker utilities. |
| void textMarkerDataForVisiblePosition(TextMarkerData&, const VisiblePosition&); |
| VisiblePosition visiblePositionForTextMarkerData(TextMarkerData&); |
| |
| enum AXNotification { |
| AXActiveDescendantChanged, |
| AXAutocorrectionOccured, |
| AXCheckedStateChanged, |
| AXChildrenChanged, |
| AXFocusedUIElementChanged, |
| AXLayoutComplete, |
| AXLoadComplete, |
| AXSelectedChildrenChanged, |
| AXSelectedTextChanged, |
| AXValueChanged, |
| AXScrolledToAnchor, |
| AXLiveRegionChanged, |
| AXMenuListItemSelected, |
| AXMenuListValueChanged, |
| AXMenuClosed, |
| AXMenuOpened, |
| AXRowCountChanged, |
| AXRowCollapsed, |
| AXRowExpanded, |
| AXInvalidStatusChanged, |
| AXTextChanged, |
| AXAriaAttributeChanged, |
| AXElementBusyChanged |
| }; |
| |
| void postNotification(RenderObject*, AXNotification, PostTarget = TargetElement, PostType = PostAsynchronously); |
| void postNotification(Node*, AXNotification, PostTarget = TargetElement, PostType = PostAsynchronously); |
| void postNotification(AccessibilityObject*, Document*, AXNotification, PostTarget = TargetElement, PostType = PostAsynchronously); |
| |
| enum AXTextChange { |
| AXTextInserted, |
| AXTextDeleted, |
| }; |
| |
| void nodeTextChangeNotification(Node*, AXTextChange, unsigned offset, const String&); |
| |
| enum AXLoadingEvent { |
| AXLoadingStarted, |
| AXLoadingReloaded, |
| AXLoadingFailed, |
| AXLoadingFinished |
| }; |
| |
| void frameLoadingEventNotification(Frame*, AXLoadingEvent); |
| |
| void clearTextMarkerNodesInUse(Document*); |
| |
| void startCachingComputedObjectAttributesUntilTreeMutates(); |
| void stopCachingComputedObjectAttributes(); |
| |
| AXComputedObjectAttributeCache* computedObjectAttributeCache() { return m_computedObjectAttributeCache.get(); } |
| |
| Document& document() const { return m_document; } |
| |
| protected: |
| void postPlatformNotification(AccessibilityObject*, AXNotification); |
| void platformHandleFocusedUIElementChanged(Node* oldFocusedNode, Node* newFocusedNode); |
| |
| void nodeTextChangePlatformNotification(AccessibilityObject*, AXTextChange, unsigned offset, const String&); |
| void frameLoadingEventPlatformNotification(AccessibilityObject*, AXLoadingEvent); |
| void textChanged(AccessibilityObject*); |
| void labelChanged(Element*); |
| |
| // This is a weak reference cache for knowing if Nodes used by TextMarkers are valid. |
| void setNodeInUse(Node* n) { m_textMarkerNodes.add(n); } |
| void removeNodeForUse(Node* n) { m_textMarkerNodes.remove(n); } |
| bool isNodeInUse(Node* n) { return m_textMarkerNodes.contains(n); } |
| |
| private: |
| Document& m_document; |
| HashMap<AXID, RefPtr<AccessibilityObject>> m_objects; |
| HashMap<RenderObject*, AXID> m_renderObjectMapping; |
| HashMap<Widget*, AXID> m_widgetObjectMapping; |
| HashMap<Node*, AXID> m_nodeObjectMapping; |
| HashSet<Node*> m_textMarkerNodes; |
| std::unique_ptr<AXComputedObjectAttributeCache> m_computedObjectAttributeCache; |
| static bool gAccessibilityEnabled; |
| static bool gAccessibilityEnhancedUserInterfaceEnabled; |
| |
| HashSet<AXID> m_idsInUse; |
| |
| Timer<AXObjectCache> m_notificationPostTimer; |
| Vector<std::pair<RefPtr<AccessibilityObject>, AXNotification>> m_notificationsToPost; |
| void notificationPostTimerFired(Timer<AXObjectCache>&); |
| void handleMenuOpened(Node*); |
| void handleMenuItemSelected(Node*); |
| |
| static AccessibilityObject* focusedImageMapUIElement(HTMLAreaElement*); |
| |
| AXID getAXID(AccessibilityObject*); |
| }; |
| |
| class AXAttributeCacheEnabler |
| { |
| public: |
| explicit AXAttributeCacheEnabler(AXObjectCache *cache); |
| ~AXAttributeCacheEnabler(); |
| |
| private: |
| AXObjectCache* m_cache; |
| }; |
| |
| bool nodeHasRole(Node*, const String& role); |
| // This will let you know if aria-hidden was explicitly set to false. |
| bool isNodeAriaVisible(Node*); |
| |
| #if !HAVE(ACCESSIBILITY) |
| inline AccessibilityObjectInclusion AXComputedObjectAttributeCache::getIgnored(AXID) const { return DefaultBehavior; } |
| inline void AXComputedObjectAttributeCache::setIgnored(AXID, AccessibilityObjectInclusion) { } |
| inline AXObjectCache::AXObjectCache(Document& document) : m_document(document), m_notificationPostTimer(this, (Timer<AXObjectCache>::TimerFiredFunction) nullptr) { } |
| inline AXObjectCache::~AXObjectCache() { } |
| inline AccessibilityObject* AXObjectCache::focusedUIElementForPage(const Page*) { return 0; } |
| inline AccessibilityObject* AXObjectCache::get(RenderObject*) { return 0; } |
| inline AccessibilityObject* AXObjectCache::get(Node*) { return 0; } |
| inline AccessibilityObject* AXObjectCache::get(Widget*) { return 0; } |
| inline AccessibilityObject* AXObjectCache::getOrCreate(AccessibilityRole) { return 0; } |
| inline AccessibilityObject* AXObjectCache::getOrCreate(RenderObject*) { return 0; } |
| inline AccessibilityObject* AXObjectCache::getOrCreate(Node*) { return 0; } |
| inline AccessibilityObject* AXObjectCache::getOrCreate(Widget*) { return 0; } |
| inline AccessibilityObject* AXObjectCache::rootObject() { return 0; } |
| inline AccessibilityObject* AXObjectCache::rootObjectForFrame(Frame*) { return 0; } |
| inline Element* AXObjectCache::rootAXEditableElement(Node*) { return 0; } |
| inline bool nodeHasRole(Node*, const String&) { return false; } |
| inline void AXObjectCache::startCachingComputedObjectAttributesUntilTreeMutates() { } |
| inline void AXObjectCache::stopCachingComputedObjectAttributes() { } |
| inline bool isNodeAriaVisible(Node*) { return true; } |
| inline const Element* AXObjectCache::rootAXEditableElement(const Node*) { return 0; } |
| inline void AXObjectCache::attachWrapper(AccessibilityObject*) { } |
| inline void AXObjectCache::checkedStateChanged(Node*) { } |
| inline void AXObjectCache::childrenChanged(RenderObject*, RenderObject*) { } |
| inline void AXObjectCache::childrenChanged(Node*) { } |
| inline void AXObjectCache::childrenChanged(AccessibilityObject*) { } |
| inline void AXObjectCache::textChanged(RenderObject*) { } |
| inline void AXObjectCache::textChanged(Node*) { } |
| inline void AXObjectCache::textChanged(AccessibilityObject*) { } |
| inline void AXObjectCache::updateCacheAfterNodeIsAttached(Node*) { } |
| inline void AXObjectCache::detachWrapper(AccessibilityObject*, AccessibilityDetachmentType) { } |
| inline void AXObjectCache::frameLoadingEventNotification(Frame*, AXLoadingEvent) { } |
| inline void AXObjectCache::frameLoadingEventPlatformNotification(AccessibilityObject*, AXLoadingEvent) { } |
| inline void AXObjectCache::handleActiveDescendantChanged(Node*) { } |
| inline void AXObjectCache::handleAriaExpandedChange(Node*) { } |
| inline void AXObjectCache::handleAriaRoleChanged(Node*) { } |
| inline void AXObjectCache::handleFocusedUIElementChanged(Node*, Node*) { } |
| inline void AXObjectCache::handleScrollbarUpdate(ScrollView*) { } |
| inline void AXObjectCache::handleAttributeChanged(const QualifiedName&, Element*) { } |
| inline void AXObjectCache::recomputeIsIgnored(RenderObject*) { } |
| inline void AXObjectCache::handleScrolledToAnchor(const Node*) { } |
| inline void AXObjectCache::nodeTextChangeNotification(Node*, AXTextChange, unsigned, const String&) { } |
| inline void AXObjectCache::nodeTextChangePlatformNotification(AccessibilityObject*, AXTextChange, unsigned, const String&) { } |
| inline void AXObjectCache::postNotification(AccessibilityObject*, Document*, AXNotification, PostTarget, PostType) { } |
| inline void AXObjectCache::postNotification(RenderObject*, AXNotification, PostTarget, PostType) { } |
| inline void AXObjectCache::postNotification(Node*, AXNotification, PostTarget, PostType) { } |
| inline void AXObjectCache::postPlatformNotification(AccessibilityObject*, AXNotification) { } |
| inline void AXObjectCache::remove(AXID) { } |
| inline void AXObjectCache::remove(RenderObject*) { } |
| inline void AXObjectCache::remove(Node*) { } |
| inline void AXObjectCache::remove(Widget*) { } |
| inline void AXObjectCache::selectedChildrenChanged(RenderObject*) { } |
| inline void AXObjectCache::selectedChildrenChanged(Node*) { } |
| #endif |
| |
| } |
| |
| #endif |