WPF: триггер для ListBoxItem.IsSelected не работает для свойства Background - PullRequest
17 голосов
/ 19 марта 2011

Я пытаюсь изменить свойство Background для моих ListBoxItem с помощью триггеров в ItemContainerStyle моего ListBox следующим образом:

    <ListBox Height="100" HorizontalAlignment="Left" Margin="107,59,0,0" Name="listBox1" VerticalAlignment="Top" Width="239">
        <ListBox.ItemContainerStyle>
            <Style TargetType="{x:Type ListBoxItem}">
                <Setter Property="Background" Value="Lightblue"/>
                <Style.Triggers>
                    <Trigger Property="IsSelected" Value="true">
                        <Setter Property="Background" Value="Red"/>
                    </Trigger>
                    <Trigger Property="IsMouseOver" Value="true">
                        <Setter Property="Background" Value="Yellow"/>
                    </Trigger>
                </Style.Triggers>
            </Style>
        </ListBox.ItemContainerStyle>
        <ListBox.Items>
            <ListBoxItem Content="First Item"/>
            <ListBoxItem Content="SecondItem"/>
            <ListBoxItem Content="Third Item"/>
        </ListBox.Items>
    </ListBox>

Я бы ожидал, что невыбранные элементы будут иметь светло-синий фон, элементы наведения (то есть, когда курсор мыши над ними) будет желтым, а выбранные элементы - красным.

Для невыбранных и всплывающих элементов это работает должным образом, но выбранные элементы по-прежнему имеют свой стандартный цвет фона (то есть синий, если список имеет фокус и светло-серый в противном случае).

Есть что-то, что я пропускаю? Задокументировано ли это поведение где-нибудь?

Спасибо за любую подсказку!

EDIT

Мне известно о решении переопределения системных цветов по умолчанию (как описано в Измените выбранный и не сфокусированный стиль списка, чтобы он не был серым , в любом случае спасибо всем, кто опубликовал это как ответ). Однако это не то, что я хочу сделать. Меня больше интересует, почему мое решение не работает.

Я подозреваю, что стандартный ControlTemplate из ListItem определяет его собственные триггеры, которые, кажется, имеют приоритет над триггерами, определенными стилем (возможно, кто-то мог бы подтвердить это и указать мне на некоторый ресурс, где это поведение определено) ,

Тем временем мое решение состоит в том, чтобы определить ControlTemplate для моих ListItem s как:

        <ListBox.ItemContainerStyle>
            <Style TargetType="{x:Type ListBoxItem}">
                <Setter Property="Template">
                    <Setter.Value>
                        <ControlTemplate TargetType="ListBoxItem">
                            <Border Name="Border" Padding="2" SnapsToDevicePixels="true" Background="LightBlue" Margin="0">
                                <ContentPresenter/>
                            </Border>
                            <ControlTemplate.Triggers>
                                <Trigger Property="IsSelected" Value="true">
                                    <Setter TargetName="Border" Property="Background" Value="Red"/>
                                </Trigger>
                            </ControlTemplate.Triggers>
                        </ControlTemplate>
                    </Setter.Value>
                </Setter>
            </Style>
        </ListBox.ItemContainerStyle>

Ответы [ 4 ]

14 голосов
/ 23 марта 2011

Немного размышлений о стиле Aero дает нам объяснение, почему этот простой параметр триггера не работает.

ListBoxItem имеет ControlTemplate с триггерами, который имеет приоритет над нашим триггером.По крайней мере, похоже, что это верно для MultiTrigger.Мне удалось переопределить простой триггер Selected = true, но для мультитриггера мне пришлось создать собственный ControlTemplate.

Это шаблон из стиля Aero, который показывает проблемный MultiTrigger:

<ControlTemplate TargetType="{x:Type ListBoxItem}">
    <ControlTemplate.Triggers>
        <Trigger Property="IsSelected" Value="true">
            <Setter TargetName="Bd" Value="{DynamicResource {x:Static HighlightBrush}}" Property="Background" />
            <Setter Value="{DynamicResource {x:Static HighlightTextBrush}}" Property="Foreground" />
        </Trigger>
        <MultiTrigger>
            <MultiTrigger.Conditions>
                <Condition Property="IsSelected" Value="true" />
                <Condition Property="IsSelectionActive" Value="false" />
            </MultiTrigger.Conditions>
            <Setter TargetName="Bd" Value="{DynamicResource {x:Static ControlBrush}}" Property="Background" />
            <Setter Value="{DynamicResource {x:Static ControlTextBrush}}" Property="Foreground" />
        </MultiTrigger>
        <Trigger Property="IsEnabled" Value="false">
            <Setter Value="{DynamicResource {x:Static GrayTextBrush}}" Property="Foreground" />
        </Trigger>
    </ControlTemplate.Triggers>
    <Border Name="Bd" Background="{TemplateBinding Background}" BorderBrush="{TemplateBinding BorderBrush}" BorderThickness="{TemplateBinding BorderThickness}" Padding="{TemplateBinding Padding}" SnapsToDevicePixels="true">
        <ContentPresenter HorizontalAlignment="{TemplateBinding HorizontalContentAlignment}" VerticalAlignment="{TemplateBinding VerticalContentAlignment}" SnapsToDevicePixels="{TemplateBinding SnapsToDevicePixels}" />
    </Border>
</ControlTemplate>

Надеюсь, это немного прояснит ситуацию.Я не могу понять, почему они так сильно усложнили стиль.

7 голосов
/ 19 марта 2011

удалить IsSelected триггер И добавить в список:

<ListBox.Resources>
    <SolidColorBrush x:Key="{x:Static SystemColors.HighlightBrushKey}"
                         Color="Red" />
    <SolidColorBrush x:Key="{x:Static SystemColors.ControlBrushKey}"
                         Color="Red" />
</ListBox.Resources>

Первая кисть для сфокусированной секунды, в противном случае

3 голосов
/ 19 марта 2011

Попробуйте добавить это к своим ресурсам окна -

 <Window.Resources>
            <SolidColorBrush x:Key="{x:Static SystemColors.HighlightBrushKey}"
                             Color="Red" />
 </Window.Resources>

И удалите триггер IsSelected из вашего кода, он не будет работать, потому что каждая система имеет свою кисть выделения по умолчанию в зависимости от вашей системной темы.

Вам нужно переопределить кисть выделения в вашем коде, чтобы она заработала.

0 голосов
/ 19 марта 2011

Попробуйте использовать Selector.IsSelected в вашем триггере, а не IsSelected.

...