IdComponent.pas 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  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); {$IFNDEF OVERLOADED_OPENARRAY_BUG}overload;{$ENDIF}
  95. procedure {$IFDEF OVERLOADED_OPENARRAY_BUG}DoStatusArr{$ELSE}DoStatus{$ENDIF}(
  96. AStatus: TIdStatus; const AArgs: array of const); {$IFNDEF OVERLOADED_OPENARRAY_BUG}overload;{$ENDIF}
  97. procedure InitComponent; override;
  98. {$IFNDEF USE_OBJECT_ARC}
  99. procedure Notification(AComponent: TComponent; Operation: TOperation); override;
  100. {$ENDIF}
  101. procedure SetWorkTarget(AValue: TIdComponent);
  102. //
  103. property OnWork: TWorkEvent read FOnWork write FOnWork;
  104. property OnWorkBegin: TWorkBeginEvent read FOnWorkBegin write FOnWorkBegin;
  105. property OnWorkEnd: TWorkEndEvent read FOnWorkEnd write FOnWorkEnd;
  106. public
  107. procedure BeginWork(AWorkMode: TWorkMode; const ASize: Int64 = 0); virtual;
  108. destructor Destroy; override;
  109. procedure DoWork(AWorkMode: TWorkMode; const ACount: Int64); virtual;
  110. procedure EndWork(AWorkMode: TWorkMode); virtual;
  111. //
  112. property WorkTarget: TIdComponent read FWorkTarget write SetWorkTarget;
  113. published
  114. property OnStatus: TIdStatusEvent read FOnStatus write FOnStatus;
  115. end;
  116. implementation
  117. { TIdComponent }
  118. destructor TIdComponent.Destroy;
  119. begin
  120. inherited Destroy;
  121. // After inherited - do at last possible moment
  122. TIdStack.DecUsage;
  123. end;
  124. procedure TIdComponent.DoStatus(AStatus: TIdStatus);
  125. begin
  126. {$IFDEF OVERLOADED_OPENARRAY_BUG}DoStatusArr{$ELSE}DoStatus{$ENDIF}(AStatus, []);
  127. end;
  128. procedure TIdComponent.{$IFDEF OVERLOADED_OPENARRAY_BUG}DoStatusArr{$ELSE}DoStatus{$ENDIF}(
  129. AStatus: TIdStatus; const AArgs: array of const);
  130. begin
  131. // We do it this way because Format() can sometimes cause an AV if the
  132. // variable array is blank and there is something like a %s or %d. This
  133. // is why there was sometimes an AV in TIdFTP
  134. if Assigned(OnStatus) then begin
  135. if Length(AArgs) = 0 then begin
  136. OnStatus(Self, AStatus, IndyFormat(IdStati[AStatus], [''])); {Do not Localize}
  137. end else begin
  138. OnStatus(Self, AStatus, IndyFormat(IdStati[AStatus], AArgs));
  139. end;
  140. end;
  141. end;
  142. procedure TIdComponent.BeginWork(AWorkMode: TWorkMode; const ASize: Int64 = 0);
  143. var
  144. // under ARC, convert a weak reference to a strong reference before working with it
  145. LWorkTarget: TIdComponent;
  146. begin
  147. LWorkTarget := FWorkTarget;
  148. if LWorkTarget <> nil then begin
  149. LWorkTarget.BeginWork(AWorkMode, ASize);
  150. end else begin
  151. Inc(FWorkInfos[AWorkMode].Level);
  152. if FWorkInfos[AWorkMode].Level = 1 then begin
  153. FWorkInfos[AWorkMode].Max := ASize;
  154. FWorkInfos[AWorkMode].Current := 0;
  155. if Assigned(OnWorkBegin) then begin
  156. OnWorkBegin(Self, AWorkMode, ASize);
  157. end;
  158. end;
  159. end;
  160. end;
  161. procedure TIdComponent.DoWork(AWorkMode: TWorkMode; const ACount: Int64);
  162. var
  163. // under ARC, convert a weak reference to a strong reference before working with it
  164. LWorkTarget: TIdComponent;
  165. begin
  166. LWorkTarget := FWorkTarget;
  167. if LWorkTarget <> nil then begin
  168. LWorkTarget.DoWork(AWorkMode, ACount);
  169. end else begin
  170. if FWorkInfos[AWorkMode].Level > 0 then begin
  171. Inc(FWorkInfos[AWorkMode].Current, ACount);
  172. if Assigned(OnWork) then begin
  173. OnWork(Self, AWorkMode, FWorkInfos[AWorkMode].Current);
  174. end;
  175. end;
  176. end;
  177. end;
  178. procedure TIdComponent.EndWork(AWorkMode: TWorkMode);
  179. var
  180. // under ARC, convert a weak reference to a strong reference before working with it
  181. LWorkTarget: TIdComponent;
  182. begin
  183. LWorkTarget := FWorkTarget;
  184. if LWorkTarget <> nil then begin
  185. LWorkTarget.EndWork(AWorkMode);
  186. end else begin
  187. if FWorkInfos[AWorkMode].Level = 1 then begin
  188. if Assigned(OnWorkEnd) then begin
  189. OnWorkEnd(Self, AWorkMode);
  190. end;
  191. end;
  192. Dec(FWorkInfos[AWorkMode].Level);
  193. end;
  194. end;
  195. procedure TIdComponent.InitComponent;
  196. begin
  197. inherited InitComponent;
  198. TIdStack.IncUsage;
  199. end;
  200. // under ARC, all weak references to a freed object get nil'ed automatically
  201. {$IFNDEF USE_OBJECT_ARC}
  202. procedure TIdComponent.Notification(AComponent: TComponent; Operation: TOperation);
  203. begin
  204. if (Operation = opRemove) and (AComponent = FWorkTarget) then begin
  205. FWorkTarget := nil;
  206. end;
  207. inherited Notification(AComponent, Operation);
  208. end;
  209. {$ENDIF}
  210. procedure TIdComponent.SetWorkTarget(AValue: TIdComponent);
  211. begin
  212. {$IFDEF USE_OBJECT_ARC}
  213. // under ARC, all weak references to a freed object get nil'ed automatically
  214. FWorkTarget := AValue;
  215. {$ELSE}
  216. if FWorkTarget <> AValue then begin
  217. if Assigned(FWorkTarget) then begin
  218. FWorkTarget.RemoveFreeNotification(Self);
  219. end;
  220. FWorkTarget := AValue;
  221. if Assigned(AValue) then begin
  222. AValue.FreeNotification(Self);
  223. end;
  224. end;
  225. {$ENDIF}
  226. end;
  227. end.