来自 ItemContainerStyle 的样式 Setter 中的 UWP 绑定不起作用

UWP Binding in Style Setter from ItemContainerStyle not working

我创建了一个 BindingHelper 来设置绑定样式 setter 就像这个例子....

但它不起作用。 谁能帮我完成这个例子或找出我的错误?

<Page
x:Class="eve.TerminView"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:eve"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d">

<Page.Resources>
    <ResourceDictionary Source="Styles.xaml" />

</Page.Resources>

<Grid Background="{StaticResource ResourceKey=StyleBackground}">

    <ItemsControl ItemsSource="{Binding Path=termindata}">
        <ItemsControl.ItemsPanel>
            <ItemsPanelTemplate>

                <Grid HorizontalAlignment="Stretch" VerticalAlignment="Stretch" Margin="50" BorderBrush="White" BorderThickness="1">
                    <Grid.RowDefinitions>
                        <RowDefinition Height="*"></RowDefinition>
                        <RowDefinition Height="*"></RowDefinition>
                        <RowDefinition Height="*"></RowDefinition>
                    </Grid.RowDefinitions>
                    <Grid.ColumnDefinitions>
                        <ColumnDefinition Width="*"></ColumnDefinition>
                        <ColumnDefinition Width="*"></ColumnDefinition>
                        <ColumnDefinition Width="*"></ColumnDefinition>
                    </Grid.ColumnDefinitions>
                </Grid>
            </ItemsPanelTemplate>
        </ItemsControl.ItemsPanel>
        <ItemsControl.ItemTemplate>
            <DataTemplate>
                <Border Background="Green">
                    <TextBlock Text="{Binding text}"></TextBlock>
                </Border>
            </DataTemplate>
        </ItemsControl.ItemTemplate>

        <ItemsControl.ItemContainerStyle>
            <Style TargetType="ContentPresenter">
                <Setter Property="local:BindingHelper.GridColumnBindingPath" Value="column"/>
                <Setter Property="local:BindingHelper.GridRowBindingPath" Value="row"/>
            </Style>
        </ItemsControl.ItemContainerStyle>

    </ItemsControl>


</Grid>

这里是 public class BindingHelper

后面的代码
namespace test
{

public sealed partial class TerminView : Page
{

    public List<TerminDate> termindata;


    public TerminView()
    {
        this.InitializeComponent();

        termindata = new List<TerminDate>();

        termindata.Add(new TerminDate(1, 1, "Test 1"));
        termindata.Add(new TerminDate(1, 2, "Test 2"));
    }
}


public class TerminDate
{

    public TerminDate(int row, int column, string text)
    {
        this.row = row;
        this.column = column;
        this.text = text;
    }

    public int row { get; set; }
    public int column { get; set; }
    public string text { get; set; }
}

public class BindingHelper
{
    public static readonly DependencyProperty GridColumnBindingPathProperty =
        DependencyProperty.RegisterAttached(
            "GridColumnBindingPath", typeof(string), typeof(BindingHelper),
            new PropertyMetadata(null, GridBindingPathPropertyChanged));

    public static readonly DependencyProperty GridRowBindingPathProperty =
        DependencyProperty.RegisterAttached(
            "GridRowBindingPath", typeof(string), typeof(BindingHelper),
            new PropertyMetadata(null, GridBindingPathPropertyChanged));

    public static string GetGridColumnBindingPath(DependencyObject obj)
    {
        return (string)obj.GetValue(GridColumnBindingPathProperty);
    }

    public static void SetGridColumnBindingPath(DependencyObject obj, string value)
    {
        obj.SetValue(GridColumnBindingPathProperty, value);
    }

    public static string GetGridRowBindingPath(DependencyObject obj)
    {
        return (string)obj.GetValue(GridRowBindingPathProperty);
    }

    public static void SetGridRowBindingPath(DependencyObject obj, string value)
    {
        obj.SetValue(GridRowBindingPathProperty, value);
    }

    private static void GridBindingPathPropertyChanged(
        DependencyObject obj, DependencyPropertyChangedEventArgs e)
    {
        var propertyPath = e.NewValue as string;

        if (propertyPath != null)
        {
            var gridProperty =
                e.Property == GridColumnBindingPathProperty
                ? Grid.ColumnProperty
                : Grid.RowProperty;

            BindingOperations.SetBinding(
                obj,
                gridProperty,
                new Binding { Path = new PropertyPath(propertyPath) });
        }
    }
}

}

尝试添加 TerminView 代码隐藏构造函数

(this.Content as FrameworkElement).DataContext = this;

那么你需要实现INotifyPropertyChanged。这是如果 从 ViewModel 或 CodeBehind 通知视图 。仅调用 OnPropertyChanged(nameof(TerminData)) 或 OnPropertyChanged("TerminData")

public event PropertyChangedEventHandler PropertyChanged;

private void OnPropertyChanged([CallerMemberName] string propertyName = null)
{
    PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
}

之后,将您的 TerminView.cs 属性(termindata)转换为依赖项 属性。

public static readonly DependencyProperty TerminDataProperty = DependencyProperty.Register(
        "TerminData", typeof (List<TerminDate>), typeof (TerminView), new PropertyMetadata(default(List<TerminDate>), TerminDataPropertyChanged));

    private static void TerminDataPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
    {
        var control = d as TerminView;
        if (control != null)
            control.TerminData = // Do here the logic if you what when this propety changes.
    }

    public List<TerminDate> TerminData
    {
        get { return (List<TerminDate>) GetValue(TerminDataProperty); }
        set { SetValue(TerminDataProperty, value); }
    }

