Silverlight开发中的疑难杂症-控件设计篇-如何实现一个NumericBox(上)

在最近的Silverlight开发中,发现一个需求,需要一个只能够输入数字,并且能够对小数点后的位数进行控制并填充的控件,原有的TextBox并不能实现这一个功能,于是就决定自己实现一个,要包含的具体的功能如下:

只能输入0-9的数字和至多一个小数点;

能够屏蔽通过非正常途径的不正确输入(输入法,粘贴等);

能够控制小数点后的最大位数,超出位数则无法继续输入;

能够选择当小数点数位数不足时是否补0

去除开头部分多余的0(为方便处理,当在开头部分输入0时,自动在其后添加一个小数点);

由于只能输入一个小数点,当在已有的小数点前再次按下小数点,能够跳过小数点;

在实现前参考了许多网上的相关资料,虽然没有找到具体的实现代码,但是还是得到了很多有用的思路,最终决定将上述功能分成两个部分来实现:其中控制只能输入数字的部分作为一个通用的TextBoxFilterBehavior进行设计;而格式化的部分则在NumericBox内部实现,下面首先介绍TextBoxFilterBehavior的实现。

在实现之前,首先查看了TextBox的相关事件,发现Silverlight中的事件比WPF来说有了很大的减少,没得选择,可用的事件只有KeyDownTextChanged。这里使用KeyDown事件来处理按键产生的输入,TextChanged事件来处理通过其它方式进行的输入(其实完全可以都在TextChanged事件中进行处理,我也想不通为什么自己选择分开进行~)。

由于要做成一个通用的TextBox筛选行为,所以这里添加了一个筛选类型的枚举选项TextBoxFilterOptions,里面暂时只有数字、小数点跟字母,留待以后扩充,实现代码如下:

TextBoxFilterOptions
 1  ///   <summary>
 2       ///  TextBox筛选选项
 3       ///   </summary>
 4      [Flags]
 5       public   enum  TextBoxFilterOptions
 6      {
 7           ///   <summary>
 8           ///  不采用任何筛选
 9           ///   </summary>
10          None  =   0 ,
11           ///   <summary>
12           ///  数字类型不参与筛选
13           ///   </summary>
14          Numeric  =   1 ,
15           ///   <summary>
16           ///  字母类型不参与筛选
17           ///   </summary>
18          Character  =   2 ,
19           ///   <summary>
20           ///  小数点不参与筛选
21           ///   </summary>
22          Dot  =   4 ,
23           ///   <summary>
24           ///  其它类型不参与筛选
25           ///   </summary>
26          Other  =   8
27      }
28 
29       ///   <summary>
30       ///  TextBox筛选选项枚举扩展方法
31       ///   </summary>
32       public   static   class  TextBoxFilterOptionsExtension
33      {
34           ///   <summary>
35           ///  在全部的选项中是否包含指定的选项
36           ///   </summary>
37           ///   <param name="allOptions"> 所有的选项 </param>
38           ///   <param name="option"> 指定的选项 </param>
39           ///   <returns></returns>
40           public   static   bool  ContainsOption( this  TextBoxFilterOptions allOptions, TextBoxFilterOptions option)
41          {
42               return  (allOptions  &  option)  ==  option;
43          }
44      }
45 

其中ContainsOption方法是一个提供了方便的枚举运算的扩展方法。

       另外还增加了一个键盘操作的帮助类KeyboardHelper,在实现时发现Silverlight中的按键比WPF中少了很多,甚至连主键盘上的小数点键(句号)都无法识别,这里使用了一个平台相关的keycode进行判别,如果有更好的方法请给我留言。代码如下:

