コンテナコントロールのすべてのコントロールを取得する方法はありますか?
質問
多数のコントロールを含むフォームがあり、特定のパネルのすべてのコントロールを反復処理して、それらを有効/無効にしたい
これを試しました:
var component: TComponent;
begin
for component in myPanel do
(component as TControl).Enabled := Value;
end;
しかし、それは何もしませんでした。すべてのコンポーネントが、親オブジェクトではなく、フォームのコンポーネントコレクションにあることがわかります。それでは、コントロール内のすべてのコントロールを取得する方法があるかどうか誰もが知っていますか? (このようない回避策に加えて、私がやらなければならなかったものです):
var component: TComponent;
begin
for component in myPanel do
if (component is TControl) and (TControl(component).parent = myPanel) then
TControl(component).Enabled := Value;
end;
誰かもっと良い方法があると教えてください...
解決
TWinControl.Controls
配列とそれに付随する ControlCount
プロパティを探しています。これらは、コントロールの直接の子供向けです。孫などを取得するには、標準の再帰的手法を使用します。
Components
配列( for
- in
ループが繰り返し処理する配列)は必要ありません。一般的に、親子関係で行います。コンポーネントは子関係のないものを所有でき、コントロールは所有していない子を所有できます。
コントロールを無効にすると、その子もすべて暗黙的に無効になることにも注意してください。無効なコントロールの子と対話することはできません。 OSは入力メッセージを送信しません。ただし、これらを無効にするには外見、個別に無効にする必要があります。つまり、ボタンにグレーのテキストを表示するには、ボタンがマウスクリックに応答しない場合でも、その親を無効にするだけでは不十分です。ボタンを「無効」にペイントするには、ボタン自体を無効にする必要があります。
他のヒント
パネルを無効にすると、そのパネルのすべてのコントロールも無効になります。
匿名メソッドを使用した再帰的ソリューション:
type
TControlProc = reference to procedure (const AControl: TControl);
procedure TForm6.ModifyControl(const AControl: TControl;
const ARef: TControlProc);
var
i : Integer;
begin
if AControl=nil then
Exit;
if AControl is TWinControl then begin
for i := 0 to TWinControl(AControl).ControlCount-1 do
ModifyControl(TWinControl(AControl).Controls[i], ARef);
end;
ARef(AControl);
end;
procedure TForm6.Button1Click(Sender: TObject);
begin
ModifyControl(Panel1,
procedure (const AControl: TControl)
begin
AControl.Enabled := not Panel1.Enabled;
end
);
end;
Delphi 2007の方法は次のとおりです。
procedure TForm6.ModifyControl(const AControl: TControl; const value: Boolean);
var
i: Integer;
begin
if AControl=nil then Exit;
if AControl is TWinControl then begin
for i := 0 to TWinControl(AControl).ControlCount-1 do
ModifyControl(TWinControl(AControl).Controls[i], value);
end;
Acontrol.Enabled := value;
end;
procedure TForm6.Button1Click(Sender: TObject);
begin
ModifyControl(Panel1, true); // true or false
end;
単純に
Panel.Enabled := Value;
この投稿が少し古いことは知っていますが、同じ情報を検索してここに来ました。ここに興味のある人のために考え出したC ++コードがあります。
// DEV-NOTE: GUIForm flattens the VCL controls
// VCL controls are nested. I.E. Controls on a
// Panel would have the Panel as a parent and if
// that Panel is on a TForm, TForm's control count
// does not account for the nested controls on the
// Panel.
//
// GUIControl is passed a Form pointer and an index
// value, the index value will walk the controls on the
// form and any child controls counting up to the idx
// value passed in. In this way, every control has a
// unique index value
//
// You can use this to iterate over every single control
// on a form. Here is example code:
//
// int count = 0;
// TForm *pTForm = some_form
// TControl *pCtrl = 0;
// do
// {
// pCtrl = GUIControl(pTForm, count++);
//
// }while(pCtrl);
TControl *GUIControl(TForm *F, int idx)
{
TControl *rval = 0;
int RunCount = 0;
for(int i=0; i<F->ControlCount && !rval; i++)
{
TControl *pCtl = F->Controls[i];
if(RunCount == idx )
rval = pCtl;
else
rval = GUIChildControl( pCtl, RunCount, idx);
RunCount++;
}
return(rval);
}
TControl *GUIChildControl(TControl *C, int &runcount, int idx)
{
TControl *rval = 0;
TWinControl *pC = dynamic_cast<TWinControl *>(C);
if(pC)
{
for(int i=0; i<pC->ControlCount && !rval; i++)
{
TControl *pCtrl = pC->Controls[i];
runcount++;
if( runcount == idx)
rval = pCtrl;
else
{
TWinControl *pCC = dynamic_cast<TWinControl *>(pCtrl);
if(pCC)
{
if( pCC->ControlCount )
rval = GUIChildControl(pCtrl, runcount, idx);
}
}
}
}
return(rval);
}