2

我真的是 WPF 的新手,我需要你的帮助。我的应用程序允许用户检查大陆并查看包含的国家/地区。Country 有两个属性:名称和区域。问题是我需要显示所有大陆国家的平均面积。我的数据模型如下所示:

       <XmlDataProvider x:Key="CountryStoreDataSource" XPath="CountryStore">
            <x:XData>
                <CountryStore xmlns="">
                    <Continents Continent="Europe">
                        <Countries Country="Italy" Area="300"/>
                        <Countries Country="Iceland" Area="350"/>
                    </Continents>
                    <Continents Continent="Asia">
                        <Countries Country="China" Area="700"/>
                        <Countries Country="India" Area="650"/>
                    </Continents>
                    <Continents Continent="Africa">
                        <Countries Country="South Africa" Area="550"/>
                        <Countries Country="Egypt" Area="500"/>
                    </Continents>
                </CountryStore>
            </x:XData>
        </XmlDataProvider>

我也有模板将列表框与我的数据模型连接起来:

        <Grid.Resources>
            <DataTemplate x:Key="countryItemTemplate">
                <Label Content="{Binding XPath=@Country}"/>
            </DataTemplate>
            <DataTemplate x:Key="areaItemTemplate">
                <Label Content="{Binding XPath=@Area}"/>
            </DataTemplate>
        </Grid.Resources>

最后我有我的列表框的实现:

           <ListBox
                Grid.Row="1"
                ItemsSource="{Binding XPath=Countries}"
                ItemTemplate="{StaticResource countryItemTemplate}"
                Margin="0,0,0,0" />
           <ListBox
                Grid.Row="1"
                ItemsSource="{Binding XPath=Countries}"
                ItemTemplate="{StaticResource areaItemTemplate}"
                Margin="0,0,0,0"
                Grid.Column="1" 
                Name="listBoxAreas"
                />

实际上我不知道如何从 c# 代码中的列表框中获取我的值,有没有办法获取值并在 xml 中对它们进行处理?谢谢你。

4

1 回答 1

1

有趣的问题!

首先,您使用的 XPath 似乎并不完全正确。如果我理解正确,您希望显示一个带有国家/地区的列表框、一个带有区域的列表框和一个带有Avg(areas).

首先,让我们简化 XPath 绑定。您应该更新您的 XmlDataSource 上的 XPath,以便您只获得一个独立于大陆的国家列表:

    <XmlDataProvider x:Key="CountryStoreDataSource" XPath="/CountryStore/Continents/Countries">
        <x:XData>
            <CountryStore xmlns="">
                <Continents Continent="Europe">
                    <Countries Country="Italy" Area="300"/>
                    <Countries Country="Iceland" Area="350"/>
                </Continents>
                <Continents Continent="Asia">
                    <Countries Country="China" Area="700"/>
                    <Countries Country="India" Area="650"/>
                </Continents>
                <Continents Continent="Africa">
                    <Countries Country="South Africa" Area="550"/>
                    <Countries Country="Egypt" Area="500"/>
                </Continents>
            </CountryStore>
        </x:XData>
    </XmlDataProvider>

然后,设置 Grid 的 DataContext:

<Grid DataContext="{StaticResource CountryStoreDataSource}"  ../>

之后,ListBoxes 的两个绑定都可以更新为以下内容:ItemsSource="{Binding}".


现在,回到问题:显示区域的平均值。为此,您需要绑定到同一个列表并应用转换器来进行计算。
首先,让我们创建转换器,它只是解析列表并使用 LINQ 计算平均值:

public class AreaConverter : MarkupExtension, IValueConverter
{
    public override object ProvideValue(IServiceProvider serviceProvider)
    {
        return this;
    }

    public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
    {
        if (value == null) return 0;

        //you can do this in one line, but I split it for clarity.
        var xml = value as IEnumerable<XmlNode>;
        var areas = xml.Select(x => x.Attributes["Area"].Value);
        var avg = areas.Average(a => int.Parse(a));

        return avg;
    }

    public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
    {
        throw new NotSupportException();
    }
}

之后,您可以使用此 Binding 创建 TextBlock 来保存此值:

 <TextBlock Text="{Binding Converter={local:AreaConverter}}" />

现在 TextBlock 应该显示平均值!

提示:要测试给定的 XPath,可以使用 Notepad++ 和 XML Tools 插件,该插件提供了Evaluate XPath Expression工具

于 2013-03-04T16:26:21.713 回答