一、WPF为何需要命令
我们已经知道WPF里已经有了路由事件,可以发布及传播一些消息,那为什么还需要命令呢?这是因为事件指负责发送消息,对消息如何处理则不管,而命令是有约束力,每个接收者对命令执行统一的行为,比如菜单上的保存,工具栏上的保存都必须是执行同样的保存。
二、命令系统的基本元素
命令(Command):实现了ICommand接口的类,经常使用的有RoutedCommand类
命令源: 是命令的发送者,是实现了ICommandSource接口的类,大部分界面的控件都实现了这个接口,Button, MenuItem 等等。
命令目标:命令的接收者,命令目标是视线了IInputElement接口的类。
命令关联:负责一些逻辑与命令关联起来,比如判断命令是否可以执行,以及执行完毕后做一些处理。
三、四个命令元素之间的关系
四、命令示例
我们让一个按钮发送Hello命令给文本框,文本框接收这个命令后显示“Nice to meet you”.
01 |
< Window x:Class = "DeepXAML.MainWindow" |
04 |
xmlns:local = "clr-namespace:DeepXAML" |
05 |
xmlns:sys = "clr-namespace:System;assembly=mscorlib" |
06 |
Title = "MainWindow" Height = "250" Width = "450" > |
08 |
< StackPanel x:Name = "stackPanel" > |
09 |
< TextBox x:Name = "textBox1" Margin = "10" ></ TextBox > |
10 |
< TextBox x:Name = "textBox2" Margin = "10" ></ TextBox > |
11 |
< Button x:Name = "btnHello" Margin = "10" >Hello</ Button > |
后台代码
02 |
using System.Collections.Generic; |
04 |
using System.Windows.Data; |
05 |
using System.Windows.Documents; |
06 |
using System.Windows.Controls; |
07 |
using System.Windows.Input; |
11 |
public partial class MainWindow : Window |
13 |
private RoutedCommand helloCmd = new RoutedCommand( "Hello" , typeof (MainWindow)); |
17 |
InitializeComponent(); |
19 |
this .btnHello.Command = this .helloCmd; |
20 |
this .helloCmd.InputGestures.Add( new KeyGesture(Key.H, ModifierKeys.Alt)); |
21 |
this .btnHello.CommandTarget = this .textBox1; |
22 |
this .btnHello.CommandTarget = this .textBox2; |
24 |
CommandBinding cb = new CommandBinding(); |
25 |
cb.Command = this .helloCmd; |
26 |
cb.CanExecute += new CanExecuteRoutedEventHandler(cb_CanExecute); |
27 |
cb.Executed += new ExecutedRoutedEventHandler(cb_Executed); |
29 |
this .stackPanel.CommandBindings.Add(cb); |
32 |
void cb_Executed( object sender, ExecutedRoutedEventArgs e) |
34 |
this .textBox1.Text = "Nice to meet you" ; |
35 |
this .textBox2.Text = "Nice to meet you" ; |
40 |
void cb_CanExecute( object sender, CanExecuteRoutedEventArgs e) |
42 |
if (! string .IsNullOrEmpty(textBox1.Text) || ! string .IsNullOrEmpty(textBox2.Text)) |
执行时
一开始按钮状态可用,也就是命令可执行
点击按钮后:
清空一个TextBox,按钮仍然不可执行,因为没达到可执行的条件,两个都清空时,按钮重回可执行状态
五、WPF的命令库
WPF提供常用应用程序所用的命令集,常用的命令集包括:ApplicationCommands, ComponentCommands, NavigationCommands, MediaCommands和EditingCommands。
ApplicationCommands(应用程序命令):
CancelPrint:取消打印
Close:关闭
ContextMenu:上下文菜单
Copy:复制
CorrectionList: Gets the value that represents the Correction List command.
Cut:剪切
Delete:删除
Find:查找
Help:帮助
New:新建
NotACommand:不是命令,被忽略
Open:打开
Paste:粘贴
Print:打印
PrintPreview:打印预览
Properties:属性
Redo:重做
Replace:取代
Save:保存
SaveAs:另存为
SelectAll:选择所有的
Stop:停止
Undo:撤消
ComponentCommands(组件命令):
ExtendSelection:后接Down/Left/Right/Up, 比如:ExtendSelectionDown(Shift+Down,Extend Selection Down),ExtendSelectionLeft等
Move:后接Down/Left/Right/Up, 如:MoveDown
MoveFocus:后接Down/Forward/Back/Up, 如:MoveFocusDown
MoveFocusPage:后接Down/Up,如:MoveFocusPageUp
MoveTo:后接End/Home/PageDown/PageUp,比如:MoveToPageDown
ScrollByLine
ScrollPage:后接Down/Left/Right/Up,比如:ScrollPageLeft
SelectTo:End/Home/PageDown/PageUp,比如:SelectToEnd
NavigationCommands(导航命令):
Browse浏览: 后接Back/Forward/Home/Stop, 比如:BrowseBack
缩放显示:DecreaseZoom, IncreaseZoom, Zoom
Favorites(收藏)
页面:FirstPage, LastPage, PreviousPage, NextPage,GoToPage
NavigateJournal
Refresh(刷新)
Search(搜索)
MediaCommands(多媒体控制命令):
Treble高音:DecreaseTreble,IncreaseTreble
Bass低音:BoostBass,DecreaseBass,IncreaseBass
Channel频道:ChannelDown,ChannelUp
MicrophoneVolume麦克风音量调节:DecreaseMicrophoneVolume,IncreaseMicrophoneVolume,MuteMicrophoneVolume
ToggleMicrophoneOnOff:麦克风开关
Volume音量: DecreaseVolume,IncreaseVolume,MuteVolume
Rewind, FastForward(回放,快进)
Track轨道:PreviousTrack,NextTrack [上一段(节)]
Play,Pause,Stop,Record(播放,暂停,停止,录制)
TogglePlayPause
Select选择
EditingCommands(编辑/排版类命令):
Align对齐:AlignCenter,AlignJustify,AlignLeft,AlignRight(居中,撑满,左对齐,右对齐)
Backspace退格
TabForward,TabBackward(Tab前缩,Tab向后)
FontSize字体大小:DecreaseFontSize,IncreaseFontSize
Indentation缩排:DecreaseIndentation, IncreaseIndentation
Delete删除: Delete选中部分,DeleteNextWord:删除后一字,DeletePreviousWord:删除前一字
EnterLineBreak:换行
EnterParagraphBreak:换段
CorrectSpellingError/IgnoreSpellingError:纠正/忽略拼写错误
MoveUpByLine,MoveDownByLine: 上/下移一行,
MoveUpByPage,MoveDownByPage: 上/下移一页
MoveUpByParagraph,MoveDownByParagraph: 上/下移一段
MoveLeftByCharacter/MoveRightByCharacter:左/右移一字符
MoveLeftByWord/MoveRightByWord 左/右移一词
MoveToDocumentStart/MoveToDocumentEnd:到文章开头/结尾
MoveToLineStart/MoveToLineEnd:到一行的开头/结尾
SelectUpByLine,SelectDownByLine:向上/下选一行
SelectUpByPage,SelectDownByPage:向上/下选一页
SelectUpByParagraph,SelectDownByParagraph:向上/下选一段
SelectLeftByCharacter,SelectRightByCharacter:向左/右选中一字
SelectLeftByWord,SelectRightByWord:向左/右选中一词
SelectToDocumentStart,SelectToDocumentEnd: 选中到篇头/篇尾
SelectToLineStart/SelectToLineEnd:选中到行首/行尾
ToggleBold, ToggleItalic, ToggleUnderline(加粗,斜体,下划线)
ToggleBullets, ToggleNumbering(列表:加点,加数字)
ToggleInsert:插入
ToggleSuperscript,ToggleSubscript(上标字,下标字)
六、命令参数
命令大部分都是类的静态属性,也就是示例只有一个,那么如果两个按钮使用同一个命令如何区分呢?比如新建一个Project,如何区分新建的是Library还是WPF呢?我们可以使用CommandParameter
XAML:
01 |
< Window x:Class = "DeepXAML.MainWindow" |
04 |
xmlns:local = "clr-namespace:DeepXAML" |
05 |
xmlns:sys = "clr-namespace:System;assembly=mscorlib" |
06 |
Title = "MainWindow" Height = "250" Width = "450" > |
07 |
< Window.CommandBindings > |
08 |
< CommandBinding Command = "New" CanExecute = "New_CanExecute" Executed = "New_Executed" ></ CommandBinding > |
09 |
</ Window.CommandBindings > |
10 |
< StackPanel x:Name = "stackPanel" > |
11 |
< TextBox x:Name = "textBox" Margin = "10" ></ TextBox > |
12 |
< Button x:Name = "btnWPF" Margin = "10" Command = "New" CommandParameter = "WPF" >新建WPF</ Button > |
13 |
< Button x:Name = "btnLibrary" Margin = "10" Command = "New" CommandParameter = "Library" >新建Library</ Button > |
后台代码:
02 |
using System.Collections.Generic; |
04 |
using System.Windows.Data; |
05 |
using System.Windows.Documents; |
06 |
using System.Windows.Controls; |
07 |
using System.Windows.Input; |
11 |
public partial class MainWindow : Window |
16 |
InitializeComponent(); |
19 |
private void New_CanExecute( object sender, CanExecuteRoutedEventArgs e) |
21 |
e.CanExecute = string .IsNullOrEmpty(textBox.Text); |
24 |
private void New_Executed( object sender, ExecutedRoutedEventArgs e) |
26 |
if (e.Parameter.ToString() == "Library" ) |
28 |
this .textBox.Text+= "建一个Library" ; |
30 |
if (e.Parameter.ToString() == "WPF" ) |
32 |
this .textBox.Text += "建一个WPF" ; |
运行:
点击第一个按钮
清空文本框再点击第二个按钮