WPF привязка строкового свойства и Button.Content
Я хочу связать строковое свойство с Button.Content.
Но почему это не сработало?
Класс данных:
namespace test4
{
public class Test : INotifyPropertyChanged
{
string _Text = "Begin";
public string Text
{
get{return _Text;}
protected set { _Text = value; }
}
public void Start()
{
Text = "Begin";
if (PropertyChanged != null)
PropertyChanged(this, new PropertyChangedEventArgs(Text));
}
public void End()
{
Text = "End";
if (PropertyChanged != null)
PropertyChanged(this, new PropertyChangedEventArgs(Text));
}
public event PropertyChangedEventHandler PropertyChanged;
}
}
Логический код:
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
test4.Test ttt = new test4.Test();
private void Window_Loaded(object sender, RoutedEventArgs e)
{
System.Windows.Data.CollectionViewSource testViewSource = ((System.Windows.Data.CollectionViewSource)(this.FindResource("testViewSource")));
testViewSource.Source = new object[]{ttt};
}
private void button1_Click(object sender, RoutedEventArgs e)
{
if (ttt.Text == "Begin")
ttt.End();
else
ttt.Start();
}
}
XAML:
<Window x:Class="test5.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="MainWindow" Height="350" Width="525" mc:Ignorable="d" xmlns:d="http://schemas.microsoft.com/expression/blend/2008" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" xmlns:my="clr-namespace:test4" Loaded="Window_Loaded">
<Window.Resources>
<CollectionViewSource x:Key="testViewSource" d:DesignSource="{d:DesignInstance my:Test, CreateList=true}" />
</Window.Resources>
<Grid DataContext="{StaticResource testViewSource}">
<Button Content="{Binding Path=Text, Mode=OneWay, UpdateSourceTrigger=PropertyChanged}" Height="23" HorizontalAlignment="Left" Margin="158,95,0,0" Name="button1" VerticalAlignment="Top" Width="75" Click="button1_Click" />
</Grid>
</Window>
4 ответа
Кавычки вокруг Text в вызове конструктора PropertyChangedEventArgs отсутствуют:
Text = "Begin";
if (PropertyChanged != null)
PropertyChanged(this, new PropertyChangedEventArgs("Text"));
PropertyChanged
хочет увидеть имя свойства, которое изменилось, а не значение. Измените вызов события на:
PropertyChanged(this, new PropertyChangedEventArgs("Text"));
и это сделает работу. Однако я бы также изменил конструкцию на
public string Text {
get{return _Text;}
protected set {
_Text = value;
if(null != PropertyChanged){
PropertyChanged(this,new PropertyChangedEventArgs("Text"));
}
}
}
и затем не вызывайте PropertyChanged-событие из начала и конца.
И чтобы пойти еще дальше, создайте методы вызова, такие как:
protected virtual void OnPropertyChanged(string propertyName) {
OnPropertyChanged(new PropertyChangedEventArgs(propertyName));
}
protected virtual void OnPropertyChanged(PropertyChangedEventArgs e) {
if (null != PropertyChanged) {
PropertyChanged(this,e);
}
}
а затем позвоните им из вашего свойства-установщик.
Вы должны использовать кавычки вокруг свойства:
public void Start()
{
Text = "Begin";
if (PropertyChanged != null)
PropertyChanged(this, new PropertyChangedEventArgs("Text"));
}
Я предпочитаю использовать метод nameof(), поскольку вы не можете переименовать свойство, если оно находится между кавычками:
public void Start()
{
Text = "Begin";
if (PropertyChanged != null)
PropertyChanged(this, new PropertyChangedEventArgs(nameof(Text)));
}
Прежде всего, я использовал objectdataproviding, но это не объект "ttt", это две разные вещи.
Во-вторых, PropertyChanged(this,new PropertyChangedEventArgs("Text"));
"Текст" - это имя, а не переменная.
Итак, код следующий, может быть полезным для других.
Класс данных:
namespace test3
{
public class Test : INotifyPropertyChanged
{
string _Text = "Begin";
public string Text
{
get{return _Text;}
protected set { _Text = value;
NotifyPropertyChanged("Text");
}
}
public void Start()
{
Text = "Begin";
}
public void End()
{
Text = "End";
}
public event PropertyChangedEventHandler PropertyChanged;
public void NotifyPropertyChanged(string propertyName)
{
if (PropertyChanged != null)
{
PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
}
}
}
}
Логический CS:
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
button1.DataContext=ttt;
}
Test ttt = new Test();
private void button1_Click(object sender, RoutedEventArgs e)
{
if (ttt.Text == "Begin")
ttt.End();
else
ttt.Start();
}
}
}
XAML:
<Window x:Class="test3.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:test3"
Title="MainWindow" Height="350" Width="525" >
<Grid>
<Button Content="{Binding Path=Text,UpdateSourceTrigger=PropertyChanged}" Height="23" HorizontalAlignment="Left" Margin="121,69,0,0" Name="button1" VerticalAlignment="Top" Width="75" Click="button1_Click" />
</Grid>
</Window>