unit formPostProcess; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs, ExtCtrls, Render, controlpoint, StdCtrls, ComCtrls; type TfrmPostProcess = class(TForm) Panel1: TPanel; ScrollBox1: TScrollBox; Image: TImage; btnSave: TButton; Label1: TLabel; pnlBackColor: TPanel; ColorDialog: TColorDialog; ProgressBar1: TProgressBar; btnApply: TButton; txtFilterRadius: TEdit; txtGamma: TEdit; txtVibrancy: TEdit; txtContrast: TEdit; txtBrightness: TEdit; pnlGamma: TPanel; pnlBrightness: TPanel; pnlContrast: TPanel; pnlVibrancy: TPanel; pnlFilter: TPanel; procedure btnSaveClick(Sender: TObject); procedure btnApplyClick(Sender: TObject); procedure FormDestroy(Sender: TObject); procedure pnlBackColorClick(Sender: TObject); procedure FormClose(Sender: TObject; var Action: TCloseAction); procedure FormShow(Sender: TObject); procedure DragPanelMouseDown(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer); procedure DragPanelMouseMove(Sender: TObject; Shift: TShiftState; X, Y: Integer); procedure DragPanelMouseUp(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer); procedure DragPanelDblClick(Sender: TObject); private { Private declarations } FRenderer: TBaseRenderer; FCP: TControlPoint; FImagename: string; pnlDragMode, pnlDragged, pnlMM: boolean; pnlDragPos, pnlDragOld: integer; pnlDragValue: double; mousepos: TPoint; BkgColor: TColor; Filter, Gamma, Brightness, Contrast, Vibrancy: double; procedure UpdateFlame; procedure SetDefaultValues; procedure OnProgress(prog: double); public procedure SetRenderer(Renderer: TBaseRenderer); procedure SetControlPoint(CP: TControlPoint); procedure SetImageName(imagename: string); end; var frmPostProcess: TfrmPostProcess; implementation uses Registry, Global; {$R *.dfm} { TfrmPostProcess } /////////////////////////////////////////////////////////////////////////////// procedure TfrmPostProcess.SetRenderer(Renderer: TBaseRenderer); begin if assigned(FRenderer) then FRenderer.Free; FRenderer := Renderer; Frenderer.OnProgress := OnProgress; Image.Picture.Graphic := FRenderer.GetImage; end; /////////////////////////////////////////////////////////////////////////////// procedure TfrmPostProcess.FormShow(Sender: TObject); var Registry: TRegistry; begin { Read posution from registry } Registry := TRegistry.Create; try Registry.RootKey := HKEY_CURRENT_USER; if Registry.OpenKey('Software\' + APP_NAME + '\Forms\PostProcess', False) then begin if Registry.ValueExists('Left') then Left := Registry.ReadInteger('Left'); if Registry.ValueExists('Top') then Top := Registry.ReadInteger('Top'); if Registry.ValueExists('Width') then Width := Registry.ReadInteger('Width'); if Registry.ValueExists('Height') then Height := Registry.ReadInteger('Height'); end; Registry.CloseKey; finally Registry.Free; end; end; /////////////////////////////////////////////////////////////////////////////// procedure TfrmPostProcess.FormClose(Sender: TObject; var Action: TCloseAction); var Registry: TRegistry; begin { Write position to registry } Registry := TRegistry.Create; try Registry.RootKey := HKEY_CURRENT_USER; if Registry.OpenKey('\Software\' + APP_NAME + '\Forms\PostProcess', True) then begin Registry.WriteInteger('Top', Top); Registry.WriteInteger('Left', Left); Registry.WriteInteger('Width', Width); Registry.WriteInteger('Height', Height); end; finally Registry.Free; end; end; /////////////////////////////////////////////////////////////////////////////// procedure TfrmPostProcess.SetDefaultValues; begin BkgColor := RGB(Fcp.background[0], Fcp.background[1], Fcp.background[2]); pnlBackColor.Color := BkgColor; Filter := FCP.spatial_filter_radius; txtFilterRadius.Text := FloatTostr(Filter); Gamma := FCP.gamma; txtGamma.Text := FloatTostr(Gamma); Vibrancy := FCP.vibrancy; txtVibrancy.Text := FloatTostr(Vibrancy); Contrast := FCP.contrast; txtContrast.Text := FloatTostr(Contrast); Brightness := FCP.brightness; txtBrightness.Text := FloatTostr(brightness); end; /////////////////////////////////////////////////////////////////////////////// procedure TfrmPostProcess.SetControlPoint(CP: TControlPoint); begin if assigned(FCP) then FCP.Free; FCP := cp.Clone; SetDefaultValues; end; /////////////////////////////////////////////////////////////////////////////// procedure TfrmPostProcess.pnlBackColorClick(Sender: TObject); var col: Longint; begin ColorDialog.Color := pnlBackColor.Color; if ColorDialog.Execute then begin pnlBackColor.Color := ColorDialog.Color; col := ColorToRGB(ColorDialog.Color); Fcp.background[0] := col and 255; Fcp.background[1] := (col shr 8) and 255; Fcp.background[2] := (col shr 16) and 255; UpdateFlame; end; end; /////////////////////////////////////////////////////////////////////////////// procedure TfrmPostProcess.UpdateFlame; begin FRenderer.UpdateImage(FCP); Image.Picture.Graphic := FRenderer.GetImage; end; /////////////////////////////////////////////////////////////////////////////// procedure TfrmPostProcess.FormDestroy(Sender: TObject); begin if assigned(FRenderer) then FRenderer.Free; if assigned(FCP) then FCP.Free; end; /////////////////////////////////////////////////////////////////////////////// procedure TfrmPostProcess.OnProgress(prog: double); begin ProgressBar1.Position := round(100 * prog); end; /////////////////////////////////////////////////////////////////////////////// procedure TfrmPostProcess.btnApplyClick(Sender: TObject); begin TryStrToFloat(txtFilterRadius.Text, FCP.spatial_filter_radius); if FCP.spatial_filter_radius > 2 then begin FCP.spatial_filter_radius := 2; txtFilterRadius.Text := '2'; end else if FCP.spatial_filter_radius < 0 then begin FCP.spatial_filter_radius := 0.01; txtFilterRadius.Text := FloatTostr(0.01); end; TryStrToFloat(txtGamma.Text, FCP.gamma); if FCP.gamma > 10 then begin FCP.gamma := 10; txtGamma.Text := '10'; end else if FCP.gamma < 0.01 then begin FCP.gamma := 0.01; txtGamma.Text := FloatTostr(0.01); end; TryStrToFloat(txtVibrancy.Text, FCP.vibrancy); if FCP.vibrancy > 10 then begin FCP.vibrancy := 10; txtVibrancy.Text := '10'; end else if FCP.vibrancy < 0.01 then begin FCP.vibrancy := 0.01; txtVibrancy.Text := FloatTostr(0.01); end; TryStrToFloat(txtContrast.Text, FCP.contrast); if FCP.contrast > 10 then begin FCP.contrast := 10; txtContrast.Text := '10'; end else if FCP.contrast < 0.01 then begin FCP.contrast := 0.01; txtContrast.Text := FloatTostr(0.01); end; TryStrToFloat(txtBrightness.Text, FCP.brightness); if FCP.brightness > 100 then begin FCP.brightness := 100; txtBrightness.Text := '100'; end else if FCP.brightness < 0.01 then begin FCP.brightness := 0.01; txtBrightness.Text := FloatTostr(0.01); end; UpdateFlame; end; /////////////////////////////////////////////////////////////////////////////// procedure TfrmPostProcess.btnSaveClick(Sender: TObject); begin FRenderer.SaveImage(FImagename); end; /////////////////////////////////////////////////////////////////////////////// procedure TfrmPostProcess.SetImageName(imagename: string); begin FImagename := imagename; end; // ----------------------------------------------------------------------------- procedure TfrmPostProcess.DragPanelMouseDown(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer); begin if Button <> mbLeft then exit; if (Sender = pnlFilter) then pnlDragValue := fcp.spatial_filter_radius * 10 else if (Sender = pnlGamma) then pnlDragValue := fcp.gamma else if (Sender = pnlBrightness) then pnlDragValue := fcp.brightness else if (Sender = pnlContrast) then pnlDragValue := fcp.contrast else if (Sender = pnlVibrancy) then pnlDragValue := fcp.vibrancy else assert(false); pnlDragMode := true; pnlDragPos := 0; pnlDragOld := x; pnlMM := false; SetCaptureControl(TControl(Sender)); Screen.Cursor := crHSplit; GetCursorPos(mousepos); // hmmm pnlDragged := false; end; procedure TfrmPostProcess.DragPanelMouseMove(Sender: TObject; Shift: TShiftState; X, Y: Integer); var v: double; pEdit: ^TEdit; begin if pnlMM then // hack: to skip MouseMove event begin pnlMM:=false; end else if pnlDragMode and (x <> pnlDragOld) then begin Inc(pnlDragPos, x - pnlDragOld); if GetKeyState(VK_MENU) < 0 then v := 100000 else if GetKeyState(VK_CONTROL) < 0 then v := 10000 else if GetKeyState(VK_SHIFT) < 0 then v := 100 else v := 1000; v := Round6(pnlDragValue + pnlDragPos / v); SetCursorPos(MousePos.x, MousePos.y); // hmmm pnlMM:=true; if (Sender = pnlFilter) then begin v := v / 10; if v > 2 then v := 2 else if v < 0.01 then v := 0.01; fcp.spatial_filter_radius := v; pEdit := @txtFilterRadius; end else if (Sender = pnlGamma) then begin if v > 10 then v := 10 else if v < 0.01 then v := 0.01; fcp.gamma := v; pEdit := @txtGamma; end else if (Sender = pnlBrightness) then begin if v > 100 then v := 100 else if v < 0.01 then v := 0.01; fcp.brightness := v; pEdit := @txtBrightness; end else if (Sender = pnlContrast) then begin if v > 10 then v := 10 else if v < 0.01 then v := 0.01; fcp.contrast := v; pEdit := @txtContrast; end else if (Sender = pnlVibrancy) then begin if v > 10 then v := 10 else if v < 0.01 then v := 0.01; fcp.vibrancy := v; pEdit := @txtVibrancy; end; pEdit^.Text := FloatToStr(v); //pEdit.Refresh; pnlDragged := True; // TODO: image preview (?) //DrawPreview; end; end; procedure TfrmPostProcess.DragPanelMouseUp(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer); begin if Button <> mbLeft then exit; if pnlDragMode then begin SetCaptureControl(nil); pnlDragMode := false; Screen.Cursor := crDefault; if pnlDragged then begin UpdateFlame; pnlDragged := False; end; end; end; procedure TfrmPostProcess.DragPanelDblClick(Sender: TObject); var pValue: ^double; pDefaultValue: ^double; pEdit: ^TEdit; begin if (Sender = pnlFilter) then begin pValue := @fcp.spatial_filter_radius; pDefaultValue := @Filter; pEdit := @txtFilterRadius; end else if (Sender = pnlGamma) then begin pValue := @fcp.gamma; pDefaultValue := @Gamma; pEdit := @txtGamma; end else if (Sender = pnlBrightness) then begin pValue := @fcp.brightness; pDefaultValue := @Brightness; pEdit := @txtBrightness; end else if (Sender = pnlContrast) then begin pValue := @fcp.contrast; pDefaultValue := @Contrast; pEdit := @txtContrast end else if (Sender = pnlVibrancy) then begin pValue := @fcp.vibrancy; pDefaultValue := @Vibrancy; pEdit := @txtVibrancy; end else assert(false); if pValue^ = pDefaultValue^ then exit; pValue^ := pDefaultValue^; pEdit^.Text := FloatToStr(pValue^); UpdateFlame; end; end.