WinForm控件开发总结(十二)-----让控件处理导航键

简介:
 最近真的真的太忙了,以至于一个多月都没哟更新我的 blog 。昨天晚上,一个网上的朋友看了我的 ToolBox 的文章,问我一个问题,他说如何让 ToolBox 控件也能响应键盘操作,也就是用 Up down 按键来选择工具箱控件里的 Item ,他添加了键盘事件,但是不起作用。一开始做这个控件的时候也只是演示一下控件的制作过程,只用了很短的时间做了一个,只考虑了用鼠标选取,没有考虑键盘操作,我想要添加键盘操作无非重载 KeyDown 事件,针对 Up Down 做一些响应就可以了。可是添加了重载了 OnKeyDown 事件后,结果和那位朋友所说的一样,没有任何作用,我设了断点,调试了一下,发现 KeyDown 根本捕获不到 Up Down 按键的点击,是什么原因呢,是不是忘记设控件的风格以便让它能够获得焦点?于是,我使用了语句:
   
SetStyle(ControlStyles.Selectable,  true );
依然没有效果,当我们在控件上按下Down键的时候,另一个控件获得了焦点。这时UpDown按钮只是起到了导航的作用就像Tab键一样。
      接下来,我在测试工程的窗体上放置了一个ListBox控件做一个对比,其实ToolBoxListBox在界面表现上有相似之处,就是都有子Item,并且在ListBox上点击Down是起作用的,ListBox并没有失去焦点,这说明这时UpDown按键没有成为导航键。我想Windows一定是对默认的导航键UpDown,Left,Right有默认的处理,除非你希望你的控件希望自己处理这些键。用反汇编工具看了一下ListBoxControl控件的源代码,发现一个有趣的函数:
protected   override   bool  IsInputKey(Keys keyData)
{
    
if ((keyData & Keys.Alt) == Keys.Alt)
    
{
        
return false;
    }

    
switch ((keyData & Keys.KeyCode))
    
{
        
case Keys.Prior:
        
case Keys.Next:
        
case Keys.End:
        
case Keys.Home:
            
return true;
    }

    
return base.IsInputKey(keyData);
}


在这里面,ListBoxControl允许PriorNextEndHome成为有效的输入键,接着一路跟下去,看看WinForm控件的基类Control的这个函数是如何处理的:
[UIPermission(SecurityAction.InheritanceDemand, Window = UIPermissionWindow.AllWindows)]
protected   virtual   bool  IsInputKey(Keys keyData)
{
    
if ((keyData & Keys.Alt) != Keys.Alt)
    
{
        
int num = 4;
        
switch ((keyData & Keys.KeyCode))
        
{
            
case Keys.Left:
            
case Keys.Up:
            
case Keys.Right:
            
case Keys.Down:
                num 
= 5;
                
break;

            
case Keys.Tab:
                num 
= 6;
                
break;
        }

        
if (this.IsHandleCreated)
        
{
            
return ((((intthis.SendMessage(0x8700)) & num) != 0);
        }

    }

    
return false;
}


      注意这一行return ((((intthis.SendMessage(0x8700)) & num) != 0);0x87是什么windows消息呢,打开WinUser.h文件,发现是WM_GETDLGCODE,MSDN中的描述是这样的:
      The WM_GETDLGCODE message is sent to the window procedure associated with a control. By default, the system handles all keyboard input to the control; the system interprets certain types of keyboard input as dialog box navigation keys. To override this default behavior, the control can respond to the WM_GETDLGCODE message to indicate the types of input it wants to process itself.
      也就是说windows用这个消息来判断哪些类型的输入交给控件本身来处理。然后,我注意到,对于方向导航键,函数都给于一个值5this.SendMessage(0x8700))的返回值进行与操作,那么this.SendMessage(0x8700))的返回值都可能是什么值呢,WinUser.h中是这样声明的:
   
/*
 * Dialog Codes
 
*/

