无法在移动到UserControl时使绑定工作

huangapple go评论115阅读模式
英文:

Can't get binding to work when moving to UserControl

问题

这里是翻译好的部分:

  1. 我们有一些想要重用的XAML。在这个示例中,我将它简化成了一个文本框。
  2. ```xaml
  3. <TextBlock Text="{Binding MyProperty.Value}"/>

我们想要重用这个组件,所以我将它放在了一个UserControl中。(不确定这是否是正确的方法)请注意,数据绑定仍然是相同的;"{Binding MyProperty.Value}"

  1. <controls:MyControl ControlValue="{Binding MyProperty.Value}"/>

这个组件工作正常,如果我将一个常数(例如"25")放在绑定的位置,它会按预期更新。
但它不会响应模型的更改。

为什么原始XAML中可以工作,但在UserControl中不行?我漏掉了什么?

MyControl.xaml

  1. <UserControl
  2. x:Class="Project.UI.Elements.Common.Controls.MyControl"
  3. xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
  4. xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
  5. xmlns:local="using:Project.UI.Elements.Common.Controls"
  6. xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
  7. xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
  8. mc:Ignorable="d"
  9. d:DesignHeight="25"
  10. d:DesignWidth="100">
  11. <TextBlock Text="{Binding ControlProperty}"/>
  12. </UserControl>

MyControl.xaml.cs

  1. using Windows.UI.Xaml;
  2. using Windows.UI.Xaml.Controls;
  3. // 用户控件项目模板在 https://go.microsoft.com/fwlink/?LinkId=234236 中有文档。
  4. namespace Project.UI.Elements.Common.Controls
  5. {
  6. public sealed partial class MyControl : UserControl
  7. {
  8. public static readonly DependencyProperty ControlValueProperty =
  9. DependencyProperty.Register("ControlValue", typeof(double), typeof(MyControl), new PropertyMetadata(0.0));
  10. public MyControl()
  11. {
  12. this.InitializeComponent();
  13. DataContext = this;
  14. }
  15. public double ControlValue
  16. {
  17. get { return (double)GetValue(ControlValueProperty); }
  18. set { SetValue(ControlValueProperty, value); }
  19. }
  20. }
  21. }
英文:

We had a bit of XAML that we wanted to reuse. In this example, I've reduced it to a textbox.

  1. &lt;TextBlock Text=&quot;{Binding MyProperty.Value}&quot;/&gt;

We want to reuse the component, so I put it in a UserControl. (Not sure if this is the right way) Note that the data binding is still the same; &quot;{Binding MyProperty.Value}&quot;

  1. &lt;controls:MyControl ControlValue=&quot;{Binding MyProperty.Value}&quot;/&gt;

The component works, in that if I put a constant (e.g. &quot;25&quot;) instead of the binding, it is updated as expected.
But it doesn't respond to changes in the model.

Why does it work in the original XAML, but not with the UserControl? What am I missing here?

MyControl.xaml

  1. &lt;UserControl
  2. x:Class=&quot;Project.UI.Elements.Common.Controls.MyControl&quot;
  3. xmlns=&quot;http://schemas.microsoft.com/winfx/2006/xaml/presentation&quot;
  4. xmlns:x=&quot;http://schemas.microsoft.com/winfx/2006/xaml&quot;
  5. xmlns:local=&quot;using:Project.UI.Elements.Common.Controls&quot;
  6. xmlns:d=&quot;http://schemas.microsoft.com/expression/blend/2008&quot;
  7. xmlns:mc=&quot;http://schemas.openxmlformats.org/markup-compatibility/2006&quot;
  8. mc:Ignorable=&quot;d&quot;
  9. d:DesignHeight=&quot;25&quot;
  10. d:DesignWidth=&quot;100&quot;&gt;
  11. &lt;TextBlock Text=&quot;{Binding ControlProperty}&quot;/&gt;
  12. &lt;/UserControl&gt;

MyControl.xaml.cs

  1. using Windows.UI.Xaml;
  2. using Windows.UI.Xaml.Controls;
  3. // The User Control item template is documented at https://go.microsoft.com/fwlink/?LinkId=234236
  4. namespace Project.UI.Elements.Common.Controls
  5. {
  6. public sealed partial class MyControl : UserControl
  7. {
  8. public static readonly DependencyProperty ControlValueProperty =
  9. DependencyProperty.Register(&quot;ControlValue&quot;, typeof(double), typeof(MyControl), new PropertyMetadata(0.0));
  10. public MyControl()
  11. {
  12. this.InitializeComponent();
  13. DataContext = this;
  14. }
  15. public double ControlValue
  16. {
  17. get { return (double)GetValue(ControlValue); }
  18. set { SetValue(ControlValue, value); }
  19. }
  20. }
  21. }

答案1

得分: 1

根据您提供的当前场景,建议在UserControl中使用**{x:bind}**。

MainPage.xaml

  1. <TextBlock Text="{Binding MainPageControlValue, Mode=OneWay}"/>
  2. <local:MyControl ControlValue="{Binding MainPageControlValue, Mode=OneWay}"/>

MainPage.xaml.cs

  1. public static readonly DependencyProperty MainPageValueProperty =
  2. DependencyProperty.Register("MainPageControlValue", typeof(string), typeof(MainPage), new PropertyMetadata("default"));
  3. public MainPage()
  4. {
  5. this.InitializeComponent();
  6. this.DataContext = this;
  7. }
  8. public string MainPageControlValue
  9. {
  10. get { return (string)GetValue(MainPageValueProperty); }
  11. set { SetValue(MainPageValueProperty, value); }
  12. }

MyControl.xaml

  1. <Grid>
  2. <TextBlock Text="{x:Bind ControlValue, Mode=OneWay}"/>
  3. </Grid>

MyControl.xaml.cs

  1. public sealed partial class MyControl : UserControl
  2. {
  3. public static readonly DependencyProperty ControlValueProperty =
  4. DependencyProperty.Register("ControlValue", typeof(string), typeof(MyControl), new PropertyMetadata("default"));
  5. public MyControl()
  6. {
  7. this.InitializeComponent();
  8. }
  9. public MyViewModel VM { get; set; }
  10. public string ControlValue
  11. {
  12. get { return (string)GetValue(ControlValueProperty); }
  13. set { SetValue(ControlValueProperty, value); }
  14. }
  15. }

在尝试使用x:bind时,请不要在UserControl中使用DataContext = this,否则将会出现以下绑定错误。

MainPageControlValue UserControlTest.MyControl.ControlValue, Name='myControl' String BindingExpression path error: 'MainPageControlValue' property not found on 'UserControlTest.MyControl'.

对于x:bind,DataContext是当前页面,它将在当前Code-Behind类中查找属性。但是对于Binding,默认情况下,它假定您正在绑定到标记页的DataContext。

如果您需要使用Binding,建议使用自定义控件。您可以参考这个线程

顺便说一下,在Visual Studio中调试时,有XAML Binding Failures Window和APP的顶部提示。

无法在移动到UserControl时使绑定工作

无法在移动到UserControl时使绑定工作

英文:

According to the current scenario you provided, it is recommended to use {x:bind} in UserControl.

MainPage.xaml

  1. &lt;TextBlock Text=&quot;{Binding MainPageControlValue, Mode=OneWay}&quot;/&gt;
  2. &lt;local:MyControl ControlValue=&quot;{Binding MainPageControlValue, Mode=OneWay}&quot;/&gt;

MainPage.xaml.cs

  1. public static readonly DependencyProperty MainPageValueProperty =
  2. DependencyProperty.Register(&quot;MainPageControlValue&quot;, typeof(string), typeof(MainPage), new PropertyMetadata(&quot;default&quot;));
  3. public MainPage()
  4. {
  5. this.InitializeComponent();
  6. this.DataContext = this;
  7. }
  8. public string MainPageControlValue
  9. {
  10. get { return (string)GetValue(MainPageValueProperty); }
  11. set { SetValue(MainPageValueProperty, value); }
  12. }

MyControl.xaml

  1. &lt;Grid&gt;
  2. &lt;TextBlock Text=&quot;{x:Bind ControlValue, Mode=OneWay}&quot;/&gt;
  3. &lt;/Grid&gt;

MyControl.xaml.cs

  1. public sealed partial class MyControl : UserControl
  2. {
  3. public static readonly DependencyProperty ControlValueProperty =
  4. DependencyProperty.Register(&quot;ControlValue&quot;, typeof(string), typeof(MyControl), new PropertyMetadata(&quot;default&quot;));
  5. public MyControl()
  6. {
  7. this.InitializeComponent();
  8. }
  9. public MyViewModel VM { get; set; }
  10. public string ControlValue
  11. {
  12. get { return (string)GetValue(ControlValueProperty); }
  13. set { SetValue(ControlValueProperty, value); }
  14. }
  15. }

Please don't use DataContext = this in UserControl When you try to use x:bind, otherwise you will get the following binding error.

> MainPageControlValue UserControlTest.MyControl.ControlValue,
> Name='myControl' String BindingExpression path error:
> 'MainPageControlValue' property not found on
> 'UserControlTest.MyControl'

For x:bind, the DataContext is current page, it will find the property in the current Code-Bebind class. But for Binding, assumes, by default, that you're binding to the DataContext of your markup page.

If you need to use Binding, it is recommended that you use Custom Control. You can refer to this thread.

By the way, there are XAML Binding Failures Window in Visual Studio and Top tips on APP when debugging.

无法在移动到UserControl时使绑定工作

无法在移动到UserControl时使绑定工作

答案2

得分: 0

BatteryLevelControl类中,您将数据上下文设置为this。但是,由于该类是从UserControl派生的,因此不支持INotifyPropertyChanged,因此不会接收到任何更改。尝试将数据上下文设置为您从INotifyPropertyChanged派生的类,该类包含您的依赖属性。

英文:

in the BatteryLevelControl class, you set the data context to this. However, since the class is derived from UserControl, it does not support the INotifyPropertyChanged and thus does not receive any changes. Try setting the data context to your class that was derived from INotifyPropertyChanged and that contains your dependency properties.

huangapple
  • 本文由 发表于 2023年8月9日 14:23:02
  • 转载请务必保留本文链接:https://go.coder-hub.com/76865080-2.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定