I have a UserControl
which I'm using to display a list of UIElement
s. The control consists of a single ItemsControl
with it's ItemPanelTemplate
switched for a horizontal StackPanel
, its ItemsSource
bound to a DependencyProperty
exposed by the UserControl
and its ItemTemplate
set in the UserControl.Resources
.
Everything works fine except the ItemTemplate
never get's applied and I can't see why. The full source is below.
UserControl.xaml -
<UserControl x:Name="UC" x:FieldModifier="private" x:Class="ContentSliderControl.ContentSlider"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
<UserControl.Resources>
<DataTemplate x:Key="pageTemplate">
<Border CornerRadius="10" Padding="5" Height="200" Width="200" Background="#333">
<ContentControl Content="{Binding}"/>
</Border>
</DataTemplate>
<ItemsPanelTemplate x:Key="template">
<StackPanel IsItemsHost="True"
Orientation="Horizontal"
ScrollViewer.HorizontalScrollBarVisibility="Disabled"
ScrollViewer.VerticalScrollBarVisibility="Disabled"/>
</ItemsPanelTemplate>
</UserControl.Resources>
<ItemsControl ItemsPanel="{StaticResource template}"
ItemTemplate="{StaticResource pageTemplate}"
ItemsSource="{Binding ElementName=UC,Path=Pages}"/>
UserControl.xaml.cs -
[ContentProperty("Pages")]
public partial class ContentSlider : UserControl
{
public List<UIElement> Pages
{
get { return (List<UIElement>)GetValue(PagesProperty); }
//set { SetValue(PagesProperty, value); }
}
// Using a DependencyProperty as the backing store for Pages. This enables animation, styling, binding, etc...
public static readonly DependencyProperty PagesProperty =
DependencyProperty.Register("Pages", typeof(List<UIElement>), typeof(ContentSlider), new UIPropertyMetadata(null));
public ContentSlider()
{
InitializeComponent();
}
}
}
I consume the control in my main window like this -
<slider:ContentSlider >
<slider:ContentSlider.Pages>
<Button>1</Button>
<Button>2</Button>
<Button>3</Button>
<Button>4</Button>
</slider:ContentSlider.Pages>
</slider:ContentSlider>
The buttons appear fine but not inside the 200px square border.
Any help would be greatlly appriciated. Thanks.
It's because it's a list of UIElement, the item template is only applied if the items can't be displayed directly.
Robert Macnee nailed the reason on the head. His solution involves using the control template which might be overkill for a given scenario. Alternatively, use a
ListBox
- set theItemContainerStyle
to a new Style forListBoxItem
, and in that style, set theContentTemplate
to theDataTemplate
that you wanted to use in theListBox
ItemTemplate
.Nir is correct,
ItemsControl
will add item directly to itsPanel
if they are UIElements. I couldn't find any mention of this behavior in MSDN, but Dr. WPF mentions it in his article on item containers:Your solution is probably to use a
ListBox
instead, and setItemContainerStyle
to a newStyle
forListBoxItem
, and in that style, use aControlTemplate
with yourBorder
in it.Nir is right, this custom ItemsControl implementation will solve the issue and let use your own ItemTemplate:
If you set the
DataType
property on theDataTemplate
it would start working.