wpf绑定全局静态变量并且实现变更通知(mvvm)

照搬代老师的博客,这篇文章能解决一个大问题:单例怎么实现WPF数据绑定。

在实际的开发中,有一些集合或者属性可能是全局的,比如当你做一个oa的时候,可能需要展示所有的人员,这时这个所有的人员列表显然可以作为全局参数,比如这里有一个全局的静态属性UserList。而你在使用mvvm做wpf开发的时候,一般每个view都已经指定好了viewmodel。而viewmodel显然是不包含UserList。这时如果你想在绑定了viewmodel里面绑定UserList给某个ListBox该怎么办呢。首先你可以在你的viewmodel里面定义一个UserList指向全局静态变量的引用,就像下面这样:

public ObservableCollection<User> UserList{ get=>return Global.UserList; }

这是一个方法,但这个方法会让你的viewmodel变得臃肿。实际上wpf是支持静态参数的绑定的,在xaml里面这样写:

ItemsSource="{Binding Source={x:Static local:GlobalData.UserList}}"

或者:

ItemsSource="{x:Static local:GlobalData.UserList}"

这样,当你的任何一个模块,改变了UserList里面的东西,所有前端用到这个列表的,都会收到属性变更通知。

我会建议这个全局静态类写成单例模式的,比如下面这样写:

public class GlobalData : INotifyPropertyChanged
{
    private GlobalData()
    { }
    public static GlobalData Instance { private set; get; } = new GlobalData();
    public event PropertyChangedEventHandler PropertyChanged;
    private string title;
    public string Title
    {
        get => title;
        set
        {
            title = value;
            PropertyChanged?.Invoke(null, 
                new PropertyChangedEventArgs(nameof(Title)));
        }
    }
}

绑定时如下:

Text="{Binding Title, Source={x:Static local:GlobalData.Instance}}"

这样做的好处是,当Title被修改了,前端能够自动修改。
再比如,你要在某个件上绑定System.DateTime.Now,则这么写:

<TextBlock Grid.Row="1" Text="{Binding Source={x:Static 
system:DateTime.Now},StringFormat='yyyy-MM-dd HH:mm:ss '}"/>

当然,要定义一下system:

xmlns:system="clr-namespace:System;assembly=mscorlib"

事实上这种绑定是无意义的,因为Now只是普通的属性,并没有实现属性变更通知,因此你的textblock显示的时间不过是第一次加载的时间。它并不会随着时间的变化而变化。
这里面,source的作用其实就是指定源,通常我们绑定,只需要设定path,比如path=Name或者直接绑定某个属性,比如Text=“{Binding Name}”,这样系统会自动从你设定的DataContext里面找Name这个属性。而如果你指定了Source,系统便不会在你的DataContext里面查找,而是从指定的Source里面查找。因此,假设你的GlobalData里面有一个静态实体(Model),它里面有一个UserList,那么你应该这么写:

ItemsSource="{Binding UserList, Source={x:Static local:GlobalData.Model}}"

上面的单例也可以写成传统的带GetInstance方法的单例方式:

public class GlobalData : INotifyPropertyChanged
{
    private GlobalData()
    { }
    private static GlobalData instance = new GlobalData();
    public static GlobalData GetInstance() => instance;
    public event PropertyChangedEventHandler PropertyChanged;
    private string title;
    public string Title
    {
        get => title;
        set
        {
            title = value;
            PropertyChanged?.Invoke(null, 
                new PropertyChangedEventArgs(nameof(Title)));
        }
    }
}

那么此时绑定的时候要绑定到具体的方法上,如下:

<Window.Resources>
        <ObjectDataProvider ObjectType="{x:Type local:GlobalData}" 
                            MethodName="GetInstance" x:Key="GlobalData"/>
    </Window.Resources>
<TextBlock Text="{Binding Source={StaticResource GlobalData}, Path=Title}"/>

自定义控件绑定会出现失败的情况,这不是单例绑定的问题,而是自定义控件的问题

//这种写法只能应对ViewModel,遇到单例会失败,系统提示在ViewModel中找不到属性
Text="{Binding FeedSpeed,Mode=TwoWay}"
//这种控件就可以应对,具体原理不清除,先照抄代老师的代码解决问题。
Text="{Binding FeedSpeed,RelativeSource={RelativeSource Mode=FindAncestor,AncestorType=local:BladeActionData},Mode=TwoWay}"

————————————————
版权声明:本文为CSDN博主「returnTrue999」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/dap769815768/article/details/82149363

你可能感兴趣的:(WPF,C#,wpf)