既存MyControl1.Controls.OfType<RadioButton>()
の検索は初期コレクションのみで、子には入力されません。
独自の再帰メソッドを使用してEnumerable.OfType<T>()
、または使用せずに、特定のタイプのすべての子コントロールを見つけることは可能ですか? このLINQ
ように。
拡張メソッドを使用してコントロール階層をフラット化し、フィルターを適用するため、独自の再帰メソッドを使用しています。
メソッドは次のようになります
public static IEnumerable<Control> FlattenChildren(this Control control)
{
var children = control.Controls.Cast<Control>();
return children.SelectMany(c => FlattenChildren(c)).Concat(children);
}
上記の回答を改善するには、戻り値の型を次のように変更するのが理にかなっています
//Returns all controls of a certain type in all levels:
public static IEnumerable<TheControlType> AllControls<TheControlType>( this Control theStartControl ) where TheControlType : Control
{
var controlsInThisLevel = theStartControl.Controls.Cast<Control>();
return controlsInThisLevel.SelectMany( AllControls<TheControlType> ).Concat( controlsInThisLevel.OfType<TheControlType>() );
}
//(Another way) Returns all controls of a certain type in all levels, integrity derivation:
public static IEnumerable<TheControlType> AllControlsOfType<TheControlType>( this Control theStartControl ) where TheControlType : Control
{
return theStartControl.AllControls().OfType<TheControlType>();
}
私はこの一般的な再帰的方法を使用します:
このメソッドの前提は、コントロールが T の場合、メソッドはその子を検索しないということです。その子も参照する必要がある場合は、それに応じて簡単に変更できます。
public static IList<T> GetAllControlsRecusrvive<T>(Control control) where T :Control
{
var rtn = new List<T>();
foreach (Control item in control.Controls)
{
var ctr = item as T;
if (ctr!=null)
{
rtn.Add(ctr);
}
else
{
rtn.AddRange(GetAllControlsRecusrvive<T>(item));
}
}
return rtn;
}