KeyboardHelper
  1  ///   <summary>
  2       ///  键盘操作帮助类
  3       ///   </summary>
  4       public   class  KeyboardHelper
  5      {
  6           ///   <summary>
  7           ///  键盘上的句号键
  8           ///   </summary>
  9           public   const   int  OemPeriod  =   190 ;
 10 
 11           #region  Fileds
 12 
 13           ///   <summary>
 14           ///  控制键
 15           ///   </summary>
 16           private   static   readonly  List < Key >  _controlKeys  =   new  List < Key >
 17                                                              {
 18                                                                  Key.Back,
 19                                                                  Key.CapsLock,
 20                                                                  Key.Ctrl,
 21                                                                  Key.Down,
 22                                                                  Key.End,
 23                                                                  Key.Enter,
 24                                                                  Key.Escape,
 25                                                                  Key.Home,
 26                                                                  Key.Insert,
 27                                                                  Key.Left,
 28                                                                  Key.PageDown,
 29                                                                  Key.PageUp,
 30                                                                  Key.Right,
 31                                                                  Key.Shift,
 32                                                                  Key.Tab,
 33                                                                  Key.Up
 34                                                              };
 35 
 36           #endregion
 37 
 38           ///   <summary>
 39           ///  是否是数字键
 40           ///   </summary>
 41           ///   <param name="key"> 按键 </param>
 42           ///   <returns></returns>
 43           public   static   bool  IsDigit(Key key)
 44          {
 45               bool  shiftKey  =  (Keyboard.Modifiers  &  ModifierKeys.Shift)  !=   0 ;
 46               bool  retVal;
 47               // 按住shift键后,数字键并不是数字键
 48               if  (key  >=  Key.D0  &&  key  <=  Key.D9  &&   ! shiftKey)
 49              {
 50                  retVal  =   true ;
 51              }
 52               else
 53              {
 54                  retVal  =  key  >=  Key.NumPad0  &&  key  <=  Key.NumPad9;
 55              }
 56               return  retVal;
 57          }
 58 
 59           ///   <summary>
 60           ///  是否是控制键
 61           ///   </summary>
 62           ///   <param name="key"> 按键 </param>
 63           ///   <returns></returns>
 64           public   static   bool  IsControlKeys(Key key)
 65          {
 66               return  _controlKeys.Contains(key);
 67          }
 68 
 69           ///   <summary>
 70           ///  是否是小数点
 71           ///  Silverlight中无法识别问号左边的那个小数点键
 72           ///  只能识别小键盘中的小数点
 73           ///   </summary>
 74           ///   <param name="key"> 按键 </param>
 75           ///   <returns></returns>
 76           public   static   bool  IsDot(Key key)
 77          {
 78               return  key  ==  Key.Decimal;
 79          }
 80 
 81           ///   <summary>
 82           ///  是否是小数点
 83           ///   </summary>
 84           ///   <param name="key"> 按键 </param>
 85           ///   <param name="keyCode"> 平台相关的按键代码 </param>
 86           ///   <returns></returns>
 87           public   static   bool  IsDot(Key key,  int  keyCode)
 88          {
 89               return  IsDot(key)  ||  (key  ==  Key.Unknown  &&  keyCode  ==  OemPeriod); 
 90          }
 91 
 92           ///   <summary>
 93           ///  是否是字母键
 94           ///   </summary>
 95           ///   <param name="key"> 按键 </param>
 96           ///   <returns></returns>
 97           public   static   bool  IsCharacter(Key key)
 98          {
 99               return  key  >=  Key.A  &&  key  <=  Key.Z;
100          }
101      }
102 

最后是TextBoxFilterBehavior的实现,关于Behavior的相关概念,大家可以参见Blend的帮助文档以及MSDN的相关文章,园子里也有很多大牛对它进行了详细的阐述,我就不深入讲解了。在这里,我们首先添加了一个TextBoxFilterOptions类型的依赖属性,用来设置过滤的选项,如下:

