Add a DataTemplate for a ListView in an UserControl

565 Views Asked by At

I have a simple usercontrol in WPF, that contains some buttons, textboxes and a listview. I simplified that control a little bit:

<UserControl x:Class="Example.SearchListControl"
         ...
         mc:Ignorable="d">
    <Grid>
        <Grid.RowDefinitions>
            <RowDefinition Height="Auto" />
            <RowDefinition Height="*" />
            <RowDefinition Height="Auto" />
        </Grid.RowDefinitions>

        <TextBox Grid.Row="0" Text="{Binding SearchText, UpdateSourceTrigger=PropertyChanged}" />
        <ListView Grid.Row="1" ItemsSource="{Binding Items}" SelectedItem="{Binding Current}" />
        <Button Grid.Row="2" Command="{Binding SomeSortOfCommand}">
   </Grid>
</UserControl>

The ViewModel for that usercontrol looks like this:

public class SearchListViewModel<T> : ViewModelBase
{
    private ObservableCollection<T> _items;

    public ObservableCollection<T> Items
    {
        get
        {
            return _items;
        }
        set
        {
            _items = value;
            OnPropertyChanged();
        }
    }

    private T _current;

    public T Current
    {
        get
        {
            return _current;
        }
        set
        {
            _current = value;
            OnPropertyChanged();
        }
    }

    ...
}

I put that control in another window like that:

<Window x:Class="Example.TestWindow"
        ...>

    <Grid>
        <local:SearchListControl DataContext="{Binding GenericSearchListViewModel}" />
    </Grid>
</Window>

For more flexibility, i want to set the datatemplate for the ListView in that control from the window? Something like:

<Window ...>

    <Window.Resources>
        <DataTemplate x:Key="ListViewTemplate">
            <Grid>
                <Grid.ColumnDefinitions>
                    <ColumnDefinition Width="100" />
                    <ColumnDefinition Width="*" />
                </Grid.ColumnDefinitions>
                <TextBlock Grid.Column="0" Text="{Binding Path=ShortCaption}" />
                <TextBlock Grid.Column="1" Text="{Binding Path=Caption}" />
            </Grid>
        </DataTemplate>

    </Window.Resources>
    <Grid>
        <local:SearchListControl DataContext="{Binding GenericSearchListViewModel}" 
              DataTemplate="{StaticResource ListViewTemplate}" />
    </Grid>
</Window>

Is that possible, or is the way im going a totally wrong one?

1

There are 1 best solutions below

0
On BEST ANSWER

If you give the ListView in your SearchListUserControl.xaml file a name:

<ListView x:Name="lv" ... />

...you could easily expose its ItemTemplate property through a property of the SearchListControl class:

public abstract class SearchListControl : UserControl
{
    public SearchListControl()
    {
        InitializeComponent();
    }

    public DataTemplate DataTemplate
    {
        get { return lv.ItemTemplate; }
        set { lv.ItemTemplate = value; }
    }
}

...and set it from the window:

<local:SearchListControl DataContext="{Binding GenericSearchListViewModel}" 
          DataTemplate="{StaticResource ListViewTemplate}" /