实在是百思不得其解,发在精华区,以期园中高手解答,罪过罪过(关于WPF的Command的思考)

WPF当中的Command,我就不再解释了。这篇文章需要读者对Command模型有所了解,才能做探讨。
我到现在不能睡觉的罪魁祸首出在MSDN当中的一句话。

"ButtonBase, MenuItem, and Hyperlink invoke a command when they are clicked, and an InputBinding invokes a command when the InputGesture associated with it is performed. "
这是MSDN在解释Command Source,它列举了四种可以成为Command Source的类型,包括Button Base,MenuItem,Hyperlink和InputBinding。
前三种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快捷键是,它是否也是通过键盘事件,然后再转到命令?如果不是
它的第一道门在何处,整个来龙去脉是如何?还请高手们指点?


 

posted @ 2009-10-14 02:01  helloj2ee  阅读(3242)  评论(23编辑  收藏  举报