2010-08-18 65 views
0

本文的最终目标是覆盖通用基类的具体实现的ToString()方法,同时仍然能够使用Linq展平技术来搜索实现。所以,如果你阅读这个,并看到更好的方式让我知道。我为Silverlight使用Telerik控件,并且它们不会更改它们的api,以允许它们的某些控件属性被数据绑定,而是依赖于绑定到的任何对象的ToString()方法。是啊,愚蠢..无论如何,这是我得到的。通用基类和扩展方法的具体实现

RadTreeView控件在我的页面上。 TreeView中每个节点的FullPath属性使用绑定到的每个项目的ToString()方法(所以这是我需要重写的)。

我不得不创建一个“中介”类来增强我的基础模型类,以便在树视图中将它绑定为层次结构,然后重写ToString()以覆盖该泛型类的具体实现。现在问题是我有一个Linq扩展,因为它不能将具体实现转换回基本泛型类,所以爆炸。我喜欢泛型,但这对我来说太过分了。 需要帮助解决扩展方法问题。

中介通用类:

public class HeirarchicalItem<T> : NotifyPropertyChangedBase, INotifyCollectionChanged where T : class 
{ 
    public event NotifyCollectionChangedEventHandler CollectionChanged; 

    public virtual void OnCollectionChanged(NotifyCollectionChangedEventArgs ea) 
    { 
     if (CollectionChanged != null) 
      CollectionChanged(this, ea); 
    } 

    public HeirarchicalItem() { } 

    public HeirarchicalItem(T item) 
    { 
     Item = item; 
    } 

    public HeirarchicalItem(IEnumerable<T> collection) 
    { 
     CopyFrom(collection); 
    } 

    private T _item; 
    public T Item 
    { 
     get 
     { 
      return _item; 
     } 
     set 
     { 
      _item = value; 
      RaisePropertyChanged<HeirarchicalItem<T>>(a => a.Item); 
     } 
    } 

    private ObservableCollection<HeirarchicalItem<T>> _children = new ObservableCollection<HeirarchicalItem<T>>(); 
    public virtual ObservableCollection<HeirarchicalItem<T>> Children 
    { 
     get { return _children; } 
     set 
     { 
      _children = value; 
      RaisePropertyChanged<HeirarchicalItem<T>>(a => a.Children); 
      OnCollectionChanged(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset)); 
     } 
    } 

    private void CopyFrom(IEnumerable<T> collection) 
    { 
     if ((collection != null)) 
     { 
      using (IEnumerator<T> enumerator = collection.GetEnumerator()) 
      { 
       while (enumerator.MoveNext()) 
       { 
        HeirarchicalItem<T> newHeirarchicalItem = new HeirarchicalItem<T>(enumerator.Current); 
        Children.Add(newHeirarchicalItem); 
        RaisePropertyChanged<HeirarchicalItem<T>>(a => a.Children); 
        OnCollectionChanged(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add)); 
       } 
      } 
     } 
    } 
} 

基模型类:(数据被使用此类穿梭向和从WCF RIA服务)

public class tbl_Path : EntityBase, IFullPath, IEquatable<tbl_Path>, IEqualityComparer<tbl_Path> 
{ 
    public tbl_Path(); 
    public int GetHashCode(tbl_Path obj); 
    public override string ToString(); 

    public DateTime CreateDate { get; set; } 
    public short Depth { get; set; } 
    public string FullPath { get; set; } 
    public bool IsAuthorized { get; set; } 
    public bool IsSelected { get; set; } 
    public string Name { get; set; } 
    public override IEnumerable<Operation> Operations { get; } 
    public int? ParentPathID { get; set; } 
    public int PathID { get; set; } 
    public Guid SecurityKey { get; set; } 
    public EntityCollection<tbl_Configuration> tbl_Configuration { get; set; } 
    public EntityCollection<tbl_Key> tbl_Key { get; set; } 
    public EntityCollection<tbl_SecurityACL> tbl_SecurityACL { get; set; } 
    public EntityCollection<tbl_SecurityInheriting> tbl_SecurityInheriting { get; set; } 
    public EntityCollection<tbl_Variable> tbl_Variable { get; set; } 
} 

具体实现,使得我可以覆盖ToString():

public class HeirarchicalPath : HeirarchicalItem<tbl_Path> 
{ 
    public HeirarchicalPath() 
    { 

    } 
    public HeirarchicalPath(tbl_Path item) 
     : base(item) 
    { 

    } 
    public HeirarchicalPath(IEnumerable<tbl_Path> collection) 
     : base(collection) 
    { 

    } 

    public override string ToString() 
    { 
     return Item.Name; **// we override here so Telerik is happy** 
    } 
} 

最后,这里是Linq扩展方法,它在编译期间爆炸,因为我介绍了泛型基类的具体实现。

public static IEnumerable<T> Traverse<T>(this IEnumerable<T> source, Func<T, IEnumerable<T>> fnRecurse) 
    { 
     foreach (T item in source) 
     { 
      yield return item; 

      IEnumerable<T> seqRecurse = fnRecurse(item); 

      if (seqRecurse != null) 
      { 
       foreach (T itemRecurse in Traverse(seqRecurse, fnRecurse)) 
       { 
        yield return itemRecurse; 
       } 
      } 
     } 
    } 

被破坏实际的代码:(x.Children突出显示与该错误)

Cannot implicitly convert type 
'System.Collections.ObjectModel.ObservableCollection<HeirarchicalItem<tbl_Path>>' to 
'System.Collections.Generic.IEnumerable<HeirarchicalPath>'. An explicit conversion 
exists (are you missing a cast?) 


      HeirarchicalPath currentItem = this.Paths.Traverse(x => x.Children).Where(x => x.Item.FullPath == "$/MyFolder/Hello").FirstOrDefault(); 

回答

0

了它。在发布问题后,我一直在这一整天和几分钟内工作,我一如既往地解决它。

只需要将此位添加到我的具体实现中,不再有编译器错误。

private ObservableCollection<HeirarchicalPath> _children = new ObservableCollection<HeirarchicalPath>(); 
    public new ObservableCollection<HeirarchicalPath> Children 
    { 
     get 
     { 
      return _children; 
     } 
     set 
     { 

      if (value == null) 
       return; 

      _children = value; 
      RaisePropertyChanged<HeirarchicalPath>(a => a.Children); 
      OnCollectionChanged(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset)); 
     } 
    }