WPF 在改写ItemTemplate、 DataTemplate中,控件命令事件触发不起作用

 

问题描述:MVVMLight框架中,改写ListBox、ListBox.ItemTemplate模板时,模板里面的Button/Image控件绑定命令后,触发不起作用,但在ListBox外的命令正常绑定到指定的ViewModel;

VieweModel 为CarPicturesViewModel,绑定上下文DataContext="{Binding CarPicturesVM ,Source={StaticResource Locator}}"

问题代码:

1.声明绑定数据类CarInfo

public class CarInfo : BaseNotifyProperty
{
    private int carIndex;
    public int CarIndex
    {
        get { return carIndex; }
        set
        {
            carIndex = value;
            OnPropertyChanged("CarIndex");
        }
    }
 
    private string carName;
    public string CarName
    {
        get { return carName; }
        set
        {
            carName = value;
            OnPropertyChanged("CarName");
        }
    }
}

2.界面设计代码,整个界面是UserControl

<UserControl x:Class="CarRental.View.CarPicturesView"
    DataContext="{Binding CarPicturesVM ,Source={StaticResource Locator}}">
    <Grid>
        <ListBox x:Name="listboxImages" HorizontalAlignment="Left" VerticalAlignment="Top"  Width="1780" Height="870" ItemsSource="{Binding DisplayImageList}">
            <ListBox.ItemsPanel>
                <ItemsPanelTemplate>
                    <WrapPanel Orientation="Horizontal" HorizontalAlignment="Center" VerticalAlignment="Top" Background="Transparent"/>
                </ItemsPanelTemplate>
            </ListBox.ItemsPanel>
            <ListBox.ItemTemplate>
                <DataTemplate x:Name="gridDataTemplate">
                    <Border x:Name="DataTemplateBr" CornerRadius="5"  HorizontalAlignment="Left" BorderThickness="2" VerticalAlignment="Top">
                        <Border.Background>
                            <ImageBrush ImageSource="Pack://application:,,,/CarRental;component/Images/CarInfo/white.png" Opacity="0.32">
                            </ImageBrush>
                        </Border.Background>
                        <Grid Margin="5" Height="400" Width="250">
                            <Button Command="{Binding AddImageRecord}" />
                            <Grid Visibility="{Binding IsShowAddImage,Converter={StaticResource boolToNVisibilityConverter}}">
                                <Grid.RowDefinitions>
                                    <RowDefinition Height="250"/>
                                    <RowDefinition Height="auto"/>
                                </Grid.RowDefinitions>
                                <Image x:Name="image_Check" Grid.Row="0" Margin="5" Height="30" Width="30" HorizontalAlignment="Left" VerticalAlignment="Top" Panel.ZIndex="9999"  >
                                    <Image.Style>
                                        <Style TargetType="{x:Type Image}">
                                            <Setter Property="Source" Value="Pack://application:,,,/CarRental;component/Images/CarInfo/check.png"/>
                                            <Style.Triggers>
                                                <Trigger Property="IsMouseOver" Value="true">
                                                    <Setter Property="Source" Value="/CarRental;component/Images/CarInfo/check_sel.png"/>
                                                </Trigger>
                                            </Style.Triggers>
                                        </Style>
                                    </Image.Style>
                                    <i:Interaction.Triggers>
                                        <i:EventTrigger EventName="MouseDown">
                                            <i:InvokeCommandAction Command="{Binding AddRentalRecord}"/>
                                        </i:EventTrigger>
                                    </i:Interaction.Triggers>
                                </Image> 
                                <StackPanel Margin="0,5,0,0" Grid.Row="1">
                                    <Grid>
                                        <TextBlock Margin="0,5,0,0" Text="名称:" Style="{StaticResource txtblock}"/>
                                        <TextBlock Margin="45,5,0,0" Text="{Binding CarName}" Style="{StaticResource txtblock}" Width="200"/>
                                    </Grid>
                                </StackPanel>
                            </Grid>
                        </Grid>
                    </Border>
                </DataTemplate>
            </ListBox.ItemTemplate>
        </ListBox>
    </Grid>
</UserControl>

问题原因:因为 ListBox 的ItemsSource绑定了 DisplayImageList,DisplayImageList 是一个 CarInfo 类的列表,ListBox里面控件的DataContext就成了CarInfo也就是里面控件的Binding都是CarInfo的属性,比如名称(Binding="{Binding Name}")。而CarInfo里没有AddImageRecord,所以就不能触发操作了。

