【问题标题】:How to extend existing interface IMessageFilter with TInterfacedObject?如何使用 TInterfacedObject 扩展现有接口 IMessageFilter?
【发布时间】:2018-04-05 10:50:10
【问题描述】:

我想按照这里的描述实现IOleMessageFilter

How to: Fix 'Application is Busy' and 'Call was Rejected By Callee' Errors

我找到了一个运行良好的 Delphi 实现:

`EOleException: Call was rejected by callee` while iterating through `Office.Interop.Word.Documents`

(请参阅答案中的更新#1)

实现如下所示:

type
  TOleMessageFilter = class(TInterfacedObject, IMessageFilter)
  public
    // IMessageFilter
    function HandleInComingCall(dwCallType: Longint; htaskCaller: HTask;
      dwTickCount: Longint; lpInterfaceInfo: PInterfaceInfo): Longint;stdcall;
    function RetryRejectedCall(htaskCallee: HTask; dwTickCount: Longint;
      dwRejectType: Longint): Longint;stdcall;
    function MessagePending(htaskCallee: HTask; dwTickCount: Longint;
      dwPendingType: Longint): Longint;stdcall;

    // TOleMessageFilter
    procedure RegisterFilter;
    procedure RevokeFilter;
  end;

implementation

function TOleMessageFilter.HandleInComingCall(dwCallType: Integer; htaskCaller: HTask; dwTickCount: Integer; lpInterfaceInfo: PInterfaceInfo): Longint;
begin
  Result := 0;
end;

function TOleMessageFilter.MessagePending(htaskCallee: HTask; dwTickCount, dwPendingType: Integer): Longint;
begin
  Result := 2 //PENDINGMSG_WAITDEFPROCESS
end;        

function TOleMessageFilter.RetryRejectedCall(htaskCallee: HTask; dwTickCount, dwRejectType: Integer): Longint;
begin
  Result := -1;
  if dwRejectType = 2 then
    Result := 99;
end;

procedure TOleMessageFilter.RegisterFilter;
var
  OldFilter: IMessageFilter;
  NewFilter: IMessageFilter;
begin
  OldFilter := nil;
  NewFilter := TOleMessageFilter.Create as IMessageFilter;
  CoRegisterMessageFilter(NewFilter,OldFilter);
end;

procedure TOleMessageFilter.RevokeFilter;
var
  OldFilter: IMessageFilter;
  NewFilter: IMessageFilter;
begin
  OldFilter := nil;
  NewFilter := nil;
  CoRegisterMessageFilter(NewFilter,OldFilter);
end;

这个确切的 Delphi 代码可以在网络上的许多其他网站上找到。到目前为止,一切都很好。我只将类名更改为TOleMessageFilter 而不是IOleMessageFilter

然而,这种用法有点烦人。

var
  Filter: TOleMessageFilter;

Filter := TOleMessageFilter.Create;
Filter.RegisterFilter;    
...    
Filter.RevokeFilter;
Filter.Free;

我想要的是,Filter 被声明为接口,例如IOleMessageFilter

var
  Filter: IOleMessageFilter;

Filter := TOleMessageFilter.Create as IOleMessageFilter;
Filter.RegisterFilter;
...
Filter.RevokeFilter;
Filter := nil;

并且有自动释放TInterfacedObject的好处。

如何创建一个新的IOleMessageFilter,它“派生”自IMessageFilter,但仍有新方法RegisterFilter()RevokeFilter(),实现为TOleMessageFilter = class(TInterfacedObject, IOleMessageFilter),并且仍然可以与@ 一起使用987654337@ 需要 IMessageFilter(在 RegisterFilter() 方法中使用)?

我已尝试声明:

IOleMessageFilter = interface(IMessageFilter)
  procedure RegisterFilter;
  procedure RevokeFilter;
end;

TOleMessageFilter = class(TInterfacedObject, IOleMessageFilter)
...
end; 

但随后调用CoRegisterMessageFilter 会引发错误:

不支持接口。

编辑:

我也尝试将TOleMessageFilter 声明为:

TOleMessageFilter = class(TInterfacedObject, IMessageFilter, IOleMessageFilter)

哪种“似乎”可行,但我不确定这是正确的方法。

【问题讨论】:

  • @whosrdaddy,你的意思是TOleMessageFilter = class(TInterfacedObject, IMessageFilter, IOleMessageFilter)?和IOleMessageFilter 包含RegisterFilterRevokeFilter 方法?
  • 不行,请耐心等待,我正在写一个答案;)

