如何编写具有某些功能的DataTemplate作为资源

4

我的应用程序中有一些ListBox,它们将包含电话簿联系人。我希望所有这些ListBox都具有DataTemplate作为它们的ItemsTemplates,并具有一些功能,如编辑,删除和查看,就像下面的代码所示:

<DataTemplate x:Key="ContactItemTemplate">
    <TextBlock Foreground="Black" Background="{StaticResource DataTemplateBackgroundBrush}" Padding="5,10" Margin="4,3">
        <TextBlock.Text>
            <MultiBinding StringFormat="{}{0} {1}">
                <Binding Path="FirstName"/>
                <Binding Path="LastName"/>
            </MultiBinding>
        </TextBlock.Text>
        <TextBlock.ContextMenu>
            <ContextMenu FontFamily="B Yekan">
                <MenuItem Header="Edit" Click="btn_EditContact_Click"/>
                <MenuItem Header="Delete" Click="btn_DeleteContact_Click"/>
                <MenuItem Header="View" Click="btn_EditContact_Click"/>
            </ContextMenu>
        </TextBlock.ContextMenu>
    </TextBlock>
</DataTemplate>

这个模板不能被写成一个资源字典中的样式并添加到控件中,因为事件处理程序无法在资源字典中呈现。所以一种方法是在每个具有 Contact ListBox 的窗口/页面中复制此模板及其处理程序,如下所示:

<Page.Resources>
    <ResourceDictionary>
        <ResourceDictionary.MergedDictionaries>
            <ResourceDictionary Source="pack://application:,,,/Styles/Controls_Style.xaml"/>
        </ResourceDictionary.MergedDictionaries>
        <DataTemplate x:Key="AttachmentsTemplate">
            <Border Height="150" Width="120" BorderThickness="1" BorderBrush="{StaticResource DefaultBorderBrush}" Margin="2">
                <Grid>
                    <Grid.RowDefinitions>
                        <RowDefinition Height="Auto"/>
                        <RowDefinition/>
                    </Grid.RowDefinitions>
                    <TextBlock Text="{Binding Title}" HorizontalAlignment="Center" VerticalAlignment="Bottom" TextWrapping="WrapWithOverflow"/>
                    <Image Grid.Row="1" Source="{Binding Image}" Margin="5,0"/>
                </Grid>
            </Border>
        </DataTemplate>
    </ResourceDictionary>
</Page.Resources>

有没有其他方法可以编写一个DataTemplate,使其具备某些功能,并在任何地方都可以使用?我应该编写UserControl而不是DataTemplate吗?


用命令绑定替换点击处理程序。将菜单项的 Command 属性绑定到视图模型中的 ICommand 属性。 - Clemens
2个回答

2

请尝试下一个解决方案:

Xaml代码:

<Window x:Class="DataTemplateInResDectReuseHelpAttempt.MainWindow"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:dataTemplateInResDectReuseHelpAttempt="clr-namespace:DataTemplateInResDectReuseHelpAttempt"
    Title="MainWindow" Height="350" Width="525">
<Window.DataContext>
    <dataTemplateInResDectReuseHelpAttempt:MainDataContext/>
</Window.DataContext>
<Window.Resources>
    <ResourceDictionary>
        <ResourceDictionary.MergedDictionaries>
            <ResourceDictionary Source="ContactsDataTemplateResourceDictionary.xaml"></ResourceDictionary>
        </ResourceDictionary.MergedDictionaries>
    </ResourceDictionary>
</Window.Resources>
<Grid>
    <ListBox ItemsSource="{Binding BaseModels}">
        <ListBox.ItemContainerStyle>
            <Style TargetType="ListBoxItem">
                <Setter Property="ContentTemplate">
                    <Setter.Value>
                        <DataTemplate>
                            <ContentControl Content="{Binding }" ContentTemplate="{StaticResource ContactItemTemplate}"></ContentControl>
                        </DataTemplate>
                    </Setter.Value>
                </Setter>
            </Style>
        </ListBox.ItemContainerStyle>
    </ListBox>
</Grid></Window>

ViewModel and model

    class MainDataContext
{
    private ICommand _viewCommand;
    private ICommand _deleteCommand;
    private ICommand _editCommand;

    public MainDataContext()
    {
        BaseModels = new ObservableCollection<BaseModel>
        {
            new BaseModel(EditCommand, DeleteCommand, ViewCommand){LastName = "Omar", FirstName = "Khayyam"},
            new BaseModel(EditCommand, DeleteCommand, ViewCommand){LastName = "Chekhov", FirstName = "Anton"},
            new BaseModel(EditCommand, DeleteCommand, ViewCommand){LastName = "Lau", FirstName = "Meir"},
        };
    }

    public ICommand ViewCommand
    {
        get { return _viewCommand ?? (_viewCommand = new RelayCommand<object>(View)); }
    }

    private void View(object obj)
    {

    }

    public ICommand DeleteCommand
    {
        get { return _deleteCommand ?? (_deleteCommand = new RelayCommand<object>(Delete)); }
    }

    private void Delete(object obj)
    {

    }

    public ICommand EditCommand
    {
        get { return _editCommand ?? (_editCommand = new RelayCommand<object>(Edit)); }
    }

    private void Edit(object obj)
    {

    }