我建议你阅读this article from Jerry Nixon

感谢解答!!!有效.....

这里我把完整的代码加在后面了

public sealed partial class TerminView : Page
{


    public TerminView()
    {
        this.InitializeComponent();

        (this.Content as FrameworkElement).DataContext = this;


        TerminData = new List<TerminDate>();

        TerminData.Add(new TerminDate(1, 1, "Test 1"));
        TerminData.Add(new TerminDate(1, 2, "Test 2"));
    }

    public List<TerminDate> TerminData
    {
        get { return (List<TerminDate>)GetValue(TerminDataProperty); }
        set { SetValue(TerminDataProperty, value); }
    }

    public static readonly DependencyProperty TerminDataProperty = DependencyProperty.Register(
    "TerminData", typeof(List<TerminDate>), typeof(TerminView), new PropertyMetadata(default(List<TerminDate>), TerminDataPropertyChanged));

    private static void TerminDataPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
    {
        var control = d as TerminView;
    }

    public event PropertyChangedEventHandler PropertyChanged;

    private void OnPropertyChanged([CallerMemberName] string propertyName = null)
    {
        PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
    }
}


public class TerminDate
{

    public TerminDate(int row, int column, string text)
    {
        this.row = row;
        this.column = column;
        this.text = text;
    }

    public int row { get; set; }
    public int column { get; set; }
    public string text { get; set; }
}

public class BindingHelper
{
    public static readonly DependencyProperty GridColumnBindingPathProperty =
        DependencyProperty.RegisterAttached(
            "GridColumnBindingPath", typeof(string), typeof(BindingHelper),
            new PropertyMetadata(null, GridBindingPathPropertyChanged));

    public static readonly DependencyProperty GridRowBindingPathProperty =
        DependencyProperty.RegisterAttached(
            "GridRowBindingPath", typeof(string), typeof(BindingHelper),
            new PropertyMetadata(null, GridBindingPathPropertyChanged));

    public static string GetGridColumnBindingPath(DependencyObject obj)
    {
        return (string)obj.GetValue(GridColumnBindingPathProperty);
    }

    public static void SetGridColumnBindingPath(DependencyObject obj, string value)
    {
        obj.SetValue(GridColumnBindingPathProperty, value);
    }

    public static string GetGridRowBindingPath(DependencyObject obj)
    {
        return (string)obj.GetValue(GridRowBindingPathProperty);
    }

    public static void SetGridRowBindingPath(DependencyObject obj, string value)
    {
        obj.SetValue(GridRowBindingPathProperty, value);
    }

    private static void GridBindingPathPropertyChanged(
        DependencyObject obj, DependencyPropertyChangedEventArgs e)
    {
        var propertyPath = e.NewValue as string;

        if (propertyPath != null)
        {
            var gridProperty =
                e.Property == GridColumnBindingPathProperty
                ? Grid.ColumnProperty
                : Grid.RowProperty;

            BindingOperations.SetBinding(
                obj,
                gridProperty,
                new Binding { Path = new PropertyPath(propertyPath) });
        }
    }
}

为了完整起见,这里 XAML

<Page.Resources>
    <ResourceDictionary Source="Styles.xaml" />

</Page.Resources>

<Grid Background="{StaticResource ResourceKey=StyleBackground}">

    <ItemsControl ItemsSource="{Binding Path=TerminData}">
        <ItemsControl.ItemsPanel>
            <ItemsPanelTemplate>

                <Grid HorizontalAlignment="Stretch" VerticalAlignment="Stretch" Margin="50" BorderBrush="White" BorderThickness="1">
                    <Grid.RowDefinitions>
                        <RowDefinition Height="*"></RowDefinition>
                        <RowDefinition Height="*"></RowDefinition>
                        <RowDefinition Height="*"></RowDefinition>
                        <RowDefinition Height="*"></RowDefinition>
                        <RowDefinition Height="*"></RowDefinition>
                        <RowDefinition Height="*"></RowDefinition>
                    </Grid.RowDefinitions>
                    <Grid.ColumnDefinitions>
                        <ColumnDefinition Width="*"></ColumnDefinition>
                        <ColumnDefinition Width="*"></ColumnDefinition>
                        <ColumnDefinition Width="*"></ColumnDefinition>
                        <ColumnDefinition Width="*"></ColumnDefinition>
                        <ColumnDefinition Width="*"></ColumnDefinition>
                        <ColumnDefinition Width="*"></ColumnDefinition>
                        <ColumnDefinition Width="*"></ColumnDefinition>
                    </Grid.ColumnDefinitions>

                </Grid>
            </ItemsPanelTemplate>
        </ItemsControl.ItemsPanel>
        <ItemsControl.ItemTemplate>
            <DataTemplate>
                <Border Background="Green">
                    <TextBlock Text="{Binding text}"></TextBlock>
                </Border>
            </DataTemplate>
        </ItemsControl.ItemTemplate>

        <ItemsControl.ItemContainerStyle>
            <Style TargetType="ContentPresenter">
                <Setter Property="local:BindingHelper.GridColumnBindingPath" Value="column"/>
                <Setter Property="local:BindingHelper.GridRowBindingPath" Value="row"/>
            </Style>
        </ItemsControl.ItemContainerStyle>

    </ItemsControl>


</Grid>