#wpf #data-binding #resources #controltemplate #attached-properties
#wpf #привязка данных #Ресурсы #controltemplate #прикрепленные свойства
Вопрос:
У меня есть простой элемент управления, который расширяет возможности управления WPF TextBox
. Основная идея заключается в экономии места в пользовательском интерфейсе, позволяя TextBox
отображать свою собственную метку внутри него, когда она пуста.
Я объявил в нем один DependencyProperty
вызываемый Label
элемент и установил для TextBox.Template
свойства значение ControlTemplate
. ControlTemplate
Это стандартный Windows Aero XAML с добавлением TextBlock
, размещенного за элементом управления, который отображает значение TextBox.Text
свойства. Его Visibility
свойство связано с Converter
, чтобы быть видимым всякий раз, когда Text
свойство пустое. (Там много и в основном неинтересного, поэтому, пожалуйста, извините за полосу прокрутки).
<ControlTemplate x:Key="LabelTextBoxTemplate" TargetType="{x:Type TextBoxBase}" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:s="clr-namespace:System;assembly=mscorlib" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:Aero="clr-namespace:Microsoft.Windows.Themes;assembly=PresentationFramework.Aero">
<Aero:ListBoxChrome Background="{TemplateBinding Panel.Background}" BorderBrush="{TemplateBinding Border.BorderBrush}" BorderThickness="{TemplateBinding Border.BorderThickness}" RenderMouseOver="{TemplateBinding UIElement.IsMouseOver}" RenderFocused="{TemplateBinding UIElement.IsKeyboardFocusWithin}" Name="Border" SnapsToDevicePixels="True">
<Grid>
<TextBlock Text="{Binding Label, ElementName=This}" Visibility="{Binding Text, ElementName=This, Converter={StaticResource StringToVisibilityConverter}}" SnapsToDevicePixels="{TemplateBinding UIElement.SnapsToDevicePixels}" FontWeight="Normal" FontStyle="Italic" Foreground="#99000000" Padding="3,0,0,0" />
<ScrollViewer Name="PART_ContentHost" SnapsToDevicePixels="{TemplateBinding UIElement.SnapsToDevicePixels}" />
</Grid>
</Aero:ListBoxChrome>
<ControlTemplate.Triggers>
<Trigger Property="UIElement.IsEnabled" Value="False">
<Setter TargetName="Border" Property="Panel.Background" Value="{DynamicResource {x:Static SystemColors.ControlBrushKey}}" />
<Setter Property="TextElement.Foreground" Value="{DynamicResource {x:Static SystemColors.GrayTextBrushKey}}" />
</Trigger>
</ControlTemplate.Triggers>
</ControlTemplate>
(Для Name
свойства элемента управления установлено значение This
для привязки)
В любом случае, все это работает хорошо, но мне было интересно, могу ли я заменить эту функциональность прикрепленным свойством. Я создал прикрепленное свойство с именем Label
и добавил делегат обратного вызова, который вызывается при установке свойства. В этом методе я планировал найти ControlTemplate
из Application.Resources
и установить для него TextBox.Template
свойство… Я не смог найти способ получить доступ к ControlTemplate
так что это мой первый вопрос.
Как я могу получить доступ к ControlTemplate
в Application.Resources
в App.xaml из прикрепленного свойства?
Остальная часть вопроса заключается в том, как я могу затем привязаться к прикрепленному свойству в ‘ControlTemplate’?
Я пробовал несколько вещей, таких как
Text="{Binding Path=(Attached:TextBoxProperties.Label),
Source={x:Static Attached:TextBoxProperties}, FallbackValue=''}"
но TextBoxProperties
класс, который содержит прикрепленное свойство, не является статическим.
ОБНОВИТЬ >>>
Благодаря H.B. я обнаружил, что могу получить доступ к ControlTemplate
с помощью следующего кода.
ControlTemplate labelControlTemplate =
(ControlTemplate)Application.Current.FindResource("LabelTextBoxTemplate");
Ответ №1:
Вы должны иметь доступ к Application.Resources
из: Application.Current.Resources
.
Привязка, вероятно, должна быть такой:
{Binding (Attached:TextBoxProperties.Label),
RelativeSource={RelativeSource AncestorType=TextBox}}
Не должно иметь значения, является ли класс, содержащий свойство, статическим, механизм привязки увидит, установлено ли свойство на TextBox
, его не волнует объявляющий класс.
Комментарии:
1. Это работает отлично! Большое спасибо. Я не перестаю удивляться тому, что вы можете создать в WPF без создания пользовательских элементов управления!