检测从一组WPF控件丢失的焦点

Jon*_*rop 6 c# wpf f# focus

我正在使用具有可编辑内容的代码构建GUI.当用户点击静态显示时,控件将换出允许编辑的控件.在某些情况下,显示控件(例如Label)被替换为一组控件,例如a ComboBoxTextBox.

我想检测何时从我的可编辑控件组中丢失焦点,以便将界面从编辑器切换回该项的显示表示.

例如,我可能在树中有一个GUI Panel1(Panel2(Button1, Button2), Button3),我想检测焦点丢失的时间Panel2.我尝试了以下(F#代码):

open System.Windows

let button1 = Controls.Button(Content="1")
let button2 = Controls.Button(Content="2")
let button3 = Controls.Button(Content="3")

[<System.STAThreadAttribute>]
do
  let panel1 = Controls.StackPanel()
  let panel2 = Controls.StackPanel()
  panel2.Children.Add button1 |> ignore
  panel2.Children.Add button2 |> ignore
  panel1.Children.Add panel2 |> ignore
  panel1.Children.Add button3 |> ignore
  panel2.LostFocus.Add(fun _ ->
    printfn "Panel2 lost focus")
  Application().Run(Window(Content=panel1))
  |> ignore
Run Code Online (Sandbox Code Playgroud)

panel2.LostFocus在触发事件button3被点击后,button2已经被点击,符合市场预期,因为焦点,搬出panel1button3.但是,即使从未失去焦点,button2点击后点击也会触发.button1panel2

阅读有关WPF焦点的MSDN文档我尝试添加:

Input.FocusManager.SetIsFocusScope(panel2, true)
Run Code Online (Sandbox Code Playgroud)

但这实际上使问题变得更糟!现在panel2.LostFocus只有当焦点从一个孩子转移panel2到另一个孩子而不是panel2实际失去焦点时才会触发事件.

我该如何获得理想的效果?

Jon*_*rop 6

感谢Ian Voyce在Twitter上我能够获得使用该IsKeyboardFocusWithinChanged事件所需的功能.这是一个演示:

open System.Windows

let Button x =
  Controls.Button(Content=x, Width=64.0, Margin=Thickness 3.0)

let Panel ctrls =
  let panel = Controls.StackPanel()
  for ctrl in ctrls do
    panel.Children.Add ctrl
    |> ignore
  panel

let label = Controls.Label(Content="Edit 12")
let button1 = Button "1"
let button2 = Button "2"
let button3 = Button "3"

let panel = Panel[button1; button2]

[<System.STAThreadAttribute>]
do
  label.HorizontalContentAlignment <- HorizontalAlignment.Center
  label.MouseLeftButtonDown.Add(fun _ ->
    label.Content <- panel)
  panel.IsKeyboardFocusWithinChanged.Add(fun e ->
    if not(unbox e.NewValue) then
      label.Content <- "Edit 12")
  Application().Run(Window(Content=Panel[label :> UIElement; button3 :> UIElement]))
  |> ignore
Run Code Online (Sandbox Code Playgroud)