А пока было интересно потратить вечер и добиться такого результата с минимальными изменениями рабочего кода. Забегая вперед, скажу что результат, как и планировалось, получился довольно чистым: фрагмент следующего кода, если не считать нескольких аттрибутов пропущенных для наглядности, это и есть окно с первого скриншота. Все изменения ограничились заданием стиля. Обновление 3 декабря: в репозиторий добавлена альтернативная имплементация использующая новые классы в . Net 4. 5 (проект Window.
Chrome. Demo), что позволило избежать существенной части нативного программирования с Win. API.< Window ..
Демонстрационный проект доступен на github'е, если вы захотите поразбираться с исходниками самостоятельно или же просто использовать этот стиль не вдаваясь в подробности. Основная проблема. WPF не работает с NC- area. NC, она же «Non- client area», она же «не- клиентская часть», она же хром, обрабатывается на более низком уровне. Если вам захотелось изменить какой- то из элементов окна — бордюр, иконку, заголовок или кнопку, то первый совет, который попадается при поиске — это убрать стиль окна и переделать все самому. Целиком.< Window. Allows. Transparency=.
К счастью, у меня были исходники из старинного поста Алекса Яхнина по стилизации под Офис 2. WPF для Микрософта, так что с нуля начинать мне не грозило. В итоге нам надо получить один стиль, и по возможности, без дополнительных контролов: в дереве проекта XAML и код стиля расположились в директории Customized. Window, а основное окно в корне проекта. Мне хотелось избежать добавления новых библиотек в проект, но сохранить возможность легко перенести стиль в другое приложение, что и определило такую структуру. Создаем стиль. Стиль для окна, как и для любого другого контрола в WPF задается при помощи Control. Template. Содержимое окна будет показываться Content.
Как можно заметить, я решил потихоньку переходит с построения приложений WinForms на приложения WPF. Почти заканчиваю . Темы WPF представляют собой наборы стилизованных шаблонов элементов управления WPF. Содержит векторные иконки, написанные на XAML.
Presenter'ом, а функциональность которую проще сделать в коде c#, подключится через x: Class атрибут в Resource. Dictionary. Все очень стандартно для XAML'а.< Resource. Dictionary. x: Class=. Это будет единственный дополнительный глобальный стиль на случай если потом возникнет желание использовать такие кнопки в приложении. Нам нужна функциональность обычной кнопки, но с очень примитивной отрисовкой — фактически только фон и содержимое. XAML стиля кнопки< Style x: Key=. Например, вот так выглядит maximize: < Path Stroke. Исходник Скайпа На Devel Studio.
Thickness=. Единственная особенность здесь — убедиться, что текст отрисован без размытия, иначе заголовок окна будет выглядеть. Windows жульничает, масштабируя окно так, что бордюр целиком уходит за границу экрана, оставляя на мониторе только клиентскую часть окна. Естественно, что Windows при этом больше не отрисовывает бордюр и для стандартных окон все работает как ожидалось. WPF это никак не отрабатывает и, для таких окон как у нас, есть риск потерять часть изображения или начать рисовать на соседнем мониторе, если он подключен. Остальные детали менее существенны, но если интересно, добро пожаловать в исходники. Оживляем окно. Net 4.
Помимо реакции на кнопки и иконку, окно должно перемещаться и изменять размер при drag'е за заголовок, за края и уголки. Соответствующие горячие зоны проще всего задать при помощи невидимых контролов. Пример для левого верхнего (северо- западного) угла.< Rectangle. Name=. Сами обработчики, например Min.
Button. Click и On. Size. North. West, выглядят примерно так: void Min. Button. Click(object sender, Routed. Event. Args e) . При добавлении к окну, Window. Chrome берет на себя функции изменения размера, положения и состояния окна, оставляя нам более «глобальные» проблемы.
Зададим триггеры для контроля изменений интерфейса при изменении состояния окна. Вернемся в XAML и, например, заставим Status. Bar'ы изменять цвет в зависимости от значения Window. Is. Active. XAML для Status. Bar'а< Style. Resources>.
Style Target. Type=. Помните самый первый фрагмент с пользовательским кодом?< Window .. При желании и времени так же можно задать и стиль для других классов контролов, например подправить Scroll. Bar, чтобы он тоже соответствовал нужному стилю. Но это уже будет упражнение на следующий свободный вечер. Собираем все вместе. Все. Нам осталось только подключить стиль к проекту через ресурсы приложения: < Application ..
Еще раз ссылка на исходники на github'е для тех кто сразу прокрутил вниз ради нее.