Главная страница
    Top.Mail.Ru    Яндекс.Метрика
Форум: "Corba";
Текущий архив: 2007.10.28;
Скачать: [xml.tar.bz2];

Вниз

Создание обработчика событий   Найти похожие ветки 

 
kutnul   (2006-02-01 18:49) [0]

Доброго времени всем

Имеем библиотеку типов, в ней


*********************************************************************//
// DispIntf:   DSonicEvents
// Flags:     (4112) Hidden Dispatchable
// GUID:      {35D54C21-5894-F12A-CD54-00402513A157}
// *********************************************************************//
DSonicEvents = dispinterface
  ["{35D54C21-5894-F12A-CD54-00402513A157}"]
   function FileOpenNotify(const FileName: WideString): Integer; dispid 1;
.............
   function ActiveDocChangeNotify: Integer; dispid 4;
.............
 end;


Разъясните начинающему
1. что надо сделать для того чтобы перехватить события на сервере
2. как активировать на сервере

С уважением!


 
Набережных С. ©   (2006-02-02 08:41) [1]

Ссылаешься в uses на модуль OleAuto и объявляешь класс - наследник от объявленного там TAutoObject. В этом классе заводишь секцию automated, в которой объявляешь методы твоего диспинтерфейса с теми же dispid. реализуешь эти мтоды, разумеется. Дальше, куогда надо подключиться к событиям, создаешь этот объект примерно так

var
 Disp: IDispatch;
 Disp:= TMyDispObject.Create.AutoDispatch

Все, далее можно обычным образом передать этот интерфейс серверу в качестве приемника событий.

Модуль OleAuto объявлен устаревшим, однако мне его использование в ряде случаев представляется достаточно удобным.

Все то же самое можно проделать и унаследовавшись от TAutoIntfObject или ComObj.TAutoObject, но там еще потребуется загрузить библиотеку типов, что, впрочем, не является препятствием, когда эта библиотека имеется, как в данном случае.

Есть и другие способы, но там возни заметно больше.


 
kutnul   (2006-02-02 16:02) [2]

Большое спасибо за содействие
буду пробовать

Уважаемый Набережных С

не прими за наглость
но можно ли о других способах услышать (или где почитать)
на емайл или в рамках данного форуму

ещё раз спасибо


 
kutnul   (2006-02-02 19:23) [3]

Брррр
не идет процесс

Во-первых
var
Disp: IDispatch;
Disp:= TMyDispObject.Create.AutoDispatch
пишет что несовместимы типы TAutoDispatch и IDispatch
пробовал и так и сяк

var
DispEv : iDispatch;
AutoMobil : TAutoDispatch;
begin
DispEv := TMyEvent.Create.AutoDispatch;
MyEvent := TMyEvent.Create;
AutoMobil := TMyEvent.Create.CreateAutoDispatch;
Automobil := TAutoDispatch.Create(MyEvent);


Во-вторых, правильно ли я понял, что
>> Все, далее можно обычным образом передать этот интерфейс серверу в качестве приемника событий.
то для этого необходимо -  InterfaceConnect (MySonicSrv, ISonicSrv, MyEvent, FCookie);
опят же не идет

что делать как быть?
привожу исходник

uses
 Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
 Dialogs, ComObj, OleAuto, SonicSrv_TLB;

type
   TMyEvent = class ;
 TForm1 = class(TForm)
   procedure FormCreate(Sender: TObject);
 private
   { Private declarations }
   MySonicSrv : ISonicSrv;
   FCookie : Integer;
 public
   { Public declarations }
 end;
 TMyEvent = class (TAutoObject)
   public
   { Public declarations }
   automated
   function DestroyNotify: Integer; dispid 3;
   function ActiveDocChangeNotify: Integer; dispid 4;
 end;

var
 Form1: TForm1;
 MyEvent : TMyEvent;

implementation

{$R *.dfm}

function TMyEvent.DestroyNotify():Integer ;
begin
ShowMessage ("Удаление ");
end;

function TMyEvent.ActiveDocChangeNotify(): Integer ;
begin
ShowMessage ("Произошло изменение активного документа");
end;

procedure TForm1.FormCreate(Sender: TObject);
var
DispEv : iDispatch;
AutoMobil : TAutoDispatch;
begin
MyEvent := TMyEvent.Create;
AutoMobil := TMyEvent.Create.CreateAutoDispatch;
Automobil := TAutoDispatch.Create(MyEvent);

MySonicSrv := CoSonicSrv_.Create;
MySonicSrv.Visible := True;
InterfaceConnect (MySonicSrv, ISonicSrv, MyEvent, FCookie);

end;

end.


 
Набережных С. ©   (2006-02-03 08:57) [4]


> kutnul   (02.02.06 19:23) [3]
> Брррр
> не идет процесс

