我正在使用具有可编辑内容的代码构建GUI。当用户点击静态显示时,控件将换出允许编辑的控件。在某些情况下,显示控件(例如Label
)会换出一组控件,例如ComboBox
和TextBox
。
我想检测何时从我的可编辑控件组中丢失焦点,以便将界面从编辑器切换回该项目的显示表示。
例如,我可能在Panel1(Panel2(Button1, Button2), Button3)
这样的树中有一个GUI,我想检测焦点何时从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
在panel2.LostFocus
被点击后点击button3
时会触发button2
事件,因为焦点已移出panel1
到button3
}。但是,即使button2
永远不会失去焦点,在button1
被点击后点击panel2
时也会触发它。
阅读关于WPF焦点的MSDN文档我尝试添加:
Input.FocusManager.SetIsFocusScope(panel2, true)
但实际上这使问题变得更糟!现在panel2.LostFocus
事件仅在焦点从panel2
的一个孩子转移到另一个孩子而不是panel2
实际失去焦点时触发。
我应该如何获得理想的效果?
答案 0 :(得分:6)
感谢Twitter上的Ian Voyce,我能够使用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