#define  DLGC_WANTARROWS     0x0001      /* Control wants arrow keys         */
#define  DLGC_WANTTAB        0x0002      /* Control wants tab keys           */
#define  DLGC_WANTALLKEYS    0x0004      /* Control wants all keys           */
#define  DLGC_WANTMESSAGE    0x0004      /* Pass message to control          */
#define  DLGC_HASSETSEL      0x0008      /* Understands EM_SETSEL message    */
#define  DLGC_DEFPUSHBUTTON  0x0010      /* Default pushbutton               */
#define  DLGC_UNDEFPUSHBUTTON 0x0020     /* Non-default pushbutton           */
#define  DLGC_RADIOBUTTON    0x0040      /* Radio button                     */
#define  DLGC_WANTCHARS      0x0080      /* Want WM_CHAR messages            */
#define  DLGC_STATIC         0x0100      /* Static item: don't include       */
#define  DLGC_BUTTON         0x2000      /* Button item: can be checked      */
      5最贴切的表达就是DLGC_WANTMESSAGE | DLGC_WANTARROWS,也就是将方向键发送给控件处理,对于6呢,也就是DLGC_WANTMESSAGE| DLGC_WANTTAB,将Tab键发送给控件处理。
   
    从这段代码里和控件实际的行为我们可以得出一个结论,那就是,控件本身是不处理方向键和Tab键的,因为他们有默认的行为,也就是支持焦点在窗体的控件之间转换。如果你想要处理这些导航键,那么结论很简单,就是重载IsInputKey方法,它是一个保护类型的虚方法。       
      在ToolBox控件的代码里重载IsinputKey方法:
         protected   override   bool  IsInputKey(Keys keyData)
        
{
            
if ((keyData & Keys.Alt) == Keys.Alt)
            
{
                
return false;
            }

            
switch ((keyData & Keys.KeyCode))
            
{
                
case Keys.Up:
                
case Keys.Down:                
                    
return true;
            }

            
return base.IsInputKey(keyData);

        }

       

       当用户点击的键是UpDown的时候,返回true,这时我们的OnKeyDown方法里就可以捕获到UpDown的点击事件了。






本文转自纶巾客博客园博客,原文链接:http://www.cnblogs.com/guanjinke/archive/2007/05/11/743464.html,如需转载请自行联系原作者
目录
相关文章
|
C# Windows
wpf怎么使用WindowsFormsHost(即winform控件)
原文:wpf怎么使用WindowsFormsHost(即winform控件) 使用方法:   1、首先,我们需要向项目中的引用(reference)中添加两个动态库dll,一个是.
5477 0
|
Java C# 索引
C#之 十九 使用WinForm控件
C#之 十九 使用WinForm控件
220 0
|
Shell C++ C语言
38【WinForm】WinForm常见窗体技术汇总
- 窗体调用外部程序与渐变窗体 - 按回车键跳转窗体中的光标焦点 - 剪切板操作
84 0
|
Windows
艾伟:WinForm控件开发总结(一)------开篇
我本人不是专业的控件开发人员,只是在平常的工作中,需要自己开发一些控件。在自己开发WinForm控件的时候,没有太多可以借鉴的资料,只能盯着MSDN使劲看,还好总算有些收获。现在我会把这些经验陆陆续续的总结出来,写成一系列方章,希望对看到的朋友有所帮助。
996 0
|
Web App开发
艾伟:WinForm控件开发总结(三)------认识WinForm控件常用的Attribute
在前面的文章里我们制作了一个非常简单的控件。现在我们回过头来看看这些代码透露出什么信息。   这个类是直接从Control类派生出来的,自定义控件都是直接从Control类派生出来的。这个类定义了一个属性TextAlignment,用来控制文本在控件中显示的位置:           ...
1018 0
|
测试技术
艾伟:WinForm控件开发总结(二)------使用和调试自定义控件
在上一篇文章里我们创建了一个简单的控件FirstControl,现在我来介绍一下怎么使用和调试自己的控件。我希望将过程写的尽可能的详细,让想学习控件开发的朋友容易上手,高手们见谅。       在同一个solution里添加一个Windows Application工程(在Solution Explorer里右键点击CustomControlSample solution选择Add->New Project…),命名为TestControl。
895 0
|
C# Windows 安全
WinForm控件与WPF控件的交互
原文:WinForm控件与WPF控件的交互 这个问题其实也可以理解为:怎样在WPF/XAML中使用Winform中的控件(如PictureBox)?首先看看XAML代码:(注意下面加粗的部分)              ...
1091 0
|
C# 容器
在WPF中使用WinForm控件方法
原文:在WPF中使用WinForm控件方法 1、      首先添加对如下两个dll文件的引用:WindowsFormsIntegration.dll,System.Windows.Forms.dll。
1266 0
|
C#
在Winform窗体中使用WPF控件(附源码)
原文:在Winform窗体中使用WPF控件(附源码) 今天是礼拜6,下雨,没有外出,闲暇就写一篇博文讲下如何在Winform中使用WPF控件。原有是我在百度上搜索相关信息无果,遂干脆动手自己实现。 WPF控件的漂亮是Winform无法匹及的,本文主旨是在Winform工程中如何使用WPF控件。
1936 0
|
存储 开发框架 .NET