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…
Reference in a new issue