IdComponent.pas 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261
  1. {
  2. $Project$
  3. $Workfile$
  4. $Revision$
  5. $DateUTC$
  6. $Id$
  7. This file is part of the Indy (Internet Direct) project, and is offered
  8. under the dual-licensing agreement described on the Indy website.
  9. (http://www.indyproject.org/)
  10. Copyright:
  11. (c) 1993-2005, Chad Z. Hower and the Indy Pit Crew. All rights reserved.
  12. }
  13. {
  14. $Log$
  15. }
  16. {
  17. Rev 1.4 1/17/2005 7:26:12 PM JPMugaas
  18. Moved stack management code to IdStack.
  19. Rev 1.3 2004.06.06 5:18:14 PM czhower
  20. OnWork bug fix
  21. Rev 1.2 2004.06.05 9:46:38 AM czhower
  22. IOHandler OnWork fix
  23. Rev 1.1 2004.02.03 3:15:52 PM czhower
  24. Updates to move to System.
  25. Rev 1.0 2004.02.03 2:28:28 PM czhower
  26. Move
  27. Rev 1.7 2004.01.22 5:59:10 PM czhower
  28. IdCriticalSection
  29. Rev 1.6 2004.01.20 10:03:24 PM czhower
  30. InitComponent
  31. Rev 1.5 2003.10.14 1:26:42 PM czhower
  32. Uupdates + Intercept support
  33. Rev 1.4 2003.10.01 9:11:16 PM czhower
  34. .Net
  35. Rev 1.3 2003.10.01 11:16:30 AM czhower
  36. .Net
  37. Rev 1.2 2003.09.30 1:22:54 PM czhower
  38. Stack split for DotNet
  39. Rev 1.1 2003.09.18 5:17:58 PM czhower
  40. Implemented OnWork
  41. Rev 1.0 11/13/2002 08:41:12 AM JPMugaas
  42. }
  43. unit IdComponent;
  44. interface
  45. {$i IdCompilerDefines.inc}
  46. uses
  47. {$IFNDEF USE_OBJECT_ARC}
  48. Classes,
  49. {$ENDIF}
  50. IdBaseComponent, IdGlobal, IdResourceStrings,
  51. IdStack;
  52. type
  53. TIdStatus = ( hsResolving,
  54. hsConnecting,
  55. hsConnected,
  56. hsDisconnecting,
  57. hsDisconnected,
  58. hsStatusText,
  59. ftpTransfer, // These are to eliminate the TIdFTPStatus and the
  60. ftpReady, // coresponding event
  61. ftpAborted); // These can be use din the other protocols to.
  62. const
  63. IdStati: array[TIdStatus] of string = (
  64. RSStatusResolving,
  65. RSStatusConnecting,
  66. RSStatusConnected,
  67. RSStatusDisconnecting,
  68. RSStatusDisconnected,
  69. RSStatusText,
  70. RSStatusText,
  71. RSStatusText,
  72. RSStatusText);
  73. type
  74. TIdStatusEvent = procedure(ASender: TObject; const AStatus: TIdStatus;
  75. const AStatusText: string) of object;
  76. TWorkMode = (wmRead, wmWrite);
  77. TWorkInfo = record
  78. Current: Int64;
  79. Max: Int64;
  80. Level: Integer;
  81. end;
  82. TWorkBeginEvent = procedure(ASender: TObject; AWorkMode: TWorkMode; AWorkCountMax: Int64) of object;
  83. TWorkEndEvent = procedure(ASender: TObject; AWorkMode: TWorkMode) of object;
  84. TWorkEvent = procedure(ASender: TObject; AWorkMode: TWorkMode; AWorkCount: Int64) of object;
  85. TIdComponent = class(TIdBaseComponent)
  86. protected
  87. FOnStatus: TIdStatusEvent;
  88. FOnWork: TWorkEvent;
  89. FOnWorkBegin: TWorkBeginEvent;
  90. FOnWorkEnd: TWorkEndEvent;
  91. FWorkInfos: array[TWorkMode] of TWorkInfo;
  92. {$IFDEF USE_OBJECT_ARC}[Weak]{$ENDIF} FWorkTarget: TIdComponent;
  93. //
  94. procedure DoStatus(AStatus: TIdStatus); overload;
  95. procedure DoStatus(AStatus: TIdStatus; const AArgs: array of const); overload;
  96. procedure InitComponent; override;
  97. {$IFNDEF USE_OBJECT_ARC}
  98. procedure Notification(AComponent: TComponent; Operation: TOperation); override;
  99. {$ENDIF}
  100. procedure SetWorkTarget(AValue: TIdComponent);
  101. //
  102. property OnWork: TWorkEvent read FOnWork write FOnWork;
  103. property OnWorkBegin: TWorkBeginEvent read FOnWorkBegin write FOnWorkBegin;
  104. property OnWorkEnd: TWorkEndEvent read FOnWorkEnd write FOnWorkEnd;
  105. public
  106. procedure BeginWork(AWorkMode: TWorkMode; const ASize: Int64 = 0); virtual;
  107. destructor Destroy; override;
  108. procedure DoWork(AWorkMode: TWorkMode; const ACount: Int64); virtual;
  109. procedure EndWork(AWorkMode: TWorkMode); virtual;
  110. //
  111. property WorkTarget: TIdComponent read FWorkTarget write SetWorkTarget;
  112. published
  113. property OnStatus: TIdStatusEvent read FOnStatus write FOnStatus;
  114. end;
  115. implementation
  116. { TIdComponent }
  117. destructor TIdComponent.Destroy;
  118. begin
  119. inherited Destroy;
  120. // After inherited - do at last possible moment
  121. TIdStack.DecUsage;
  122. end;
  123. procedure TIdComponent.DoStatus(AStatus: TIdStatus);
  124. begin
  125. DoStatus(AStatus, []);
  126. end;
  127. procedure TIdComponent.DoStatus(AStatus: TIdStatus; const AArgs: array of const);
  128. begin
  129. // We do it this way because Format() can sometimes cause an AV if the
  130. // variable array is blank and there is something like a %s or %d. This
  131. // is why there was sometimes an AV in TIdFTP
  132. if Assigned(OnStatus) then begin
  133. if Length(AArgs) = 0 then begin
  134. OnStatus(Self, AStatus, IndyFormat(IdStati[AStatus], [''])); {Do not Localize}
  135. end else begin
  136. OnStatus(Self, AStatus, IndyFormat(IdStati[AStatus], AArgs));
  137. end;
  138. end;
  139. end;
  140. procedure TIdComponent.BeginWork(AWorkMode: TWorkMode; const ASize: Int64 = 0);
  141. var
  142. // under ARC, convert a weak reference to a strong reference before working with it
  143. LWorkTarget: TIdComponent;
  144. begin
  145. LWorkTarget := FWorkTarget;
  146. if LWorkTarget <> nil then begin
  147. LWorkTarget.BeginWork(AWorkMode, ASize);
  148. end else begin
  149. Inc(FWorkInfos[AWorkMode].Level);
  150. if FWorkInfos[AWorkMode].Level = 1 then begin
  151. FWorkInfos[AWorkMode].Max := ASize;
  152. FWorkInfos[AWorkMode].Current := 0;
  153. if Assigned(OnWorkBegin) then begin
  154. OnWorkBegin(Self, AWorkMode, ASize);
  155. end;
  156. end;
  157. end;
  158. end;
  159. procedure TIdComponent.DoWork(AWorkMode: TWorkMode; const ACount: Int64);
  160. var
  161. // under ARC, convert a weak reference to a strong reference before working with it
  162. LWorkTarget: TIdComponent;
  163. begin
  164. LWorkTarget := FWorkTarget;
  165. if LWorkTarget <> nil then begin
  166. LWorkTarget.DoWork(AWorkMode, ACount);
  167. end else begin
  168. if FWorkInfos[AWorkMode].Level > 0 then begin
  169. Inc(FWorkInfos[AWorkMode].Current, ACount);
  170. if Assigned(OnWork) then begin
  171. OnWork(Self, AWorkMode, FWorkInfos[AWorkMode].Current);
  172. end;
  173. end;
  174. end;
  175. end;
  176. procedure TIdComponent.EndWork(AWorkMode: TWorkMode);
  177. var
  178. // under ARC, convert a weak reference to a strong reference before working with it
  179. LWorkTarget: TIdComponent;
  180. begin
  181. LWorkTarget := FWorkTarget;
  182. if LWorkTarget <> nil then begin
  183. LWorkTarget.EndWork(AWorkMode);
  184. end else begin
  185. if FWorkInfos[AWorkMode].Level = 1 then begin
  186. if Assigned(OnWorkEnd) then begin
  187. OnWorkEnd(Self, AWorkMode);
  188. end;
  189. end;
  190. Dec(FWorkInfos[AWorkMode].Level);
  191. end;
  192. end;
  193. procedure TIdComponent.InitComponent;
  194. begin
  195. inherited InitComponent;
  196. TIdStack.IncUsage;
  197. end;
  198. // under ARC, all weak references to a freed object get nil'ed automatically
  199. {$IFNDEF USE_OBJECT_ARC}
  200. procedure TIdComponent.Notification(AComponent: TComponent; Operation: TOperation);
  201. begin
  202. if (Operation = opRemove) and (AComponent = FWorkTarget) then begin
  203. FWorkTarget := nil;
  204. end;
  205. inherited Notification(AComponent, Operation);
  206. end;
  207. {$ENDIF}
  208. procedure TIdComponent.SetWorkTarget(AValue: TIdComponent);
  209. begin
  210. {$IFDEF USE_OBJECT_ARC}
  211. // under ARC, all weak references to a freed object get nil'ed automatically
  212. FWorkTarget := AValue;
  213. {$ELSE}
  214. if FWorkTarget <> AValue then begin
  215. if Assigned(FWorkTarget) then begin
  216. FWorkTarget.RemoveFreeNotification(Self);
  217. end;
  218. FWorkTarget := AValue;
  219. if Assigned(AValue) then begin
  220. AValue.FreeNotification(Self);
  221. end;
  222. end;
  223. {$ENDIF}
  224. end;
  225. end.