标签: delphi interface delphi-7


【解决方案1】:

拆分两个接口并让 TOleMessageFilter 保留对实际消息过滤器的引用,作为奖励,您不必再调用 RegisterFilter 和 RevokeFilter,因为这将通过构造函数/析构函数完成:

program SO46913922;

{$APPTYPE CONSOLE}

{$R *.res}

uses
  ActiveX,
  Windows,
  SysUtils;


type
  IOleMessageFilter = interface
  ['{0ECA5DA7-F6C7-4D21-8FD3-872558F88CBE}']
    procedure RegisterFilter;
    procedure RevokeFilter;
  end;

  TMessageFilter = class(TInterfacedObject, IMessageFilter)
  public
    // IMessageFilter
    function HandleInComingCall(dwCallType: Longint; htaskCaller: HTask;
      dwTickCount: Longint; lpInterfaceInfo: PInterfaceInfo): Longint;stdcall;
    function RetryRejectedCall(htaskCallee: HTask; dwTickCount: Longint;
      dwRejectType: Longint): Longint;stdcall;
    function MessagePending(htaskCallee: HTask; dwTickCount: Longint;
      dwPendingType: Longint): Longint;stdcall;
  end;

  TOleMessageFilter = class(TInterfacedObject, IOleMessageFilter)
  private
    Filter : IMessageFilter;
    procedure RegisterFilter;
    procedure RevokeFilter;
  public
    constructor Create;
    destructor Destroy; override;
  end;


function TMessageFilter.HandleInComingCall(dwCallType: Integer; htaskCaller: HTask; dwTickCount: Integer; lpInterfaceInfo: PInterfaceInfo): Longint;
begin
  Result := 0;
end;

function TMessageFilter.MessagePending(htaskCallee: HTask; dwTickCount, dwPendingType: Integer): Longint;
begin
  Result := 2 //PENDINGMSG_WAITDEFPROCESS
end;

function TMessageFilter.RetryRejectedCall(htaskCallee: HTask; dwTickCount, dwRejectType: Integer): Longint;
begin
  Result := -1;
  if dwRejectType = 2 then
    Result := 99;
end;

procedure TOleMessageFilter.RegisterFilter;
var
  OldFilter: IMessageFilter;

begin
  OldFilter := nil;
  Filter := TMessageFilter.Create;
  CoRegisterMessageFilter(Filter,OldFilter);
end;

procedure TOleMessageFilter.RevokeFilter;
var
  OldFilter: IMessageFilter;
  NewFilter: IMessageFilter;
begin
  OldFilter := nil;
  NewFilter := nil;
  CoRegisterMessageFilter(NewFilter,OldFilter);
  Filter := nil;
end;

constructor TOleMessageFilter.Create;
begin
 RegisterFilter;
end;

destructor TOleMessageFilter.Destroy;
begin
 RevokeFilter;
 inherited;
end;

var
  Filter :  IOleMessageFilter;

begin
  try
   CoInitialize(nil);
   Filter := TOleMessageFilter.Create;
   Readln; // do something
   Filter := nil;
  finally
   CoUninitialize();
  end;
  Readln;
end.

【讨论】:

  • 谢谢。我投了赞成票,因为它是一个优雅的解决方案,但它并没有完全回答我提出的问题。我想用另外 2 个方法扩展 IMessageFilter 接口。不要分裂他们。声明 TOleMessageFilter = class(TInterfacedObject, IMessageFilter, IOleMessageFilter) 也有效。但我不明白为什么TOleMessageFilter = class(TInterfacedObject, IOleMessageFilter)CoRegisterMessageFilter 一起失败,因为我声明了IOleMessageFilter = interface(IMessageFilter)
  • 如果没有人回答我提出的问题,我会接受你的,因为也许这根本无法做到。
  • 原因很简单,IOleMessagefilter != IMessagefilter 即使IOleMessageFilter继承自IMessageFilter,也不是同一个接口。您可以选择在 TOleMessageFilter 上实现这两个接口,或者使用我在帖子中描述的方法(更简洁)
猜你喜欢
  • 2018-08-22
  • 1970-01-01
  • 1970-01-01
  • 2016-07-19
  • 2012-09-01
  • 2021-11-07
  • 2022-01-14
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多