实在是百思不得其解,发在精华区,以期园中高手解答,罪过罪过(关于WPF的Command的思考)
WPF当中的Command,我就不再解释了。这篇文章需要读者对Command模型有所了解,才能做探讨。
我到现在不能睡觉的罪魁祸首出在MSDN当中的一句话。
前三种Button Base,MenuItem和Hyperlink,它们都是通过点击,然后触发了Command。而InputBinding则是通过和它关联的InputGesture执行,从而触发Command。
我到现在不能睡觉的第二大祸根出在我的好奇心,真是好奇害死猫。
我只是想通过Reflector看看,是否诚如MSDN所言。“前三种Button Base,MenuItem和Hyperlink,它们都是通过点击,然后触发了Command” 很快得到了认证。以ButtonBase为例。在它的OnClick函数里找到了如下源码:
protected virtual void OnClick()
{
RoutedEventArgs e = new RoutedEventArgs(ClickEvent, this);
base.RaiseEvent(e);
CommandHelpers.ExecuteCommandSource(this);
}
注意最后一句话,CommandHelpers.ExecuteCommandSource(this);这一句话正是OnClick会触发Command的明证。
有好奇者可以继续往下剖析源码,我就暂且不表了。
可是第二个我伤破脑筋也没有搞明白。“而InputBinding则是通过和它关联的InputGesture执行,从而触发Command”
关键性的问题是InputGesture什么时候执行?
当然首先我们是通过键盘和鼠标执行,以如下情景为例。
<Window x:Class="mumu_command.Window1"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="Window1" Height="300" Width="300">
<Grid>
<TextBox AcceptsReturn="True" />
</Grid>
</Window>
就是一个Grid里面放置了一个TextBox,当你运行程序的时候,按下Ctrl+C。就会神奇地进行复制,而不需写任何代码。这当中的原因是TextBox内置了对ApplicationCommands.Copy命令的支持。如果没有什么InputGestrue的话,我们也是可以完全捕获到Ctrl+C这般的快捷键,见如下代码:
private void TextBox_PreviewKeyDown(object sender, KeyEventArgs e)
{
bool isCtrldown = Keyboard.IsKeyDown(Key.LeftCtrl) | Keyboard.IsKeyDown(Key.RightCtrl);
if (isCtrldown)
{
if (e.Key == Key.C)
{
MessageBox.Show("复制");
e.Handled = true;
}
else if (e.Key == Key.V)
{
MessageBox.Show("粘贴");
e.Handled = true;
}
else if (e.Key == Key.X)
{
MessageBox.Show("剪切");
e.Handled = true;
}
}
}
这是没有什么劳子InputGestrue或者说是KeyGestrue的情况。我自己也可以通过previewkeydown事件来捕获到Ctrl+C这般的快捷键,然后我也可以执行Command的Execute。我本想WPF的思路和我一样。但是通过Reflector,去查找
TextBox没有OnPreviewKeyDown函数.......
父类TextBoxBase依旧没有.....
UIElement有了,但里面全然是空......
于是再去找他们的keydown事件的处理函数,依然一无所获。
MSDN只说InputGesture会执行,但是它全然不告诉我何时执行。害我这个苦命人,打着哈欠,流着眼泪,在这儿写博客,诉苦.......
罗嗦了这么多,不知道高手们看懂了我的问题没有。以拷贝命令为例,当我们在TextBox按下Ctrl+C快捷键是,它是否也是通过键盘事件,然后再转到命令?如果不是
它的第一道门在何处,整个来龙去脉是如何?还请高手们指点?