WPF Reset Focus on Button Click
Solution 1
The problem is that the toolbar places your button in a different FocusManager.FocusScope
. That means that both the Button
and the TextBox
can receive logical focus at the same time, each in its own scope. This is normally a good thing, since you usually don't want to lose focus in your main window area when you select menu items and ToolBar buttons, but in your case it is preventing what you are doing from working.
Although you could override the FocusManager.IsFocusScope
property on the toolbar and get the effect you want, this is probably not the best plan since it would make all the other toolbar buttons also steal focus from your main window area.
Instead you could use one of several easy solutions:
- Put your button outside the Toolbar
- Add a Focusable="true" control to your main window area and focus it when the button is clicked
- Manually force the update by calling
textBox.GetBindingExpression(TextBox.TextProperty).UpdateSource()
- Temporarily set Focusable="true" on a control in the main window, set focus to it, then immediately set Focusable="false" again
Solution 2
I encountered a similar issue. I need to unfocus a textbox when enter is pressed. I end up with this code:
var scope = FocusManager.GetFocusScope(elem); // elem is the UIElement to unfocus
FocusManager.SetFocusedElement(scope, null); // remove logical focus
Keyboard.ClearFocus(); // remove keyboard focus
I think it is cleaner than creating dummy controls and it is reusable. I'm not confident with this solution though. But it seems work well.
Solution 3
I had an issue leaving a calendar and needing to click a button twice (WPF, .Net 5.0), I tried some of the suggested solutions above but no luck, however this worked - quoted from someone on the Microsoft site:
"My solution to this problem was to insert into the main form, but it's probably overkill."
protected override void OnPreviewMouseUp(MouseButtonEventArgs e)
{
base.OnPreviewMouseUp(e);
if (Mouse.Captured is Calendar || Mouse.Captured is System.Windows.Controls.Primitives.CalendarItem)
{
Mouse.Capture(null);
}
}
HaxElit
Updated on May 22, 2021Comments
-
HaxElit almost 3 years
I have a
TextBox
and aToolBar
with aButton
. If I'm typing in theTextBox
and I click theButton
I want theTextBox
to loseFocus
so the binding gets updated. I don't want to add aUpdateSourceTrigger=PropertyChanged
to myTextBox
. But instead when I click on theButton
I resetFocus
to the main window so what ever I'm on losesFocus
and updates the bindings.I've tried adding a
OnClick
to the button with the following, but it doesn't seem to work:private void Button_Click(object sender, RoutedEventArgs e) { FocusManager.SetFocusedElement(this, null); }
Any tips would be appreciated.
Thanks, Raul
-
HaxElit over 14 yearsI've tried a Keyboard.Focus(mainWindow); but that doesn't seem to be changing the focus either. It remains on the textbox or what ever element was selected. For now I just explicitly force the element to UpdateSource() but it doesn't seem like a long term sustainable solution. I'll try adding a ghost control to set focus to.
-
Ray Burns over 14 years
Keyboard.Focus(mainWindow)
will only work if "mainWindow" is Focusable (by default it isn't) and if it is in the sameFocusScope
as theTextBox
. Try creating aFocusable="true"
control in the same container as the TextBox and focusing that. The other control can be a simple<Control Focusable="true" IsTabStop="false"/>
. SinceControl
has no default template it will be invisible. -
Aphex over 12 yearsThanks for mentioning
FocusManager.IsFocusScope
. In my specific scenario, I wanted to steal focus from my main window area, so I just setFocusManager.IsFocusScope="False"
on my Toolbar. -
Dave about 11 yearsThis worked for me when nothing else would force a DatePicker to parse the entered text on clicking the save button in toolbar. The code below uses the view (MVVM) as this.owner, called from ViewModel. var t =(DependencyObject)FocusManager.GetFocusedElement(this.Owner); var scope = FocusManager.GetFocusScope(t); FocusManager.SetFocusedElement(scope, null);