05_Березина_DependencyProperties_Base

advertisement
Классы Windows Presentation Foundation
System.Object
DispatcherObject
Application
DependencyObject
NavigationService
FrameworkTemplate
Visual
ContentElement
UIElement
FrameworkContentElement
Style
FrameworkElement
Page
Shape
TextBlock
ContentControl
RangeBase
TextBoxBase
Frame
Slider
TextBox
Window
ProgressBar
RichTextBox
Grid
ItemsControl
Menu
ContextMenu
HeaderedItemsControl
MenuItem
Button
Toolbar
TreeViewItem
Selector
RepeatButton
TabControl
ToggleButton
CheckBox
Canvas
MenuBase
NavigationWindow
ButtonBase
Panel
ToolBarTray
Control
RadioButton
ListBox
ListView
TreeView
ComboBox
Свойства зависимостей (dependency properties)
 Система свойств WPF (WPF property system ) - набор сервисов, которые расширяют
функциональность свойств CLR.
 Свойства, которые управляются системой свойств WPF, называются свойствами
зависимостей (dependency properties).
 Основной функцией системы свойств WPF является вычисление значений свойств и
уведомление об изменении значений.
 Изменение значения свойства зависимостей может быть связано с различными
участниками системы свойств WPF – стилями, триггерами (triggers), наследованием
значений свойств, установкой локальных значений.
 Свойства зависимостей (dependency properties) поддерживают
• проверку корректности значений (property invalidation)
• приведение типа (dependent-value coercion)
• значения по умолчанию (default values)
• наследование (inheritance)
• привязку данных (data binding)
• анимацию (animation)
• оповещение об изменении значений (property change notification)
• cтили (styling)
Свойства зависимостей и свойства CLR
 Для свойств зависимостей из библиотек WPF определены методы {get; set},
позволяющие использовать для них обычный синтаксис CLR для свойств.
 В документации WPF есть информация о том, что свойство реализовано как свойство
зависимостей.
Например, свойство string Text { get; set; } класса TextBox определено как свойство
зависимостей, а свойство string SelectedText { get; set; } класса TextBox не является
свойством зависимостей.
 В WPF определены два класса для поддержки свойств зависимостей:
• DependencyProperty поддерживает регистрацию свойства зависимостей в сиcтеме
свойств WPF и информацию о свойствах зависимостей;
• DependencyObject используется как базовый для всех типов, в которых
определяются свойства зависимостей.
Класс DependencyProperty
System.Object
 Класс поддерживает регистрацию свойства зависимостей
в сиcтеме свойств WPF.
System.Windows.DependencyProperty
 Чтобы определить пользовательское свойство зависимости, необходимо,
• зарегистрировать его с помощью статического метода Register класса
DependencyProperty в системе свойств WPF;
• сохранить в открытом статическом поле класса (доступном только для чтения)
ссылку на объект DependencyProperty, который возвращает метод Register при
успешной регистрации.
 При регистрации можно
• определить для свойства обычные CLR методы доступа {get; set} (рекомендуется);
• с помощью объекта типа PropertyMetadata передать информацию о том, какие
службы (привязка данных, ведение журнала, …) будут использоваться с
регистрируемым свойством зависимостей.
Регистрация свойства зависимости
 Регистрацию свойства зависимостей выполняет статический метод Register класса
DependencyProperty. Метод перегружен (3). Перегруженный вариант с наибольшим числом
параметров
public static DependencyProperty Register( string name,
Type propertyType,
Type ownerType,
PropertyMetadata typeMetadata,
ValidateValueCallback validateValueCallback );
 При регистрации необходимо указать
• имя свойства (string name);
• тип свойства (Type propertyType);
• тип-владелец, в котором регистрируется свойство зависимости (Type ownerType).
 Дополнительно можно определить метаданные свойства зависимостей
(PropertyMetadata typeMetadata) и метод обратного вызова для пользовательской проверки
корректности значений свойства .
 В метаданных свойства зависимостей можно предусмотреть методы обратного вызова,
