| OLD | NEW |
| 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #ifndef CHROME_BROWSER_GTK_ACCESSIBILITY_EVENT_ROUTER_GTK_H_ | 5 #ifndef CHROME_BROWSER_GTK_ACCESSIBILITY_EVENT_ROUTER_GTK_H_ |
| 6 #define CHROME_BROWSER_GTK_ACCESSIBILITY_EVENT_ROUTER_GTK_H_ | 6 #define CHROME_BROWSER_GTK_ACCESSIBILITY_EVENT_ROUTER_GTK_H_ |
| 7 | 7 |
| 8 #include <gtk/gtk.h> | 8 #include <gtk/gtk.h> |
| 9 | 9 |
| 10 #include <string> | 10 #include <string> |
| (...skipping 18 matching lines...) Expand all Loading... |
| 29 } // namespace __gnu_cxx | 29 } // namespace __gnu_cxx |
| 30 | 30 |
| 31 // Struct to keep track of event listener hook ids to remove them later. | 31 // Struct to keep track of event listener hook ids to remove them later. |
| 32 struct InstalledHook { | 32 struct InstalledHook { |
| 33 InstalledHook(guint _signal_id, gulong _hook_id) | 33 InstalledHook(guint _signal_id, gulong _hook_id) |
| 34 : signal_id(_signal_id), hook_id(_hook_id) { } | 34 : signal_id(_signal_id), hook_id(_hook_id) { } |
| 35 guint signal_id; | 35 guint signal_id; |
| 36 gulong hook_id; | 36 gulong hook_id; |
| 37 }; | 37 }; |
| 38 | 38 |
| 39 // NOTE: This class is part of the Accessibility Extension API, which lets |
| 40 // extensions receive accessibility events. It's distinct from code that |
| 41 // implements platform accessibility APIs like MSAA or ATK. |
| 42 // |
| 39 // Singleton class that adds a signal emission hook to many gtk events, and | 43 // Singleton class that adds a signal emission hook to many gtk events, and |
| 40 // then sends an accessibility notification whenever a relevant event is | 44 // then sends an accessibility notification whenever a relevant event is |
| 41 // sent to an accessible control. | 45 // sent to an accessible control. |
| 42 // | 46 // |
| 43 // Gtk widgets are not accessible by default. When you register a root widget, | 47 // Gtk widgets are not accessible by default. When you register a root widget, |
| 44 // that widget and all of its descendants will start sending accessibility | 48 // that widget and all of its descendants will start sending accessibility |
| 45 // event notifications. You can then override the default behavior for | 49 // event notifications. You can then override the default behavior for |
| 46 // specific descendants using other methods. | 50 // specific descendants using other methods. |
| 47 // | 51 // |
| 48 // You can use Profile::PauseAccessibilityEvents to prevent a flurry | 52 // You can use Profile::PauseAccessibilityEvents to prevent a flurry |
| 49 // of accessibility events when a window is being created or initialized. | 53 // of accessibility events when a window is being created or initialized. |
| 50 class AccessibilityEventRouter { | 54 class AccessibilityEventRouterGtk { |
| 51 public: | 55 public: |
| 52 // Internal information about a particular widget to override the | 56 // Internal information about a particular widget to override the |
| 53 // information we get directly from gtk. | 57 // information we get directly from gtk. |
| 54 struct WidgetInfo { | 58 struct WidgetInfo { |
| 55 WidgetInfo() : ignore(false) { } | 59 WidgetInfo() : ignore(false) { } |
| 56 | 60 |
| 57 // If nonempty, will use this name instead of the widget's label. | 61 // If nonempty, will use this name instead of the widget's label. |
| 58 std::string name; | 62 std::string name; |
| 59 | 63 |
| 60 // If true, will ignore this widget and not send accessibility events. | 64 // If true, will ignore this widget and not send accessibility events. |
| 61 bool ignore; | 65 bool ignore; |
| 62 }; | 66 }; |
| 63 | 67 |
| 64 // Get the single instance of this class. | 68 // Get the single instance of this class. |
| 65 static AccessibilityEventRouter* GetInstance(); | 69 static AccessibilityEventRouterGtk* GetInstance(); |
| 66 | 70 |
| 67 // Start sending accessibility events for this widget and all of its | 71 // Start sending accessibility events for this widget and all of its |
| 68 // descendants. Notifications will go to the specified profile. | 72 // descendants. Notifications will go to the specified profile. |
| 69 void AddRootWidget(GtkWidget* root_widget, Profile* profile); | 73 void AddRootWidget(GtkWidget* root_widget, Profile* profile); |
| 70 | 74 |
| 71 // Stop sending accessibility events for this widget and all of its | 75 // Stop sending accessibility events for this widget and all of its |
| 72 // descendants. | 76 // descendants. |
| 73 void RemoveRootWidget(GtkWidget* root_widget); | 77 void RemoveRootWidget(GtkWidget* root_widget); |
| 74 | 78 |
| 75 // Don't send any events for this widget. | 79 // Don't send any events for this widget. |
| 76 void IgnoreWidget(GtkWidget* widget); | 80 void IgnoreWidget(GtkWidget* widget); |
| 77 | 81 |
| 78 // Use the following string as the name of this widget, instead of the | 82 // Use the following string as the name of this widget, instead of the |
| 79 // gtk label associated with the widget. | 83 // gtk label associated with the widget. |
| 80 void SetWidgetName(GtkWidget* widget, std::string name); | 84 void SetWidgetName(GtkWidget* widget, std::string name); |
| 81 | 85 |
| 82 // Forget all information about this widget. | 86 // Forget all information about this widget. |
| 83 void RemoveWidget(GtkWidget* widget); | 87 void RemoveWidget(GtkWidget* widget); |
| 84 | 88 |
| 85 // | 89 // |
| 86 // The following methods are only for use by gtk signal handlers. | 90 // The following methods are only for use by gtk signal handlers. |
| 87 // | 91 // |
| 88 | 92 |
| 89 // Returns true if this widget is a descendant of one of our registered | |
| 90 // root widgets and not in the set of ignored widgets. If |profile| is | |
| 91 // not null, return the profile where notifications associated with this | |
| 92 // widget should be sent. | |
| 93 bool IsWidgetAccessible(GtkWidget* widget, Profile** profile); | |
| 94 | |
| 95 // Return the name of a widget. | |
| 96 std::string GetWidgetName(GtkWidget* widget); | |
| 97 | |
| 98 // Called by the signal handler. Checks the type of the widget and | 93 // Called by the signal handler. Checks the type of the widget and |
| 99 // calls one of the more specific Send*Notification methods, below. | 94 // calls one of the more specific Send*Notification methods, below. |
| 100 void DispatchAccessibilityNotification( | 95 void DispatchAccessibilityNotification( |
| 101 GtkWidget* widget, NotificationType type); | 96 GtkWidget* widget, NotificationType type); |
| 102 | 97 |
| 103 // Post a task to call DispatchAccessibilityNotification the next time | 98 // Post a task to call DispatchAccessibilityNotification the next time |
| 104 // through the event loop. | 99 // through the event loop. |
| 105 void PostDispatchAccessibilityNotification( | 100 void PostDispatchAccessibilityNotification( |
| 106 GtkWidget* widget, NotificationType type); | 101 GtkWidget* widget, NotificationType type); |
| 107 | 102 |
| 103 private: |
| 104 AccessibilityEventRouterGtk(); |
| 105 virtual ~AccessibilityEventRouterGtk(); |
| 106 |
| 107 // Given a widget, determine if it's the descendant of a root widget |
| 108 // that's mapped to a profile and if so, if it's marked as accessible. |
| 109 void FindWidget(GtkWidget* widget, Profile** profile, bool* is_accessible); |
| 110 |
| 111 // Return the name of a widget. |
| 112 std::string GetWidgetName(GtkWidget* widget); |
| 113 |
| 108 // Each of these methods constructs an AccessibilityControlInfo object | 114 // Each of these methods constructs an AccessibilityControlInfo object |
| 109 // and sends a notification of a specific accessibility event. | 115 // and sends a notification of a specific accessibility event. |
| 110 void SendButtonNotification( | 116 void SendButtonNotification( |
| 111 GtkWidget* widget, NotificationType type, Profile* profile); | 117 GtkWidget* widget, NotificationType type, Profile* profile); |
| 112 void SendCheckboxNotification( | 118 void SendCheckboxNotification( |
| 113 GtkWidget* widget, NotificationType type, Profile* profile); | 119 GtkWidget* widget, NotificationType type, Profile* profile); |
| 114 void SendComboBoxNotification( | 120 void SendComboBoxNotification( |
| 115 GtkWidget* widget, NotificationType type, Profile* profile); | 121 GtkWidget* widget, NotificationType type, Profile* profile); |
| 116 void SendListBoxNotification( | 122 void SendListBoxNotification( |
| 117 GtkWidget* widget, NotificationType type, Profile* profile); | 123 GtkWidget* widget, NotificationType type, Profile* profile); |
| 124 void SendMenuItemNotification( |
| 125 GtkWidget* widget, NotificationType type, Profile* profile); |
| 118 void SendRadioButtonNotification( | 126 void SendRadioButtonNotification( |
| 119 GtkWidget* widget, NotificationType type, Profile* profile); | 127 GtkWidget* widget, NotificationType type, Profile* profile); |
| 120 void SendTabNotification( | 128 void SendTabNotification( |
| 121 GtkWidget* widget, NotificationType type, Profile* profile); | 129 GtkWidget* widget, NotificationType type, Profile* profile); |
| 122 void SendTextBoxNotification( | 130 void SendTextBoxNotification( |
| 123 GtkWidget* widget, NotificationType type, Profile* profile); | 131 GtkWidget* widget, NotificationType type, Profile* profile); |
| 124 | 132 |
| 125 void InstallEventListeners(); | 133 void InstallEventListeners(); |
| 126 void RemoveEventListeners(); | 134 void RemoveEventListeners(); |
| 127 | 135 |
| 128 // Start and stop listening to signals. | 136 // Start and stop listening to signals. |
| 129 void StartListening(); | 137 void StartListening(); |
| 130 void StopListening(); | 138 void StopListening(); |
| 131 | 139 |
| 132 private: | |
| 133 AccessibilityEventRouter(); | |
| 134 virtual ~AccessibilityEventRouter(); | |
| 135 | |
| 136 // Add a signal emission hook for one particular signal name and | 140 // Add a signal emission hook for one particular signal name and |
| 137 // widget type, and save the hook_id in installed_hooks so we can | 141 // widget type, and save the hook_id in installed_hooks so we can |
| 138 // remove it later. | 142 // remove it later. |
| 139 void InstallEventListener( | 143 void InstallEventListener( |
| 140 const char *signal_name, | 144 const char *signal_name, |
| 141 GType widget_type, | 145 GType widget_type, |
| 142 GSignalEmissionHook hook_func); | 146 GSignalEmissionHook hook_func); |
| 143 | 147 |
| 144 friend struct DefaultSingletonTraits<AccessibilityEventRouter>; | 148 friend struct DefaultSingletonTraits<AccessibilityEventRouterGtk>; |
| 145 | 149 |
| 146 // The set of all root widgets; only descendants of these will generate | 150 // The set of all root widgets; only descendants of these will generate |
| 147 // accessibility notifications. | 151 // accessibility notifications. |
| 148 base::hash_map<GtkWidget*, Profile*> root_widget_profile_map_; | 152 base::hash_map<GtkWidget*, Profile*> root_widget_profile_map_; |
| 149 | 153 |
| 150 // Extra information about specific widgets. | 154 // Extra information about specific widgets. |
| 151 base::hash_map<GtkWidget*, WidgetInfo> widget_info_map_; | 155 base::hash_map<GtkWidget*, WidgetInfo> widget_info_map_; |
| 152 | 156 |
| 153 // Installed event listener hook ids so we can remove them later. | 157 // Installed event listener hook ids so we can remove them later. |
| 154 std::vector<InstalledHook> installed_hooks_; | 158 std::vector<InstalledHook> installed_hooks_; |
| 155 | 159 |
| 156 // True if we are currently listening to signals. | 160 // True if we are currently listening to signals. |
| 157 bool listening_; | 161 bool listening_; |
| 158 | 162 |
| 159 // Used to schedule invocations of StartListening(). | 163 // The profile associated with the most recent window event - used to |
| 160 ScopedRunnableMethodFactory<AccessibilityEventRouter> method_factory_; | 164 // figure out where to route a few events that can't be directly traced |
| 165 // to a window with a profile (like menu events). |
| 166 Profile* most_recent_profile_; |
| 167 |
| 168 // Used to schedule invocations of StartListening() and to defer handling |
| 169 // of some events until the next time through the event loop. |
| 170 ScopedRunnableMethodFactory<AccessibilityEventRouterGtk> method_factory_; |
| 161 }; | 171 }; |
| 162 | 172 |
| 163 #endif // CHROME_BROWSER_GTK_ACCESSIBILITY_EVENT_ROUTER_GTK_H_ | 173 #endif // CHROME_BROWSER_GTK_ACCESSIBILITY_EVENT_ROUTER_GTK_H_ |
| OLD | NEW |