天天看点

第七章:XAML vs. code(3)

属性元素语法

这里有一些C#与第4章中的FramedText代码相似。在一个语句中,它实例化一个Frame和一个Label,并将Label设置为Frame的Content属性:

new Frame
{
    OutlineColor = Color.Accent,
    HorizontalOptions = LayoutOptions.Center,
    VerticalOptions = LayoutOptions.Center,
    Content = new Label
    {
        Text = "Greetings, Xamarin.Forms!"
    }
};           

但是当你开始在XAML中复制它时,你可能会在设置Content属性的时候变得有点困难:

<Frame OutlineColor="Accent"
       HorizontalOptions="Center"
       VerticalOptions="Center"
       Content=" what goes here? " />           

如何将Content属性设置为整个Label对象?

这个问题的解决方案是XAML语法的最基本特征。 第一步是将Frame标签分成开始标签和结束标签:

<Frame OutlineColor="Accent"
       HorizontalOptions="Center"
       VerticalOptions="Center">
 
</Frame>           

在这些标签中,添加两个由元素(Frame)和您想要设置的属性(Content)组成的标签,并添加一个句点:

<Frame OutlineColor="Accent"
       HorizontalOptions="Center"
       VerticalOptions="Center">
    <Frame.Content>
 
    </Frame.Content>
</Frame>           

现在把标签放在这些标签中:

<Frame OutlineColor="Accent"
       HorizontalOptions="Center"
       VerticalOptions="Center">
    <Frame.Content>
        <Label Text="Greetings, Xamarin.Forms!" />
    </Frame.Content>
</Frame>           

该语法是如何将Label设置为Frame的Content属性。

您可能想知道这个XAML功能是否违反了XML语法规则。 它不是。 这段时间在XML中没有特殊含义,所以Frame.Content是一个完全有效的XML标签。 然而,XAML强加了关于这些标签的规则:Frame.Content标签必须出现在Frame标签中,并且不能在Frame.Content标签中设置属性。 设置为Content属性的对象显示为这些标记的XML内容。

一旦引入了这个语法,就需要一些术语。 在上面显示的最终XAML代码片段中:

  • 框架和标签是用XML元素表示的C#对象。 它们被称为对象元素。
  • OutlineColor,HorizontalOptions,VerticalOptions和Text是作为XML属性提供的C#属性。 他们被称为财产属性。
  • Frame.Content是以XML元素表示的C#属性,因此称为属性元素。

属性元素在现实生活中很常见。 本章和未来章节中将会看到许多示例,您很快就会发现属性元素成为您使用XAML的第二本质。 但要小心:有时开发人员必须记得太多,以至于我们忘记了基本知识。 即使在您使用XAML一段时间后,您可能会遇到这样的情况,即似乎无法将特定对象设置为特定属性。 解决方案通常是一个属性元素。

您还可以对简单的属性使用属性元素语法,例如:

<Frame HorizontalOptions="Center">
    <Frame.VerticalOptions>
        Center
    </Frame.VerticalOptions>
    <Frame.OutlineColor>
        Accent
    </Frame.OutlineColor>
    <Frame.Content>
        <Label>
            <Label.Text>
                Greetings, Xamarin.Forms!
            </Label.Text>
        </Label>
    </Frame.Content>
</Frame>           

现在,Frame的VerticalOptions和OutlineColor属性以及Label的Text属性都已成为属性元素。 这些属性的值始终是不带引号的属性元素的内容。

当然,将这些属性定义为属性元素没有多大意义。 这是不必要的,一切都很冗长。 但它的工作原理应该如此。

让我们进一步:不是将HorizontalOptions设置为“Center”(对应于静态属性LayoutOptions.Center),您可以将HorizontalOptions表示为属性元素,并将其设置为LayoutOptions值并设置其各个属性:

<Frame>
    <Frame.HorizontalOptions>
        <LayoutOptions Alignment="Center"
                       Expands="False" />
    </Frame.HorizontalOptions>
    <Frame.VerticalOptions>
        Center
    </Frame.VerticalOptions>
    <Frame.OutlineColor>
        Accent
    </Frame.OutlineColor>
    <Frame.Content>
       <Label>
            <Label.Text>
                Greetings, Xamarin.Forms!
            </Label.Text>
        </Label>
    </Frame.Content>
</Frame>```

您还可以将LayoutOptions的这些属性表示为属性元素:
           
<Frame.HorizontalOptions>
    <LayoutOptions>
         <LayoutOptions.Alignment>
             Center
         </LayoutOptions.Alignment>
         <LayoutOptions.Expands>
             False
           </LayoutOptions.Expands>
    </LayoutOptions>
</Frame.HorizontalOptions>
           

`

您不能将属性设置为属性属性和属性元素。 这是设置属性两次,这是不允许的。 请记住,没有其他内容可以出现在属性元素标签中。 设置为属性的值始终是这些标记的XML内容。

现在你应该知道如何在XAML中使用StackLayout。 首先将Children属性表示为属性元素StackLayout.Children,然后将StackLayout的子元素作为属性元素标记的XML内容。 下面是一个例子,第一个StackLayout的每个孩子都是另一个具有水平方向的StackLayout:

<StackLayout>
    <StackLayout.Children>
        <StackLayout Orientation="Horizontal">
            <StackLayout.Children>
                <BoxView Color="Red" />
                <Label Text="Red"
                            VerticalOptions="Center" />
            </StackLayout.Children>
        </StackLayout>
        <StackLayout Orientation="Horizontal">
            <StackLayout.Children>
                <BoxView Color="Green" />
                <Label Text="Green"
                       VerticalOptions="Center" />
            </StackLayout.Children>
         </StackLayout>
         <StackLayout Orientation="Horizontal">
             <StackLayout.Children>
                 <BoxView Color="Blue" />
                 <Label Text="Blue"
                        VerticalOptions="Center" />
             </StackLayout.Children>
         </StackLayout>
    </StackLayout.Children>
</StackLayout>           

每个水平的StackLayout都有一个带有颜色的BoxView和一个带有该颜色名称的标签。

当然,这里的重复标记看起来相当可怕! 如果你想要显示16种颜色怎么办? 还是140? 一开始可能会有很多复制和粘贴成功,但如果您需要稍微改进视觉效果,那么您的状态会很糟糕。 在代码中,您可以在循环中执行此操作,但XAML没有此功能。

当标记威胁过于重复时,您可以随时使用代码。 在XAML中定义一些用户界面而在代码中定义一些用户界面是非常合理的。 但是还有其他的解决方案,你会在后面的章节中看到。

继续阅读