которые будут вызываться при изменении значения свойства и для анализа допустимости
(coerce) значения свойства.
Пример регистрации свойства зависимостей
public class DoubleValueTextBox : StackPanel
{
public static readonly DependencyProperty DoubleValueProperty;
static DoubleValueTextBox()
{
PropertyMetadata metadata = new PropertyMetadata(OnDoubleValueChanged);
metadata.DefaultValue = (double) double.NaN;
DoubleValueProperty =
DependencyProperty.Register("DoubleValue", typeof(double),
typeof(DoubleValueTextBox), metadata);
}
public double DoubleValue
{
get { return (double)this.GetValue(DoubleValueProperty); }
set { this.SetValue(DoubleValueProperty, value); }
}
 В примере в классе DoubleValueTextBox в статическом конструкторе зарегистрировано свойство
зависимостей с именем DoubleValue.
 Свойство имеет тип double, его владельцем является тип DoubleValueTextBox.
 Ссылка на объект DependencyProperty находится в поле DoubleValueProperty класса-владельца (как
имя статического поля класса принято использовать имя свойства, к которому добавлен корень
Property).
 Для свойства определены методы {get; set}, позволяющие использовать для него обычный синтаксис
CLR для свойств (методы GetValue и SetValue определены в классе DependencyObject).
 При регистрации для свойства задано значение NaN по умолчанию.
Класс DependencyObject
System.Object
System.Windows.Threading.DispatcherObject
System.Windows.DependencyObject
 Класс является базовым для типов, использующих
свойства зависимостей WPF.
 Метод SetValue класса DependencyObject задает
локальное значение свойства зависимостей. Метод
бросает исключение , если значение value имеет тип,
который не может быть приведен к типу, указанному при
регистрации свойства.
public Object GetValue ( DependencyProperty dp );
public void SetValue ( DependencyProperty dp, Object value );
 Метод GetValue класса DependencyObject возвращает текущее действующее значение
свойства зависимостей. Действующее значение вычисляется системой свойств WPF с
учетом всех данных, которые влияют на значение данного свойства зависимости.
Проверка корректности значения свойства
 Для проверки корректности значения свойства зависимости следует определить
отдельный метод с сигнатурой, отвечающей делегату ValidateValueCallback, который
указывается при регистрации свойства зависимости и работает как метод обратного
вызова.
public delegate bool ValidateValueCallback( Object value );
Метод выполняет пользовательскую проверку значения свойства зависимостей после
обычной проверки типа, и должен возвращать значение true, если значение свойства
принимается, и false в противном случае. Метод не предполагает корректировку значения
свойства – новое значение принимается или нет.
 Проверку корректности значения следует выполнять в отдельном методе, а не в
упаковщиках свойства { get; set}, так как
• код WPF может вызвать методы SetValue или GetValue напрямую без вызова { get;
set} ;
• значение может зависеть от других объектов, например, свойство может
унаследовать свое значение через дерево элементов или получить его в результате
привязки данных.
Присоединенные (attached) свойства зависимостей
 Присоединенные (вложенные) свойства зависимостей - это механизм, который дает
возможность присваивать значения свойствам других классов.
 Например, присоединенными свойствами зависимости являются свойства Row и Column
в классе Grid.
 Эти свойства зарегистрированы в класе Grid как присоединенные (вложенные), вызовы
методов GetValue и SetValue класса DependencyObject находятся в статических методах
класса Grid:
public static void SetRow( UIElement element, int value );
public static int GetRow( UIElement element );
 Пример использования в разметке:
<Grid Name="grid1" ShowGridLines="True">
<Grid.RowDefinitions>
<RowDefinition Height="1*" />
<RowDefinition Height="1*" />
</Grid.RowDefinitions>
<Button Grid.Row="1" Margin="5,29,5,50" Name="button1“
Click="button1_Click"> Add </Button>
</Grid>
 Пример использования в коде:
Button btn = new Button();
btn.Content = "Set";
Grid.SetRow(btn, 1);
Grid.SetColumn(btn, 1);
grid_1.Children.Add(btn);
Регистрация присоединенного свойства зависимостей
 Для регистрации присоединенного свойства зависимостей используется одна из
перегрузок метода RegisterAttached класса DependencyProperty. Перегруженный вариант с
наибольшим числом параметров
public static DependencyProperty RegisterAttached( string name,
Type propertyType,
Type ownerType,
PropertyMetadata defaultMetadata,
ValidateValueCallback validateValueCallback );
 Параметры метода имеют такой же смысл как в методе Register().
 Регистрация присоединенного свойства отличается от регистрации свойства
зависимости тем, что вызовы методов GetValue и SetValue упаковываются не в методы
{get; set }, а в статические методы с именами SetИмяСвойства() и GetИмяСвойства().
Download