Dependency Properties
 1  #region  Dependency Properties
 2 
 3           ///   <summary>
 4           ///  TextBox筛选选项,这里选择的为过滤后剩下的按键
 5           ///  控制键不参与筛选,可以多选组合
 6           ///   </summary>
 7           public  TextBoxFilterOptions TextBoxFilterOptions
 8          {
 9               get  {  return  (TextBoxFilterOptions)GetValue(TextBoxFilterOptionsProperty); }
10               set  { SetValue(TextBoxFilterOptionsProperty, value); }
11          }
12 
13           //  Using a DependencyProperty as the backing store for TextBoxFilterOptions.  This enables animation, styling, binding, etc...
14           public   static   readonly  DependencyProperty TextBoxFilterOptionsProperty  =
15              DependencyProperty.Register( " TextBoxFilterOptions " typeof (TextBoxFilterOptions),  typeof (TextBoxFilterBehavior),  new  PropertyMetadata(TextBoxFilterOptions.None));
16 
17           #endregion
18 

然后注册了关联的TextBox控件的KeyDownTextChanged事件,在里面对输入的文本进行了验证,如果没有通过验证,则将文本充值回上次正确的文本。由于在TextChanged事件中并没有WPF里面的TextChange数组,所以无法拿到本次变更的文本,只能选择处理完整的Text完整的代码如下:

