为什么不同的实例具有相同的依赖属性值?

6

在WPF/.NET 3.5中开发自定义控件时,存在一个严重的问题。

详细信息

1.当我在同一窗口中添加两个实例时,似乎第二个实例总会有第一个实例的依赖属性值,即使test2只是被构造而没有应用模板(e.g: tes2.TopItems/CenterItems/BottomItems和tes1的一样,包括Count、Items等)。

2.当我删除其中任何一个时,问题就解决了。

3.TopItems/CenterItems/BottomItems是在OnApplyTemplate()中初始化的。

4.更改"Calendar"和"CalendarViewType"属性会调用PropertyChangedCallBack()。

5."TopItems"、"CenterItems"、"BttomItems"在控件的模板中使用{TemplateBinding}。

6.我尝试使用"普通属性"“{Binding RelativeSource=TemplatedParent}”代替"依赖属性"和{TemplateBinding},它正常工作!多么奇怪!

请问,有人能帮忙吗?非常感谢!

类窗口

<Window>
 <Grid>
  <common:CalendarTitle x:Name="test1" Calendar="{Binding Calendar}" CalendarViewType="Month_Week"></common:CalendarTitle>
  <common:CalendarTitle x:Name="test2" Calendar="{Binding Calendar}"></common:CalendarTitle>
 </Grid>
</Window>

班级日历瓷砖控件:Control

    public static readonly DependencyProperty CalendarProperty = DependencyProperty.Register("Calendar", typeof(ICalendar), typeof(CalendarTitle), new FrameworkPropertyMetadata(null, FrameworkPropertyMetadataOptions.AffectsRender, PropertyChangedCallback));
    public static readonly DependencyProperty CalendarViewTypeProperty = DependencyProperty.Register("CalendarViewType", typeof(CalendarViewType), typeof(CalendarTitle), new FrameworkPropertyMetadata(CalendarViewType.Week_Day, FrameworkPropertyMetadataOptions.AffectsRender, PropertyChangedCallback));
    public static readonly DependencyProperty TopItemsProperty = DependencyProperty.Register("TopItems", typeof(IEnumerable<ICalendarItem>), typeof(CalendarTitle), new FrameworkPropertyMetadata(new ObservableCollection<ICalendarItem>()));
    public static readonly DependencyProperty CenterItemsProperty = DependencyProperty.Register("CenterItems", typeof(IEnumerable<ICalendarItem>), typeof(CalendarTitle), new FrameworkPropertyMetadata(new ObservableCollection<ICalendarItem>()));
    public static readonly DependencyProperty BottomItemsProperty = DependencyProperty.Register("BottomItems", typeof(IEnumerable<ICalendarItem>), typeof(CalendarTitle), new FrameworkPropertyMetadata(new ObservableCollection<ICalendarItem>()));
    public IEnumerable<ICalendarItem> TopItems
    {
        get { return (IEnumerable<ICalendarItem>)GetValue(TopItemsProperty); }
        set { SetValue(TopItemsProperty, value); }
    }
    public IEnumerable<ICalendarItem> CenterItems
    {
        get { return (IEnumerable<ICalendarItem>)GetValue(CenterItemsProperty); }
        set { SetValue(CenterItemsProperty, value); }
    }
    public IEnumerable<ICalendarItem> BottomItems
    {
        get { return (IEnumerable<ICalendarItem>)GetValue(BottomItemsProperty); }
        set { SetValue(BottomItemsProperty, value); }
    }

PropertyCallBack()

 static void PropertyChangedCallback(DependencyObject d, DependencyPropertyChangedEventArgs e)
    {
        var calendarTile = (CalendarTitle)d;
        calendarTile.isLoaded = false;
        calendarTile.OnApplyTemplate();
    }

OnApplyTemplate()

 public override void OnApplyTemplate()
    {
        if (!isLoaded)
        {
            ClearItems();
            if (Calendar != null)
                PopulateItems();
            isLoaded = true;
        }
    }

PopulateItems()

 private void PopulateItems()
    {
        switch (CalendarViewType)
        {
            case CalendarViewType.Week_Day:
                {
                    foreach (var item in Calendar.Items)
                    {
                        if (item.Date.DayOfWeek == DayOfWeek.Sunday)
                            (TopItems as IList<ICalendarItem>).Add(item);
                    }
                    CenterItems = BottomItems = Calendar.Items;
                    break;
                }

            case CalendarViewType.Month_Week:
                {
                    foreach (var item in Calendar.Items)
                    {
                        if (item.Date.DayOfYear == 1)
                            (TopItems as IList<ICalendarItem>).Add(item);
                        if (item.Date.Day == 1)
                            (CenterItems as IList<ICalendarItem>).Add(item);
                        if (item.Date.DayOfWeek == DayOfWeek.Monday)
                            (BottomItems as IList<ICalendarItem>).Add(item);
                    }
                    break;
                }

            case CalendarViewType.Year_Month:
                {
                    foreach (var item in Calendar.Items)
                    {
                        if (item.Date.DayOfYear == 1)
                            (TopItems as IList<ICalendarItem>).Add(item);
                        if (item.Date.Day == 1)
                            (CenterItems as IList<ICalendarItem>).Add(item);
                        if (item.Date.DayOfWeek == DayOfWeek.Monday)
                            (BottomItems as IList<ICalendarItem>).Add(item);
                    }
                    break;
                }
        }

    }

你的哪个依赖属性在窗口的两个实例中似乎具有相同的值? - O. R. Mapper
OnApplyTemplate() 不会操作任何依赖属性。PopulateItems() 的内容是什么?实际输出与您的预期有何不同? - Nuffin
1个回答

18
当您指定依赖属性的默认值时,此默认值将在所有实例之间共享。因此,如果它是可变引用类型(例如集合),则对该集合所做的任何更改都将反映在类的所有实例中。
您应该将默认值保留为null,并在类构造函数中初始化集合。这样,类的所有实例都将拥有自己的集合。

谢谢!但事情可能会有所不同。例如:当加载test1时,test1.Top/Center/Bottom有1/8/37项。然后开始构建test2,我发现test2.Top/Center/Bottom也有1/8/37项!有什么建议吗?谢谢! - Claw
非常感谢你,伙计!你完全正确!!!你刚刚救了我的生命和午餐!!再次感谢!! - Claw
纯净的小宝石! - dotNET
哇!!!!!!太棒了!我没想到能这么快在谷歌上找到答案!! - Carl Chang

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接