Как изменить стиль кнопки по умолчанию, не возвращая WPF из Aero в Classic? - PullRequest
6 голосов
/ 25 мая 2010

Я добавил PresentationFramework.Aero в свои объединенные словари App.xaml, как в ...

<Application
    x:Class="TestApp.App"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
    <Application.Resources>
        <ResourceDictionary>
            <ResourceDictionary.MergedDictionaries>
                <ResourceDictionary
                    Source="/PresentationFramework.Aero;component/themes/Aero.NormalColor.xaml" />
                <ResourceDictionary
                     Source="pack://application:,,,/WPFToolkit;component/Themes/Aero.NormalColor.xaml" />
                <ResourceDictionary
                    Source="/CommonLibraryWpf;component/ResourceDictionaries/ButtonResourceDictionary.xaml" />
                    <!-- Note, ButtonResourceDictionary.xaml is defined in an external class library-->
            </ResourceDictionary.MergedDictionaries>
        </ResourceDictionary>
    </Application.Resources>
</Application>

Я пытаюсь слегка изменить вид кнопок по умолчанию. Я поместил этот стиль в свой ButtonResourceDictionary:

<Style TargetType="Button">
    <Setter Property="Padding" Value="3" />
    <Setter Property="FontWeight" Value="Bold" />
</Style>

Все кнопки теперь имеют правильный отступ и жирный текст, но они выглядят как «Классический», а не «Аэро». Как мне исправить этот стиль, чтобы все мои кнопки выглядели Aero, но также имели эти незначительные изменения? Я бы предпочел не устанавливать свойство Style для каждой кнопки.

Обновление

Я должен был упомянуть об этом в первую очередь, но если я попытаюсь использовать BasedOn, как показано ниже, я получу StackOverflowException:

<Style BasedOn="{StaticResource {x:Type Button}}" TargetType="{x:Type Button}">
    <Setter Property="Padding" Value="3" />
    <Setter Property="FontWeight" Value="Bold" />
</Style>

Обычно это работает, но не с объединенными словарями Aero. Если я закомментирую эти словари, исключение исчезнет.

Обновление 2

Если я добавлю атрибут x:Key и вручную установлю стиль, он будет работать правильно (стиль Aero с отступами и полужирным шрифтом), но, как я уже сказал, я бы предпочел, чтобы стиль автоматически применялся глобально ко всем кнопкам.

Обновление 3

Я только что обнаружил новую морщину. В моем приложении ButtonResourceDictionary.xaml помещается в библиотеку классов (т. Е. Во внешний проект). Если я переместить этот файл в локальную папку, все работает нормально. Таким образом, проблема заключается в плохом взаимодействии, вызванном ссылками на различные словари внешних ресурсов. Я исправляю свой фрагмент кода App.xaml (см. Выше), чтобы отразить, что ButtonResourceDictionary фактически определен внешне.

Ответы [ 3 ]

8 голосов
/ 22 июня 2010

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

UPDATE

Вот решение, которое я придумал:

<Style TargetType="TextBox" BasedOn="{Common:StaticApplicationResource {x:Type TextBox}}">
    <Setter Property="Height" Value="21"/>
</Style>

Где StaticApplicationResource - это написанное мной пользовательское расширение MarkupExtension, которое просто вызывает TryFindResource:

[MarkupExtensionReturnType(typeof(object))]
public class StaticApplicationResource : MarkupExtension
{
    public StaticApplicationResource(object pResourceKey)
    {
        mResourceKey = pResourceKey;
    }

    private object _ResourceKey;

    [ConstructorArgument("pResourceKey")]
    public object mResourceKey
    {
        get { return _ResourceKey; }
        set { _ResourceKey = value; }
    }

    public override object ProvideValue(IServiceProvider serviceProvider)
    {
        if (mResourceKey == null)
            return null;

        object o = Application.Current.TryFindResource(mResourceKey);

        return o;
    }
}

Таким образом, мне не нужно ссылаться на мои словари ресурсов вне моего файла App.xaml, что мне нравится :). Вы также можете использовать более сложную логику, позволяя вам разрешать стиль BasedOn любым удобным для вас способом. Вот отличная статья , показывающая, как загружать словари ресурсов (и те, которые среда разрешает автоматически) из кода.

0 голосов
/ 25 мая 2010

На основании ваших обновлений вы можете сделать это (правда, это ужасно некрасиво):

<Style x:Key="_buttonStyleBase"
       BasedOn="{StaticResource {x:Type Button}}"
       TargetType="{x:Type Button}"> 
    <Setter Property="Padding" Value="3" /> 
    <Setter Property="FontWeight" Value="Bold" /> 
</Style>

<Style TargetType="{x:Type Button}"
       BasedOn="{StaticResource _buttonStyleBase}" />
0 голосов
/ 25 мая 2010

Используйте атрибут BasedOn для наследования свойств от стиля Aero. Это должно решить вашу проблему.

<Style
  BasedOn="{StaticResource {x:Type Button}}"
  TargetType="{x:Type Button}"> 
    <Setter Property="Padding" Value="3" /> 
    <Setter Property="FontWeight" Value="Bold" /> 
</Style>
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...