Еще бы:( Дело в том, что у меня модуль OleAuto основательно переделан, и не только он. А я об этом напрочь забыл:(( Еще хуже то, что я не помню что и где я изменил, давно я им не пользовался. А разбираться с этим сейчас мне не с руки, у меня была тяжелая ночь, почти не спал. Так что извини, виноват.

Я тут выкопал у себя вот этот класс:

unit DispatchEventsUnt;

interface

uses
 Windows, Messages, SysUtils, ComObj, ActiveX;

type
 TDispatchEvents = class(TObject, IUnknown, IDispatch)
 private
   FDispIID: TGUID;
   FConnectionID: integer;
   FSource: Pointer;
 protected
   { IUnknown }
   function QueryInterface(const IID: TGUID; out Obj): HResult; stdcall;
   function _AddRef: Integer; stdcall;
   function _Release: Integer; stdcall;

   { IDispatch }
   function GetIDsOfNames(const IID: TGUID; Names: Pointer;
     NameCount, LocaleID: Integer; DispIDs: Pointer): HResult; virtual; stdcall;
   function GetTypeInfo(Index, LocaleID: Integer; out TypeInfo): HResult; virtual; stdcall;
   function GetTypeInfoCount(out Count: Integer): HResult; virtual; stdcall;
   function Invoke(DispID: Integer; const IID: TGUID; LocaleID: Integer;
     Flags: Word; var Params; VarResult, ExcepInfo, ArgErr: Pointer): HResult; virtual; stdcall;

   function DoInvoke (DispID: Integer; LocaleID: Integer;
     var Params: TDispParams; VarResult, ExcepInfo,
     ArgErr: Pointer): HResult; virtual; abstract;
 public
   constructor Create(const DisIntfID: TGUID);
   destructor Destroy; override;

   procedure Connect(const Source: IUnknown);
   procedure Disconnect;
   
   property DispIID: TGUID read FDispIID;
 end;

implementation

{ TDispatchEvents }

function TDispatchEvents._AddRef: Integer;
begin
 Result:= 1;
end;

function TDispatchEvents._Release: Integer;
begin
 Result:= 1;
end;

function TDispatchEvents.GetIDsOfNames(const IID: TGUID; Names: Pointer;
 NameCount, LocaleID: Integer; DispIDs: Pointer): HResult;
begin
 Result:= E_NOTIMPL;
end;

function TDispatchEvents.GetTypeInfo(Index, LocaleID: Integer;
 out TypeInfo): HResult;
begin
 Result:= E_NOTIMPL;
end;

function TDispatchEvents.GetTypeInfoCount(out Count: Integer): HResult;
begin
 Result:= S_OK;
 Count:= 0;
end;

function TDispatchEvents.Invoke(DispID: Integer; const IID: TGUID;
 LocaleID: Integer; Flags: Word; var Params; VarResult, ExcepInfo,
 ArgErr: Pointer): HResult;
begin
 if Flags and DISPATCH_METHOD = 0 then
   raise EOleError.CreateFmt
     ("Class %s only dispatcing methods support.", [ClassName]);

 Result:= DoInvoke(DispID, LocaleID, TDispParams(Params),
   VarResult, ExcepInfo, ArgErr);
end;

function TDispatchEvents.QueryInterface(const IID: TGUID;
 out Obj): HResult;
begin
 if IsEqualGUID(IID, FDispIID) then
 begin
   Result:= S_OK;
   IDispatch(Obj):= Self;
 end else
 begin
   if GetInterface(IID, Obj) then Result:= S_OK
   else Result:= E_NOINTERFACE;
 end;
end;

constructor TDispatchEvents.Create(const DisIntfID: TGUID);
begin
 FDispIID:= DisIntfID;
end;

procedure TDispatchEvents.Connect(const Source: IUnknown);
begin
 Disconnect;
 if nil = Source then Exit;
 InterfaceConnect(Source, FDispIID, Self, FConnectionID);
 FSource:= Pointer(Source);
end;

destructor TDispatchEvents.Destroy;
begin
 Disconnect;
 inherited;
end;

procedure TDispatchEvents.Disconnect;
begin
 if nil <> FSource then
   InterfaceDisconnect(IUnknown(FSource), FDispIID, FConnectionID);
 FSource:= nil;
end;

end.


Это базовый класс, наследуясь от которого можно создавать реализаторы дисинтерфейсов. Создаешь наследника, в нем объявляешь и реализуешь методы твоего диспинтерфейса, также перекрываешь DoInvoke, в котором и вызываешь свои методы. Правда, диспетчирование нужно делать вручную, но это несложно. Например, для такого метода

procedure SomeMethod(const Value1: WideString; Value2: Integer);

c DispId = 333, реализация DoInvoke может быть такой:

function TMyEventSink.DoInvoke(DispID, LocaleID: Integer;
 var Params: TDispParams; VarResult, ExcepInfo, ArgErr: Pointer): HResult;
begin
 Result:= DISP_E_MEMBERNOTFOUND;
 case DispID of
   333:
   begin
     if Params.cArgs <> 2 then Exit;
     with Params do
       SomeMethod(rgvarg[1].bstrVal, rgvarg[0].iVal);
   end;
 end;
end;

Или можно, например, объявить в этом наследике событие и его генерировать. Обрати внимания, что аргументы в списке передаются в обратном порядке.

Вот это и есть один из других способов. Честно говоря, я не собирался выкладывать код, но раз уж ввел в заблуждение, то это как бы компенсация:)) Другой способ - сделать то же самое, но не в отдельном объете, а прямо внутри клиентского объекта, инкапсулирующего работу с сервером, если конечно таковой объект имеется.

Еще один - воспользоваться Import Type Library с генерацией компонента-обертки.


 
kutnul   (2006-02-06 18:28) [5]

что-то не так
а что кто его знает

кое-какие сдвиги есть но....
что они значат - для моего темного разума не вдамек

используем класс (без изменений)  
TMyEventsSink = class (TDispatchEvents)
MyEventSonic : TMyEventsSink;
создаем экземпляр          MyEventSonic := MyEventsSink.Create(MyDiID_SonicSrv);
но при подключении       MyEventSonic.Connect(SonicSrv);
1. получаем ошибку
2. а при живом тыкание в приложении-сервере Sonic видим сообщение от Windows

Сервер занят
Действие не может быть завершено, так как
программа "Project1" не отвечает на запросы.
Перейдите в нужное окно с помощью кнопки
"Переключиться" и устраните ошибку.


т.е. получается сообщение от сервера получаю а обработать не могу - наверное?

Есть сомнение те ли параметры передаю в конструкторы
1. в TMyEventsSink.Create(MyDiID_SonicSrv) передаём  
из билиотеки типов
DSonicEvents = dispinterface
 ["{35D54C21-5894-F12A-CD54-00402513A157}"]
DIID_DSonicEvents: TGUID = "{35D54C21-5894-F12A-CD54-00402513A157}";
в клиенте
MyDiID_SonicSrv : TGUID;
MyDiID_SonicSrv := DIID_DSonicEvents;

2. MyEventSonic.Connect (SonicSrv);
где SonicSrv := CoSonicSrv_.Create;

3.1 На всякий случай - ни каких действий в методе DoInvoke - не делал - может и здесь ошибка
function TMyEventsSink.DoInvoke(DispID: Integer; LocaleID: Integer;
    var Params: TDispParams; VarResult, ExcepInfo,
    ArgErr: Pointer): HResult;
begin
  Result:= DISP_E_MEMBERNOTFOUND;
end;


3.2 да и в
function TMyEventsSink.Invoke то же - ничего
Result := S_ok; end;

С уважением!



Страницы: 1 вся ветка

Форум: "Corba";
Текущий архив: 2007.10.28;
Скачать: [xml.tar.bz2];

Наверх




Память: 0.49 MB
Время: 0.048 c
2-1191480014
KLAUS
2007-10-04 10:40
2007.10.28
Поиск по Реестру


11-1164116139
Psychedelic
2006-11-21 16:35
2007.10.28
Прозрачный Tab


2-1191413729
DimOk
2007-10-03 16:15
2007.10.28
CreateOleObject + 1Cv77+Vista


6-1172807693
ACSBaks
2007-03-02 06:54
2007.10.28
Как получить IP адрес на компоненте TIdHTTPServer


11-1175009897
ElectriC
2007-03-27 19:38
2007.10.28
a-la Application.MessageBox()





Afrikaans Albanian Arabic Armenian Azerbaijani Basque Belarusian Bulgarian Catalan Chinese (Simplified) Chinese (Traditional) Croatian Czech Danish Dutch English Estonian Filipino Finnish French
Galician Georgian German Greek Haitian Creole Hebrew Hindi Hungarian Icelandic Indonesian Irish Italian Japanese Korean Latvian Lithuanian Macedonian Malay Maltese Norwegian
Persian Polish Portuguese Romanian Russian Serbian Slovak Slovenian Spanish Swahili Swedish Thai Turkish Ukrainian Urdu Vietnamese Welsh Yiddish Bengali Bosnian
Cebuano Esperanto Gujarati Hausa Hmong Igbo Javanese Kannada Khmer Lao Latin Maori Marathi Mongolian Nepali Punjabi Somali Tamil Telugu Yoruba
Zulu
Английский Французский Немецкий Итальянский Португальский Русский Испанский