我看到朋友的代码出现绑定了 一个 TextBlock 的 Text 的 length ,那时候我觉得 length 不是依赖属性,绑定了是无法通知的。最后我做了实验才发现,原因有 Text 可以通知。
请看简单的代码,界面就是一个 TextBlock 和两个按钮,其中一个按钮是绑定了 length 如果大于 0 才可以使用。一个按钮是把 TextBlock 的文字设置为空或者设置为任意字符串。
<Grid>
<Grid.Resources>
<local:LengthToBoolenConverter x:Key="LengthToBoolenConverter"/>
</Grid.Resources>
<Grid.RowDefinitions>
<RowDefinition Height="79*"/>
<RowDefinition Height="28*"/>
</Grid.RowDefinitions>
<TextBlock x:Name="TitleBlock" Margin="0,0,-0.6,-0.2" ></TextBlock>
<Grid Grid.Row="1">
<Button Margin="10,10,10,10" HorizontalAlignment="Left" Content="确定" Click="ButtonBase_OnClick"></Button>
<Button Margin="10,10,10,10" HorizontalAlignment="Right" IsEnabled="{Binding ElementName=TitleBlock,Path=Text.Length,Converter={StaticResource LengthToBoolenConverter}}" Content="确定"></Button>
</Grid>
</Grid>
public class LengthToBoolenConverter : IValueConverter
{
public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
{
var length = (int?) value;
return length > 0;
}
public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
{
throw new NotImplementedException();
}
}
private void ButtonBase_OnClick(object sender, RoutedEventArgs e)
{
if (string.IsNullOrEmpty(TitleBlock.Text))
{
TitleBlock.Text = "1";
}
else
{
TitleBlock.Text = "";
}
}
看到IsEnabled="{Binding ElementName=TitleBlock,Path=Text.Length,Converter={StaticResource LengthToBoolenConverter}}"
绑定就是 Text.Length ,但是 Length 不是依赖属性,没有通知,那么在 Text 变化时是否会通知?会的,因为使用的是 Text 的 Length,所以在 Text 修改时就会改变了 Length 。
如果有一个 Model 类,这个类是没有继承通知的,那么如何在里面的属性修改时,可以通知?一个方法就是在 ViewModel 使用 Model 属性,每次都是修改整个 Model
class Model
{
public string Foo { set; get}
//其他就不写了
}
class ViewModel : INotifyPropertyChanged
{
public void Foo()
{
var model = new Model()
{
Foo = Model.Foo,
//复制所有属性
};
model.Foo = "新的";
Model = model;
}
private Model _model;
public Model Model
{
get { return _model; }
set
{
_model = value;
OnPropertyChanged();
}
}
public event PropertyChangedEventHandler PropertyChanged;
protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null)
{
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
}
}
这样可以不修改 Model 就可以在修改属性通知,但是可以看到这需要复制所有属性,不过我有写了C# 使用Emit深克隆使用这个就可以做快复制,而且还是深复制,但是复制不是真的深,代码复制是一层,如果需要实现真的复制,还需要自己去写
那么不复制是否可以做?可以,只需要调用OnPropertyChanged
请把上面的代码做修改,添加一个函数,这个函数更新会自动通知。
public void Foo(string str)
{
Model.Foo = str;
OnPropertyChanged(nameof(Model));
}
在按钮点击就把代码修改为
private void ButtonBase_OnClick(object sender, RoutedEventArgs e)
{
if (string.IsNullOrEmpty(TitleBlock.Text))
{
ViewModel.Foo("1");
}
else
{
ViewModel.Foo("");
}
}
需要添加一些绑定,请看代码
public ViewModel ViewModel { get; set; } = new ViewModel();
设置了 DataContext 之后就可以在界面绑定,这时可以看到和直接设置TextBlock的文字看起来是一样
<TextBlock x:Name="TitleBlock" Margin="0,0,-0.6,-0.2" Text="{Binding Model.Foo}"></TextBlock>
本文会经常更新,请阅读原文: https://dotnet-campus.github.io//post/wpf-%E7%BB%91%E5%AE%9A-TextLength.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。
本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名 lindexi (包含链接: https://dotnet-campus.github.io/ ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系 。