| 1 | unit UJobProgressView;
|
|---|
| 2 |
|
|---|
| 3 | {$MODE Delphi}
|
|---|
| 4 |
|
|---|
| 5 | interface
|
|---|
| 6 |
|
|---|
| 7 | uses
|
|---|
| 8 | SysUtils, Variants, Classes, Graphics, Controls, Forms, Syncobjs,
|
|---|
| 9 | Dialogs, ComCtrls, StdCtrls, ExtCtrls, Contnrs, UThreading, Math,
|
|---|
| 10 | DateUtils;
|
|---|
| 11 |
|
|---|
| 12 | const
|
|---|
| 13 | EstimatedTimeShowTreshold = 4;
|
|---|
| 14 | EstimatedTimeShowTresholdTotal = 1;
|
|---|
| 15 | UpdateInterval = 100; // ms
|
|---|
| 16 |
|
|---|
| 17 | type
|
|---|
| 18 |
|
|---|
| 19 | { TProgress }
|
|---|
| 20 |
|
|---|
| 21 | TProgress = class
|
|---|
| 22 | private
|
|---|
| 23 | FLock: TCriticalSection;
|
|---|
| 24 | FOnChange: TNotifyEvent;
|
|---|
| 25 | FText: string;
|
|---|
| 26 | FValue: Integer;
|
|---|
| 27 | FMax: Integer;
|
|---|
| 28 | procedure SetMax(const AValue: Integer);
|
|---|
| 29 | procedure SetText(AValue: string);
|
|---|
| 30 | procedure SetValue(const AValue: Integer);
|
|---|
| 31 | public
|
|---|
| 32 | procedure Increment;
|
|---|
| 33 | procedure Reset;
|
|---|
| 34 | constructor Create;
|
|---|
| 35 | destructor Destroy; override;
|
|---|
| 36 | property Value: Integer read FValue write SetValue;
|
|---|
| 37 | property Max: Integer read FMax write SetMax;
|
|---|
| 38 | property Text: string read FText write SetText;
|
|---|
| 39 | property OnChange: TNotifyEvent read FOnChange write FOnChange;
|
|---|
| 40 | end;
|
|---|
| 41 |
|
|---|
| 42 | TFormJobProgressView = class;
|
|---|
| 43 | TJobProgressView = class;
|
|---|
| 44 | TJobThread = class;
|
|---|
| 45 | TJob = class;
|
|---|
| 46 |
|
|---|
| 47 | TJobProgressViewMethod = procedure(Job: TJob) of object;
|
|---|
| 48 |
|
|---|
| 49 | { TJob }
|
|---|
| 50 |
|
|---|
| 51 | TJob = class
|
|---|
| 52 | private
|
|---|
| 53 | FTerminate: Boolean;
|
|---|
| 54 | procedure SetTerminate(const AValue: Boolean);
|
|---|
| 55 | public
|
|---|
| 56 | StartTime: TDateTime;
|
|---|
| 57 | EndTime: TDateTime;
|
|---|
| 58 | ProgressView: TJobProgressView;
|
|---|
| 59 | Title: string;
|
|---|
| 60 | Method: TJobProgressViewMethod;
|
|---|
| 61 | NoThreaded: Boolean;
|
|---|
| 62 | WaitFor: Boolean;
|
|---|
| 63 | Progress: TProgress;
|
|---|
| 64 | Thread: TJobThread;
|
|---|
| 65 | ResultString: string;
|
|---|
| 66 | Finished: Boolean;
|
|---|
| 67 | procedure AddLogItem(Value: string);
|
|---|
| 68 | constructor Create;
|
|---|
| 69 | destructor Destroy; override;
|
|---|
| 70 | property Terminate: Boolean read FTerminate write SetTerminate;
|
|---|
| 71 | end;
|
|---|
| 72 |
|
|---|
| 73 | TJobs = class(TObjectList)
|
|---|
| 74 | end;
|
|---|
| 75 |
|
|---|
| 76 | TJobThread = class(TListedThread)
|
|---|
| 77 | procedure Execute; override;
|
|---|
| 78 | public
|
|---|
| 79 | ProgressView: TJobProgressView;
|
|---|
| 80 | Job: TJob;
|
|---|
| 81 | end;
|
|---|
| 82 |
|
|---|
| 83 | { TFormJobProgressView }
|
|---|
| 84 |
|
|---|
| 85 | TFormJobProgressView = class(TForm)
|
|---|
| 86 | ImageList1: TImageList;
|
|---|
| 87 | LabelText: TLabel;
|
|---|
| 88 | Label2: TLabel;
|
|---|
| 89 | LabelOperation: TLabel;
|
|---|
| 90 | LabelEstimatedTimePart: TLabel;
|
|---|
| 91 | LabelEstimatedTimeTotal: TLabel;
|
|---|
| 92 | ListViewJobs: TListView;
|
|---|
| 93 | MemoLog: TMemo;
|
|---|
| 94 | PanelText: TPanel;
|
|---|
| 95 | PanelProgressTotal: TPanel;
|
|---|
| 96 | PanelOperationsTitle: TPanel;
|
|---|
| 97 | PanelLog: TPanel;
|
|---|
| 98 | PanelOperations: TPanel;
|
|---|
| 99 | PanelProgress: TPanel;
|
|---|
| 100 | ProgressBarPart: TProgressBar;
|
|---|
| 101 | ProgressBarTotal: TProgressBar;
|
|---|
| 102 | TimerUpdate: TTimer;
|
|---|
| 103 | procedure FormHide(Sender: TObject);
|
|---|
| 104 | procedure FormShow(Sender: TObject);
|
|---|
| 105 | procedure ReloadJobList;
|
|---|
| 106 | procedure FormClose(Sender: TObject; var CloseAction: TCloseAction);
|
|---|
| 107 | procedure FormDestroy(Sender: TObject);
|
|---|
| 108 | procedure ListViewJobsData(Sender: TObject; Item: TListItem);
|
|---|
| 109 | procedure TimerUpdateTimer(Sender: TObject);
|
|---|
| 110 | procedure FormCreate(Sender: TObject);
|
|---|
| 111 | procedure FormCloseQuery(Sender: TObject; var CanClose: Boolean);
|
|---|
| 112 | procedure UpdateHeight;
|
|---|
| 113 | public
|
|---|
| 114 | JobProgressView: TJobProgressView;
|
|---|
| 115 | end;
|
|---|
| 116 |
|
|---|
| 117 | { TJobProgressView }
|
|---|
| 118 |
|
|---|
| 119 | TJobProgressView = class(TComponent)
|
|---|
| 120 | private
|
|---|
| 121 | FAutoClose: Boolean;
|
|---|
| 122 | Finished: Boolean;
|
|---|
| 123 | FOnJobFinish: TJobProgressViewMethod;
|
|---|
| 124 | FOnOwnerDraw: TNotifyEvent;
|
|---|
| 125 | FOwnerDraw: Boolean;
|
|---|
| 126 | FShowDelay: Integer;
|
|---|
| 127 | FTerminate: Boolean;
|
|---|
| 128 | FormList: TList;
|
|---|
| 129 | TotalStartTime: TDateTime;
|
|---|
| 130 | Log: TStringList;
|
|---|
| 131 | FForm: TFormJobProgressView;
|
|---|
| 132 | procedure SetTerminate(const AValue: Boolean);
|
|---|
| 133 | procedure UpdateProgress;
|
|---|
| 134 | procedure JobProgressChange(Sender: TObject);
|
|---|
| 135 | public
|
|---|
| 136 | Jobs: TJobs;
|
|---|
| 137 | CurrentJob: TJob;
|
|---|
| 138 | CurrentJobIndex: Integer;
|
|---|
| 139 | constructor Create(TheOwner: TComponent); override;
|
|---|
| 140 | destructor Destroy; override;
|
|---|
| 141 | procedure Clear;
|
|---|
| 142 | function AddJob(Title: string; Method: TJobProgressViewMethod;
|
|---|
| 143 | NoThreaded: Boolean = False; WaitFor: Boolean = False): TJob;
|
|---|
| 144 | procedure Start;
|
|---|
| 145 | procedure Stop;
|
|---|
| 146 | procedure TermSleep(Delay: Integer);
|
|---|
| 147 | property Form: TFormJobProgressView read FForm;
|
|---|
| 148 | property Terminate: Boolean read FTerminate write SetTerminate;
|
|---|
| 149 | published
|
|---|
| 150 | property OwnerDraw: Boolean read FOwnerDraw write FOwnerDraw;
|
|---|
| 151 | property ShowDelay: Integer read FShowDelay write FShowDelay;
|
|---|
| 152 | property AutoClose: Boolean read FAutoClose write FAutoClose;
|
|---|
| 153 | property OnJobFinish: TJobProgressViewMethod read FOnJobFinish
|
|---|
| 154 | write FOnJobFinish;
|
|---|
| 155 | property OnOwnerDraw: TNotifyEvent read FOnOwnerDraw
|
|---|
| 156 | write FOnOwnerDraw;
|
|---|
| 157 | end;
|
|---|
| 158 |
|
|---|
| 159 | //var
|
|---|
| 160 | // FormJobProgressView: TFormJobProgressView;
|
|---|
| 161 |
|
|---|
| 162 | procedure Register;
|
|---|
| 163 |
|
|---|
| 164 | resourcestring
|
|---|
| 165 | SExecuted = 'Executed';
|
|---|
| 166 |
|
|---|
| 167 | implementation
|
|---|
| 168 |
|
|---|
| 169 | {$R *.lfm}
|
|---|
| 170 |
|
|---|
| 171 | resourcestring
|
|---|
| 172 | SPleaseWait = 'Please wait...';
|
|---|
| 173 | STerminate = 'Termination';
|
|---|
| 174 | SEstimatedTime = 'Estimated time: %s';
|
|---|
| 175 | STotalEstimatedTime = 'Total estimated time: %s';
|
|---|
| 176 | SFinished = 'Finished';
|
|---|
| 177 |
|
|---|
| 178 | procedure Register;
|
|---|
| 179 | begin
|
|---|
| 180 | RegisterComponents('Common', [TJobProgressView]);
|
|---|
| 181 | end;
|
|---|
| 182 |
|
|---|
| 183 | { TJobThread }
|
|---|
| 184 |
|
|---|
| 185 | procedure TJobThread.Execute;
|
|---|
| 186 | begin
|
|---|
| 187 | try
|
|---|
| 188 | try
|
|---|
| 189 | //raise Exception.Create('Exception in job');
|
|---|
| 190 | ProgressView.CurrentJob.Method(Job);
|
|---|
| 191 | except
|
|---|
| 192 | on E: Exception do begin
|
|---|
| 193 | ProgressView.Terminate := True;
|
|---|
| 194 | raise;
|
|---|
| 195 | end;
|
|---|
| 196 | end;
|
|---|
| 197 | finally
|
|---|
| 198 | Terminate;
|
|---|
| 199 | end;
|
|---|
| 200 | end;
|
|---|
| 201 |
|
|---|
| 202 | { TFormJobProgressView }
|
|---|
| 203 |
|
|---|
| 204 | procedure TFormJobProgressView.UpdateHeight;
|
|---|
| 205 | var
|
|---|
| 206 | H: Integer;
|
|---|
| 207 | PanelOperationsVisible: Boolean;
|
|---|
| 208 | PanelOperationsHeight: Integer;
|
|---|
| 209 | PanelProgressVisible: Boolean;
|
|---|
| 210 | PanelProgressTotalVisible: Boolean;
|
|---|
| 211 | PanelLogVisible: Boolean;
|
|---|
| 212 | MemoLogHeight: Integer = 200;
|
|---|
| 213 | I: Integer;
|
|---|
| 214 | ItemRect: TRect;
|
|---|
| 215 | MaxH: Integer;
|
|---|
| 216 | begin
|
|---|
| 217 | H := PanelOperationsTitle.Height;
|
|---|
| 218 | PanelOperationsVisible := JobProgressView.Jobs.Count > 0;
|
|---|
| 219 | if PanelOperationsVisible <> PanelOperations.Visible then
|
|---|
| 220 | PanelOperations.Visible := PanelOperationsVisible;
|
|---|
| 221 | if ListViewJobs.Items.Count > 0 then begin
|
|---|
| 222 | Maxh := 0;
|
|---|
| 223 | for I := 0 to ListViewJobs.Items.Count - 1 do
|
|---|
| 224 | begin
|
|---|
| 225 | ItemRect := ListViewJobs.Items[i].DisplayRect(drBounds);
|
|---|
| 226 | Maxh := Max(Maxh, ItemRect.Top + (ItemRect.Bottom - ItemRect.Top));
|
|---|
| 227 | end;
|
|---|
| 228 | PanelOperationsHeight := Scale96ToScreen(12) + Maxh;
|
|---|
| 229 | end else PanelOperationsHeight := Scale96ToScreen(8);
|
|---|
| 230 | if PanelOperationsHeight <> PanelOperations.Height then
|
|---|
| 231 | PanelOperations.Height := PanelOperationsHeight;
|
|---|
| 232 | if PanelOperationsVisible then
|
|---|
| 233 | H := H + PanelOperations.Height;
|
|---|
| 234 |
|
|---|
| 235 | PanelProgressVisible := (JobProgressView.Jobs.Count > 0) and not JobProgressView.Finished;
|
|---|
| 236 | if PanelProgressVisible <> PanelProgress.Visible then
|
|---|
| 237 | PanelProgress.Visible := PanelProgressVisible;
|
|---|
| 238 | if PanelProgressVisible then
|
|---|
| 239 | H := H + PanelProgress.Height;
|
|---|
| 240 | PanelProgressTotalVisible := (JobProgressView.Jobs.Count > 1) and not JobProgressView.Finished;
|
|---|
| 241 | if PanelProgressTotalVisible <> PanelProgressTotal.Visible then
|
|---|
| 242 | PanelProgressTotal.Visible := PanelProgressTotalVisible;
|
|---|
| 243 | if PanelProgressTotalVisible then
|
|---|
| 244 | H := H + PanelProgressTotal.Height;
|
|---|
| 245 | Constraints.MinHeight := H;
|
|---|
| 246 | PanelLogVisible := MemoLog.Lines.Count > 0;
|
|---|
| 247 | if PanelLogVisible <> PanelLog.Visible then
|
|---|
| 248 | PanelLog.Visible := PanelLogVisible;
|
|---|
| 249 | if PanelLogVisible then
|
|---|
| 250 | H := H + Scale96ToScreen(MemoLogHeight);
|
|---|
| 251 | if PanelText.Visible then
|
|---|
| 252 | H := H + PanelText.Height;
|
|---|
| 253 | if Height <> H then begin
|
|---|
| 254 | Height := H;
|
|---|
| 255 | Top := (Screen.Height - H) div 2;
|
|---|
| 256 | end;
|
|---|
| 257 | end;
|
|---|
| 258 |
|
|---|
| 259 | procedure TFormJobProgressView.TimerUpdateTimer(Sender: TObject);
|
|---|
| 260 | var
|
|---|
| 261 | ProgressBarPartVisible: Boolean;
|
|---|
| 262 | ProgressBarTotalVisible: Boolean;
|
|---|
| 263 | begin
|
|---|
| 264 | JobProgressView.UpdateProgress;
|
|---|
| 265 | if Visible and (not ProgressBarPart.Visible) and
|
|---|
| 266 | Assigned(JobProgressView.CurrentJob) and
|
|---|
| 267 | (JobProgressView.CurrentJob.Progress.Value > 0) then begin
|
|---|
| 268 | ProgressBarPartVisible := True;
|
|---|
| 269 | if ProgressBarPartVisible <> ProgressBarPart.Visible then
|
|---|
| 270 | ProgressBarPart.Visible := ProgressBarPartVisible;
|
|---|
| 271 | ProgressBarTotalVisible := True;
|
|---|
| 272 | if ProgressBarTotalVisible <> ProgressBarTotal.Visible then
|
|---|
| 273 | ProgressBarTotal.Visible := ProgressBarTotalVisible;
|
|---|
| 274 | end;
|
|---|
| 275 | if not Visible then begin
|
|---|
| 276 | TimerUpdate.Interval := UpdateInterval;
|
|---|
| 277 | if not JobProgressView.OwnerDraw then Show;
|
|---|
| 278 | end;
|
|---|
| 279 | if Assigned(JobProgressView.CurrentJob) then begin
|
|---|
| 280 | LabelText.Caption := JobProgressView.CurrentJob.Progress.Text;
|
|---|
| 281 | if LabelText.Caption <> '' then begin
|
|---|
| 282 | PanelText.Visible := True;
|
|---|
| 283 | UpdateHeight;
|
|---|
| 284 | end;
|
|---|
| 285 | end;
|
|---|
| 286 | end;
|
|---|
| 287 |
|
|---|
| 288 | procedure TFormJobProgressView.FormDestroy(Sender:TObject);
|
|---|
| 289 | begin
|
|---|
| 290 | end;
|
|---|
| 291 |
|
|---|
| 292 | procedure TFormJobProgressView.ListViewJobsData(Sender: TObject; Item: TListItem);
|
|---|
| 293 | begin
|
|---|
| 294 | if (Item.Index >= 0) and (Item.Index < JobProgressView.Jobs.Count) then
|
|---|
| 295 | with TJob(JobProgressView.Jobs[Item.Index]) do begin
|
|---|
| 296 | Item.Caption := Title;
|
|---|
| 297 | if Item.Index = JobProgressView.CurrentJobIndex then Item.ImageIndex := 1
|
|---|
| 298 | else if Finished then Item.ImageIndex := 0
|
|---|
| 299 | else Item.ImageIndex := 2;
|
|---|
| 300 | Item.Data := JobProgressView.Jobs[Item.Index];
|
|---|
| 301 | end;
|
|---|
| 302 | end;
|
|---|
| 303 |
|
|---|
| 304 | procedure TFormJobProgressView.FormClose(Sender: TObject;
|
|---|
| 305 | var CloseAction: TCloseAction);
|
|---|
| 306 | begin
|
|---|
| 307 | end;
|
|---|
| 308 |
|
|---|
| 309 | procedure TFormJobProgressView.FormCreate(Sender: TObject);
|
|---|
| 310 | begin
|
|---|
| 311 | Caption := SPleaseWait;
|
|---|
| 312 | try
|
|---|
| 313 | //Animate1.FileName := ExtractFileDir(UTF8Encode(Application.ExeName)) +
|
|---|
| 314 | // DirectorySeparator + 'horse.avi';
|
|---|
| 315 | //Animate1.Active := True;
|
|---|
| 316 | except
|
|---|
| 317 |
|
|---|
| 318 | end;
|
|---|
| 319 | end;
|
|---|
| 320 |
|
|---|
| 321 | procedure TFormJobProgressView.ReloadJobList;
|
|---|
| 322 | begin
|
|---|
| 323 | // Workaround for not showing first line
|
|---|
| 324 | //Form.ListViewJobs.Items.Count := Jobs.Count + 1;
|
|---|
| 325 | //Form.ListViewJobs.Refresh;
|
|---|
| 326 |
|
|---|
| 327 | if ListViewJobs.Items.Count <> JobProgressView.Jobs.Count then
|
|---|
| 328 | ListViewJobs.Items.Count := JobProgressView.Jobs.Count;
|
|---|
| 329 | ListViewJobs.Refresh;
|
|---|
| 330 | Application.ProcessMessages;
|
|---|
| 331 | UpdateHeight;
|
|---|
| 332 | end;
|
|---|
| 333 |
|
|---|
| 334 | procedure TFormJobProgressView.FormShow(Sender: TObject);
|
|---|
| 335 | begin
|
|---|
| 336 | ReloadJobList;
|
|---|
| 337 | end;
|
|---|
| 338 |
|
|---|
| 339 | procedure TFormJobProgressView.FormHide(Sender: TObject);
|
|---|
| 340 | begin
|
|---|
| 341 | JobProgressView.Jobs.Clear;
|
|---|
| 342 | ReloadJobList;
|
|---|
| 343 | end;
|
|---|
| 344 |
|
|---|
| 345 | procedure TFormJobProgressView.FormCloseQuery(Sender: TObject; var CanClose: Boolean);
|
|---|
| 346 | begin
|
|---|
| 347 | CanClose := JobProgressView.Finished;
|
|---|
| 348 | JobProgressView.Terminate := True;
|
|---|
| 349 | Caption := SPleaseWait + STerminate;
|
|---|
| 350 | end;
|
|---|
| 351 |
|
|---|
| 352 |
|
|---|
| 353 | { TJobProgressView }
|
|---|
| 354 |
|
|---|
| 355 | function TJobProgressView.AddJob(Title: string; Method: TJobProgressViewMethod;
|
|---|
| 356 | NoThreaded: Boolean = False; WaitFor: Boolean = False): TJob;
|
|---|
| 357 | begin
|
|---|
| 358 | Result := TJob.Create;
|
|---|
| 359 | Result.ProgressView := Self;
|
|---|
| 360 | Result.Title := Title;
|
|---|
| 361 | Result.Method := Method;
|
|---|
| 362 | Result.NoThreaded := NoThreaded;
|
|---|
| 363 | Result.WaitFor := WaitFor;
|
|---|
| 364 | Result.Progress.Max := 100;
|
|---|
| 365 | Result.Progress.Reset;
|
|---|
| 366 | Result.Progress.OnChange := JobProgressChange;
|
|---|
| 367 | Jobs.Add(Result);
|
|---|
| 368 | //ReloadJobList;
|
|---|
| 369 | end;
|
|---|
| 370 |
|
|---|
| 371 | procedure TJobProgressView.Start;
|
|---|
| 372 | var
|
|---|
| 373 | I: Integer;
|
|---|
| 374 | begin
|
|---|
| 375 | Terminate := False;
|
|---|
| 376 |
|
|---|
| 377 | if not OwnerDraw then Form.BringToFront;
|
|---|
| 378 |
|
|---|
| 379 | Finished := False;
|
|---|
| 380 | Form.Caption := SPleaseWait;
|
|---|
| 381 | try
|
|---|
| 382 | FormList := Screen.DisableForms(Form);
|
|---|
| 383 | Log.Clear;
|
|---|
| 384 | Form.MemoLog.Clear;
|
|---|
| 385 |
|
|---|
| 386 | Form.PanelText.Visible := False;
|
|---|
| 387 | Form.LabelEstimatedTimePart.Visible := False;
|
|---|
| 388 | Form.LabelEstimatedTimeTotal.Visible := False;
|
|---|
| 389 |
|
|---|
| 390 | CurrentJob := nil;
|
|---|
| 391 | if ShowDelay = 0 then begin
|
|---|
| 392 | Form.TimerUpdate.Interval := UpdateInterval;
|
|---|
| 393 | Form.TimerUpdate.Enabled := True;
|
|---|
| 394 | Form.TimerUpdateTimer(Self);
|
|---|
| 395 | end else begin
|
|---|
| 396 | Form.TimerUpdate.Interval := ShowDelay;
|
|---|
| 397 | Form.TimerUpdate.Enabled := True;
|
|---|
| 398 | end;
|
|---|
| 399 |
|
|---|
| 400 | TotalStartTime := Now;
|
|---|
| 401 | Form.ProgressBarTotal.Position := 0;
|
|---|
| 402 | Form.ProgressBarTotal.Visible := False;
|
|---|
| 403 | //UpdateHeight;
|
|---|
| 404 |
|
|---|
| 405 | I := 0;
|
|---|
| 406 | while I < Jobs.Count do
|
|---|
| 407 | with TJob(Jobs[I]) do begin
|
|---|
| 408 | CurrentJobIndex := I;
|
|---|
| 409 | CurrentJob := TJob(Jobs[I]);
|
|---|
| 410 | JobProgressChange(Self);
|
|---|
| 411 | StartTime := Now;
|
|---|
| 412 | Form.LabelEstimatedTimePart.Caption := Format(SEstimatedTime, ['']);
|
|---|
| 413 | Form.ProgressBarPart.Position := 0;
|
|---|
| 414 | Form.ProgressBarPart.Visible := False;
|
|---|
| 415 | //Show;
|
|---|
| 416 | Form.ReloadJobList;
|
|---|
| 417 | Application.ProcessMessages;
|
|---|
| 418 | if NoThreaded then begin
|
|---|
| 419 | Thread := nil;
|
|---|
| 420 | Method(CurrentJob);
|
|---|
| 421 | end else begin
|
|---|
| 422 | try
|
|---|
| 423 | Thread := TJobThread.Create(True);
|
|---|
| 424 | with Thread do begin
|
|---|
| 425 | FreeOnTerminate := False;
|
|---|
| 426 | Job := CurrentJob;
|
|---|
| 427 | Name := 'Job: ' + Job.Title;
|
|---|
| 428 | ProgressView := Self;
|
|---|
| 429 | Start;
|
|---|
| 430 | while not Terminated do begin
|
|---|
| 431 | Application.ProcessMessages;
|
|---|
| 432 | Sleep(1);
|
|---|
| 433 | end;
|
|---|
| 434 | WaitFor;
|
|---|
| 435 | end;
|
|---|
| 436 | finally
|
|---|
| 437 | FreeAndNil(Thread);
|
|---|
| 438 | end;
|
|---|
| 439 | end;
|
|---|
| 440 | Form.ProgressBarPart.Hide;
|
|---|
| 441 | if Assigned(FOnJobFinish) then
|
|---|
| 442 | FOnJobFinish(CurrentJob);
|
|---|
| 443 | if Terminate then Break;
|
|---|
| 444 | EndTime := Now;
|
|---|
| 445 | Finished := True;
|
|---|
| 446 | Inc(I);
|
|---|
| 447 | end;
|
|---|
| 448 | finally
|
|---|
| 449 | CurrentJob := nil;
|
|---|
| 450 | Form.TimerUpdate.Enabled := False;
|
|---|
| 451 | Screen.EnableForms(FormList);
|
|---|
| 452 | //if Visible then Hide;
|
|---|
| 453 | Form.MemoLog.Lines.Assign(Log);
|
|---|
| 454 | if (Form.MemoLog.Lines.Count = 0) and FAutoClose then begin
|
|---|
| 455 | Form.Hide;
|
|---|
| 456 | end;
|
|---|
| 457 | if not Form.Visible then Clear;
|
|---|
| 458 | Form.Caption := SFinished;
|
|---|
| 459 | //LabelEstimatedTimePart.Visible := False;
|
|---|
| 460 | Finished := True;
|
|---|
| 461 | CurrentJobIndex := -1;
|
|---|
| 462 | Form.ReloadJobList;
|
|---|
| 463 | end;
|
|---|
| 464 | end;
|
|---|
| 465 |
|
|---|
| 466 | procedure TJobProgressView.JobProgressChange(Sender: TObject);
|
|---|
| 467 | begin
|
|---|
| 468 | if Assigned(FOnOwnerDraw) then
|
|---|
| 469 | FOnOwnerDraw(Self);
|
|---|
| 470 | end;
|
|---|
| 471 |
|
|---|
| 472 | procedure TJobProgressView.Stop;
|
|---|
| 473 | begin
|
|---|
| 474 | Terminate := True;
|
|---|
| 475 | end;
|
|---|
| 476 |
|
|---|
| 477 | procedure TJobProgressView.TermSleep(Delay: Integer);
|
|---|
| 478 | const
|
|---|
| 479 | Quantum = 100;
|
|---|
| 480 | var
|
|---|
| 481 | I: Integer;
|
|---|
| 482 | begin
|
|---|
| 483 | Sleep(Delay mod Quantum);
|
|---|
| 484 | for I := 1 to (Delay div Quantum) do begin
|
|---|
| 485 | if Terminate then Break;
|
|---|
| 486 | Sleep(Quantum);
|
|---|
| 487 | end;
|
|---|
| 488 | end;
|
|---|
| 489 |
|
|---|
| 490 | procedure TJobProgressView.SetTerminate(const AValue: Boolean);
|
|---|
| 491 | var
|
|---|
| 492 | I: Integer;
|
|---|
| 493 | begin
|
|---|
| 494 | if AValue = FTerminate then Exit;
|
|---|
| 495 | for I := 0 to Jobs.Count - 1 do
|
|---|
| 496 | TJob(Jobs[I]).Terminate := AValue;
|
|---|
| 497 | FTerminate := AValue;
|
|---|
| 498 | end;
|
|---|
| 499 |
|
|---|
| 500 | procedure TJobProgressView.UpdateProgress;
|
|---|
| 501 | const
|
|---|
| 502 | OneJobValue: Integer = 100;
|
|---|
| 503 | var
|
|---|
| 504 | TotalMax: Integer;
|
|---|
| 505 | TotalValue: Integer;
|
|---|
| 506 | EstimatedTimePart: TDateTime;
|
|---|
| 507 | RemainingTime: TDateTime;
|
|---|
| 508 | begin
|
|---|
| 509 | if Assigned(CurrentJob) then
|
|---|
| 510 | with CurrentJob, Form do begin
|
|---|
| 511 | // Part progress
|
|---|
| 512 | ProgressBarPart.Max := Progress.Max;
|
|---|
| 513 | ProgressBarPart.Position := Progress.Value;
|
|---|
| 514 | if (Progress.Value >= EstimatedTimeShowTreshold) then begin
|
|---|
| 515 | EstimatedTimePart := (Now - StartTime) / Progress.Value * (Progress.Max - Progress.Value);
|
|---|
| 516 | LabelEstimatedTimePart.Caption := Format(SEstimatedTime, [
|
|---|
| 517 | TimeToStr(EstimatedTimePart)]);
|
|---|
| 518 | LabelEstimatedTimePart.Visible := True;
|
|---|
| 519 | end;
|
|---|
| 520 |
|
|---|
| 521 | // Total progress
|
|---|
| 522 | TotalMax := Jobs.Count * OneJobValue;
|
|---|
| 523 | TotalValue := Int64(CurrentJobIndex) * OneJobValue +
|
|---|
| 524 | Round(Progress.Value / Progress.Max * OneJobValue);
|
|---|
| 525 | ProgressBarTotal.Max := TotalMax;
|
|---|
| 526 | ProgressBarTotal.Position := TotalValue;
|
|---|
| 527 | if (TotalValue >= EstimatedTimeShowTresholdTotal) then begin
|
|---|
| 528 | // Project estimated time according part estimated time plus
|
|---|
| 529 | // estimated time by elapsed time divided by elapsed ticks mutiplied by rest ticks
|
|---|
| 530 | RemainingTime := EstimatedTimePart +
|
|---|
| 531 | (Now - TotalStartTime + EstimatedTimePart) /
|
|---|
| 532 | ((CurrentJobIndex + 1) * OneJobValue) *
|
|---|
| 533 | ((Jobs.Count - 1 - CurrentJobIndex) * OneJobValue);
|
|---|
| 534 | if (RemainingTime > 0) and (RemainingTime < EncodeDate(2100, 1, 1)) then begin
|
|---|
| 535 | LabelEstimatedTimeTotal.Caption := Format(STotalEstimatedTime, [
|
|---|
| 536 | TimeToStr(RemainingTime)]);
|
|---|
| 537 | LabelEstimatedTimeTotal.Visible := True;
|
|---|
| 538 | end else begin
|
|---|
| 539 | LabelEstimatedTimeTotal.Visible := False;
|
|---|
| 540 | end;
|
|---|
| 541 | end;
|
|---|
| 542 | end;
|
|---|
| 543 | end;
|
|---|
| 544 |
|
|---|
| 545 | constructor TJobProgressView.Create(TheOwner: TComponent);
|
|---|
| 546 | begin
|
|---|
| 547 | inherited;
|
|---|
| 548 | if not (csDesigning in ComponentState) then begin
|
|---|
| 549 | FForm := TFormJobProgressView.Create(Self);
|
|---|
| 550 | FForm.JobProgressView := Self;
|
|---|
| 551 | end;
|
|---|
| 552 | Jobs := TJobs.Create;
|
|---|
| 553 | Log := TStringList.Create;
|
|---|
| 554 | //PanelOperationsTitle.Height := 80;
|
|---|
| 555 | AutoClose := True;
|
|---|
| 556 | ShowDelay := 0;
|
|---|
| 557 | end;
|
|---|
| 558 |
|
|---|
| 559 | procedure TJobProgressView.Clear;
|
|---|
| 560 | begin
|
|---|
| 561 | Jobs.Clear;
|
|---|
| 562 | Log.Clear;
|
|---|
| 563 | //ReloadJobList;
|
|---|
| 564 | end;
|
|---|
| 565 |
|
|---|
| 566 | destructor TJobProgressView.Destroy;
|
|---|
| 567 | begin
|
|---|
| 568 | FreeAndNil(Log);
|
|---|
| 569 | FreeAndNil(Jobs);
|
|---|
| 570 | inherited;
|
|---|
| 571 | end;
|
|---|
| 572 |
|
|---|
| 573 | { TProgress }
|
|---|
| 574 |
|
|---|
| 575 | procedure TProgress.SetMax(const AValue: Integer);
|
|---|
| 576 | begin
|
|---|
| 577 | try
|
|---|
| 578 | FLock.Acquire;
|
|---|
| 579 | FMax := AValue;
|
|---|
| 580 | if FMax < 1 then FMax := 1;
|
|---|
| 581 | if FValue >= FMax then FValue := FMax;
|
|---|
| 582 | finally
|
|---|
| 583 | FLock.Release;
|
|---|
| 584 | end;
|
|---|
| 585 | end;
|
|---|
| 586 |
|
|---|
| 587 | procedure TProgress.SetText(AValue: string);
|
|---|
| 588 | begin
|
|---|
| 589 | try
|
|---|
| 590 | FLock.Acquire;
|
|---|
| 591 | if FText = AValue then Exit;
|
|---|
| 592 | FText := AValue;
|
|---|
| 593 | finally
|
|---|
| 594 | FLock.Release;
|
|---|
| 595 | end;
|
|---|
| 596 | end;
|
|---|
| 597 |
|
|---|
| 598 | procedure TProgress.SetValue(const AValue: Integer);
|
|---|
| 599 | var
|
|---|
| 600 | Change: Boolean;
|
|---|
| 601 | begin
|
|---|
| 602 | try
|
|---|
| 603 | FLock.Acquire;
|
|---|
| 604 | if AValue < Max then begin
|
|---|
| 605 | Change := AValue <> FValue;
|
|---|
| 606 | FValue := AValue;
|
|---|
| 607 | if Change and Assigned(FOnChange) then
|
|---|
| 608 | try
|
|---|
| 609 | FLock.Release;
|
|---|
| 610 | FOnChange(Self);
|
|---|
| 611 | finally
|
|---|
| 612 | FLock.Acquire;
|
|---|
| 613 | end;
|
|---|
| 614 | end;
|
|---|
| 615 | finally
|
|---|
| 616 | FLock.Release;
|
|---|
| 617 | end;
|
|---|
| 618 | end;
|
|---|
| 619 |
|
|---|
| 620 | procedure TProgress.Increment;
|
|---|
| 621 | begin
|
|---|
| 622 | try
|
|---|
| 623 | FLock.Acquire;
|
|---|
| 624 | Value := Value + 1;
|
|---|
| 625 | finally
|
|---|
| 626 | FLock.Release;
|
|---|
| 627 | end;
|
|---|
| 628 | end;
|
|---|
| 629 |
|
|---|
| 630 | procedure TProgress.Reset;
|
|---|
| 631 | begin
|
|---|
| 632 | try
|
|---|
| 633 | FLock.Acquire;
|
|---|
| 634 | FValue := 0;
|
|---|
| 635 | finally
|
|---|
| 636 | FLock.Release;
|
|---|
| 637 | end;
|
|---|
| 638 | end;
|
|---|
| 639 |
|
|---|
| 640 | constructor TProgress.Create;
|
|---|
| 641 | begin
|
|---|
| 642 | FMax := 100;
|
|---|
| 643 | FLock := TCriticalSection.Create;
|
|---|
| 644 | end;
|
|---|
| 645 |
|
|---|
| 646 | destructor TProgress.Destroy;
|
|---|
| 647 | begin
|
|---|
| 648 | FLock.Free;
|
|---|
| 649 | inherited Destroy;
|
|---|
| 650 | end;
|
|---|
| 651 |
|
|---|
| 652 | { TJob }
|
|---|
| 653 |
|
|---|
| 654 | procedure TJob.SetTerminate(const AValue: Boolean);
|
|---|
| 655 | begin
|
|---|
| 656 | if FTerminate = AValue then Exit;
|
|---|
| 657 | FTerminate := AValue;
|
|---|
| 658 | if AValue then begin
|
|---|
| 659 | ProgressView.Terminate := AValue;
|
|---|
| 660 | if Assigned(Thread) then Thread.Terminate;
|
|---|
| 661 | end;
|
|---|
| 662 | end;
|
|---|
| 663 |
|
|---|
| 664 | procedure TJob.AddLogItem(Value: string);
|
|---|
| 665 | begin
|
|---|
| 666 | with ProgressView do begin
|
|---|
| 667 | Log.Add(Value);
|
|---|
| 668 | end;
|
|---|
| 669 | end;
|
|---|
| 670 |
|
|---|
| 671 | constructor TJob.Create;
|
|---|
| 672 | begin
|
|---|
| 673 | Progress := TProgress.Create;
|
|---|
| 674 | Terminate := False;
|
|---|
| 675 | Finished := False;
|
|---|
| 676 | end;
|
|---|
| 677 |
|
|---|
| 678 | destructor TJob.Destroy;
|
|---|
| 679 | begin
|
|---|
| 680 | Progress.Free;
|
|---|
| 681 | inherited;
|
|---|
| 682 | end;
|
|---|
| 683 |
|
|---|
| 684 | end.
|
|---|