2010-08-30 33 views
2

在Silverlight MVVMLight 4.0应用程序中,我有一个列表框,一个文本框和一个复选框。 列表框的ItemsSource绑定到viewmodel中的对象列表。 列表框的SelectedItem对视图模型中的对象(SelectedActivity)进行双向绑定。Silverlight MVVM - Twoway绑定未在Listbox上触发点击

文本框的Text和复选框的IsSelected属性都是双向绑定到ViewModel中的SelectedActivity对象(Name和Selected properties)。 没有隐藏代码。

这工作正常:更改文本框中的名称或选中/取消选中复选框,然后选项卡将更改该对象的基础属性。

但是,当我更改名称(或选中状态),然后立即单击列表中的另一个项目时,更改未注册。

有没有人有这方面的解决方法?

亲切的问候,

卡雷尔

这是XAML:

<ListBox Height="251" HorizontalAlignment="Left" Margin="11,39,0,0" Name="activitiesListBox" ItemsSource="{Binding Activities.Items}" VerticalAlignment="Top" Width="139" 
      SelectedItem="{Binding Activities.SelectedActivity, Mode=TwoWay}"> 

这是活动类保持绑定到列表中的项目:

public class CLJActivitiesViewModel : ViewModelBase 
{ 
    /// <summary> 
    /// Initializes a new instance of the ActivitiesViewModel class. 
    /// </summary> 
    public CLJActivitiesViewModel() 
    { 
     ////if (IsInDesignMode) 
     ////{ 
     //// // Code runs in Blend --> create design time data. 
     ////} 
     ////else 
     ////{ 
     //// // Code runs "for real": Connect to service, etc... 
     ////} 
    } 


    #region items 
    /// <summary> 
    /// The <see cref="Items" /> property's name. 
    /// </summary> 
    public const string ItemsPropertyName = "Items"; 

    private ObservableCollection<CLJActivityViewModel> m_Items = null; 

    /// <summary> 
    /// Gets the Items property. 
    /// TODO Update documentation: 
    /// Changes to that property's value raise the PropertyChanged event. 
    /// This property's value is broadcasted by the Messenger's default instance when it changes. 
    /// </summary> 
    public ObservableCollection<CLJActivityViewModel> Items 
    { 
     get 
     { 
      return m_Items; 
     } 

     set 
     { 
      if (m_Items == value) 
      { 
       return; 
      } 

      var oldValue = m_Items; 
      m_Items = value; 

      RaisePropertyChanged(ItemsPropertyName, oldValue, value, true); 
     } 
    } 
    #endregion 

    #region SelectedActivity 
    /// <summary> 
    /// The <see cref="SelectedActivity" /> property's name. 
    /// </summary> 
    public const string SelectedActivityPropertyName = "SelectedActivity"; 

    private CLJActivityViewModel m_SelectedActivity = null; 

    /// <summary> 
    /// Gets the SelectedActivity property. 
    /// TODO Update documentation: 
    /// Changes to that property's value raise the PropertyChanged event. 
    /// This property's value is broadcasted by the Messenger's default instance when it changes. 
    /// </summary> 
    public CLJActivityViewModel SelectedActivity 
    { 
     get 
     { 
      return m_SelectedActivity; 
     } 

     set 
     { 
      if (m_SelectedActivity == value) 
      { 
       return; 
      } 

      var oldValue = m_SelectedActivity; 
      m_SelectedActivity = value; 

      RaisePropertyChanged(SelectedActivityPropertyName, oldValue, value, true); 
     } 
    } 
    #endregion 



    public override void Cleanup() 
    { 
     // Clean own resources if needed 

     base.Cleanup(); 
    } 
}   
+0

这将有助于如果您发布至少是你的viewmodel/view代码的相关部分。您的列表是ObservableCollection,而您的视图模型是否实现INotifyPropertyChanged? – 2010-08-30 16:44:37

+0

是的,列表绑定到一个ObservableCollection并且视图模型正在实现INotifyPropertyChanged。我使用了MVVMLight模板。 我会用相关的代码更新这个问题。 – Karel 2010-08-31 08:13:07

回答

1

我遇到了同样的问题。我必须在用户输入文本时触发更新,以便我可以进行一些验证。

实现这一目的的一种简单方法是创建一个自定义行为,然后您可以将其添加到任何TextBox

煤矿如下:

public static class TextChangedBindingBehavior 
{ 
    public static readonly DependencyProperty InstanceProperty = 
     DependencyProperty.RegisterAttached("Instance", typeof(object), typeof(TextChangedBindingBehavior), new PropertyMetadata(OnSetInstanceCallback)); 


    public static object GetInstance(DependencyObject obj) 
    { 
     return (object)obj.GetValue(InstanceProperty); 
    } 

    public static void SetInstance(DependencyObject obj, object value) 
    { 
     obj.SetValue(InstanceProperty, value); 
    } 

    private static void OnSetInstanceCallback(DependencyObject d, DependencyPropertyChangedEventArgs e) 
    { 
     var textBox = d as TextBox; 
     if (textBox != null) 
     { 
      textBox.TextChanged -= OnTextChanged; 
      textBox.TextChanged += OnTextChanged; 
     } 
    } 

    private static void OnTextChanged(object sender, TextChangedEventArgs e) 
    { 
     var textBox = (TextBox)sender; 

     if(!DesignerProperties.GetIsInDesignMode(textBox)) 
     { 
      textBox.GetBindingExpression(TextBox.TextProperty).UpdateSource(); 
     } 
    } 
} 

,你将其设置为TextBox像(Behaviors是我把类上面的命名空间):

<TextBox Behaviors:TextChangedBindingBehavior.Instance="" Text="{Binding Name, Mode=TwoWay, ValidatesOnExceptions=true, NotifyOnValidationError=true}" /> 
+0

谢谢,我会试试看。 – Karel 2010-09-01 15:42:05

+0

太棒了,完美的作品! – Karel 2010-09-01 15:47:26

1

我已经用TextBox碰到类似的问题,但没有看到它影响复选框。 TextBox问题发生,因为绑定文本得到更新,然后焦点丢失。这就是为什么如果你先选中,然后改变你的选择,它会像你期望的那样工作。如果您直接更改选择,则由于焦点丢失消息到达太迟,所以文本不会更新。

处理此问题的一种方法是每次用户键入文本框中的文本时强制绑定更新。你可以自定义行为来保持mvvm。

相关问题