popup menu accessibility
(cherry picked from commit 2b6c2f0ecaa8953ae7f126b562e58b87f98d6d3f)
This commit is contained in:
		
							parent
							
								
									d040615c6e
								
							
						
					
					
						commit
						c783578bdf
					
				
					 2 changed files with 69 additions and 10 deletions
				
			
		|  | @ -62,15 +62,15 @@ | ||||||
|                                         <ColumnDefinition Width="Auto" /> |                                         <ColumnDefinition Width="Auto" /> | ||||||
|                                     </Grid.ColumnDefinitions> |                                     </Grid.ColumnDefinitions> | ||||||
|                                     <TextBlock Grid.Column="0" x:Name="ZoomText" HorizontalAlignment="Left" Margin="10,0" VerticalAlignment="Center" /> |                                     <TextBlock Grid.Column="0" x:Name="ZoomText" HorizontalAlignment="Left" Margin="10,0" VerticalAlignment="Center" /> | ||||||
|                                     <Button Grid.Column="1" x:Name="ZoomInButton" Margin="5" Padding="5" Template="{StaticResource BrowserButton}"> |                                     <Button Grid.Column="1" x:Name="ZoomInButton" Margin="5" Padding="5" Template="{StaticResource BrowserButton}" TabIndex="30"> | ||||||
|                                         <fa:ImageAwesome Icon="SearchPlus" /> |                                         <fa:ImageAwesome Icon="SearchPlus" /> | ||||||
|                                     </Button> |                                     </Button> | ||||||
|                                     <Button Grid.Column="2" x:Name="ZoomResetButton" Margin="5" Padding="5" Template="{StaticResource BrowserButton}"> |                                     <Button Grid.Column="2" x:Name="ZoomResetButton" Margin="5" Padding="5" Template="{StaticResource BrowserButton}" TabIndex="31"> | ||||||
|                                         <Viewbox Stretch="Uniform"> |                                         <Viewbox Stretch="Uniform"> | ||||||
|                                             <TextBlock x:Name="ZoomLevel" /> |                                             <TextBlock x:Name="ZoomLevel" /> | ||||||
|                                         </Viewbox> |                                         </Viewbox> | ||||||
|                                     </Button> |                                     </Button> | ||||||
|                                     <Button Grid.Column="3" x:Name="ZoomOutButton" Margin="5" Padding="5" Template="{StaticResource BrowserButton}"> |                                     <Button Grid.Column="3" x:Name="ZoomOutButton" Margin="5" Padding="5" Template="{StaticResource BrowserButton}" TabIndex="32"> | ||||||
|                                         <fa:ImageAwesome Icon="SearchMinus" /> |                                         <fa:ImageAwesome Icon="SearchMinus" /> | ||||||
|                                     </Button> |                                     </Button> | ||||||
|                                 </Grid> |                                 </Grid> | ||||||
|  | @ -80,7 +80,7 @@ | ||||||
|                                         <ColumnDefinition Width="Auto" /> |                                         <ColumnDefinition Width="Auto" /> | ||||||
|                                     </Grid.ColumnDefinitions> |                                     </Grid.ColumnDefinitions> | ||||||
|                                     <TextBlock Grid.Column="0" x:Name="FindMenuText" HorizontalAlignment="Left" Margin="10,0" VerticalAlignment="Center" /> |                                     <TextBlock Grid.Column="0" x:Name="FindMenuText" HorizontalAlignment="Left" Margin="10,0" VerticalAlignment="Center" /> | ||||||
|                                     <Button Grid.Column="1" x:Name="FindMenuButton" Margin="5" Padding="5" Template="{StaticResource BrowserButton}"> |                                     <Button Grid.Column="1" x:Name="FindMenuButton" Margin="5" Padding="5" Template="{StaticResource BrowserButton}" TabIndex="33"> | ||||||
|                                         <fa:ImageAwesome Icon="Search" /> |                                         <fa:ImageAwesome Icon="Search" /> | ||||||
|                                     </Button> |                                     </Button> | ||||||
|                                 </Grid> |                                 </Grid> | ||||||
|  | @ -90,7 +90,7 @@ | ||||||
|                                         <ColumnDefinition Width="Auto" /> |                                         <ColumnDefinition Width="Auto" /> | ||||||
|                                     </Grid.ColumnDefinitions> |                                     </Grid.ColumnDefinitions> | ||||||
|                                     <TextBlock Grid.Column="0" x:Name="DeveloperConsoleText" HorizontalAlignment="Left" Margin="10,0" VerticalAlignment="Center" /> |                                     <TextBlock Grid.Column="0" x:Name="DeveloperConsoleText" HorizontalAlignment="Left" Margin="10,0" VerticalAlignment="Center" /> | ||||||
|                                     <Button Grid.Column="1" x:Name="DeveloperConsoleButton" Margin="5" Padding="5" Template="{StaticResource BrowserButton}"> |                                     <Button Grid.Column="1" x:Name="DeveloperConsoleButton" Margin="5" Padding="5" Template="{StaticResource BrowserButton}" TabIndex="34"> | ||||||
|                                         <fa:ImageAwesome Icon="Wrench" /> |                                         <fa:ImageAwesome Icon="Wrench" /> | ||||||
|                                     </Button> |                                     </Button> | ||||||
|                                 </Grid> |                                 </Grid> | ||||||
|  |  | ||||||
|  | @ -8,6 +8,7 @@ | ||||||
| 
 | 
 | ||||||
