7

德尔福部分:

我有一个带有该事件的类,并且从该事件中我需要调用一个将接口对象传递给它的过程。它在 Delphi 中运行良好,但我在 Pascal 脚本中声明它时遇到问题。

到后台 -IGPGraphics接口是其中的一部分,Delphi GDI+ library并且没有方法定义如下:

type
  IGdiplusBase = interface
  ['{24A5D3F5-4A9B-42A2-9F60-20825E2740F5}']
  IGPGraphics = interface(IGdiPlusBase)
  ['{57F85BA4-CB01-4466-8441-948D03588F54}']

以下只是我需要在 Pascal 脚本中执行的简化 Delphi 伪代码:

type
  TRenderEvent = procedure(Sender: TObject; const GPGraphics: IGPGraphics) of object;
  TRenderClass = class(TGraphicControl)
  private
    FOnRender: TRenderEvent;
  public
    property OnRender: TRenderEvent read FOnRender write FOnRender;
  end;

// when the TRenderClass object instance fires its OnRender event I want to call 
// the RenderObject procedure passing the IGPGraphics interfaced object to it; I
// hope I'm doing it right, I'm just a newbie to this stuff - but it works so far
// in Delphi (since I didn't get it to work in Pascal Script)

procedure TForm1.RenderClass1Render(Sender: TObject; const GPGraphics: IGPGraphics);
begin
  RenderObject(GPGraphics, 10, 10);
end;

// what I need in Pascal Script is between these two lines; just pass the interface
// object from the event fired by component to the procedure called from inside it

procedure RenderObject(const GPGraphics: IGPGraphics; X, Y);
begin
  // and here to work with the interfaced object somehow
end;

Pascal 脚本编译部分:

我的目标是让带有事件的类在 Pascal 脚本中可用,并且需要将该接口对象传递给该过程,就像上面一样,所以我首先尝试在编译时声明这个(但我什至不确定是否这是这样做的正确方法):

// the interface
PS.AddInterface(Cl.FindInterface('IUnknown'), StringToGuid('{57F85BA4-CB01-4466-8441-948D03588F54}'), 'IGPGraphics');
// the type for the event
PS.AddTypeS('TRenderEvent', 'procedure(Sender: TObject; const GPGraphics: IGPGraphics)');
// and the class with the event itself
with PS.AddClassN(PS.FindClass('TGraphicControl'), 'TRenderClass') do
begin
  RegisterProperty('OnRender', 'TRenderEvent', iptrw);
end;

Pascal 脚本运行时部分:

我绝对迷失的地方是运行时部分。我不知道如何从调用堆栈中获取接口对象并将其传递给我的 RenderObject 过程:

function RenderClassProc(Caller: TPSExec; Proc: TPSExternalProcRec; Global, 
  Stack: TPSStack): Boolean;
var
  PStart: Cardinal;
begin
  PStart := Stack.Count-1;
  Result := True;
  if Proc.Name = 'RENDEROBJECT' then
  begin
    // how do I get the interfaced object from Stack (or whatever else) and pass 
    // it to the RenderObject proc here ? I can't find anything related about it
    // except functions that has no parameter index
    RenderObject(Stack.Get ?, Stack.GetInt(PStart-2), Stack.GetInt(PStart-3));
  end;
end;

问题是:

谁能建议我如何为这种情况正确定义编译和运行时部分,或者通过某种方式传递接口对象来纠正我?

PS 对那个 Inno-Setup 标签感到抱歉,但也许那里有人试图以这种方式自定义 InnoSetup。

非常感谢!

4

2 回答 2

1

如果我理解您的要求,您希望将接口作为参数传递给方法。不幸的是,我对此没有确切的答案,但我确实知道如何将接口分配给 PascalScript 的全局变量。这是我在 Castalia 的做法:

在 PSScript OnCompile 事件中,使用 PS.Comp.AddInterface 添加接口,然后添加每个必要的方法。之后,添加接口类型的变量。它看起来像这样,例如:

with PS.Comp.AddInterface(ARunner.Comp.FindInterface('IUnknown'),
  StringToGUID('{0346F7DF-CA7B-4B15-AEC9-2BDD680EE7AD}'),
  'ICastaliaMacroClipboardAccess') do
begin
  RegisterMethod('function GetText: string', cdRegister);
  RegisterMethod('procedure SetText(AText: string)', cdRegister);
end;
PS.AddRegisteredVariable('Clipboard', 'ICastaliaMacroClipboardAccess');

然后,在 OnExectute 事件中,将之前创建的变量绑定到实例:

P := PS.GetVariable('Clipboard'); //P is type PIFVariant
SetVariantToInterface(P, Implementing object as ICastaliaMacroClipboardAccess);    

完成此操作后,脚本可以通过变量访问接口对象,因此在这种情况下,脚本可以包含对 Clipboard.GetText 的调用,并且它可以按预期工作。

这是经过测试和工作的。

现在,我推测您可能能够使用 TPSScript.ExecuteFunction,从上面传入 PIFVariant,以更接近您想要的。然而,这不是我测试过的东西。

祝你好运!

于 2012-07-16T17:07:32.603 回答
1

很难相信但我发现了怎么做

procedure TApp.CallProcedureWithClassArg(x: TPSExec);
var
  o: TSampleObject;
  argumentList: TPSList;
  arg1: TPSVariantClass;
  proc: Integer;
begin
  o := TSampleObject.Create;
  o.X := 1; // do something with the object maybe
  proc := x.GetProc('TakeThis');
  argumentList := TPSList.Create;
  arg1.VI.FType := x.FindType2(btClass);
  arg1.Data := o;
  argumentList.Add(@arg1);
  x.RunProc(argumentList, proc);
  argumentList.Free;
end;

这基本上是应该做的。

  • 使用您的 TPSExec 实例,比如说 x
  • 然后使用 x.GetProc 方法获取过程号
  • 然后创建 TPSList 类型的参数列表
  • 创建 TPSVariantClass var,将您的类实例(应该传递)分配给它的 Data 字段
  • 还将 x.FindType2(btClass) 分配给它的 VI.FType 字段(完全不知道为什么会这样)
  • 将指向 TPSVariantClass 变量的指针添加到 TPSList 列表
  • 并且....... 调用过程 x.RunProc(argList, proc); 其中 proc 是前面获得的过程的编号

这适用于类,但对于接口应该没有太大区别,只需使用 TPSVariantInterface 类型作为参数变量而不是 TPSVariantClass;其他一切都应该是一样的。

我希望这可能会对某人有所帮助。

于 2013-11-08T17:15:08.703 回答