Как оформить заголовок экспандера WPF? - PullRequest
49 голосов
/ 19 марта 2009

Я хотел бы применить стиль к заголовку расширителя WPF. В следующем XAML у меня есть Expander, но стиль для всего этого не только для заголовка.

Спасибо.

<Page
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Width="640"
>
    <StackPanel>
        <StackPanel.Resources>
            <Style TargetType="Expander">
                <Style.Resources>
                    <LinearGradientBrush x:Key="BackBrush" StartPoint="0.5,0" EndPoint="0.5,1">
                        <GradientStop Color="#EF3132" Offset="0.1" />
                        <GradientStop Color="#D62B2B" Offset="0.9" />
                    </LinearGradientBrush>
                </Style.Resources>
                <Setter Property="Background" Value="{StaticResource BackBrush}"/>
            </Style>
        </StackPanel.Resources>
        <Expander>
            <StackPanel>
                <TextBlock>Bike</TextBlock>
                <TextBlock>Car</TextBlock>
                <TextBlock>Truck</TextBlock>
            </StackPanel>
        </Expander>
    </StackPanel>
</Page>

Ответы [ 3 ]

53 голосов
/ 19 марта 2009

Я объединил немного XAML из Джоша Смита и MSDN и придумал решение. Действительно, элемент управления (по крайней мере, заголовок) должен быть повторно помечен.

<Page
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Width="400">
    <StackPanel>
        <StackPanel.Resources>

            <Style TargetType="Border" x:Key="RacePitBorderStyle" >
                <Style.Resources>
                    <LinearGradientBrush x:Key="BackBrush" StartPoint="0.5,0" EndPoint="0.5,1">
                        <GradientStop Color="#EF3132" Offset="0.1" />
                        <GradientStop Color="#D62B2B" Offset="0.9" />
                    </LinearGradientBrush>
                </Style.Resources>
                <Setter Property="Background" Value="{StaticResource BackBrush}"/>
            </Style>

            <DataTemplate x:Key="titleText">
                <Border Style="{StaticResource RacePitBorderStyle}" Height="24">
                    <TextBlock Text="{Binding}" 
                        Margin="4 0"
                        VerticalAlignment="Center"
                        Foreground="White"
                        FontSize="11" 
                        FontWeight="Normal"
                        Width="{Binding
                        RelativeSource={RelativeSource
                        Mode=FindAncestor,
                        AncestorType={x:Type Expander}},
                        Path=ActualWidth}"
                        TextWrapping="Wrap"/>
                </Border>
            </DataTemplate>

            <Style TargetType="{x:Type Expander}">
                <Setter Property="HeaderTemplate" Value="{StaticResource titleText}"/>
            </Style>

        </StackPanel.Resources>

        <Expander Name="hcontCtrl" Header="This is the header.">
            <StackPanel>
                <TextBox>This is a textbox</TextBox>
                <Button>A button</Button>
            </StackPanel>
        </Expander>

    </StackPanel>
</Page>
15 голосов
/ 16 мая 2014

Я думаю, что ответ Василе на правильном пути, но кажется, что он делает намного больше, чем нужно оригинальному постеру. Все, что требовалось сделать, это изменить фон заголовка. В то время как представленное изменение делает это, оно также делает другие вещи.

Одной из этих вещей является замена реализации по умолчанию, я полагаю, ContentPresenter, на TextBlock. Так что же произойдет, когда позже мы добавим второй расширитель к этой панели стека? Может быть что-то вроде:

<Expander>
    <Expander.Header>
        <StackPanel>
            <Border height="5" width="5" Foreground="Blue"/>
            <TextBlock>Ha!</TextBlock>
        </StackPanel>
    </Expander.Header>
</Expander>

Не знаю, но это нехорошо. Вместо этого я думаю, что мы хотим сохранить это простым.

<DataTemplate x:Key="expanderHeader">
    <ContentPresenter
        Content={Binding}
        TextBlock.Background={StaticResource myBrush}/>
</DataTemplate>

<Style TargetType="Expander">
    <Setter Property="HeaderTemplate" Value="{StaticResource expanderHeader}"/>
</Style>

Таким образом, когда кто-то помещает что-то, что не просто текст, в наш стилизованный расширитель, мы не ломаемся. Если вы хотите убедиться, что вы обернули все, что они делают с этим фоном, что, вероятно, желательно, это будет выглядеть так:

<DataTemplate x:Key="expanderHeader">
    <Border Background={StaticResource myBrush}>
        <ContentPresenter Content={Binding}/>
    </Border>
</DataTemplate>
6 голосов
/ 19 марта 2009

Зависит от того, что вы хотите стилизовать - вы можете стилизовать любую его часть. Если вы хотите изменить содержимое в заголовке, просто поместите весь ваш пользовательский интерфейс в свойство Expander.Header, и оно будет отображаться в области заголовка.

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

...