【问题标题】:Binding to a DependencyProperty via ViewModel has no effect通过 ViewModel 绑定到 DependencyProperty 无效
【发布时间】:2012-09-17 17:18:55
【问题描述】:

我已经简化了我的问题(意思是原理差不多,但是我不想把一个int转换成字符串……):

MainWindow.xaml:

<Window x:Class="WpfApplication1.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:WpfApplication1="clr-namespace:WpfApplication1"
        Title="MainWindow" Height="500" Width="500">
    <Grid>

        <WpfApplication1:UserControl1 CurrentNumber="{Binding Path=TheSpecialNumber, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}" />

    </Grid>
</Window>

在主窗口中出现了 UserControl1。用户控件有一个属性 Current Number。这个我想绑定属性TheSpecialNumber(ViewModel)。

MainWindow.xaml.cs:

namespace WpfApplication1
{
    public partial class MainWindow
    {
        private readonly ViewModel _viewModel;

        public MainWindow()
        {
            InitializeComponent();
            _viewModel = new ViewModel();
            DataContext = _viewModel;

            _viewModel.TheSpecialNumber = "8";
            _viewModel.UpdateTheSpecialNumberBinding();
        }
    }
}

ViewModel.cs:

namespace WpfApplication1
{
    class ViewModel : INotifyPropertyChanged
    {
        public string TheSpecialNumber { get; set; }

        public event PropertyChangedEventHandler PropertyChanged;

        public void UpdateTheSpecialNumberBinding()
        {
            if (PropertyChanged != null)
            {
                PropertyChanged(this, new PropertyChangedEventArgs("TheSpecialNumber"));
            }
        }
    }
}

UserControl1.xaml:

<UserControl x:Class="WpfApplication1.UserControl1"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
             xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
             mc:Ignorable="d"
             Height="200" Width="300" Background="Aqua">
    <Grid Name="container">
        <ComboBox SelectedIndex="{Binding Path=CurrentNumberIndex, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}">
            <ComboBoxItem Height="20">0</ComboBoxItem>
            <ComboBoxItem Height="20">1</ComboBoxItem>
            <ComboBoxItem Height="20">2</ComboBoxItem>
            <ComboBoxItem Height="20">3</ComboBoxItem>
            <ComboBoxItem Height="20">4</ComboBoxItem>
            <ComboBoxItem Height="20">5</ComboBoxItem>
            <ComboBoxItem Height="20">6</ComboBoxItem>
            <ComboBoxItem Height="20">7</ComboBoxItem>
            <ComboBoxItem Height="20">8</ComboBoxItem>
            <ComboBoxItem Height="20">9</ComboBoxItem>
        </ComboBox>
    </Grid>
</UserControl>

UserControl 是一个组合框。这有 10 个项目。我想始终显示与属性当前编号具有相同编号的项目。因此,当前数字索引的绑定。

UserControl1.xaml.cs:

using System;
using System.ComponentModel;
using System.Windows;

namespace WpfApplication1
{
    public partial class UserControl1 : INotifyPropertyChanged
    {
        public static readonly DependencyProperty CurrentNumberProperty = DependencyProperty.Register("CurrentNumber", typeof(string), typeof(UserControl1));

        public UserControl1()
        {
            InitializeComponent();
            container.DataContext = this;

            CurrentNumber = "5";
        }

        public string CurrentNumber
        {
            get { return (string)GetValue(CurrentNumberProperty); }
            set
            {
                SetValue(CurrentNumberProperty, value);
                OnPropertyChanged("CurrentNumberIndex");
            }
        }

        public int CurrentNumberIndex
        {
            get
            {
                return Convert.ToInt32(CurrentNumber);
            }
            set
            {
                CurrentNumber = Convert.ToString(value);
            }
        }

        #region Implementation of INotifyPropertyChanged

        public event PropertyChangedEventHandler PropertyChanged;

        private void OnPropertyChanged(string propertyName)
        {
            if (PropertyChanged != null)
            {
                PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
            }
        }

        #endregion Implementation of INotifyPropertyChanged
    }
}

现在我的问题。如果我在 MainWindow(通过 ViewModel)设置 8,为什么只显示内容为 0 的 ComboBox 项?即使我在当前编号的设置器中设置断点,我也只停止一次,即在 5 处,而不是在 8 处。

【问题讨论】:

    标签: wpf mvvm binding dependency-properties code-behind


    【解决方案1】:

    您可以通过手动设置 CurrentNumber=5 来“破坏” TheSpecialNumber 和 CurrentNumber 之间的绑定。删除此部分,然后重试。

    编辑: 好的,如果这不起作用,请尝试以下操作:

    1. 向您的 PropertyMetaData 添加一个新的 PropertyChangedCallback:

      public static readonly DependencyProperty CurrentNumberProperty = DependencyProperty.Register("CurrentNumber", typeof(string), typeof(UserControl1), new PropertyMetadata( new PropertyChangedCallback(CurrentNumberPropertyChanged))); 
      
    2. 在此方法中设置 CurrentNumber,如:

      private static void CurrentNumberPropertyChanged(DependencyObject dependencyObject, DependencyPropertyChangedEventArgs dependencyPropertyChangedEventArgs)
      {
          UserControl1 uc1 = (UserControl1)dependencyObject;
          uc1.CurrentNumber = (string)dependencyPropertyChangedEventArgs.NewValue;
      }
      

    【讨论】:

    • 如果我删除CurrentNumber = "5";,什么都不会改变
    • 查看我的编辑(基于 ChrisDD 的解决方案)。
    • 差不多,但我的意思是您需要将 OnPropertyCHanged("CurrentIndexNumber") 添加到 PropertyChanged 回调中。
    • 但是通过 uc1.CurrentNumber = (string)dependencyPropertyChangedEventArgs.NewValue;调用 CurrentNumber 的 setter,其中包括 OnPropertyChanged("CurrentNumberIndex");
    【解决方案2】:

    CurrentNumber 是 WPF 属性,这意味着如果 WPF 使用该属性,它将传递 CurrentNumber 的 getter/setter。

    所以基本上当你设置 TheSpecialNumber 时,这部分不会被执行:

     SetValue(CurrentNumberProperty, value);
     OnPropertyChanged("CurrentNumberIndex");
    

    将自己与 WPF 属性 UI 属性元数据挂钩。

    【讨论】:

    • 我是否理解正确:我应该将 5 作为 FrameworkPropertyMetadata 吗? DependencyProperty.Register("CurrentNumber", typeof(string), typeof(UserControl1), new FrameworkPropertyMetadata("5"));
    • 查看 Florian 的答案,但将 OnPropertyChanged 添加到 CurrentNumberPropertyChanged 事件处理程序中。
    【解决方案3】:
    <ComboBox SelectedIndex="{Binding Path=CurrentNumberIndex, RelativeSource={RelativeSource Mode=Self}, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}"/>
    

    在 SelectedIndex 的绑定中,绑定系统将在您的 ViewModel 的 UserControl DataContext 中查找 CurrentNumberIndex 属性,并且它没有在那里找到此属性,因此将其工作集 RelativeSource 设置为 Self。我希望这会有所帮助。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2021-12-10
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2011-10-03
      • 2023-03-19
      相关资源
      最近更新 更多