Hallo zusammen,
in einer Applikation hatte ich bislang nur TAdpComPort verwendet und in dem Zusammenhang händisch einen TimerTrigger gesetzt. Zusätzlich ließ ich noch auf die beiden Events OnTriggerAvail und OnTriggerTimer triggern.
Da ich jedoch mitunter Kommunikationsprobleme habe (auf die ich nicht näher eingehen möchte), wollte ich (zunächst zusätzlich/optional) TApdDataPacket verwenden.
Weil jedoch (wie soll ich sagen?) von mehreren Seiten aus eigentlich auf den TApdComPort.OnTriggerAvail getriggert wird, soll der Umstieg auf TApdDataPacket fliessend erfolgen. Mit anderen Worten: Ich möchte sowohl TApdComPort als auch TApdDataPacket in der Übergangsphase einsetzen.
Das erste, was auffiel, war, dass der händisch gesetzte TimerTrigger nicht mehr fällt, wenn TApdDataPacket.ComPort auf TApdComPort verweist. Offensichtlich wird also der händische Trigger vom TApdDataPacket.OnTimeout überlagert. Ein Setzen von TApdDataPacket.Enabled und/oder TApddataPacket. AutoEnable auf FALSE brachte keine Besserung.
So dachte ich mir, ich erzeuge ein TApdDataPacket dynamisch in einer eigenen Kommunikationsroutine und weise alles von Hand zu. Und damit nicht ggf. die TApdComPort Trigger fallen, melde ich sie temporär ab.
Delphi-Quellcode:
TCOMInterface = class(TDataModule)
procedure ApdDataPacketTimeout(Sender: TObject);
procedure ApdDataPacketPacket(Sender: TObject; Data: Pointer; Size: Integer);
procedure ApdDataPacketStringPacket(Sender: TObject; Data: string);
private
{ Private-Deklarationen }
TimeOut : boolean;
RcvDataOk: boolean;
RcvDatas : array[1..4192] of AnsiChar;
function GetDatas(ACommand: string;
APacketSize: integer = 255;
AEndCond: TPacketEndCond = ecPacketSize;
AEndString: string = #10#13;
ATimeOut: word = 90): boolean;
end;
function TCOMInterface.GetDatas(ACommand: string;
APacketSize: integer = 255;
AEndCond: TPacketEndCond = ecPacketSize;
AEndString: string = #10#13;
ATimeOut: word = 90): boolean;
var ApdDataPacket: TApdDataPacked;
begin
// *** Vorarbeiten ***
Result:= false;
TimeOut:= false;
RcvDataOk:= false;
FillChar(RcvDatas, SizeOf(RcvDatas), #0);
// ApdComPort Events abmelden
ApdComPort.OnTriggerAvail:= nil;
ApdCOMPort.OnTriggerTimer:= nil;
// ApdDataPacket erzeugen
ApdDataPacket:= TApdDataPacket.Create(ApdCOMPort);
ApdDataPacket.StartCond:= scAnyData;
ApdDataPacket.PacketSize:= APacketSize;
ApdDataPacket.EndString:= AEndString;
ApdDataPacket.EndCond:= [AEndCond];
ApdDataPacket.TimeOut:= ATimeOut;
ApdDataPacket.AutoEnable:= false;
// Events
ApdDataPacket.OnPacket:= ApdDataPacketPacket;
ApdDataPacket.OnStringPacket:= ApdDataPacketStringPacket;
ApdDataPacket.OnTimeout:= ApdDataPacketTimeout;
ApdDataPacket.Enabled:= true;
// *** Kommunikation ***
ApdCOMPort.PutString(ACommand);
repeat
Application.ProcessMessages;
ApdComPort.ProcessCommunications;
until (TimeOut) or (RcvDataOk);
if (RcvDataOk) then
Result:= true;
// *** Nacharbeiten ***
// ApdDataPacket vernichten
ApdDataPacket.Free;
// ApdComPort Events wieder anmelden
ApdComPort.OnTriggerAvail:= ApdComPortTriggerAvail;
ApdComPort.OnTriggerTimer:= ApdComPortTriggerTimer;
end;
procedure TCOMInterface.ApdDataPacketPacket(Sender: TObject; Data: Pointer;
Size: Integer);
begin
if (ecString in DataPacket.EndCond) then
Exit;
Move(Data^, RcvDatas, Size);
RcvDataOk:= true;
end;
procedure TCOMInterface.ApdDataPacketStringPacket(Sender: TObject; Data: string);
var x: word;
begin
if (ecPacketSize in DataPacket.EndCond) then
Exit;
for x:= 1 to Length(Data) do
RcvDatas[x]:= Data[x];
RcvDataOk:= true;
end;
procedure TCOMInterface.ApdDataPacketTimeout(Sender: TObject);
begin
TimeOut := true;
end;
Das Blöde an der Sache ist jedoch, dass keiner der ApdDataPacket Events jemals fällt.
Frage: Hat jemand eine Idee, warum dem so ist?
Dank vorab und Gruß, Carsten