15

既存MyControl1.Controls.OfType<RadioButton>()の検索は初期コレクションのみで、子には入力されません。

独自の再帰メソッドを使用してEnumerable.OfType<T>()、または使用せずに、特定のタイプのすべての子コントロールを見つけることは可能ですか? このLINQように。

4

3 に答える 3

43

拡張メソッドを使用してコントロール階層をフラット化し、フィルターを適用するため、独自の再帰メソッドを使用しています。

メソッドは次のようになります

public static IEnumerable<Control> FlattenChildren(this Control control)
{
  var children = control.Controls.Cast<Control>();
  return children.SelectMany(c => FlattenChildren(c)).Concat(children);
}
于 2010-02-05T19:42:12.313 に答える
1

上記の回答を改善するには、戻り値の型を次のように変更するのが理にかなっています

//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>();
}
于 2013-07-25T12:49:46.097 に答える
1

私はこの一般的な再帰的方法を使用します:

このメソッドの前提は、コントロールが 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;
}
于 2014-04-22T08:51:28.780 に答える