TextBoxFilterBehavior
  1  ///   <summary>
  2       ///  TextBox筛选行为,过滤不需要的按键
  3       ///   </summary>
  4       public   class  TextBoxFilterBehavior : Behavior < TextBox >
  5      {
  6           private   string  _prevText  =   string .Empty;
  7 
  8           public  TextBoxFilterBehavior()
  9          {
 10          }
 11 
 12           #region  Dependency Properties
 13 
 14           ///   <summary>
 15           ///  TextBox筛选选项,这里选择的为过滤后剩下的按键
 16           ///  控制键不参与筛选,可以多选组合
 17           ///   </summary>
 18           public  TextBoxFilterOptions TextBoxFilterOptions
 19          {
 20               get  {  return  (TextBoxFilterOptions)GetValue(TextBoxFilterOptionsProperty); }
 21               set  { SetValue(TextBoxFilterOptionsProperty, value); }
 22          }
 23 
 24           //  Using a DependencyProperty as the backing store for TextBoxFilterOptions.  This enables animation, styling, binding, etc...
 25           public   static   readonly  DependencyProperty TextBoxFilterOptionsProperty  =
 26              DependencyProperty.Register( " TextBoxFilterOptions " typeof (TextBoxFilterOptions),  typeof (TextBoxFilterBehavior),  new  PropertyMetadata(TextBoxFilterOptions.None));
 27 
 28           #endregion
 29 
 30           protected   override   void  OnAttached()
 31          {
 32               base .OnAttached();
 33 
 34               this .AssociatedObject.KeyDown  +=   new  KeyEventHandler(AssociatedObject_KeyDown);
 35               this .AssociatedObject.TextChanged  +=   new  TextChangedEventHandler(AssociatedObject_TextChanged);
 36          }
 37 
 38           protected   override   void  OnDetaching()
 39          {
 40               base .OnDetaching();
 41 
 42               this .AssociatedObject.KeyDown  -=   new  KeyEventHandler(AssociatedObject_KeyDown);
 43               this .AssociatedObject.TextChanged  -=   new  TextChangedEventHandler(AssociatedObject_TextChanged);
 44          }
 45 
 46           #region  Events
 47 
 48           ///   <summary>
 49           ///  处理通过其它手段进行的输入
 50           ///   </summary>
 51           ///   <param name="sender"></param>
 52           ///   <param name="e"></param>
 53           void  AssociatedObject_TextChanged( object  sender, TextChangedEventArgs e)
 54          {
 55               // 如果符合规则,就保存下来
 56               if  (IsValidText( this .AssociatedObject.Text))
 57              {
 58                  _prevText  =   this .AssociatedObject.Text;
 59              }
 60               // 如果不符合规则,就恢复为之前保存的值
 61               else  
 62              {
 63                   int  selectIndex  =   this .AssociatedObject.SelectionStart  -  ( this .AssociatedObject.Text.Length  -  _prevText.Length);                
 64                   this .AssociatedObject.Text  =  _prevText;
 65                   this .AssociatedObject.SelectionStart  =  selectIndex;
 66              }
 67 
 68          }
 69 
 70           ///   <summary>
 71           ///  处理按键产生的输入
 72           ///   </summary>
 73           ///   <param name="sender"></param>
 74           ///   <param name="e"></param>
 75           void  AssociatedObject_KeyDown( object  sender, KeyEventArgs e)
 76          {
 77               bool  handled  =   true ;
 78               // 不进行过滤
 79               if  (TextBoxFilterOptions  ==  TextBoxFilterOptions.None  ||
 80                  KeyboardHelper.IsControlKeys(e.Key))
 81              {
 82                  handled  =   false ;
 83              }
 84               // 数字键
 85               if  (handled  &&  TextBoxFilterOptions.ContainsOption(TextBoxFilterOptions.Numeric))
 86              {
 87                  handled  =   ! KeyboardHelper.IsDigit(e.Key);
 88              }
 89               // 小数点
 90               if  (handled  &&  TextBoxFilterOptions.ContainsOption(TextBoxFilterOptions.Dot))
 91              {
 92                  handled  =   ! (KeyboardHelper.IsDot(e.Key, e.PlatformKeyCode)  &&   ! _prevText.Contains( " . " ));
 93                   if  (KeyboardHelper.IsDot(e.Key, e.PlatformKeyCode)  &&  _prevText.Contains( " . " ))
 94                  {
 95                       // 如果输入位置的下一个就是小数点,则将光标跳到小数点后面
 96                       if  ( this .AssociatedObject.SelectionStart <   this .AssociatedObject.Text.Length  &&  _prevText[ this .AssociatedObject.SelectionStart]  ==   ' . ' )
 97                      {
 98                           this .AssociatedObject.SelectionStart ++ ;
 99                      }                    
100                  }
101              }
102               // 字母
103               if  (handled  &&  TextBoxFilterOptions.ContainsOption(TextBoxFilterOptions.Character))
104              {
105                  handled  =   ! KeyboardHelper.IsDot(e.Key);
106              }
107              e.Handled  =  handled;
108          }
109 
110           #endregion
111 
112           #region  Private Methods
113 
114           ///   <summary>
115           ///  判断是否符合规则
116           ///   </summary>
117           ///   <param name="c"></param>
118           ///   <returns></returns>
119           private   bool  IsValidChar( char  c)
120          {
121               if  (TextBoxFilterOptions  ==  TextBoxFilterOptions.None)
122              {
123                   return   true ;
124              }
125               else   if  (TextBoxFilterOptions.ContainsOption(TextBoxFilterOptions.Numeric)  &&
126                   ' 0 '   <=  c  &&  c  <=   ' 9 ' )
127              {
128                   return   true ;
129              }
130               else   if  (TextBoxFilterOptions.ContainsOption(TextBoxFilterOptions.Dot)  &&
131                  c  ==   ' . ' )
132              {
133                   return   true ;
134              }
135               else   if  (TextBoxFilterOptions.ContainsOption(TextBoxFilterOptions.Character)) 
136              {
137                   if  (( ' A '   <=  c  &&  c  <=   ' Z ' ||  ( ' a '   <=  c  &&  c  <=   ' z ' ))
138                  {
139                       return   true ;
140                  }
141              }
142               return   false ;
143          }
144 
145           ///   <summary>
146           ///  判断文本是否符合规则
147           ///   </summary>
148           ///   <param name="text"></param>
149           ///   <returns></returns>
150           private   bool  IsValidText( string  text)
151          {
152               // 只能有一个小数点
153               if  (text.IndexOf( ' . ' !=  text.LastIndexOf( ' . ' ))
154              {
155                   return   false ;
156              }
157 
158               foreach  ( char  c  in  text)
159              {
160                   if  ( ! IsValidChar(c))
161                  {
162                       return   false ;
163                  }
164              }
165               return   true ;
166          }
167 
168           #endregion
169      }
170 

TextBoxFilterBehavior类主要的实现思路和代码就是这样,希望能给需要的朋友一定的帮助,NumericBox类的实现思路及相关代码我将在下篇文章中给出,敬请关注。

(未完待续)

你可能感兴趣的:(silverlight)