Можно ли эмулировать пограничный коллапс (ala CSS) в WPF ItemsControl?
Я стилю элементы в WPF ListBox
и хочу поместить границу вокруг каждого элемента. Например, при BorderThickness
, установленном в 1, границы верхнего края между смежными элементами рисуются и, следовательно, кажутся "более толстыми", чем боковые границы, как показано:
![ListBoxItem border example]()
Шаблон элемента, создающий эти ListBoxItems
:
<DataTemplate>
<Border BorderThickness="1" BorderBrush="DarkSlateGray" Background="DimGray" Padding="8 4 8 4">
<TextBlock Text="{Binding Name}" FontSize="16"/>
</Border>
</DataTemplate>
Я хотел бы "свернуть" эти смежные границы, как можно было бы, например, через CSS. Я знаю, что BorderThickness
может быть определен отдельно для левых/правых/верхних/нижних границ, но это также влияет на границу первого или последнего элемента, что нежелательно.
Есть ли способ сделать это с помощью WPF? Свойство Border
Мне не хватает или требуется другой подход к созданию границ?
Ответы
Ответ 1
Используйте BorderThickness="1,0,1,1"
и a DataTrigger
, который проверяет, что RelativeSource={RelativeSource Mode=PreviousData}
является null
, чтобы установить BorderThickness="1,1,1,1"
:
<Window x:Class="CollapseBordersDemo.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:sys="clr-namespace:System;assembly=mscorlib"
Title="MainWindow" Height="239" Width="525">
<Window.Resources>
<x:Array x:Key="ListBoxItems" Type="{x:Type sys:String}">
<sys:String>Alice</sys:String>
<sys:String>Bob</sys:String>
<sys:String>Colleen</sys:String>
</x:Array>
<DataTemplate x:Key="ListBoxTemplate">
<Border x:Name="Border" BorderThickness="1,0,1,1" BorderBrush="DarkSlateGray" Background="LightGray" Padding="8 4 8 4">
<TextBlock Text="{Binding}" FontSize="16"/>
</Border>
<DataTemplate.Triggers>
<DataTrigger Binding="{Binding RelativeSource={RelativeSource Mode=PreviousData}}" Value="{x:Null}">
<Setter TargetName="Border" Property="BorderThickness" Value="1,1,1,1"/>
</DataTrigger>
</DataTemplate.Triggers>
</DataTemplate>
</Window.Resources>
<Grid>
<ListBox ItemsSource="{StaticResource ListBoxItems}" ItemTemplate="{StaticResource ListBoxTemplate}" HorizontalContentAlignment="Stretch" />
</Grid>
</Window>
Ответ 2
Одна вещь, которая приходит на ум, - использовать AlternationIndex. Это потребует от вас установить что-то вроде AlternationCount="10000"
в ListBox. После этого вы можете установить BorderThickess="1,0,1,1"
и использовать DataTrigger для поиска первого списка ListBoxItem
<DataTemplate>
<Border x:Name="border" BorderThickness="1,0,1,1" BorderBrush="DarkSlateGray" Background="DimGray" Padding="8 4 8 4">
<TextBlock Text="{Binding Name}" FontSize="16"/>
</Border>
<DataTemplate.Triggers>
<DataTrigger Binding="{Binding RelativeSource={RelativeSource AncestorType={x:Type ListBoxItem}},
Path=(ItemsControl.AlternationIndex)}"
Value="0">
<Setter TargetName="border" Property="BorderThickness" Value="1,1,1,1"/>
</DataTrigger>
</DataTemplate.Triggers>
</DataTemplate>
Ответ 3
Вы можете добавить
Margin="0,0,0,-1" SnapsToDevicePixels="True"
к определению границы