Подтвердить что ты не робот

WPF - HeaderStringFormat Не работает в Expander

Я не могу найти волшебную комбинацию, чтобы заставить HeaderStringFormat работать для Expander WPF.

Вот все, что я пробовал:

<Expander Header="{Binding Path=MyProperty, StringFormat=Stuff: ({0})}"  >
    <TextBlock Text="Some Content" />
</Expander>
<Expander HeaderStringFormat="{}Stuff ({0})" Header="{Binding Path=MyProperty}">
    <TextBlock Text="Some More Content" />
</Expander>
<Expander HeaderStringFormat="{}Stuff ({0:0})" Header="{Binding Path=MyProperty}">
    <TextBlock Text="Even More Content" />
</Expander>

Единственный способ, с помощью которого я могу правильно отформатировать строку в моем коде, это сделать:

<Expander>
    <Expander.Header>
        <TextBlock Text="{Binding Path=MyProperty, StringFormat=Stuff: ({0})}" />
    </Expander.Header>
    <Expander.Content>
        A Expander with working header
    </Expander.Content>
</Expander>

Что я делаю неправильно?

4b9b3361

Ответ 1

Прежде всего следует отметить следующее:

Если вы установите HeaderTemplate или Свойство HeaderTemplateSelector HeaderedContentControl, Свойство HeaderStringFormat игнорируются. MSDN

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

Во-вторых, это не то, что:

String.Format("My string value is: {0}", myValue");

HeaderedContentControl и HeaderStringFormat используются специально для классов, которые реализуют IFormattable. HederStringFormat форматирует заголовок, а ContentStringFormat форматирует содержимое. Значение любого свойства - это формат, который передается в реализацию вашего класса, если IFormattable.ToString. Вы можете прочитать полный пример MSDN. Но вот суть того, как заставить его работать.

public class MyTestClass : IFormattable
{
    #region IFormattable Members
    public string ToString(string format, IFormatProvider formatProvider)
    {
        if(format == "n")
        {
            return "This is my formatted string";
        }
        else
        {
            return "this is my non-formatted string";
        }
    }
    #endregion
}

    <Style TargetType="{x:Type TabItem}">
        <Setter Property="HeaderStringFormat" Value="n" />
        <Setter Property="ContentStringFormat" Value="" />
    </Style>

<TabControl>
    <TabItem Header="{Binding Content, RelativeSource={RelativeSource Self}}">
        <local:MyTestClass />
    </TabItem>
</TabControl>

Теперь этот TabItem будет отображать "Это моя форматированная строка" в заголовке, а контент будет "это моя неформатированная строка".

Есть пара вещей, которые нужно иметь в виду. Обычно эти свойства будут использоваться только в контексте HeaderedItemsControl. HeaderStringFormat не будет связан таким образом и вместо этого будет иметь привязку по умолчанию, предоставленную ItemContainer HeaderedItemsControl. Например, если вы установите свойство ItemsSource для TabItem, он автоматически подключит заголовок и привязку содержимого для вас, и все, что вам нужно сделать, - это предоставить нужное значение форматирования.

Наконец, но не в последнюю очередь, мне удалось все нормально работать с GroupBox и TabItem, но не так много удачи с расширителем, и я не уверен, почему. Расширитель правильно обрабатывает ContentStringFormat, но не HeaderContentStringFormat. Это удивительно, учитывая, что оба наследуются от HeaderContentControl.