x: UWP应用程序中的绑定和嵌套类

本文关键字:绑定 嵌套 UWP 应用程序 | 更新日期: 2023-09-27 18:20:48

我目前正在开发一个UWP应用程序,我想使用编译后的绑定系统。

我有一个基,它扩展了包含属性ViewModelWindows.UI.Xaml.Controls.Page

这里是ViewModel属性的基类:

 public class BaseViewModel : INotifyPropertyChanged
 {
    public event PropertyChangedEventHandler PropertyChanged;
    protected void RaiseOnPropertyChanged([CallerMemberName] string propertyName = "")
    {
      OnPropertyChanged(propertyName);
    }
    public void OnPropertyChanged(string propertyName)
    {
      if (PropertyChanged != null)
      {
        PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
      }
    }
  }

这里是我的主页:

public abstract class BasePage : Page
{
  public BaseViewModel ViewModel { get; set; }
}

我的应用程序的页面扩展了BasePage,并包含一个扩展BaseViewModel类的nester(内部)类。这里有一个示例代码:

public sealed partial class MyPage : BasePage
{
  public sealed class MyViewModel : BaseViewModel
  {
      public string Title
      {
        get
        {
          return "Test";
        }
      }
    }
    public MyPage()
    {
      InitializeComponent();
      ViewModel = MyViewModel();
    }
}

现在,我想将MyViewModel类的属性Title绑定到我的UI。根据这篇文章和这篇文章,类似的东西应该起作用:

<TextBlock 
  Text="{x:Bind ViewModel.(namespace:MyPage+MyViewModel.Title)}"  
/>

不幸的是,我无法编译。由于"+"字符,我在生成的文件MyPage.g.cs上有几个错误。你知道UWP应用程序是否支持对嵌套(内部)类的绑定吗?也许它只在WPF应用程序上受支持?:(

谢谢你的帮助!

x: UWP应用程序中的绑定和嵌套类

解决方案是使用一个新属性,以避免对XAML文件进行强制转换和嵌套类访问。因此,解决方案是使用类似的东西:

public sealed partial class MainPage : BasePage
{
  public sealed class MyViewModel : BaseViewModel
  {
    public string Title
    {
      get
      {
        return "Test";
      }
    }
  }
  public MyViewModel LocalViewModel
  {
    get
    {
      return (MyViewModel) ViewModel;
    }
  }
  public MainPage()
  {
    this.InitializeComponent();
    ViewModel = new MyViewModel();
  }
}

因此,使用x:Bind语法,XAML看起来像:

<TextBlock Text="{x:Bind LocalViewModel.Title}" />

{x:Bind}中使用"+"字符进行强制转换时出现一些问题。当在MyPage.xaml中使用{x:Bind ViewModel.(local:MyPage+MyViewModel.Title)}时,它会在MyPage.g.cs生成如下代码来更新绑定数据:

private void Update_ViewModel(global::UWP.BaseViewModel obj, int phase)
{
    if (obj != null)
    {
        if ((phase & (NOT_PHASED | (1 << 0))) != 0)
        {
            this.Update_ViewModel__local_MyPage+MyViewModel_Title_(((global::UWP.MyPage.MyViewModel)(obj)).Title, phase);
        }
    }
}
private void Update_ViewModel__local_MyPage+MyViewModel_Title_(global::System.String obj, int phase)
{
    if((phase & ((1 << 0) | NOT_PHASED )) != 0)
    {
        XamlBindingSetters.Set_Windows_UI_Xaml_Controls_TextBlock_Text(this.obj2, obj, null);
    }
}

虽然生成的强制转换是正确的,但是方法名称Update_ViewModel__local_MyPage+MyViewModel_Title_是无效的。因此,目前还不支持在{x:Bind}中强制转换嵌套类。

如果你想使用{x:Bind},你可以使用MyViewModel而不是BaseViewModel,如下所示,它会很好地工作。

在后面的代码中:

public sealed partial class MyPage : BasePage
{
    public MyViewModel myViewModel;
    public sealed class MyViewModel : BaseViewModel
    {
        public string Title
        {
            get
            {
                return "Test";
            }
        }
    }
    public MyPage()
    {
        InitializeComponent();
        myViewModel = new MyViewModel();
    }
}

在XAML中:

<TextBlock Text="{x:Bind myViewModel.Title}" />

此外,如果要使用BaseViewModel,则可以使用{Binding}而不是{x:Bind},因为{Binding}使用通用运行时对象检查。

在后面的代码中:

public sealed partial class MyPage : BasePage
{
    public sealed class MyViewModel : BaseViewModel
    {
        public string Title
        {
            get
            {
                return "Test";
            }
        }
    }
    public MyPage()
    {
        InitializeComponent();
        ViewModel = new MyViewModel();
        this.DataContext = this;
    }
}

在XAML中:

<TextBlock Text="{Binding ViewModel.Title}" />