| using System; | using System; | ||||||
| using System.ComponentModel; | using System.ComponentModel; | ||||||
|  | using System.Reflection; | ||||||
| using System.Threading.Tasks; | using System.Threading.Tasks; | ||||||
| using System.Windows; | using System.Windows; | ||||||
| using System.Windows.Controls.Primitives; | using System.Windows.Controls.Primitives; | ||||||
|  | @ -39,6 +40,7 @@ namespace SafeExamBrowser.UserInterface.Desktop.Windows | ||||||
| 		private WindowClosedEventHandler closed; | 		private WindowClosedEventHandler closed; | ||||||
| 		private WindowClosingEventHandler closing; | 		private WindowClosingEventHandler closing; | ||||||
| 		private bool browserControlGetsFocusFromTaskbar = false; | 		private bool browserControlGetsFocusFromTaskbar = false; | ||||||
|  | 		private IInputElement tabKeyDownFocusElement = null; | ||||||
| 
 | 
 | ||||||
| 		private WindowSettings WindowSettings | 		private WindowSettings WindowSettings | ||||||
| 		{ | 		{ | ||||||
|  | @ -205,9 +207,10 @@ namespace SafeExamBrowser.UserInterface.Desktop.Windows | ||||||
| 
 | 
 | ||||||
| 		private void BrowserWindow_KeyDown(object sender, KeyEventArgs e) | 		private void BrowserWindow_KeyDown(object sender, KeyEventArgs e) | ||||||
| 		{ | 		{ | ||||||
| 			if (e.Key == Key.Tab && Toolbar.IsKeyboardFocusWithin) | 			if (e.Key == Key.Tab) | ||||||
| 			{ | 			{ | ||||||
| 				if ((Keyboard.Modifiers & ModifierKeys.Shift) == ModifierKeys.Shift) | 				var hasShift = (Keyboard.Modifiers & ModifierKeys.Shift) == ModifierKeys.Shift; | ||||||
|  | 				if (Toolbar.IsKeyboardFocusWithin && hasShift) | ||||||
| 				{ | 				{ | ||||||
| 					var firstActiveElementInToolbar = Toolbar.PredictFocus(FocusNavigationDirection.Right); | 					var firstActiveElementInToolbar = Toolbar.PredictFocus(FocusNavigationDirection.Right); | ||||||
| 					if (firstActiveElementInToolbar is System.Windows.UIElement) | 					if (firstActiveElementInToolbar is System.Windows.UIElement) | ||||||
|  | @ -220,6 +223,12 @@ namespace SafeExamBrowser.UserInterface.Desktop.Windows | ||||||
| 						} | 						} | ||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
|  | 
 | ||||||
|  | 				tabKeyDownFocusElement = FocusManager.GetFocusedElement(this); | ||||||
|  | 			} | ||||||
|  | 			else | ||||||
|  | 			{ | ||||||
|  | 				tabKeyDownFocusElement = null; | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | @ -242,9 +251,11 @@ namespace SafeExamBrowser.UserInterface.Desktop.Windows | ||||||
| 
 | 
 | ||||||
| 			if (e.Key == Key.Tab) | 			if (e.Key == Key.Tab) | ||||||
| 			{ | 			{ | ||||||
| 				if (BrowserControlHost.IsFocused && (Keyboard.Modifiers & ModifierKeys.Control) == ModifierKeys.Control) | 				var hasCtrl = (Keyboard.Modifiers & ModifierKeys.Control) == ModifierKeys.Control; | ||||||
|  | 				var hasShift = (Keyboard.Modifiers & ModifierKeys.Shift) == ModifierKeys.Shift; | ||||||
|  | 				if (BrowserControlHost.IsFocused && hasCtrl) | ||||||
| 				{ | 				{ | ||||||
| 					if (Findbar.Visibility == Visibility.Hidden || (Keyboard.Modifiers & ModifierKeys.Shift) == ModifierKeys.Shift) | 					if (Findbar.Visibility == Visibility.Hidden || hasShift) | ||||||
| 					{ | 					{ | ||||||
| 						Toolbar.Focus(); | 						Toolbar.Focus(); | ||||||
| 					} | 					} | ||||||
|  | @ -253,9 +264,51 @@ namespace SafeExamBrowser.UserInterface.Desktop.Windows | ||||||
| 						Findbar.Focus(); | 						Findbar.Focus(); | ||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
|  | 				else if (MenuPopup.IsKeyboardFocusWithin) | ||||||
|  | 				{ | ||||||
|  | 					var focusedElement = FocusManager.GetFocusedElement(this); | ||||||
|  | 					var focusedControl = focusedElement as System.Windows.Controls.Control; | ||||||
|  | 					var prevFocusedControl = tabKeyDownFocusElement as System.Windows.Controls.Control; | ||||||
|  | 					if (focusedControl != null && prevFocusedControl != null) | ||||||
|  | 					{ | ||||||
|  | 						//var commonAncestor = focusedControl.FindCommonVisualAncestor(prevFocusedControl); | ||||||
|  | 						//var nextTab = GetNextTab(MenuPopup, this, true); | ||||||
|  | 						if (!hasShift && focusedControl.TabIndex < prevFocusedControl.TabIndex) | ||||||
|  | 						{ | ||||||
|  | 							MenuPopup.IsOpen = false; | ||||||
|  | 							FocusBrowser(); | ||||||
|  | 						} | ||||||
|  | 						else if (hasShift && focusedControl.TabIndex > prevFocusedControl.TabIndex) | ||||||
|  | 						{ | ||||||
|  | 							MenuPopup.IsOpen = false; | ||||||
|  | 							MenuButton.Focus(); | ||||||
|  | 						} | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | 		/// <summary> | ||||||
|  | 		/// Get next tab order element. Copied from https://stackoverflow.com/questions/5756448/in-wpf-how-can-i-get-the-next-control-in-the-tab-order | ||||||
|  | 		/// </summary> | ||||||
|  | 		/// <param name="e">The element to get next tab order</param> | ||||||
|  | 		/// <param name="container">The container element owning 'e'. Make sure this is a container of 'e'.</param> | ||||||
|  | 		/// <param name="goDownOnly">True if search only itself and inside of 'container'; otherwise false. | ||||||
|  | 		/// If true and next tab order element is outside of 'container', result in null.</param> | ||||||
|  | 		/// <returns>Next tab order element or null if not found</returns> | ||||||
|  | 		public DependencyObject GetNextTab(DependencyObject e, DependencyObject container, bool goDownOnly) | ||||||
|  | 		{ | ||||||
|  | 			var navigation = typeof(FrameworkElement) | ||||||
|  | 				.GetProperty("KeyboardNavigation", BindingFlags.NonPublic | BindingFlags.Static) | ||||||
|  | 				.GetValue(null); | ||||||
|  | 
 | ||||||
|  | 			var method = navigation | ||||||
|  | 				.GetType() | ||||||
|  | 				.GetMethod("GetNextTab", BindingFlags.NonPublic | BindingFlags.Instance); | ||||||
|  | 
 | ||||||
|  | 			return method.Invoke(navigation, new object[] { e, container, goDownOnly }) as DependencyObject; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
| 		private void BrowserWindow_Loaded(object sender, RoutedEventArgs e) | 		private void BrowserWindow_Loaded(object sender, RoutedEventArgs e) | ||||||
| 		{ | 		{ | ||||||
| 			Handle = new WindowInteropHelper(this).Handle; | 			Handle = new WindowInteropHelper(this).Handle; | ||||||
|  | @ -359,7 +412,7 @@ namespace SafeExamBrowser.UserInterface.Desktop.Windows | ||||||
| 			ForwardButton.Click += (o, args) => ForwardNavigationRequested?.Invoke(); | 			ForwardButton.Click += (o, args) => ForwardNavigationRequested?.Invoke(); | ||||||
| 			HomeButton.Click += (o, args) => HomeNavigationRequested?.Invoke(); | 			HomeButton.Click += (o, args) => HomeNavigationRequested?.Invoke(); | ||||||
| 			Loaded += BrowserWindow_Loaded; | 			Loaded += BrowserWindow_Loaded; | ||||||
| 			MenuButton.Click += (o, args) => MenuPopup.IsOpen = !MenuPopup.IsOpen; | 			MenuButton.Click += MenuButton_Click; | ||||||
| 			MenuButton.MouseLeave += (o, args) => Task.Delay(250).ContinueWith(_ => Dispatcher.Invoke(() => MenuPopup.IsOpen = MenuPopup.IsMouseOver)); | 			MenuButton.MouseLeave += (o, args) => Task.Delay(250).ContinueWith(_ => Dispatcher.Invoke(() => MenuPopup.IsOpen = MenuPopup.IsMouseOver)); | ||||||
| 			MenuPopup.CustomPopupPlacementCallback = new CustomPopupPlacementCallback(Popup_PlacementCallback); | 			MenuPopup.CustomPopupPlacementCallback = new CustomPopupPlacementCallback(Popup_PlacementCallback); | ||||||
| 			MenuPopup.MouseLeave += (o, args) => Task.Delay(250).ContinueWith(_ => Dispatcher.Invoke(() => MenuPopup.IsOpen = MenuPopup.IsMouseOver)); | 			MenuPopup.MouseLeave += (o, args) => Task.Delay(250).ContinueWith(_ => Dispatcher.Invoke(() => MenuPopup.IsOpen = MenuPopup.IsMouseOver)); | ||||||
|  | @ -381,6 +434,12 @@ namespace SafeExamBrowser.UserInterface.Desktop.Windows | ||||||
| 			BrowserControlHost.GotKeyboardFocus += BrowserControlHost_GotKeyboardFocus; | 			BrowserControlHost.GotKeyboardFocus += BrowserControlHost_GotKeyboardFocus; | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | 		private void MenuButton_Click(object sender, RoutedEventArgs e) | ||||||
|  | 		{ | ||||||
|  | 			MenuPopup.IsOpen = !MenuPopup.IsOpen; | ||||||
|  | 			ZoomInButton.Focus(); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
| 		private void BrowserControlHost_GotKeyboardFocus(object sender, KeyboardFocusChangedEventArgs e) | 		private void BrowserControlHost_GotKeyboardFocus(object sender, KeyboardFocusChangedEventArgs e) | ||||||
| 		{ | 		{ | ||||||
| 			var forward = !this.browserControlGetsFocusFromTaskbar; | 			var forward = !this.browserControlGetsFocusFromTaskbar; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		
		Reference in a new issue
	
	 Jonas Sourlier
						Jonas Sourlier