WPF控件分类:
- 内容控件
- 标题内容控件
- 文本控件
- 列表控件
- 基于范围的控件
- 日期控件
控件类
控件是与用户交互的元素。控件可以获得焦点,能接受键盘或鼠标的输入。
所有控件的基类是System.Windows.Control类,这类包括一些基本功能:
- 对齐
- tab序列
- 背景、前景、边界
- 文本内容的字体
背景和前景刷子
控件包括两个属性Background和Foreground属性,这两个属性使用Brush对象。刷子对象的派生类包括SolidColorBrush、LinearGradientBrush、和TileBrush类。
用代码设置颜色
为名为cmd的按钮设置背景色:
cmd.Background = new SolidColorBrush(Colors.AliceBlue);
通过Colors类的静态属性获得预定义的颜色,将它传递给构造函数创建一个新的SolidColorBrush实例,将它赋值给按钮的背景属性。
也可使用系统颜色:
cmd.Background = new SolidColorBrush(SystemColors.ControlColor);
SystemColors类也提供预制的属性返回SolidColorBrush对象:
cmd.Background = SystemColors.ControlBrush;
你能创造一个颜色对象,依靠提供R,G,B值(红绿蓝)。每个值是从0到255一个数字:
int red = 0; int green = 255; int blue = 0; cmd.Foreground = new SolidColorBrush(Color.FromRgb(red, green, blue));
你能设置颜色的透明度,通过调用Color.FromArgb()方法,为其传递alpha值。alpha值为255是完全不透明,而为0是完全透明。
用XAML设置颜色
在XAML中,只需要提供颜色的名字或颜色值,其他的工作由解析器负责。
用 #rrggbb 或 #aarrggbb格式提供颜色值:
刷子支持自动改变通知。刷子从System.Windows.Freezable类派生而来。Freezable类有两个状态:可读状态,只读状态(冻结)
控件类还定义了BorderBrush和BorderThickness属性。
字体
Control类定义几个字体相关的属性。决定控件文本的外观。这些属性列在表6-1。
名字 | 描述 |
FontFamily | |
FontSize | |
FontStyle | |
FontWeight | |
FontStretch |
Control类没有定义任何使用它字体的属性。然而许多控件包括Text属性,没有定义为Control基类的成员。明显地,除非被派生类使用,字体属性没有任何意义。
字体家族
鼠标光标
内容控件
内容控件是更特殊的控件类型,它能拥有并显示一件内容。技术上,内容控件是能包含单个嵌套元素的控件。内容控件与布局容器的区别是,内容控件只能包含一个子元素,而布局容器可以拥有任意个子元素。
当然,你仍然能把多个内容包装到内容控件中。诀窍是把它们都包裹到单个容器中,诸如一个StackPanel、或一个Grid。例如,Window类本身是一个内容控件。明显地,窗口经常有大量内容,但是,它们都被包裹在一顶层的容器中(典型地,Grid)。
所有的内容控件起源于ContentControl抽象类。内容控件类主要包括:一些公共控件Label、ToolTip。各种按钮控件Button、RadioButton、CheckBox。一些更专用的ScrollViewer、UserControl。Window本身是个内容控件。
最后,存在一个内容控件子集,依靠从HeaderedContentControl类派生,增加了一层继承。包括GroupBox,TabItem,和Expander控件。这些控件有内容区域和标题区域,标题区域用于显示某种标题。
另外,用于导航的Frame、用于其他控件内部的ListBoxItem、StatusBarItem等也是内容控件。
Content属性
ContentControl类添加一个Content属性,接受单个对象。Content属性支持任何类型的对象,但是它把对象分为两组,并且区别对待:
- 不派生自UIElement的对象:内容控件调用ToString()获得这些控件的文本,然后显示文本。
- 派生自UIElement的对象:这些对象包括所有可视元素。使用UIElement.OnRender()方法,被显示在内容控件内部。
例如,一个提供简单字符串的按钮:
使用Image类放置一个图像到按钮内:
<Button Margin="3"> <Image Source="happyface.jpg" Stretch="None" /> Button>
放置一个包裹图像和文本的容器到按钮内:
<Button Margin="3"> <StackPanel> <TextBlock Margin="3">Image and text buttonTextBlock> <Image Source="happyface.jpg" Stretch="None" /> <TextBlock Margin="3">Courtesy of the StackPanelTextBlock> StackPanel> Button>
甚至可以在按钮中放置其他内容控件,如文本框,按钮、组合框。虽然这样做不太合理,但是WPF允许这样。
Window是内容控件,但是他只能是顶级容器,不能嵌套在其他元素中。
内容控件的其他属性包括:
HasContent属性,为真时表示控件有内容。
ContentTemplate属性,是一个告诉控件如何显示对象的模板。使用ContentTemplate,更智能地显示非UIElement派生来的对象。你可以获取对象的各种属性值,并整理放入更复杂的标记中。
对齐内容
HorizontalContentAlignment、VerticalContentAlignment取值为Top, Bottom, Left, Right、Center、Stretch
Padding属性指控件边界到其内容的距离。
HorizontalContentAlignment,VerticalContentAlignment,和Padding属性定义在Control类中,不在更特殊的ContentControl类。因为,一些不是内容控件的控件也有某种内容。例如,TextBox不是内容控件,它使用上述属性设置输入文本。
WPF内容哲学
内容控件减少了控件的数量,但是,增加了一点控件的复杂度。
标签(Label)
标签控件主要的功能是助记键,使链接控件获得焦点的快捷键。标签控件的Target属性指定链接控件。为了设置Target,使用绑定表达式指向另一个控件。
<Label Target="{Binding ElementName=txtA}">Choose _ALabel> <TextBox Name="txtA">TextBox> <Label Target="{Binding ElementName=txtB}">Choose _BLabel> <TextBox Name="txtB">TextBox>
标签文本下划线指明快捷键。如果真需要一个下划线,可以连续输入两个下划线转义。
同时按下Alt和所指定的快捷键,链接的控件就会获得焦点。例如,在本例中,按下Alt+A,焦点就会跳到txtA控件。
如果不需要助记键功能,可以考虑使用TextBlock。
按钮类的控件
按钮类的控件包括Button、CheckBox、和RadioButton。他们都从ButtonBase派生。
Click支持命令功能。
ClickMode属性,ClickMode.Release,ClickMode.Press,ClickMode.Hover
按钮也支持快捷键。
按钮
每个窗口可以有取消按钮和默认按钮,通过设置按钮的IsCancel、IsDefault属性。详见159页。
另外,还有一个容易混淆的IsDefaulted属性,见160页侧边条。
ToggleButton、RepeatButton
从ButtonBase派生的类还有三个:
- GridViewColumnHeader
- RepeatButton
- ToggleButton
RepeatButton 和ToggleButton都位于System.Windows.Controls.Primitives名字空间。常用于组合、或派生成其他控件,也可以单独使用。
CheckBox
CheckBox和RadioButton都是从ToggleButton派生。
ToggleButton添加了一个IsChecked属性,它是一个可空布尔值。
为了在WPF标记中分配一个空值,使用空标记扩展,如下所示:
A check box in indeterminate state
ToggleButton还有一个IsThreeState属性,它决定是否可以设置复选框为未定态。默认为false。
ToggleButton类定义三事件:Checked,Unchecked,和Indeterminate事件。
RadioButton
默认情况下,单选按钮按它们的容器分组。RadioButton的GroupName属性允许你覆盖这行为。
<StackPanel> <GroupBox Margin="5"> <StackPanel> <RadioButton>Group 1RadioButton> <RadioButton>Group 1RadioButton> <RadioButton>Group 1RadioButton> <RadioButton Margin="0,10,0,0" GroupName="Group2">Group 2RadioButton> StackPanel> GroupBox> <GroupBox Margin="5"> <StackPanel> <RadioButton>Group 3RadioButton> <RadioButton>Group 3RadioButton> <RadioButton>Group 3RadioButton> <RadioButton Margin="0,10,0,0" GroupName="Group2">Group 2RadioButton> StackPanel> GroupBox> StackPanel>
不需要使用GroupBox容器包裹单选按钮组,但这是一个普遍的约定。GroupBox显示一个边界和一个标题。
专用的容器
内容控件也包括一些专用的容器。
ScrollViewer直接从ContentControl继承。
ContentControl类派生了HeaderedContentControl类,这个类包括一个标题和一个内容。标题和内容都可以嵌套单个子元素。HeaderedContentControl类派生了几个子类:GroupBox,TabItem,和Expander。
ScrollViewer
尽管ScrollViewer能包裹任何元素,但是一般情况下,它包裹一个布局容器。
<ScrollViewer> <Grid Margin="3,3,10,3"> Grid> ScrollViewer>
VerticalScrollBarVisibility属性,此属性是ScrollBarVisibility枚举。Visible、Auto、Disabled。默认值是Visible。
HorizontalScrollBarVisibility属性,默认值是Hidden。
编程控制滚动
详见171页。
自定义滚动
详见172页。
GroupBox
GroupBox从HeaderedContentControl类派生。
<GroupBox Header="A GroupBox Test" Padding="5" Margin="5" VerticalAlignment="Top"> <StackPanel> <RadioButton Margin="3">OneRadioButton> <RadioButton Margin="3">TwoRadioButton> <RadioButton Margin="3">ThreeRadioButton> <Button Margin="3">SaveButton> StackPanel> GroupBox>
GroupBox仍然要求一个布局容器布置内容。GroupBox没有特别的功能,只是一个装饰控件。
TabItem
TabItem代表TabControl的一个选项卡。TabItem类添加了IsSelected属性,指示选项卡是否是TabControl当前显示的选项卡。
<TabControl Margin="5"> <TabItem Header="Tab One"> <StackPanel Margin="3"> <CheckBox Margin="3">Setting OneCheckBox> <CheckBox Margin="3">Setting TwoCheckBox> <CheckBox Margin="3">Setting ThreeCheckBox> StackPanel> TabItem> <TabItem Header="Tab Two"> ... TabItem> TabControl>
通过设置TabControl的TabStripPlacement属性,可以将选项卡从正常的顶部放到侧边。
正如Content属性,Header属性能接受任何类型的对象。这是一个例子:
<TabControl Margin="5"> <TabItem> <TabItem.Header> <StackPanel> <TextBlock Margin="3" >Image and Text Tab TitleTextBlock> <Image Source="happyface.jpg" Stretch="None" /> StackPanel> TabItem.Header> <StackPanel Margin="3"> <CheckBox Margin="3">Setting OneCheckBox> <CheckBox Margin="3">Setting TwoCheckBox> <CheckBox Margin="3">Setting ThreeCheckBox> StackPanel> TabItem> <TabItem Header="Tab Two">TabItem> TabControl>
Expander
见175页。
文本控件
WPF包括三文本输入控件:TextBox,RichTextBox,和PasswordBox。PasswordBox直接从Control派生。TextBox和RichTextBox控件派生自TextBoxBase。
不同于内容控件,文本框仅限于他们能包含的内容类型。TextBox永远存储一个字符串(Text属性)。PasswordBox也处理字符串内容(Password属性),尽管它使用SecureString。只有RichTextBox有能力存储更世故的内容:一个FlowDocument。
多行文本
MaxLength属性,设置TextBox允许的最大字符数。
TextWrapping属性,设置为Wrap或WrapWithOverflow,表示自动换行。
MinLines和MaxLines属性,设置TextBox的最小(最大)可见的行数。
LineCount属性,可以检索出文本框中的文本有多少行。
VerticalScrollBarVisibility、HorizontalScrollBarVisibility属性,设置滚动条的可视状态。
AcceptsReturn属性,设置为真表示TextBox接受回车。默认情况下,TextBox不接受回车。
AcceptsTab属性为真表示接受Tab键,默认情况下,TextBox不接受Tab键。
IsReadOnly属性,阻止编辑文本。
文本选择
见180页。
拼写检查
见181页。
密码框
见183页。
列表控件
列表控件的基类是ItemsControl类。
每个ItemsControl类都有项目列表。有二种方法填充项目列表。最直白的方法是直接添加项到项集合,使用代码或XAML。更常用的方法是数据绑定。这种方法是设置ItemsSource属性为要显示的数据项集。
ItemsControl类派生的一个主要分支为Selector类,包括ListBox, ComboBox,和TabControl类。可以跟踪当前选择项(SelectedItem),或它的位置(SelectedIndex)。
其余的列表类不支持当前项,直接从ItemsControl类派生。这些类包括Menu、Toolbar、TreeView等。
ListBox
设置SelectionMode属性为Multiple可以多选。在多选模式下,要使用SelectedItems集合而不是SelectedItem。
添加列表框项:
<ListBox> <ListBoxItem>GreenListBoxItem> <ListBoxItem>BlueListBoxItem> <ListBoxItem>YellowListBoxItem> <ListBoxItem>RedListBoxItem> ListBox>
ListBoxItem派生自ContentControl。
例如,创建一个图像的列表框:
<ListBox> <ListBoxItem> <Image Source="happyface.jpg">Image> ListBoxItem> <ListBoxItem> <Image Source="happyface.jpg">Image> ListBoxItem> ListBox>
可以省略上例的ListBoxItem,列表框足够智能,可以识别列表项:
<ListBox> <StackPanel Orientation="Horizontal"> <Image Source="happyface.jpg" Width="30" Height="30">Image> <Label VerticalContentAlignment="Center">A happy faceLabel> StackPanel> <StackPanel Orientation="Horizontal"> <Image Source="redx.jpg" Width="30" Height="30">Image> <Label VerticalContentAlignment="Center">A warning signLabel> StackPanel> <StackPanel Orientation="Horizontal"> <Image Source="happyface.jpg" Width="30" Height="30">Image> <Label VerticalContentAlignment="Center">A happy faceLabel> StackPanel> ListBox>
下面是一个列表项为复选框的例子:
<ListBox Name="lst" SelectionChanged="lst_SelectionChanged" CheckBox.Click="lst_SelectionChanged"> <CheckBox Margin="3">Option 1CheckBox> <CheckBox Margin="3">Option 2CheckBox> ListBox>
如果你没有使用ListBoxItem填充列表项,当你读SelectedItem值时,将不会得到ListBoxItem对象,而是你放置到列表中的对象。在上例中,SelectedItem将提供一个CheckBox对象。
下面代码获取当前的选择项,显示该项目是否被选中。
private void lst_SelectionChanged(object sender, SelectionChangedEventArgs e) { if (lst.SelectedItem == null) return; txtSelection.Text = String.Format( "You chose item at position {0}.\r\nChecked state is {1}.", lst.SelectedIndex, ((CheckBox)lst.SelectedItem).IsChecked); }
如果你希望知道哪一个项目失去选择,你能使用SelectionChangedEventArgs对象的RemovedItems属性。类似地,AddedItems属性告诉你哪一个项目被添加到选择。在单选模式,无论何时选择改变,永远是一项目被添加和一项目被移除。在multiple或extended模式,就不一定了。
下面是遍历列表项目集合的代码:
private void cmd_ExamineAllItems(object sender, RoutedEventArgs e) { var sb = new StringBuilder(); foreach (CheckBox item in lst.Items) { if (item.IsChecked == true) { sb.Append(item.Content); sb.Append(" is checked."); sb.Append("\r\n"); } } txtSelection.Text = sb.ToString(); }
ListBoxItem也有一些额外的功能:IsSelected属性、Selected事件和Unselected事件。这些功能也可以通过ListBox的SelectedItem属性和SelectionChanged事件实现。
有趣地,存在一个技术,当你使用嵌套对象方法时,获取指定对象的ListBoxItem包裹器。诀窍是ContainerFromElement()方法。这是使用这个技术的代码检查第一项目是否是列表的被选择项:
var item = (ListBoxItem)lst.ContainerFromElement( (DependencyObject)lst.SelectedItems[0]); MessageBox.Show("IsSelected: " + item.IsSelected.ToString());
ComboBox
组合框用法与列表框基本相同。
如果你允许用户通过在组合框键入文本来选择一个项目,你必须设置IsEditable属性为true,并且你必须确保你存储平凡的仅文本的ComboBoxItem对象,或一个提供有意义的ToString()表示法的对象。例如,如果你填充一个可编辑的带有图像对象组合框,文本出现在上面的部分是Image类的全名,这不是非常使用。
基于范围的控件
基于范围的控件基类是RangeBase类,从Control类派生。包括ScrollBar, Slider, 和ProgressBar类。RangeBase类的属性包括:
Value、Maximum、Minimum、SmallChange、LargeChange
因为有ScrollView控件,ScrollBar很少用。现在关注Slider和ProgressBar。
Slider
见188页
ProgressBar
见190页
日期控件
见190页。