如何正确使用TValue.AsType <tnotifyevent>?</tnotifyevent>

时间:2010-03-10 18:37:19

标签: delphi delphi-2010 rtti

我正在尝试使用RTTI向控件添加事件处理程序,该控件可能已经设置了事件处理程序。代码看起来像这样:

var
  prop: TRttiProperty;
  val: TValue;
begin
  prop := FContext.GetType(MyControl.ClassInfo).GetProperty('OnChange');
  val := prop.GetValue(MyControl);
  FOldOnChange := val.AsType<TNotifyEvent>;
  prop.SetValue(MyControl, TValue.From<TNotifyEvent>(self.MyOnChange));
end;

我想这样,所以我可以在MyOnChange中执行此操作:

begin
  if assigned(FOldOnChange) then
    FOldOnChange(Sender);
  //additional code here
end;

不幸的是,编译器似乎不喜欢行FOldOnChange := val.AsType<TNotifyEvent>;。它说

  

E2010不兼容的类型:'程序,   无类型指针或无类型参数'   和'TNotifyEvent'

任何人都知道为什么会这样或如何解决它?看起来对我来说......

2 个答案:

答案 0 :(得分:6)

FOldOnChange是方法指针类型,而AsType<TNotifyEvent>是方法。编译器认为您正在尝试将方法分配给方法指针。解决方案是将()附加到方法调用以强制它,并使用方法的返回值作为要分配给FOldOnChange的值。

这是一个完整的例子:

uses SysUtils, Rtti;

type
  TEv = procedure(Sender: TObject) of object;

  TObj = class
  private
    FEv: TEv;
  public
    property Ev: TEv read FEv write FEv;
    class procedure Meth(Sender: TObject);
  end;

class procedure TObj.Meth(Sender: TObject);
begin
end;

procedure P;
var
  ctx: TRttiContext;
  t: TRttiType;
  p: TRttiProperty;
  v: TValue;
  o: TObj;
  e: TEv;
begin
  t := ctx.GetType(TObj);
  p := t.GetProperty('Ev');
  o := TObj.Create;
  try
    // Set value explicitly
    o.Ev := TObj.Meth;
    // Get value via RTTI
    v := p.GetValue(o);
    //e := v.AsType<TEv>; // doesn't work
    e := v.AsType<TEv>(); // works
  finally
    o.Free;
  end;
end;

begin
  try
    P;
  except
    on e: Exception do
      Writeln(e.Message);
  end;
end.

答案 1 :(得分:1)

2010年推出的新RTTI基本上只是旧版TypInfo RTTI的高级包装(目前)。在TypInfo中,事件处理程序由TMethod记录表示。试试这个(未经测试):

var 
  prop: TRttiProperty;
  val: TValue;
  evt: TNotifyEvent;
begin
  prop := FContext.GetType(MyControl.ClassInfo).GetProperty('OnChange');
  val := prop.GetValue(MyControl);
  TMethod(FOldOnChange) := val.AsType<TMethod>;
  evt := Self.MyOnChange;
  prop.SetValue(MyControl, TValue.From<TMethod>(TMethod(evt));
end;