我如何在这个 ControlTemplate 中绑定?
How can I bind inside this ControlTemplate?
我有一个 ListView
,填充了一个 List<MyListItem>
,我需要使用 ControlTemplate
才能在选择项目时更改效果。现在我遇到了 {Binding MyProperty}
在 ControlTemplate
中不起作用的问题。如何访问模板内 MyListItem
的属性?
我的 XAML 看起来像这样(简化):
<ListView
Name="ListView"
Grid.Column="1"
IsSwipeEnabled="False">
<ListView.ItemContainerStyle>
<Style TargetType="ListViewItem">
<Setter Property="Template">
<Setter.Value>
<ControlTemplate TargetType="ListViewItem">
<Grid>
<VisualStateManager.VisualStateGroups>
<VisualStateGroup x:Name="CommonStates">
<VisualState x:Name="Normal"/>
</VisualStateGroup>
<VisualStateGroup x:Name="SelectionStates">
<VisualState x:Name="Unselected">
<Storyboard>
<ColorAnimation Duration="0" Storyboard.TargetName="myColoredText" Storyboard.TargetProperty="(Border.Background).(SolidColorBrush.Color)" To="Orange"/>
</Storyboard>
</VisualState>
<VisualState x:Name="SelectedUnfocused">
<Storyboard>
<ColorAnimation Duration="0" Storyboard.TargetName="myColoredText" Storyboard.TargetProperty="(Border.Background).(SolidColorBrush.Color)" To="Red"/>
</Storyboard>
</VisualState>
</VisualStateGroup>
</VisualStateManager.VisualStateGroups>
<!-- Here I have my custom layout, removed for readability -->
<TextBlock
Name="myColoredText"
Foreground="Green"
Text="{Binding MyProperty}"/><!-- This does not work -->
</Grid>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>
</ListView.ItemContainerStyle>
</ListView>
这是XAML后面的class:
public sealed partial class MyPage : Page
{
public MyPage()
{
InitializeComponent();
ListView.ItemsSource = new List<MyListItem>()
{
new MyListItem("Title1", "SubTitle1"),
new MyListItem("Title2", "SubTitle2")
}
}
}
和 MyListItem class:
class MyListItem
{
public string MyProperty { get; set; }
public string MyOtherProperty { get; set; }
public MyListItem(string myProperty, string myOtherProperty)
{
MyProperty = myProperty;
MyOtherProperty = myOtherProperty;
}
}
有问题的最小项目:
在处理自定义 ControlTemplate
时需要使用 {TemplateBinding}
表达式。这会将 属性 值绑定到与控件实例关联的任何相应属性。
<TextBlock Name="myColoredText"
Foreground="Green"
Text="{TemplateBinding MyProperty}" />
这将直接绑定到您控件中的指定实例 属性。由于WPF/WinRT控件是DependencyObjects
,所以常用DependencyProperties
:
public static readonly DependencyProperty MyPropertyProperty =
DependencyProperty.Register(
"MyProperty",
typeof(Boolean),
typeof(ListViewItem),
null
);
public bool MyProperty
{
get { return (bool)GetValue(MyPropertyProperty); }
set { SetValue(MyPropertyProperty, value); }
}
然后在声明一个新实例时,将按照 XAML 中的标准填充此 属性:
<someNamespace:ListViewItem MyProperty={Binding ViewModelProperty} />
您可以在 MSDN 阅读有关 TemplateBinding 标记表达式的更多信息。
A ControlTemplate
通常用于定义控件的外观。因此,将 XAML 放入 DataTemplate
并将其分配给 ListView.ItemTemplate
属性 会容易得多。您可以从 DataTemplate
:
访问所有自定义属性
<ListView.ItemTemplate>
<DataTemplate>
<TextBlock Name="myColoredText" Foreground="Green"
Text="{Binding MyProperty}" /><!-- This will now work -->
</DataTemplate>
</ListView.ItemTemplate>
更新>>>
您也可以像这样使用 ItemContainerStyle
属性 中的 DataTemplate
:
<Style x:Key="SomeStyle" TargetType="{x:Type ListBoxItem}">
<Setter Property="ContentTemplate">
<Setter.Value>
<DataTemplate DataType="{x:Type Your:DataType}">
<TextBlock Name="myColoredText" Foreground="Green"
Text="{Binding MyProperty}" /><!-- This will now work -->
</DataTemplate>
</Setter.Value>
</Setter>
</Style>
Wpf 仅支持绑定到属性,不支持绑定到字段。所以,你的 MyItemClass 应该是:
public class MyListItem {
public string MyProperty { get; set; }
public string MyOtherProperty { get; set; }
public MyListItem(string myProperty, string myOtherProperty) {
MyProperty = myProperty;
MyOtherProperty = myOtherProperty;
}
}
希望,有帮助
在TextBox
中,使用:
Text="{Binding Content.MyProperty,
RelativeSource={RelativeSource TemplatedParent}}"
我有一个 ListView
,填充了一个 List<MyListItem>
,我需要使用 ControlTemplate
才能在选择项目时更改效果。现在我遇到了 {Binding MyProperty}
在 ControlTemplate
中不起作用的问题。如何访问模板内 MyListItem
的属性?
我的 XAML 看起来像这样(简化):
<ListView
Name="ListView"
Grid.Column="1"
IsSwipeEnabled="False">
<ListView.ItemContainerStyle>
<Style TargetType="ListViewItem">
<Setter Property="Template">
<Setter.Value>
<ControlTemplate TargetType="ListViewItem">
<Grid>
<VisualStateManager.VisualStateGroups>
<VisualStateGroup x:Name="CommonStates">
<VisualState x:Name="Normal"/>
</VisualStateGroup>
<VisualStateGroup x:Name="SelectionStates">
<VisualState x:Name="Unselected">
<Storyboard>
<ColorAnimation Duration="0" Storyboard.TargetName="myColoredText" Storyboard.TargetProperty="(Border.Background).(SolidColorBrush.Color)" To="Orange"/>
</Storyboard>
</VisualState>
<VisualState x:Name="SelectedUnfocused">
<Storyboard>
<ColorAnimation Duration="0" Storyboard.TargetName="myColoredText" Storyboard.TargetProperty="(Border.Background).(SolidColorBrush.Color)" To="Red"/>
</Storyboard>
</VisualState>
</VisualStateGroup>
</VisualStateManager.VisualStateGroups>
<!-- Here I have my custom layout, removed for readability -->
<TextBlock
Name="myColoredText"
Foreground="Green"
Text="{Binding MyProperty}"/><!-- This does not work -->
</Grid>
</ControlTemplate>
</Setter.Value>
</Setter>
</Style>
</ListView.ItemContainerStyle>
</ListView>
这是XAML后面的class:
public sealed partial class MyPage : Page
{
public MyPage()
{
InitializeComponent();
ListView.ItemsSource = new List<MyListItem>()
{
new MyListItem("Title1", "SubTitle1"),
new MyListItem("Title2", "SubTitle2")
}
}
}
和 MyListItem class:
class MyListItem
{
public string MyProperty { get; set; }
public string MyOtherProperty { get; set; }
public MyListItem(string myProperty, string myOtherProperty)
{
MyProperty = myProperty;
MyOtherProperty = myOtherProperty;
}
}
有问题的最小项目:
在处理自定义 ControlTemplate
时需要使用 {TemplateBinding}
表达式。这会将 属性 值绑定到与控件实例关联的任何相应属性。
<TextBlock Name="myColoredText"
Foreground="Green"
Text="{TemplateBinding MyProperty}" />
这将直接绑定到您控件中的指定实例 属性。由于WPF/WinRT控件是DependencyObjects
,所以常用DependencyProperties
:
public static readonly DependencyProperty MyPropertyProperty =
DependencyProperty.Register(
"MyProperty",
typeof(Boolean),
typeof(ListViewItem),
null
);
public bool MyProperty
{
get { return (bool)GetValue(MyPropertyProperty); }
set { SetValue(MyPropertyProperty, value); }
}
然后在声明一个新实例时,将按照 XAML 中的标准填充此 属性:
<someNamespace:ListViewItem MyProperty={Binding ViewModelProperty} />
您可以在 MSDN 阅读有关 TemplateBinding 标记表达式的更多信息。
A ControlTemplate
通常用于定义控件的外观。因此,将 XAML 放入 DataTemplate
并将其分配给 ListView.ItemTemplate
属性 会容易得多。您可以从 DataTemplate
:
<ListView.ItemTemplate>
<DataTemplate>
<TextBlock Name="myColoredText" Foreground="Green"
Text="{Binding MyProperty}" /><!-- This will now work -->
</DataTemplate>
</ListView.ItemTemplate>
更新>>>
您也可以像这样使用 ItemContainerStyle
属性 中的 DataTemplate
:
<Style x:Key="SomeStyle" TargetType="{x:Type ListBoxItem}">
<Setter Property="ContentTemplate">
<Setter.Value>
<DataTemplate DataType="{x:Type Your:DataType}">
<TextBlock Name="myColoredText" Foreground="Green"
Text="{Binding MyProperty}" /><!-- This will now work -->
</DataTemplate>
</Setter.Value>
</Setter>
</Style>
Wpf 仅支持绑定到属性,不支持绑定到字段。所以,你的 MyItemClass 应该是:
public class MyListItem {
public string MyProperty { get; set; }
public string MyOtherProperty { get; set; }
public MyListItem(string myProperty, string myOtherProperty) {
MyProperty = myProperty;
MyOtherProperty = myOtherProperty;
}
}
希望,有帮助
在TextBox
中,使用:
Text="{Binding Content.MyProperty,
RelativeSource={RelativeSource TemplatedParent}}"