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