blob: 63a9ed716608a8ff0054caf68b75a71741a9c88f [file] [log] [blame]
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef CONTENT_BROWSER_ACCESSIBILITY_BROWSER_ACCESSIBILITY_MANAGER_WIN_H_
#define CONTENT_BROWSER_ACCESSIBILITY_BROWSER_ACCESSIBILITY_MANAGER_WIN_H_
#include <oleacc.h>
#include <map>
#include <memory>
#include <unordered_set>
#include <vector>
#include "base/macros.h"
#include "content/browser/accessibility/browser_accessibility_manager.h"
#include "ui/accessibility/platform/ax_platform_node_win.h"
namespace content {
class BrowserAccessibilityWin;
// Manages a tree of BrowserAccessibilityWin objects.
class CONTENT_EXPORT BrowserAccessibilityManagerWin
: public BrowserAccessibilityManager {
public:
BrowserAccessibilityManagerWin(const ui::AXTreeUpdate& initial_tree,
BrowserAccessibilityDelegate* delegate);
~BrowserAccessibilityManagerWin() override;
static ui::AXTreeUpdate GetEmptyDocument();
// Get the closest containing HWND.
HWND GetParentHWND();
// BrowserAccessibilityManager methods
void UserIsReloading() override;
BrowserAccessibility* GetFocus() const override;
bool CanFireEvents() const override;
gfx::Rect GetViewBoundsInScreenCoordinates() const override;
void FireFocusEvent(BrowserAccessibility* node) override;
void FireBlinkEvent(ax::mojom::Event event_type,
BrowserAccessibility* node) override;
void FireGeneratedEvent(ui::AXEventGenerator::Event event_type,
BrowserAccessibility* node) override;
void FireWinAccessibilityEvent(LONG win_event, BrowserAccessibility* node);
void FireUiaAccessibilityEvent(LONG uia_event, BrowserAccessibility* node);
void FireUiaPropertyChangedEvent(LONG uia_property,
BrowserAccessibility* node);
void FireUiaStructureChangedEvent(StructureChangeType change_type,
BrowserAccessibility* node);
void FireUiaTextContainerEvent(LONG uia_event, BrowserAccessibility* node);
// Do event pre-processing
void BeforeAccessibilityEvents() override;
// Do event post-processing
void FinalizeAccessibilityEvents() override;
// Track this object and post a VISIBLE_DATA_CHANGED notification when
// its container scrolls.
// TODO(dmazzoni): remove once http://crbug.com/113483 is fixed.
void TrackScrollingObject(BrowserAccessibilityWin* node);
// Called when |accessible_hwnd_| is deleted by its parent.
void OnAccessibleHwndDeleted();
protected:
// AXTreeObserver methods.
void OnSubtreeWillBeDeleted(ui::AXTree* tree, ui::AXNode* node) override;
void OnNodeWillBeDeleted(ui::AXTree* tree, ui::AXNode* node) override;
void OnAtomicUpdateFinished(
ui::AXTree* tree,
bool root_changed,
const std::vector<ui::AXTreeObserver::Change>& changes) override;
private:
struct SelectionEvents {
std::vector<BrowserAccessibility*> added;
std::vector<BrowserAccessibility*> removed;
SelectionEvents();
~SelectionEvents();
};
using SelectionEventsMap = std::map<BrowserAccessibility*, SelectionEvents>;
using IsSelectedPredicate =
base::RepeatingCallback<bool(BrowserAccessibility*)>;
using FirePlatformSelectionEventsCallback =
base::RepeatingCallback<void(BrowserAccessibility*,
BrowserAccessibility*,
const SelectionEvents&)>;
static bool IsIA2NodeSelected(BrowserAccessibility* node);
static bool IsUIANodeSelected(BrowserAccessibility* node);
void FireIA2SelectionEvents(BrowserAccessibility* container,
BrowserAccessibility* only_selected_child,
const SelectionEvents& changes);
void FireUIASelectionEvents(BrowserAccessibility* container,
BrowserAccessibility* only_selected_child,
const SelectionEvents& changes);
static void HandleSelectedStateChanged(
SelectionEventsMap& selection_events_map,
BrowserAccessibility* node,
bool is_selected);
static void FinalizeSelectionEvents(
SelectionEventsMap& selection_events_map,
IsSelectedPredicate is_selected_predicate,
FirePlatformSelectionEventsCallback fire_platform_events_callback);
// Give BrowserAccessibilityManager::Create access to our constructor.
friend class BrowserAccessibilityManager;
// Keep track of if we got a "load complete" event but were unable to fire
// it because of no HWND, because otherwise JAWS can get very confused.
// TODO(dmazzoni): a better fix would be to always have an HWND.
// http://crbug.com/521877
bool load_complete_pending_;
// Since there could be multiple aria property changes on a node and we only
// want to fire UIA_AriaPropertiesPropertyId once for that node, we use the
// unordered set here to keep track of the unique nodes that had aria property
// changes, so we only fire the event once for every node.
std::unordered_set<BrowserAccessibility*> aria_properties_events_;
// Since there could be duplicate text selection changed events on a node
// raised from both FireBlinkEvent and FireGeneratedEvent, we use an unordered
// set here to keep track of the unique nodes that had
// UIA_Text_TextSelectionChangedEventId, so we only fire the event once for
// every node.
std::unordered_set<BrowserAccessibility*> text_selection_changed_events_;
// When the ignored state changes for a node, we only want to fire the
// events relevant to the ignored state change (e.g. show / hide).
// This set keeps track of what nodes should suppress superfluous events.
std::set<BrowserAccessibility*> ignored_changed_nodes_;
// Keep track of selection changes so we can optimize UIA event firing.
// Pointers are only stored for the duration of |OnAccessibilityEvents|, and
// the map is cleared in |FinalizeAccessibilityEvents|.
SelectionEventsMap ia2_selection_events_;
SelectionEventsMap uia_selection_events_;
DISALLOW_COPY_AND_ASSIGN(BrowserAccessibilityManagerWin);
};
} // namespace content
#endif // CONTENT_BROWSER_ACCESSIBILITY_BROWSER_ACCESSIBILITY_MANAGER_WIN_H_