依存関係プロパティのメタデータ
Windows Presentation Foundation (WPF) プロパティ システムには、リフレクションや一般的な共通言語ランタイム (CLR) 特性から得られる以上の詳細なプロパティ情報を提供するメタデータ報告システムがあります。 依存関係プロパティのメタデータは、依存関係プロパティを定義するクラスで個別に割り当てることも、依存関係プロパティを別のクラスに追加する際に変更することもできます。また、依存関係プロパティをその定義元の基本クラスから継承するすべての派生クラスで明確にオーバーライドすることもできます。
前提条件
このトピックは、WPF クラスの既存の依存関係プロパティのコンシューマーの観点から依存関係プロパティを理解しており、「依存関係プロパティの概要」を読んでいることを前提としています。 このトピックの例について理解するには、XAML および WPF アプリケーションの記述方法について知っておく必要もあります。
依存関係プロパティのメタデータの使用方法
依存関係プロパティのメタデータは、依存プロパティの特性を照会して調べるためのオブジェクトとして存在します。 また、このメタデータは、特定の依存関係プロパティを処理するために、プロパティ システムによって頻繁にアクセスされます。 依存関係プロパティのメタデータ オブジェクトには、次のような情報が格納されます。
依存関係プロパティの既定値 (ローカル値、スタイル、継承などによってその他の依存関係プロパティ値が指定されない場合)。依存関係プロパティの値を割り当てる際の、既定値と、プロパティ システムで使用される優先順位の関係に関する詳細な説明については、「依存関係プロパティ値の優先順位」を参照してください。
所有者型に基づく強制型変換または変更通知の動作に影響を与えるコールバック実装への参照。 多くの場合、これらのコールバックは非パブリックなアクセス レベルで定義されます。したがって、参照が、許可されたアクセス スコープ内にない限り、一般にメタデータから実際の参照を取得することはできません。 依存関係プロパティのコールバックの詳細については、「依存関係プロパティのコールバックと検証」を参照してください。
対象の依存関係プロパティが WPF フレームワーク レベルのプロパティと見なされる場合、WPF フレームワーク レベルの依存関係プロパティ特性がメタデータに含まれる可能性があります。これは、WPF フレームワーク レベルのレイアウト エンジンやプロパティ継承ロジックなどのサービスの情報および状態を報告します。 この内容に関する依存関係プロパティのメタデータの詳細については、「フレームワーク プロパティ メタデータ」を参照してください。
メタデータ API
プロパティ システムで使用されるほとんどのメタデータ情報を報告する型は PropertyMetadata クラスです。 メタデータ インスタンスは、依存関係プロパティをプロパティ システムに登録する際に必要に応じて指定されます。それ自体を所有者として追加する追加の型、または基本クラスの依存関係プロパティ定義から継承したメタデータをオーバーライドする追加の型に再度指定することもできます。 (プロパティ登録でメタデータが指定されない場合は、そのクラスの既定値を使用して既定の PropertyMetadata が作成されます。) DependencyObject インスタンスの依存関係プロパティからメタデータを取得するさまざまな GetMetadata オーバーロードを呼び出すと、登録されているメタデータが PropertyMetadata として返されます。
その後、WPF フレームワーク レベル クラスなどのアーキテクチャ区分に対してより具体的なメタデータを提供するため、PropertyMetadata クラスが派生されます。 UIPropertyMetadata では、アニメーション報告が追加され、FrameworkPropertyMetadata では、前のセクションで説明した WPF フレームワーク レベル プロパティが提供されます。 依存関係プロパティを登録すると、これらの PropertyMetadata 派生クラスにその依存関係プロパティを登録できるようになります。 メタデータを検査すると、基本の PropertyMetadata 型が派生クラスにキャストされる場合があります。これにより、より具体的な情報プロパティを検査することができます。
注意
このドキュメントでは、FrameworkPropertyMetadata で指定できるプロパティ特性を "フラグ" と呼ぶことがあります。 依存関係プロパティの登録やメタデータのオーバーライドで使用するメタデータ インスタンスを新しく作成する場合、フラグ列挙型である FrameworkPropertyMetadataOptions を使用してこれらの値を指定し、この列挙型の値 (連結された値である可能性があります) を FrameworkPropertyMetadata コンストラクターに渡します。 ただし、メタデータが構築されると、これらのオプション特性は、構築元の列挙値ではなく、一連のブール型プロパティとして FrameworkPropertyMetadata 内で公開されます。 これらのブール型プロパティを使用すると、目的の情報を取得するためにフラグ列挙値に対してマスクを適用することなく、各条件をチェックすることができます。 コンストラクターでは、コンストラクター シグネチャを適切な長さに保つために、連結された FrameworkPropertyMetadataOptions が使用されます。一方、実際に構築されたメタデータでは、より直感的な方法でメタデータを照会できるようにするために、プロパティが個別に公開されます。
メタデータをオーバーライドする場合、クラスを派生する場合
WPF プロパティ システムには、依存関係プロパティを完全に再実装することなく、依存関係プロパティの一部の特性を変更するための機能が用意されています。 これは、特定の型に存在する依存関係プロパティについて、そのプロパティ メタデータの別のインスタンスを構築することで実現されます。 既存の依存関係プロパティの大部分は仮想プロパティではありません。したがって、厳密には、継承クラスでの依存関係プロパティの "再実装" は、既存のメンバーをシャドウすることによってのみ実現されます。
型の依存関係プロパティに対して有効にしようとしているシナリオが、既存の依存関係プロパティの特性の変更では実現できない場合、派生クラスを作成し、その派生クラスでカスタム依存関係プロパティを宣言することが必要になる場合があります。 カスタム依存関係プロパティは、WPF API で定義されている依存関係プロパティとまったく同じように動作します。 カスタム依存関係プロパティの詳細については、「カスタム依存関係プロパティ」を参照してください。
オーバーライドすることができない依存関係プロパティの代表的な特性の 1 つは、依存関係プロパティの値型です。 目的の動作にほぼ合致する依存関係プロパティを継承していても、その依存関係プロパティに別の型が必要な場合には、カスタム依存関係プロパティを実装し、型変換またはカスタム クラスのその他の実装を通じてプロパティをリンクする必要があります。 また、既存の ValidateValueCallback を置換することはできません。このコールバックが、そのメタデータ内ではなく、登録フィールド自体に存在するためです。
既存のメタデータを変更するシナリオ
既存の依存関係プロパティのメタデータを使用している場合、依存関係プロパティのメタデータを変更する一般的なシナリオの 1 つは、既定値を変更することです。 プロパティ システム コールバックの変更または追加は、より高度なシナリオです。 これは、実装している派生クラスの相互関係が依存関係プロパティごとに異なる場合に使用します。 コードと宣言的な使用方法の両方をサポートするプログラミング モデルを使用する条件の 1 つとして、プロパティを任意の順序で設定できる必要があります。 したがって、依存関係プロパティはすべて、コンテキストを使用せずに Just-In-Time で設定する必要があります。また、設定順序 (コンストラクター内の順序など) に依存することもできません。 この内容に関するプロパティ システムの詳細については、「依存関係プロパティのコールバックと検証」を参照してください。 検証コールバックは、メタデータの一部ではなく、依存関係プロパティ識別子の一部であることに注意してください。 したがって、検証コールバックは、メタデータのオーバーライドでは変更できません。
既存の依存関係プロパティに対して、WPF フレームワーク レベルのプロパティのメタデータ オプションの変更が必要になる場合があります。 これらのオプションは、WPF フレームワーク レベルのプロパティに関する特定の既知の条件を、レイアウト システムなどの他の WPF フレームワーク レベルのプロセスに伝達します。 通常、これらのオプションを設定するのは、新しい依存関係プロパティを登録するときだけです。ただし、OverrideMetadata または AddOwner の呼び出しの一部として WPF フレームワーク レベルのプロパティのメタデータを変更することもできます。 使用する具体的な値および詳細については、「フレームワーク プロパティ メタデータ」を参照してください。 新しく登録した依存関係プロパティに対してこれらのオプションを設定する方法の詳細については、「カスタム依存関係プロパティ」を参照してください。
メタデータのオーバーライド
メタデータのオーバーライドの主な目的は、型に存在する依存関係プロパティに適用される、メタデータから派生したさまざまな動作を変更できるようにすることです。 この理由については、「メタデータ」セクションで詳しく説明します。 コード例を含む詳細については、「方法 : 依存関係プロパティのメタデータをオーバーライドする」を参照してください。
プロパティのメタデータは、登録 (Register) の呼び出し時に依存関係プロパティに対して指定できます。 ただし、多くの場合、その依存関係プロパティを継承するクラスに対して、型固有のメタデータを提供する必要があります。 OverrideMetadata メソッドを呼び出すことで、これを行うことができます。 WPF API の例では、Focusable 依存関係プロパティを最初に登録する型は FrameworkElement クラスです。 ただし、Control クラスでは、独自の初期既定値を提供するために (false
から true
) 依存関係プロパティのメタデータがオーバーライドされ、それ以外の場合は元の Focusable の実装が再利用されます。
メタデータをオーバーライドすると、さまざまなメタデータ特性がマージされるか置き換えられます。
PropertyChangedCallback はマージされます。 新しい PropertyChangedCallback を追加すると、そのコールバックがメタデータに格納されます。 オーバーライドで PropertyChangedCallback を指定しないと、PropertyChangedCallback の値は、それをメタデータで指定した最も近い先祖からの参照としてレベル上げされます。
PropertyChangedCallback の実際のプロパティ システムの動作では、階層内のすべてのメタデータ所有者用の実装が維持されてテーブルに追加されます。プロパティ システムによる実行順序としては、派生回数の最も多いクラスのコールバックが最初に呼び出されます。
DefaultValue は置き換えられます。 オーバーライドで DefaultValue を指定しないと、DefaultValue の値は、それをメタデータで指定した最も近い先祖のものになります。
CoerceValueCallback の実装は置き換えられます。 新しい CoerceValueCallback を追加すると、そのコールバックがメタデータに格納されます。 オーバーライドで CoerceValueCallback を指定しないと、CoerceValueCallback の値は、それをメタデータで指定した最も近い先祖からの参照としてレベル上げされます。
プロパティ システム動作では、直接のメタデータ内の CoerceValueCallback だけが呼び出されます。 階層内の他の CoerceValueCallback の実装への参照が保持されることはありません。
この動作は Merge によって実装され、派生メタデータ クラスでオーバーライドできます。
添付プロパティのメタデータのオーバーライド
WPF では、添付プロパティは、依存関係プロパティとして実装されます。 このことは、添付プロパティもプロパティ メタデータを持ち、これを個々のクラスでオーバーライドできることを意味します。 WPF での添付プロパティのスコープに関する考慮事項は、通常、任意の DependencyObject で添付プロパティを設定できるということです。 したがって、クラスのインスタンスで設定されている場合、DependencyObject 派生クラスで添付プロパティのメタデータをオーバーライドできます。 オーバーライドできるのは、既定値、コールバック、または WPF フレームワーク レベルの特性報告プロパティです。 添付プロパティがクラスのインスタンスで設定されている場合、そのオーバーライド プロパティ メタデータ特性が適用されます。 たとえば、プロパティが他では特に指定されていないときには、オーバーライド値がクラスのインスタンス上の添付プロパティの値として報告されるように、既定値をオーバーライドできます。
注意
Inherits プロパティは、添付プロパティに関係しません。
既存の依存関係プロパティの所有者としてのクラスの追加
クラスでは、AddOwner メソッドを使用することで、それ自体を登録済みの依存関係プロパティの所有者として追加できます。 これにより、当初は別の型に対して登録された依存関係プロパティをクラスで使用できるようになります。 通常、追加するクラスは、所有者としてその依存関係プロパティを最初に登録した型の派生クラスではありません。 これにより、元の所有者クラスと追加するクラスが同一のクラス階層内にない場合でも、クラスとその派生クラスで依存関係プロパティの実装を "継承" することが事実上可能になります。 また、追加するクラスとすべての派生クラスでは、型固有のメタデータを元の依存関係プロパティに提供することができます。
追加するクラスは、プロパティ システムのユーティリティ メソッドを介してそれ自体を所有者として追加するだけでなく、追加のパブリック メンバーをそれ自体で宣言する必要があります。これは、依存関係プロパティを完全な形でプロパティ システムに登録し、コードとマークアップの両方に対して公開するためです。 既存の依存関係プロパティを追加するクラスは、その依存関係プロパティのオブジェクト モデルを公開する限り、新しいカスタム依存関係プロパティを定義するクラスと同様の役割を負います。 これらのメンバーの中で最初に公開するのは、依存関係プロパティの識別子フィールドです。 このフィールドは、AddOwner の呼び出しの戻り値に割り当てられる、DependencyProperty 型の public static readonly
フィールドである必要があります。 2 番目に定義するメンバーは、共通言語ランタイム (CLR) の "ラッパー" プロパティです。 ラッパーを使用すると、コード内での依存関係プロパティの操作がずっと簡単になります (SetValue を毎回呼び出す必要がなくなり、ラッパー内で 1 回呼び出すだけで済むようになります)。 このラッパーの実装方法は、カスタム依存関係プロパティを登録する場合の実装方法とまったく同じです。 依存関係プロパティの実装の詳細については、「カスタム依存関係プロパティ」および「依存関係プロパティの所有者の種類を追加する」を参照してください。
AddOwner および添付プロパティ
所有者クラスで添付プロパティとして定義されている依存関係プロパティに対して、AddOwner を呼び出すことができます。 通常、これは、以前の添付プロパティを非添付の依存関係プロパティとして公開する目的で行います。 次に、AddOwner の戻り値を、依存関係プロパティの識別子として使用される public static readonly
フィールドとして公開し、プロパティがメンバー テーブルに表示され、クラス内で非添付プロパティを使用できるように、適切な "ラッパー" プロパティを定義します。
関連項目
.NET Desktop feedback