1

我有一个简单的 Item-class,看起来像这样:

public class Item : DependencyObject
{
    public int No
    {
        get { return (int)GetValue(NoProperty); }
        set { SetValue(NoProperty, value); }
    }

    public string Name
    {
        get { return (string)GetValue(NameProperty); }
        set { SetValue(NameProperty, value); }
    }

    public static readonly DependencyProperty NoProperty = DependencyProperty.Register("No", typeof(int), typeof(Item));
    public static readonly DependencyProperty NameProperty = DependencyProperty.Register("Name", typeof(string), typeof(Item));
}

还有一个 ValueConverter,看起来像这样:

[ValueConversion(typeof(Item), typeof(string))]
internal class ItemToStringConverter : IValueConverter
{
    public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
    {
        if (value == null)
        {
            return null;
        }

        var item = ((Item)value);

        var sb = new StringBuilder();
        sb.AppendFormat("Item # {0}", item.No);

        if (string.IsNullOrEmpty(item.Name) == false)
        {
            sb.AppendFormat(" - [{0}]", item.Name);
        }

        return sb.ToString();
    }


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

在我的 WPF 窗口中,我声明了一个 DependencyProperty(称为 Items),它包含一个 Item 对象列表(List< Item >),并使用此 XAML 代码创建一个绑定到此 DependencyProperty 的 ComboBox:

<ComboBox ItemsSource="{Binding ElementName=mainWindow, Path=Items}">
    <ComboBox.ItemTemplate>
        <DataTemplate>
            <TextBlock Text="{Binding Converter={StaticResource itemToStringConverter}}"/>
        </DataTemplate>
    </ComboBox.ItemTemplate>
</ComboBox>

如果我执行一次下面的代码,数据绑定工作正常,但是如果我再次执行它,值转换会失败:

var item1 = new Item {Name = "Item A", No = 1};
var item2 = new Item {Name = "Item B", No = 2};
var item3 = new Item {Name = "Item C", No = 3};
Items = new List<Item> {item1, item2, item3};

问题是,ItemToStringConverter.Convert 方法现在传递一个字符串对象作为第一个参数,而不是一个项目对象?

我究竟做错了什么?

问候,肯尼斯

4

3 回答 3

0

替代方法,

  1. 您可以从 Object 而不是 DependencyObject 派生您的类
  2. 您可以实现 INotifyPropertyChange 接口
  3. 您可以实现只读属性并在 No 或 Name 中的任何一个更改时触发通知事件。

公共类项目:System.ComponentModel.INotifyPropertyChanged {

public event System.ComponentModel.PropertyChangedEventHandler PropertyChanged;


private void Notify(string p)
{
    if (PropertyChanged != null)
        PropertyChanged(this,
            new System.ComponentModel.PropertyChangedEventArgs(p));
}

private int _No = 0;
public int No
{
    get
    {
        return _No;
    }
    set
    {
        _No = value;
        Notify("No");
        Notify("DisplayName");
    }
}


private string _Name = "";
public string Name
{
    get
    {
        return _Name;
    }
    set
    {
        _Name = value;
        Notify("Name");
        Notify("DisplayName");
    }
}

public string DisplayName
{
    get
    {
        string sb = string.Format("Item # {0}", _No);
        if (!string.IsNullOrEmpty(_Name))
            sb += _Name;
        return sb;
    }
}

}

现在您只能绑定“DisplayName”属性而不是转换器..

  1. 转换器实现起来相当复杂
  2. 并且基于 DependencyObject 的类只能用于 UI 目的
于 2009-06-26T12:36:40.153 回答
0

简单的解决方法是检查传递给 ValueConverter 的类型。更改转换方法如下:

public object Convert(object value, Type targetType, object parameter, CultureInfo culture) {
        var item = value as Item;
        if(item == null) { return null; }
        var sb = new StringBuilder();
        sb.AppendFormat("Item # {0}", item.No);
        if(string.IsNullOrEmpty(item.Name) == false) {
            sb.AppendFormat(" - [{0}]", item.Name);
        }
        return sb.ToString();
    }
于 2009-06-26T12:45:07.420 回答
-1

如果您想使用数据绑定,您应该将您的集合分配给 ItemsSource,而不是 Items。我认为这可能会解决这个问题。

于 2009-06-26T12:29:08.997 回答