To bridge the gap between string values and nonstring properties, the XAML parser needs to perform a conversion. The conversion is performed by type converters, a basic piece of infrastructure that’s borrowed from the full .NET Framework.
Essentially, a type converter has one role in life—it provides utility methods that can convert a specific .NET data type to and from any other .NET type, such as a string representation in this case. The XAML parser follows two steps to find a type converter:
Some classes define a content property, which allows you to provide the property value between the start and end tags. For example, the Button class designates Content as its content property, meaning this markup:
1 <Button>Click Me!</Button>
is equivalent to this:
1 <Button Content="Click Me!"></Button>
XAML provides another option: property-element syntax. With property-element syntax, you add a child element with a name in the form Parent.PropertyName. For example, the Grid has a Background property that allows you to supply a brush that’s used to paint the area behind the elements. If you want to use a complex brush—one more advanced than a solid color fill—you’ll need to add a child tag named Grid.Background, as shown here:
1 <UserControl x:Class="EightBall.Page" 2 xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 3 xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> 4 <Grid x:Name="grid1"> 5 <Grid.Background> 6 <LinearGradientBrush> 7 <LinearGradientBrush.GradientStops> 8 <GradientStop Offset="0.00" Color="Yellow" /> 9 <GradientStop Offset="0.50" Color="White" /> 10 <GradientStop Offset="1.00" Color="Purple" /> 11 </LinearGradientBrush.GradientStops> 12 </LinearGradientBrush> 13 </Grid.Background> 14 </Grid> 15 </UserControl>
Attached properties always use a two-part name in this form: DefiningType.PropertyName. This two-part naming syntax allows the XAML parser to distinguish between a normal property and an attached property.
1 <UserControl x:Class="EightBall.Page" 2 xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 3 xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> 4 <Grid x:Name="grid1"> 5 <Grid.RowDefinitions> 6 <RowDefinition Height="*" /> 7 <RowDefinition Height="Auto" /> 8 <RowDefinition Height="*" /> 9 </Grid.RowDefinitions> 10 11 <TextBox x:Name="txtQuestion" 12 Grid.Row="0" 13 Text="[Place question here.]"> 14 </TextBox> 15 16 <Button x:Name="cmdAnswer" 17 Grid.Row="1" 18 Content="Ask the Eight Ball"> 19 </Button> 20 21 <TextBox x:Name="txtAnswer" 22 Grid.Row="2" 23 Text="[Answer will appear here.]"> 24 </TextBox> 25 </Grid> 26 </UserControl>
Attached properties aren’t really properties at all. They’re actually translated into method calls. The XAML parser calls the static method that has this form: DefiningType.SetPropertyName(). For example, in the previous XAML snippet, the defining type is the Grid class, and the property is Row, so the parser calls Grid.SetRow().
When calling SetPropertyName(), the parser passes two parameters: the object that’s being modified and the property value that’s specified. For example, when you set the Grid.Row property on the TextBox control, the XAML parser executes this code:
1 Grid.SetRow(txtQuestion, 0);
This pattern (calling a static method of the defining type) is a convenience that conceals what’s really taking place. To the casual eye, this code implies that the row number is stored in the Grid object. However, the row number is actually stored in the object that it applies to—in this case, the TextBox object.
This sleight of hand works because the TextBox derives from the DependencyObject base class, as do all Silverlight elements. The DependencyObject is designed to store a virtually unlimited collection of dependency properties (and attached properties are one type of dependency property).
In fact, the Grid.SetRow() method is actually a shortcut that’s equivalent to calling the DependencyObject.SetValue() method, as shown here:
1 txtQuestion.SetValue(Grid.RowProperty, 0);
XAML allows each element to decide how it deals with nested elements. This interaction is mediated through one of three mechanisms that are evaluated in this order:
Some properties might support more than one type of collection. In this case, you need to add a tag that specifies the collection class, like this:
1 <Grid.Background> 2 <LinearGradientBrush> 3 <LinearGradientBrush.GradientStops> 4 <GradientStopCollection> 5 <GradientStop Offset="0.00" Color="Yellow" /> 6 <GradientStop Offset="0.50" Color="White" /> 7 <GradientStop Offset="1.00" Color="Purple" /> 8 </GradientStopCollection> 9 </LinearGradientBrush.GradientStops> 10 </LinearGradientBrush> 11 </Grid.Background>
Nested content doesn’t always indicate a collection. For example, consider the Grid element, which contains several other elements:
1 <UserControl x:Class="EightBall.Page" 2 xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 3 xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> 4 <Grid x:Name="grid1"> 5 <Grid.RowDefinitions> 6 <RowDefinition Height="*" /> 7 <RowDefinition Height="Auto" /> 8 <RowDefinition Height="*" /> 9 </Grid.RowDefinitions> 10 11 <TextBox x:Name="txtQuestion" 12 Grid.Row="0" 13 Text="[Place question here.]"> 14 </TextBox> 15 16 <Button x:Name="cmdAnswer" 17 Grid.Row="1" 18 Content="Ask the Eight Ball"> 19 </Button> 20 21 <TextBox x:Name="txtAnswer" 22 Grid.Row="2" 23 Text="[Answer will appear here.]"> 24 </TextBox> 25 </Grid> 26 </UserControl>
These nested tags don’t correspond to complex properties, because they don’t include the period. Furthermore, the Grid control isn’t a collection and so it doesn’t implement IList or IDictionary. What the Grid does support is the ContentProperty attribute, which indicates the property that should receive any nested content. Technically, the ContentProperty attribute is applied to the Panel class, from which the Grid derives, and looks like this:
1 [ContentPropertyAttribute("Children")] 2 public abstract class Panel : FrameworkElement
This indicates that any nested elements should be used to set the Children property. The XAML parser treats the content property differently depending on whether it’s a collection property (in which case it implements the IList or IDictionary interface). Because the Panel.Children property returns a UIElementCollection and because UIElementCollection implements IList, the parser uses the IList.Add() method to add nested content to the grid.
In many situations, you’ll use attributes to set properties and attach event handlers on the same element. Silverlight always follows the same sequence: first it sets the Name property (if set), then it attaches any event handlers, and lastly it sets the properties. This means that any event handlers that respond to property changes will fire when the property is set for the first time.
Silverlight includes a resource system that integrates closely with XAML. Using resources, you can do the following:
Every element includes a Resources property, which stores a dictionary collection of resources. The resources collection can hold any type of object, indexed by string.
Although every element includes the Resources property, the most common way to define resources is at the page level. That’s because every element has access to the resources in its own resource collection and the resources in all of its parents’ resource collections. So if you define a resource in the page, all the elements on the page can use it.
1 <LinearGradientBrush x:Key="BackgroundBrush"> 2 <LinearGradientBrush.GradientStops> 3 <GradientStop Offset="0.00" Color="Yellow" /> 4 <GradientStop Offset="0.50" Color="White" /> 5 <GradientStop Offset="1.00" Color="Purple" /> 6 </LinearGradientBrush.GradientStops> 7 </LinearGradientBrush> 8 </UserControl.Resources>
To use a resource in your XAML markup, you need a way to refer to it. This is accomplished using a markup extension—a specialized type of syntax that sets a property in a nonstandard way. Markup extensions extend the XAML language and can be recognized by their curly braces. To use a resource, you use a markup extension named StaticResource:
1 <Grid x:Name="grid1" Background="{StaticResource BackgroundBrush}">
Every element has its own resource collection, and Silverlight performs a recursive search up your element tree to find the resource you want.
Order is important when defining a resource in markup. The rule of thumb is that a resource must appear before you refer to it in your markup.
Interestingly, resource names can be reused as long as you don’t use the same resource name more than once in the same collection. In this case, Silverlight uses the resource it finds first. This allows you to define a resource in your application resources collection and then selectively override it with a replacement in some pages with a replacement.
1 LinearGradientBrush brush = (LinearGradientBrush)this.Resources["ButtonFace"];
However, there’s one limitation. Because Silverlight doesn’t support dynamic resources, you aren’t allowed to change the resource reference. That means you can’t replace a resource with a new object. Here’s an example of code that breaks this rule and will generate a runtime error:
1 SolidColorBrush brush = new SolidColorBrush(Colors.Yellow); 2 this.Resources["ButtonFace"] = brush;
Rather than dig through the Resources collection to find the object you want, you can give your resource a name by adding the Name attribute. You can then access it directly by name in your code. However, you can’t set both a name and a key on the same object, and the StaticResource markup extension recognizes keys only. Thus, if you create a named resource, you won’t be able to use it in your markup with a StaticResource reference. For that reason, it’s more common to use keys.
If you want to share resources between multiple projects or just improve the organization of a complex, resource-laden project, you can create a resource dictionary. A resource dictionary is simply a XAML document that does nothing but store a set of resources.
1 <ResourceDictionary 2 xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 3 xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> 4 5 </ResourceDictionary>
To use a resource dictionary, you need to merge it into a resource collection somewhere in your application. You could do this in a specific page, but it’s more common to merge it into the resources collection for the application, as shown here:
1 <Application xmlns="http://schemas.microsoft.com/client/2007" 2 xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" 3 x:Class="SilverlightApplication1.App"> 4 <Application.Resources> 5 <ResourceDictionary> 6 <ResourceDictionary.MergedDictionaries> 7 <ResourceDictionary Source="ElementBrushes.xaml" /> 8 </ResourceDictionary.MergedDictionaries> 9 </ResourceDictionary> 10 </Application.Resources> 11 </Application>
One reason to use resource dictionaries is to define the styles for application skins that you can apply dynamically to your controls. Another reason is to store content that needs to be localized (such as error message strings).
1 <Slider x:Name="sliderFontSize" Margin="3" 2 Minimum="1" Maximum="40" Value="10"> 3 </Slider> 4 5 <TextBlock Margin="10" Text="Simple Text" x:Name="lblSampleText" 6 FontSize="{Binding ElementName=sliderFontSize, Path=Value}" > 7 </TextBlock>
Data binding expressions use a XAML markup extension (and hence have curly braces). You begin with the word Binding, followed by any constructor arguments and then a list of the properties you want to set by name—in this case, ElementName and Path. ElementName indicates the source element. Path indicates the property in the source element. Thus, this binding expression copies the value from the Slider.Value property to the TextBlock.FontSize property.
Tip:The Path can point to a property of a property (for example, FontFamily.Source) or an indexer used by a property (for example, Content.Children[0]). You can also refer to an attached property (a property that’s defined in another class but applied to the bound element) by wrapping the property name in parentheses. For example, if you’re binding to an element that’s placed in a Grid, the path (Grid.Row) retrieves the row number where you’ve placed it.
备注:在单向绑定中,如果用代码修改了绑定目标的属性值,目标属性的绑定就会消失(得查查为啥这样设计)。
Interestingly, there’s a way to force values to flow in both directions: from the source to the target and from the target to the source. The trick is to set the Mode property of the Binding to TwoWay.