У меня есть простой элемент управления, который расширяет 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
для привязки)
Так или иначе, все это работает хорошо, но мне было интересно, смогу ли я заменить эту функцию на Attached Property. Я создал Attached Property с именем Label
и добавил делегата обратного вызова, который вызывается при установке свойства. В этом методе я планировал найти ControlTemplate
из Application.Resources
и установить для него свойство TextBox.Template
... Я не смог найти способ доступа к ControlTemplate
, так что это мой первый вопрос.
Как я могу получить доступ к ControlTemplate
в Application.Resources
в App.xaml из присоединенного свойства?
Остальная часть вопроса заключается в том, как я могу затем привязать свойство Attached в «ControlTemplate»?
Я пробовал несколько вещей, таких как
Text="{Binding Path=(Attached:TextBoxProperties.Label),
Source={x:Static Attached:TextBoxProperties}, FallbackValue=''}"
, но класс TextBoxProperties
, содержащий свойство Attached, не является статическим.
ОБНОВЛЕНИЕ >>>
Благодаря Х.Б. Я обнаружил, что могу получить доступ к ControlTemplate
, используя следующий код.
ControlTemplate labelControlTemplate =
(ControlTemplate)Application.Current.FindResource("LabelTextBoxTemplate");