{ Apophysis Copyright (C) 2001-2004 Mark Townsend Apophysis Copyright (C) 2005-2006 Ronald Hordijk, Piotr Borys, Peter Sdobnov This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. } unit Fullscreen; interface uses Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs, Menus, ExtCtrls, ControlPoint, RenderThread; type TFullscreenForm = class(TForm) Image: TImage; Timelimiter: TTimer; FullscreenPopup: TPopupMenu; RenderStop: TMenuItem; N1: TMenuItem; Exit1: TMenuItem; RenderMore: TMenuItem; procedure FormShow(Sender: TObject); procedure FormClose(Sender: TObject; var Action: TCloseAction); procedure FormCreate(Sender: TObject); procedure FormDestroy(Sender: TObject); procedure FormKeyPress(Sender: TObject; var Key: Char); procedure ImageDblClick(Sender: TObject); procedure TimelimiterOnTimer(Sender: TObject); procedure RenderStopClick(Sender: TObject); procedure RenderMoreClick(Sender: TObject); private Remainder, StartTime, t: double; imgLeft, imgTop, imgWidth, imgHeight: integer; Closing: boolean; Renderer: TRenderThread; procedure showTaskbar; procedure hideTaskbar; procedure DrawFlame; procedure OnProgress(prog: double); procedure HandleThreadCompletion(var Message: TMessage); message WM_THREAD_COMPLETE; procedure HandleThreadTermination(var Message: TMessage); message WM_THREAD_TERMINATE; public Calculate : boolean; cp: TControlPoint; Zoom: double; center: array[0..1] of double; ActiveForm: TForm; end; var FullscreenForm: TFullscreenForm; implementation uses Main, Math, Global, Tracer; {$R *.DFM} procedure Trace1(const str: string); begin if TraceLevel >= 1 then TraceForm.FullscreenTrace.Lines.Add('. ' + str); end; procedure Trace2(const str: string); begin if TraceLevel >= 2 then TraceForm.FullscreenTrace.Lines.Add('. . ' + str); end; procedure TFullscreenForm.DrawFlame; var r: double; begin if (cp.width / cp.height) > (ClientWidth / ClientHeight) then begin imgWidth := ClientWidth; r := cp.width / imgWidth; imgHeight := round(cp.height / r); imgLeft := 1; imgTop := (ClientHeight - imgHeight) div 2; end else begin imgHeight := ClientHeight; r := cp.height / imgHeight; imgWidth := round(cp.Width / r); imgTop := 1; imgLeft := (ClientWidth - ImgWidth) div 2; end; cp.AdjustScale(imgWidth, imgHeight); // cp.Zoom := MainForm.Zoom; // cp.center[0] := MainForm.center[0]; // cp.center[1] := MainForm.center[1]; cp.sample_density := defSampleDensity; StartTime := Now; t := now; Remainder := 1; if Assigned(Renderer) then begin // hmm... Trace2('Killing previous RenderThread #' + inttostr(Renderer.ThreadID)); Renderer.Terminate; Renderer.WaitFor; while Renderer <> nil do Application.ProcessMessages; // HandleThreadTermination kinda should be called here...(?) end; assert(not assigned(renderer), 'Render thread is still running!?'); Renderer := TRenderThread.Create; // Hmm... Why do we use RenderThread here, anyway? :-\ Renderer.TargetHandle := Handle; Renderer.OnProgress := OnProgress; Renderer.NrThreads := NrTreads; if TraceLevel > 0 then Renderer.Output := TraceForm.FullscreenTrace.Lines; Renderer.SetCP(cp); Renderer.WaitForMore := true; RenderStop.Enabled := true; RenderMore.Enabled := false; Renderer.Resume; end; procedure TFullscreenForm.HandleThreadCompletion(var Message: TMessage); var bm: TBitmap; begin Trace2(MsgComplete + IntToStr(message.LParam)); if not Assigned(Renderer) then begin Trace2(MsgNotAssigned); exit; end; if Renderer.ThreadID <> message.LParam then begin Trace2(MsgAnotherRunning); exit; end; if Assigned(Renderer) then begin bm := TBitmap.Create; bm.assign(Renderer.GetImage); Image.SetBounds(imgLeft, imgTop, imgWidth, imgHeight); Image.Picture.Graphic := bm; bm.Free; end; RenderStop.Enabled := false; RenderMore.Enabled := true; TimeLimiter.Enabled := false; end; procedure TFullscreenForm.HandleThreadTermination(var Message: TMessage); var bm: TBitmap; begin Trace2(MsgTerminated + IntToStr(message.LParam)); if not Assigned(Renderer) then begin Trace2(MsgNotAssigned); exit; end; if Renderer.ThreadID <> message.LParam then begin Trace2(MsgAnotherRunning); exit; end; RenderStop.Enabled := false; RenderMore.Enabled := false; TimeLimiter.Enabled := false; end; procedure TFullscreenForm.OnProgress(prog: double); begin prog := (Renderer.Slice + Prog) / Renderer.NrSlices; try if prog >= 1 then begin Canvas.Brush.Color := clBlack; Canvas.FillRect(Rect(5, ClientHeight - 15, ClientWidth - 5, ClientHeight - 5)); end else if prog >= 0 then begin Canvas.Brush.Color := clTeal; Canvas.FrameRect(Rect(5, ClientHeight - 15, ClientWidth - 5, ClientHeight - 5)); Canvas.Brush.Color := clTeal; Canvas.Fillrect(Rect(7, ClientHeight - 13, 7 + Round(prog * (ClientWidth - 14)), ClientHeight - 7)); Canvas.Brush.Color := clBlack; Canvas.Fillrect(Rect(7 + Round(prog * (ClientWidth - 14)), ClientHeight - 13, ClientWidth - 7, ClientHeight - 7)); end; except end; Application.ProcessMessages; end; procedure TFullscreenForm.hideTaskbar; var wndHandle: THandle; wndClass: array[0..50] of Char; begin StrPCopy(@wndClass[0], 'Shell_TrayWnd'); wndHandle := FindWindow(@wndClass[0], nil); ShowWindow(wndHandle, SW_HIDE); end; procedure TFullscreenForm.showTaskbar; var wndHandle: THandle; wndClass: array[0..50] of Char; begin StrPCopy(@wndClass[0], 'Shell_TrayWnd'); wndHandle := FindWindow(@wndClass[0], nil); ShowWindow(wndHandle, SW_RESTORE); end; procedure TFullscreenForm.FormShow(Sender: TObject); begin Trace1('--- Opening Fullscreen View ---'); if Image.Width < ClientWidth then Image.Left := (ClientWidth - Image.Width) div 2; if Image.Height < ClientHeight then Image.Top := (ClientHeight - Image.Height) div 2; Closing := false; TimeLimiter.Enabled := false; RenderStop.Enabled := false; RenderMore.Enabled := false; MainForm.mnuFullScreen.enabled := true; HideTaskbar; if calculate then DrawFlame; end; procedure TFullscreenForm.FormClose(Sender: TObject; var Action: TCloseAction); begin Closing := true; if Assigned(Renderer) then begin if Renderer.Suspended then begin Renderer.WaitForMore := false; Renderer.Resume; end; Trace2('Form closing: killing RenderThread #' + inttostr(Renderer.ThreadID)); Renderer.Terminate; Renderer.WaitFor; Trace2('Destroying RenderThread #' + IntToStr(Renderer.ThreadID)); Renderer.Free; Renderer := nil; end; Trace1('--- Closing Fullscreen View ---'); Trace1(''); ShowTaskbar; ActiveForm.SetFocus; end; procedure TFullscreenForm.FormCreate(Sender: TObject); begin cp := TControlPoint.Create; end; procedure TFullscreenForm.FormDestroy(Sender: TObject); begin if assigned(Renderer) then begin Renderer.Terminate; Renderer.WaitFor; Renderer.Free; end; cp.Free; end; procedure TFullscreenForm.FormKeyPress(Sender: TObject; var Key: Char); begin if key = ' ' then begin if RenderStop.Enabled then RenderStop.Click else if RenderMore.Enabled then RenderMore.Click; end else Close; end; procedure TFullscreenForm.ImageDblClick(Sender: TObject); begin Close; end; procedure TFullscreenForm.TimelimiterOnTimer(Sender: TObject); begin //if assigned(Renderer) then Renderer.Break; TimeLimiter.Enabled := false; end; procedure TFullscreenForm.RenderStopClick(Sender: TObject); begin if assigned(Renderer) then Renderer.BreakRender; end; procedure TFullscreenForm.RenderMoreClick(Sender: TObject); begin if assigned(Renderer) and Renderer.Suspended then begin Renderer.Resume; RenderStop.Enabled := true; RenderMore.Enabled := false; end; end; end.