14

我有一个从 TextBox 控件继承的自定义控件。我想INotifyPropertyChanged在我的自定义控件中实现接口。

public class CustomTextBox : TextBox, INotifyPropertyChanged
{
    public event PropertyChangedEventHandler PropertyChanged;
    protected void NotifyPropertyChanged(string info)
    {
        if (PropertyChanged != null)
            PropertyChanged(this, new PropertyChangedEventArgs(info));
    }
}

我的问题是当我尝试引发 PropertyChanged 事件时,PropertyChanged 事件处理程序始终为空。

任何人都可以帮助我吗?

4

3 回答 3

15

PropertyChanged 事件处理程序始终为空。

PropertyChanged在某些人订阅该事件之前,这始终是正确的。

但是,通常,如果您要制作自定义控件,则不会使用INotifyPropertyChanged. 在这种情况下,您将改为创建自定义依赖属性。通常,依赖对象(即:控件)都将使用依赖属性,而 INPC 由成为DataContext这些对象的类使用。这允许绑定系统正常工作。

于 2012-09-17T19:56:15.230 回答
10

UI 代码使用 PropertyChanged 事件,但不是您正在编写的那个意义上的事件。控件从不实现 INPC(INotifyPropertyChanged 的​​缩写),它们绑定到已实现INPC 的对象。这样,某些 UI 属性,例如 TextBox 控件上的Text属性绑定到此类的属性。这是 MVVM 架构的基础。

例如,您将编写以下 XAML 代码:

<TextBlock x:Name="txtBox" Text="{Binding Title}" />

并且您以下列方式在代码中设置 TextBlock(或其任何祖先,DataContext 被传播)的数据上下文:

txtBox.DataContext = new Movie {Title = "Titanic"};

现在对于班级本身:

public class Movie : INotifyPropertyChanged
{
    public event PropertyChangedEventHandler PropertyChanged;
    protected void NotifyPropertyChanged(string info)
    {
        if (PropertyChanged != null)
            PropertyChanged(this, new PropertyChangedEventArgs(info));
    }

    private string _title;
    public string Title
    {
        get { return _title; }
        set
        {
            if (_title == value) return;

            _title = value;
            NotifyPropertyChanged("Title");
        }
    }
}

现在,无论何时更改Title property,无论是在代码中还是通过其他绑定,UI 都会自动刷新。这是有关数据绑定MVVM的更多信息。

于 2012-09-17T20:03:13.750 回答
5

像我在评论中说的那样做。它不像你那样工作。只需创建一个额外的类并Notification Interface在那里实现并将这个类应用于DataContext. UserControl它会像你需要的那样工作。

 public partial class W3 : UserControl
    {
        WeatherViewModel model = new WeatherViewModel();

        public W3()
        {
            InitializeComponent();
            this.DataContext = model;
        }

        public void UpdateUI(List<WeatherDetails> data, bool isNextDays)
        {
            model.UpdateUI(data, isNextDays);
        }
    }

    class WeatherViewModel : INotifyPropertyChanged
    {     

        public void UpdateUI(List<WeatherDetails> data, bool isNextDays)
        {
            List<WeatherDetails> weatherInfo = data;
            if (weatherInfo.Count != 0)
            {
                CurrentWeather = weatherInfo.First();
                if (isNextDays)
                    Forecast = weatherInfo.Skip(1).ToList();
            }
        }

        private List<WeatherDetails> _forecast = new List<WeatherDetails>();

        public List<WeatherDetails> Forecast
        {
            get { return _forecast; }
            set
            {
                _forecast = value;
                OnPropertyChanged("Forecast");
            }
        }

        private WeatherDetails _currentWeather = new WeatherDetails();

        public WeatherDetails CurrentWeather
        {
            get { return _currentWeather; }
            set
            {
                _currentWeather = value;
                OnPropertyChanged("CurrentWeather");
            }
        }

        public event PropertyChangedEventHandler PropertyChanged;

        private void OnPropertyChanged(string propertyName)
        {
            PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
        }
    }
于 2016-08-12T02:03:30.300 回答