解决方法:把Button的Command绑定为ViewModel里面的AddImageRecord就好了,而ListBox的DataContext就是ViewModel,那这样做就好了;

解决代码: <Button Command="{Binding CarPicturesVM.AddRentalRecord,Source={StaticResource Locator}}" />  或者

Command="{Binding DataContext.AddRentalRecord,RelativeSource={RelativeSource AncestorType={x:Type UserControl}}}" 或者

Command="{Binding DataContext.AddRentalRecord,RelativeSource={RelativeSource AncestorLevel=2, AncestorType={x:Type UserControl}}}"

AncestorLevel:模式中获取或设置要查找的上级节点的级别。 使用 1 表示与绑定目标元素最近的一个级别。

返回值:上级节点级别。 使用 1 表示与绑定目标元素最近的一个级别。

AncestorType:获取或设置要查找的上级节点的类型。

修改后正确代码

<UserControl x:Class="CarRental.View.CarPicturesView"
    DataContext="{Binding CarPicturesVM ,Source={StaticResource Locator}}">
    <Grid>
        <ListBox x:Name="listboxImages" HorizontalAlignment="Left" VerticalAlignment="Top"  Width="1780" Height="870" ItemsSource="{Binding DisplayImageList}">
            <ListBox.ItemsPanel>
                <ItemsPanelTemplate>
                    <WrapPanel Orientation="Horizontal" HorizontalAlignment="Center" VerticalAlignment="Top" Background="Transparent"/>
                </ItemsPanelTemplate>
            </ListBox.ItemsPanel>
            <ListBox.ItemTemplate>
                <DataTemplate x:Name="gridDataTemplate">
                    <Border x:Name="DataTemplateBr" CornerRadius="5"  HorizontalAlignment="Left" BorderThickness="2" VerticalAlignment="Top">
                        <Border.Background>
                            <ImageBrush ImageSource="Pack://application:,,,/CarRental;component/Images/CarInfo/white.png" Opacity="0.32">
                            </ImageBrush>
                        </Border.Background>
                        <Grid Margin="5" Height="400" Width="250">
                            <Button Command="{Binding DataContext.AddImageRecord,RelativeSource={RelativeSource AncestorType={x:Type UserControl}}}" />
                            <Grid Visibility="{Binding IsShowAddImage,Converter={StaticResource boolToNVisibilityConverter}}">
                                <Grid.RowDefinitions>
                                    <RowDefinition Height="250"/>
                                    <RowDefinition Height="auto"/>
                                </Grid.RowDefinitions>
                                <Image x:Name="image_Check" Grid.Row="0" Margin="5" Height="30" Width="30" HorizontalAlignment="Left" VerticalAlignment="Top" Panel.ZIndex="9999"  >
                                    <Image.Style>
                                        <Style TargetType="{x:Type Image}">
                                            <Setter Property="Source" Value="Pack://application:,,,/CarRental;component/Images/CarInfo/check.png"/>
                                            <Style.Triggers>
                                                <Trigger Property="IsMouseOver" Value="true">
                                                    <Setter Property="Source" Value="/CarRental;component/Images/CarInfo/check_sel.png"/>
                                                </Trigger>
                                            </Style.Triggers>
                                        </Style>
                                    </Image.Style>
                                    <i:Interaction.Triggers>
                                        <i:EventTrigger EventName="MouseDown">
                                            <i:InvokeCommandAction Command="{Binding DataContext.AddRentalRecord,RelativeSource={RelativeSource AncestorType={x:Type UserControl}}}"/>
                                        </i:EventTrigger>
                                    </i:Interaction.Triggers>
                                </Image>                            
                                <StackPanel Margin="0,5,0,0" Grid.Row="1">
                                    <Grid>
                                        <TextBlock Margin="0,5,0,0" Text="名称:" Style="{StaticResource txtblock}"/>
                                        <TextBlock Margin="45,5,0,0" Text="{Binding CarName}" Style="{StaticResource txtblock}" Width="200"/>
                                    </Grid>                            
                                </StackPanel>
                            </Grid>
                        </Grid>
                    </Border>
                </DataTemplate>
            </ListBox.ItemTemplate>
        </ListBox>
    </Grid>
</UserControl>

 

posted @ 2024-03-13 02:13  小林野夫  阅读(184)  评论(0编辑  收藏  举报
原文链接:https://www.cnblogs.com/cdaniu/