    public ObservableCollection<BaseModel> BaseModels { get; set; } 
}

public class BaseModel:BaseObservableObject
{
    private string _firstName;
    private string _lastName;
    private readonly ICommand _editCommand;
    private readonly ICommand _deleteCommand;
    private readonly ICommand _viewCommand;

    public BaseModel(ICommand editCommand, ICommand deleteCommand, ICommand viewCommand)
    {
        _editCommand = editCommand;
        _deleteCommand = deleteCommand;
        _viewCommand = viewCommand;
    }

    public string FirstName
    {
        get { return _firstName; }
        set
        {
            _firstName = value;
            OnPropertyChanged();
        }
    }

    public string LastName
    {
        get { return _lastName; }
        set
        {
            _lastName = value;
            OnPropertyChanged();
        }
    }

    public ICommand EditCommand
    {
        get { return _editCommand; }
    }

    public ICommand DeleteCommand
    {
        get { return _deleteCommand; }
    }

    public ICommand ViewCommand
    {
        get { return _viewCommand; }
    }
}

ResourceDictionary 代码:

<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
                xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
<SolidColorBrush x:Key="DataTemplateBackgroundBrush" Color="Azure"/>
<DataTemplate x:Key="ContactItemTemplate">
    <TextBlock Foreground="Black" Background="{StaticResource DataTemplateBackgroundBrush}" Padding="5,10" Margin="4,3">
        <TextBlock.Text>
            <MultiBinding StringFormat="{}{0} {1}">
                <Binding Path="FirstName"/>
                <Binding Path="LastName"/>
            </MultiBinding>
        </TextBlock.Text>
        <TextBlock.ContextMenu>
            <ContextMenu FontFamily="B Yekan">
                <MenuItem Header="Edit" Command="{Binding EditCommand}" CommandParameter="{Binding }"/>
                <MenuItem Header="Delete" Command="{Binding DeleteCommand}" CommandParameter="{Binding }"/>
                <MenuItem Header="View" Command="{Binding ViewCommand}" CommandParameter="{Binding }"/>
            </ContextMenu>
        </TextBlock.ContextMenu>
    </TextBlock>
</DataTemplate></ResourceDictionary>

如果你在代码方面遇到了问题,我很乐意提供帮助。

祝好。


1
您可以在字典中拥有事件处理程序,只要它们是字典的一部分 - 也就是说,字典后面有一些代码支持。这类似于PageUserControl;您需要添加一个x:Class属性。例如,Styles.xml可能如下所示:
<ResourceDictionary
    x:Class="WPF.Styles"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:local="clr-namespace:WpfApplication1">
    <DataTemplate x:Key="ContactItemTemplate" DataType="local:Person">
        <Grid>
            <Grid.ColumnDefinitions>
                <ColumnDefinition Width="*" />
                <ColumnDefinition Width="Auto" />
            </Grid.ColumnDefinitions>
            <TextBlock
                Grid.Column="0"
                Foreground="Black"
                Background="Yellow"
                Padding="5,10"
                Margin="4,3"
                Text="{Binding Name}"
                >
                <TextBlock.ContextMenu>
                    <ContextMenu>
                        <MenuItem Header="Edit!" Click="btn_EditContact_Click"/>
                        <MenuItem Header="Delete!" Click="btn_DeleteContact_Click"/>
                        <MenuItem Header="View!" Click="btn_EditContact_Click"/>
                    </ContextMenu>
                </TextBlock.ContextMenu>
            </TextBlock>
            <Button
                Grid.Column="1"
                Content="Edit"
                Click="btn_EditContact_Click"/>
        </Grid>
    </DataTemplate>
</ResourceDictionary>

...有一个相应的Styles.cs文件(注意partial):

public sealed partial class Styles
{
    private void btn_EditContact_Click(object sender, EventArgs args)
    {
        Debug.WriteLine(args);
    }

    private void btn_DeleteContact_Click(object sender, EventArgs args)
    {
        Debug.WriteLine(args);
    }
}

在我的窗口中,类似这样的东西:
<Grid.Resources>
    <ResourceDictionary>
        <ResourceDictionary.MergedDictionaries>
            <ResourceDictionary Source="pack://application:,,,/Styles.xaml" />
        </ResourceDictionary.MergedDictionaries>
    </ResourceDictionary>
</Grid.Resources>

...

<ItemsControl
    ItemsSource="{Binding People}"
    Grid.Row="3"
    ItemTemplate="{StaticResource ContactItemTemplate}" />

为了完整起见,模型对象:

public class Person
{
    public string Name { get; }

    public Person(string name)
    {
        Name = name;
    }
}

"...和ItemsSource:"
public Person[] People { get; } =
    {
        new Person("Donald Duck"),
        new Person("Mickey Mouse"),
        new Person("Darth Vader"),
    };

通常情况下这个工作很好 - 例如,点击按钮会调用btn_EdxtContact_Click。不幸的是,涉及TextBox的上下文菜单的场景存在问题; 在此帖子中详细了解可能的解决方法

更新:关于上下文菜单和模板

额外的一点信息:本文的评论中有一些关于ContextMenu模板的讨论,可能会有所启发。

...还有另一个TextBoxContextMenu链接,可能会很有用。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接