0

我正在用D3绘制图表。此刻我正在通过这种方式绑定来做到这一点:

<d3:ChartPlotter x:Name="plotter" ItemsSource="Charts" Margin="20">

我在图表中添加/删除项目,绘图仪会自动更新。效果很好。

问题是我需要从几个集合中绑定,但显然我不能设置 ItemsSource 两次。我读过一些关于 CompositeCollections 的文章,但几乎每篇文章都是基于 StaticResource,这不是我的情况。

 <d3:ChartPlotter x:Name="plotter"Margin="20">
 <d3:ChartPlotter.ItemsSource>
    <CompositeCollection>
      <CollectionContainer Collection="{Binding Charts}" />
      <CollectionContainer Collection="{Binding Charts2}" />
      </CompositeCollection>
  </d3:ChartPlotter.ItemsSource>

此代码编译但绑定不起作用。

我搜索了很多,但令人惊讶的是我没有找到答案。我认为这必须是 WPF 中的一项常见任务。

我对将多个集合绑定到单个 ItemsSource 的其他方式持开放态度,但是手动将每个子集合中的每个项目添加到 Charts 我认为这太麻烦了。谢谢你。

编辑:

我正在尝试通过 MultiBinding 来实现,这是 Converter 的方案

编辑2:

图表是一个ObservableCollection<LineGraph>

public class ConcatConverter : IMultiValueConverter
{
    public object Convert(object[] values, Type targetType, object parameter, CultureInfo culture)
    {
        ObservableCollection<LineGraph> enumerables = new ObservableCollection<LineGraph>();

        foreach (LineGraph line in values[0])
        {
            enumerables.Add(line);
        }

        foreach (LineGraph line in values[1])
        {
            enumerables.Add(line);
        }

        return enumerables;

    }
    public object[] ConvertBack(object value, Type[] targetType, object parameter, CultureInfo culture)
    {
        return null;
    }
}

由于此错误,我无法编译:“foreach 语句大炮在“object”类型的变量中运行,因为“object”不包含“GetEnumerator”的公共定义。

4

1 回答 1

2

使用MultiBinding. 首先,制作一个可以满足您需求的转换器:

public class ConcatConverter : IMultiValueConverter
{
    public object Convert(object[] values, Type targetType, object parameter, CultureInfo culture)
    {
        List<IEnumerable> enumerables = new List<IEnumerable>();
        foreach (object obj in values)
        {
            IEnumerable temp = obj as IEnumerable;
            if (temp == null) throw new ArgumentException();
            enumerables.Add(temp);
        }
        List<dynamic> enDynamic = new List<dynamic>();
        enDynamic.AddRange(enumerables);
        return Concat((dynamic)enDynamic);
    }
    private IEnumerable<T> Concat<T>(params IEnumerable<T>[] toConcat)
    {
        return toConcat.Aggregate((a, b) => a.Concat(b));
    }
    private IEnumerable Concat(params IEnumerable[] toConcat)
    {
        ArrayList temp = new ArrayList();
        foreach (IEnumerable x in toConcat)
        {
            foreach (object n in x)
            {
                temp.Add(n);
            }
        }
        return temp;
    }
    public object[] ConvertBack(object value, Type[] targetType, object parameter, CultureInfo culture)
    {
        return null;
    }
}

(非通用但我不想使用钝的反射量)

然后将其添加到窗口的资源中:

<!--in your window declaration-->
xmlns:local="clr-namespace:YourNameSpace"
<!--after that-->
<Window.Resources>
    <ttp:ConcatConverter x:Key="Concat"/>
</Window.Resources>
<!--finally:-->
<d3:ChartPlotter.ItemsSource>
    <MultiBinding Converter="{StaticResource ResourceKey=Concat}">    
        <Binding Source="Charts"/>
        <Binding Source="Charts2"/>
    </MultiBinding>
</d3:ChartPlotter.ItemsSource>
于 2013-06-11T18:33:43.903 回答