如果我在控件的构造函数中检查该属性,则该属性始终是默认值,即使我在设计器中指定了其他值。
正确,因为它的设计时值尚未分配。
这些已发布的属性实际上是在什么时候设置的?
构造所有者(窗体、框架或数据模块)时。它加载自己的 DFM 资源并对其进行解析,构建存储的子组件并读取它们的属性值。
例如,假设您有以下 DFM:
object Form1: TForm1
Left = 0
Top = 0
Caption = 'Form1'
...
object Edit1: TEdit
Left = 136
Top = 64
Width = 121
Height = 21
TabOrder = 0
end
object Button1: TButton
Left = 263
Top = 62
Width = 75
Height = 25
Caption = 'Button1'
TabOrder = 1
end
end
DFM 流处理大致转换为以下等效代码(为简单起见,我省略了很多内部细节):
__fastcall TCustomForm::TCustomForm(TComponent *Owner)
: TScrollingWinControl(Owner)
{
this->FFormState << fsCreating;
try
{
// locate, load, and parse the "Form1" DFM resource ...
this->FComponentState << csLoading;
this->Parent = ...;
this->Name = L"Form1":
this->FComponentState << csReading;
this->Left = 0;
this->Top = 0;
this->Caption = L"Form1";
...
TEdit *e = new TEdit(this);
try
{
e->FComponentState << csLoading;
e->Parent = this;
e->Name = L"Edit1"; // <-- sets the derived Form's 'Edit1' member to this object
e->FComponentState << csReading;
e->Left = 136;
e->Top = 64;
e->Width = 121;
e->Height = 21;
e->TabOrder = 0;
e->FComponentState >> csReading;
}
catch (...)
{
delete e;
throw;
}
TButton *b = new TButton(this);
try
{
b->FComponentState << csLoading;
b->Parent = this;
b->Name = L"Button1"; // <-- sets the derived Form's 'Button1' member to this object
b->FComponentState << csReading;
b->Left = 263;
b->Top = 62;
b->Width = 75;
b->Height = 25;
b->Caption = L"Button1";
b->TabOrder = 1;
b->FComponentState >> csReading;
}
catch (...)
{
delete b;
throw;
}
this->FComponentState >> csReading;
...
e->Loaded();
b->Loaded();
this->Loaded();
}
__finally
{
this->FFormState >> fsCreating;
}
}
因此,如您所见,当调用其构造函数时,组件的属性值尚不可用。
一旦属性设置正确,我可以/应该使用什么方法来根据这些属性在我的控件中执行一些逻辑?
这取决于属性需要做什么。如果他们需要立即执行操作,您可以直接在他们的属性设置器中执行此操作。但是如果他们需要等到其他属性首先被加载(如果一个属性依赖于另一个属性的值),然后重写虚拟Loaded()
方法,它会在 DFM 流式传输完成后自动调用。属性设置者可以检查ComponentState
属性的标志,以了解组件当前是否在设计时在表单设计器中运行,当前是否正在流式传输 DFM 等,然后根据需要采取相应的行动。
我尝试覆盖加载的方法,但仍然遇到同样的问题
究竟是什么?你没有解释你的实际问题是什么。请编辑您的问题以提供这些详细信息。
所以我不认为这正是我想要的。
最有可能的是,您可能只是没有正确使用它。