1

次のような単純な Item クラスがあります。

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 ウィンドウで、Item オブジェクト (List< Item >) のリストを保持する DependencyProperty (Items と呼ばれる) を宣言し、この XAML コードを使用してこの DependencyProperty にバインドする ComboBox を作成します。

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

以下のコードを 1 回実行すると、データバインディングは正常に機能しますが、再度実行すると値の変換が失敗します。

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 メソッドに、Item オブジェクトではなく最初のパラメーターとして文字列オブジェクトが渡されることです。

私は何を間違っていますか?

よろしく、ケネス

4

3 に答える 3

0

代替アプローチ、

  1. DependencyObjectの代わりにObjectからクラスを派生させることができます
  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;
    }
}

}

これで、converterの代わりに「DisplayName」プロパティのみをバインドできます。

  1. コンバーターの実装はかなり複雑です
  2. また、DependencyObjectベースのクラスはUIの目的でのみ使用する必要があります
于 2009-06-26T12:36:40.153 に答える
0

簡単な回避策は、ValueConverter に渡された型を確認することです。Convert メソッドを次のように変更します。

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

データバインディングを使用する場合は、コレクションをItemsではなくItemsSourceに割り当てる必要があります。それでおそらくこの問題は解決すると思います。

于 2009-06-26T12:29:08.997 に答える