9

我有从 Web 服务返回的数据,ObservableCollection<string>我想将集合绑定到只读,TextBox以便用户可以选择数据并将其复制到剪贴板。

要将集合绑定到我创建的 TextBox 的 Text 属性,该属性IValueConverter会将集合转换为文本字符串。这似乎有效,只是它只有效一次,就好像绑定无法识别对 Observable 集合的后续更改。这是一个重现问题的简单应用程序,只是为了确认绑定正常工作,我还绑定到“ListBox”

这是因为 Text binding simple 不处理集合的更改事件吗?

一种选择当然是让我处理集合更改并将这些更改传播到 TextBox 绑定到的 Text 属性,这很好,但我想了解为什么在我看来是一个明显的解决方案不起作用正如预期的那样。

XAML

<Window x:Class="WpfTextBoxBinding.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:local="clr-namespace:WpfTextBoxBinding"
        Title="MainWindow" Height="331" Width="402">
  <StackPanel>
    <StackPanel.Resources>
      <local:EnumarableToTextConverter x:Key="EnumarableToTextConverter" />
    </StackPanel.Resources>
    <TextBox Text="{Binding TextLines, Mode=OneWay, Converter={StaticResource EnumarableToTextConverter}}" Height="100" />
    <ListBox ItemsSource="{Binding TextLines}" Height="100" />
    <Button Click="Button_Click" Content="Add Line" />
  </StackPanel >
</Window>

代码背后

using System;
using System.Collections;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Text;
using System.Windows;
using System.Windows.Data;
using System.Globalization;

namespace WpfTextBoxBinding
{
  /// <summary>
  /// Interaction logic for MainWindow.xaml
  /// </summary>
  public partial class MainWindow : Window
  {
    public ObservableCollection<string> TextLines {get;set;}

    public MainWindow()
    {
      DataContext = this;

      TextLines = new ObservableCollection<string>();

      // Add some initial data, this shows that the 
      // TextBox binding works the first time      
      TextLines.Add("First Line");

      InitializeComponent();      
    }

    private void Button_Click(object sender, RoutedEventArgs e)
    {
      TextLines.Add("Line :" + TextLines.Count);
    }
  }

  public class EnumarableToTextConverter : IValueConverter
  {
    public object Convert(
      object value, Type targetType, 
      object parameter, CultureInfo culture)
    {
      if (value is IEnumerable)
      {
        StringBuilder sb = new StringBuilder();
        foreach (var s in value as IEnumerable)
        {
          sb.AppendLine(s.ToString());
        }
        return sb.ToString();
      }
      return string.Empty;
    }

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

3 回答 3

6

一种更优雅的实现方式是在 Text 属性上使用 MultiBinding 并绑定到 Collection 的 Count 属性。这将在每次集合的 Count 更改时更新绑定,并根据您定义的 MultiValueConverter 更新 Text。

<TextBox>
    <TextBox.Text>
        <MultiBinding Converter="{x:Static l:Converters.LogEntryCollectionToTextConverter}">
            <Binding Path="LogEntries" Mode="OneWay"/>
            <Binding Path="LogEntries.Count" Mode="OneWay" />
        </MultiBinding>
    </TextBox.Text>
</TextBox>

和转换器:

public static class Converters
{
    public static LogEntryCollectionToTextConverter LogEntryCollectionToTextConverter = new LogEntryCollectionToTextConverter();
}

public class LogEntryCollectionToTextConverter : IMultiValueConverter
{
    public object Convert(object[] values, Type targetType, object parameter, CultureInfo culture)
    {
        ObservableCollection<LogEntry> logEntries = values[0] as ObservableCollection<LogEntry>;

        if (logEntries != null && logEntries.Count > 0)
            return logEntries.ToString();
        else
            return String.Empty;
    }

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

在我的用例中,我不允许 TextBox 更新其源(因此是“Mode="OneWay"”),但如果需要,转换器的 ConvertBack 方法会处理它。

于 2012-01-13T08:45:09.147 回答
5

这是因为 Text binding simple 不处理集合的更改事件吗?

确实。绑定仅在其源属性更改时更新。如果您TextLines通过设置一个全新的ObservableCollection和 implement来更改属性INotifyPropertyChanged,您的绑定将按预期工作。ItemsControl.ItemsSource仅当将新元素绑定到诸如侦听集合更改的属性时,向集合添加新元素才有意义。

一种选择当然是让我处理集合更改并将这些更改传播到 TextBox 绑定到的 Text 属性,这很好。

那将是另一种解决方案。

于 2010-12-04T11:16:58.593 回答
0

更新以下代码

  private void Button_Click(object sender, RoutedEventArgs e)
    {
        TextLines.Add("Line :" + TextLines.Count);
      BindingExpression be =  BindingOperations.GetBindingExpression(txtName, TextBox.TextProperty);
      be.UpdateTarget();
    } 

其中 txtName 是您的文本框的名称

MVVM方式

1- 在 ViewModel 中定义一个字符串类型的属性,如下所示,并将此属性绑定到如下所示的文本框文本属性 a,并立即删除不需要的 ValueConverter。

public string TextLines {get;set;}

 <TextBox Text="{Binding TextLines, Mode=OneWay/> 

2-我认为,您很可能使用命令处理程序处理按钮单击事件,说您的命令是 AddMoreLines

因此,在 AddMoreLine Command Handler 中,在 OBservrableCollection 添加新对象后,创建一个 StringBuilder 并附加 Collection 的所有内容并将字符串分配给步骤 1 中创建的属性。

3-调用 PropertyChanged 处理程序。

于 2010-12-04T11:40:22.583 回答