diff --git a/2.10/Source/Adjust.pas b/2.10/Source/Adjust.pas index a92b2b6..250adfc 100644 --- a/2.10/Source/Adjust.pas +++ b/2.10/Source/Adjust.pas @@ -1,5 +1,6 @@ { Apophysis Copyright (C) 2001-2004 Mark Townsend + Apophysis Copyright (C) 2005-2006 Ronald Hordijk, Piotr Boris, 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 @@ -1903,7 +1904,8 @@ begin pnlDragPos := 0; pnlDragOld := x; pnlMM := false; - SetCaptureControl(TControl(Sender)); + //SetCaptureControl(TControl(Sender)); + Screen.Cursor := crHSplit; GetCursorPos(mousepos); // hmmm pnlDragged := false; @@ -1982,7 +1984,8 @@ begin if pnlDragMode then begin - SetCaptureControl(nil); + //SetCaptureControl(nil); + pnlDragMode := false; Screen.Cursor := crDefault; diff --git a/2.10/Source/BucketFillerThread.pas b/2.10/Source/BucketFillerThread.pas index 0890c38..889886c 100644 --- a/2.10/Source/BucketFillerThread.pas +++ b/2.10/Source/BucketFillerThread.pas @@ -93,8 +93,6 @@ end; /////////////////////////////////////////////////////////////////////////////// constructor TBucketFillerThread.Create(cp: TControlPoint); -var - i, n: integer; begin inherited Create(True); Self.FreeOnTerminate := True; diff --git a/2.10/Source/ControlPoint.pas b/2.10/Source/ControlPoint.pas index bad027b..ce65a2c 100644 --- a/2.10/Source/ControlPoint.pas +++ b/2.10/Source/ControlPoint.pas @@ -1,6 +1,7 @@ { Flame screensaver Copyright (C) 2002 Ronald Hordijk Apophysis Copyright (C) 2001-2004 Mark Townsend + Apophysis Copyright (C) 2005-2006 Ronald Hordijk, Piotr Boris, 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 @@ -88,9 +89,11 @@ type TControlPoint = class public xform: array[0..NXFORMS] of TXForm; - finalXform: ^TXForm; + + finalXform: TXForm; finalXformEnabled: boolean; useFinalXform: boolean; + variation: TVariation; cmap: TColorMap; cmapindex: integer; @@ -121,7 +124,7 @@ type pulse: array[0..1, 0..1] of double; // [i][0]=magnitute [i][1]=frequency */ wiggle: array[0..1, 0..1] of double; // frequency is /minute, assuming 30 frames/s */ - PropTable: array of ^TXForm;//Integer; + PropTable: array of TXForm;//Integer; FAngle: Double; FTwoColorDimensions: Boolean; private @@ -258,7 +261,7 @@ destructor TControlPoint.Destroy; var i: Integer; begin - for i := 0 to NXFORMS - 1 do + for i := 0 to NXFORMS do xform[i].Free; inherited; @@ -278,7 +281,7 @@ begin n := NumXforms; assert(n > 0); - finalXform := @xform[n]; + finalXform := xform[n]; finalXform.Prepare; useFinalXform := FinalXformEnabled and HasFinalXform; for i := 0 to n - 1 do begin @@ -294,7 +297,7 @@ begin inc(j); propsum := propsum + xform[j].density; until (propsum > LoopValue) or (j = n - 1); - PropTable[i] := @xform[j]; + PropTable[i] := xform[j]; LoopValue := LoopValue + TotValue / PROP_TABLE_SIZE; end; @@ -543,9 +546,6 @@ begin px := 2 * random - 1; py := 2 * random - 1; -// PreparePropTable; -// for i := 0 to NXFORMS do xform[i].prepare; - try for i := 0 to FUSE do PropTable[Random(PROP_TABLE_SIZE)].NextPointXY(px,py); @@ -556,7 +556,7 @@ if UseFinalXform then PropTable[Random(PROP_TABLE_SIZE)].NextPointXY(px,py); pPoint^.X := px; pPoint^.Y := py; - finalXform^.NextPointXY(pPoint^.X, pPoint^.y); + finalXform.NextPointXY(pPoint^.X, pPoint^.y); Inc(pPoint); end else @@ -599,9 +599,6 @@ asm end; {$ifend} -// PreparePropTable; -// for i := 0 to NXFORMS do xform[i].prepare; - try for i := 0 to FUSE do PropTable[Random(PROP_TABLE_SIZE)].NextPoint(p); @@ -611,7 +608,7 @@ end; if UseFinalXform then for i := 0 to NrPoints - 1 do begin PropTable[Random(PROP_TABLE_SIZE)].NextPoint(p); - finalXform^.NextPointTo(p, pPoint^); + finalXform.NextPointTo(p, pPoint^); Inc(pPoint); end else @@ -689,9 +686,6 @@ begin p.c1 := random; p.c2 := random; -// PreparePropTable; -// for i := 0 to NXFORMS do xform[i].prepare; - try for i := 0 to FUSE do PropTable[Random(PROP_TABLE_SIZE)].NextPoint2C(p);//px, py, pc1, pc2); @@ -704,7 +698,7 @@ if UseFinalXform then CurrentPoint.Y := p.y; CurrentPoint.C1 := p.c1; CurrentPoint.C2 := p.c2; - finalXform^.NextPoint2C(CurrentPoint^); + finalXform.NextPoint2C(CurrentPoint^); Inc(CurrentPoint); end else @@ -1158,15 +1152,6 @@ begin IterateXY(SUB_BATCH_SIZE, points); -{ if finalXformEnabled and HasFinalXform then begin - try - finalXform := @xform[NumXforms]; - for i := 0 to SUB_BATCH_SIZE - 1 do - finalXform.NextPoint(points[i]); - except - end - end;} - LimitOutSidePoints := Round(0.05 * SUB_BATCH_SIZE); minx := 1E99; @@ -1730,7 +1715,6 @@ function TControlPoint.HasFinalXForm: boolean; var i: integer; begin -// if finalXformEnabled then Result := true else with xform[NumXForms] do begin Result := (c[0,0]<>1) or (c[0,1]<>0) or(c[1,0]<>0) or (c[1,1]<>1) or (c[2,0]<>0) or (c[2,1]<>0) or diff --git a/2.10/Source/Main.dfm b/2.10/Source/Main.dfm index 4da8219..d2c4881 100644 --- a/2.10/Source/Main.dfm +++ b/2.10/Source/Main.dfm @@ -2549,7 +2549,6 @@ object MainForm: TMainForm ImageIndex = 45 ShortCut = 16456 Visible = False - OnClick = mnuPostSheepClick end object N21: TMenuItem Caption = '-' diff --git a/2.10/Source/Main.pas b/2.10/Source/Main.pas index 01b4d5b..a2ce686 100644 --- a/2.10/Source/Main.pas +++ b/2.10/Source/Main.pas @@ -37,7 +37,7 @@ const RS_XO = 2; RS_VO = 3; - AppVersionString = 'Apophysis 2.03d pre-release 2'; + AppVersionString = 'Apophysis 2.03d pre-release 3'; type TMouseMoveState = (msUsual, msZoomWindow, msZoomOutWindow, msZoomWindowMove, msZoomOutWindowMove, msDrag, msDragMove, msRotate, msRotateMove); @@ -246,15 +246,8 @@ type procedure ApplicationEventsActivate(Sender: TObject); procedure mnuPasteClick(Sender: TObject); procedure mnuCopyClick(Sender: TObject); - procedure mnuExportFLameClick(Sender: TObject); - procedure mnuPostSheepClick(Sender: TObject); -{ - procedure HTTPRedirect(Sender: TObject; var dest: string; - var NumRedirect: Integer; var Handled: Boolean; - var VMethod: TIdHTTPMethod); - procedure HTTPStatus(ASender: TObject; const AStatus: TIdStatus; - const AStatusText: string); -} + procedure mnuExportFlameClick(Sender: TObject); + procedure ListXmlScannerStartTag(Sender: TObject; TagName: string; Attributes: TAttrList); procedure XMLScannerStartTag(Sender: TObject; TagName: string; @@ -312,7 +305,6 @@ type procedure ParseXML(var cp1: TControlPoint; const params: PCHAR); function SaveFlame(cp1: TControlPoint; title, filename: string): boolean; function SaveXMLFlame(const cp1: TControlPoint; title, filename: string): boolean; - //function TrianglesFromCP(const cp1: TControlPoint; var Triangles: TTriangles): integer; procedure DisplayHint(Sender: TObject); procedure OnProgress(prog: double); procedure DrawFlame; @@ -358,8 +350,9 @@ var MainForm: TMainForm; pname, ptime: string; nxform: integer; - FinalXformLoaded: boolean; - ParseCp: TControlPoint; // For parsing; + FinalXformLoaded: boolean; // + ParseCp: TControlPoint; // For parsing; + ActiveXformSet: integer; // MainCp: TControlPoint; implementation @@ -540,6 +533,7 @@ procedure TMainForm.StopThread; begin RedrawTimer.Enabled := False; if Assigned(Renderer) then begin + assert(Renderer.Suspended = false); Renderer.Terminate; Renderer.WaitFor; end; @@ -1303,7 +1297,7 @@ function FlameToXML(const cp1: TControlPoint; sheep: boolean; compact: boolean = var t, i{, j}: integer; FileList: TStringList; - x, y{, a, b, cc, d, e, f}: double; + x, y: double; {varlist,} nick, url, pal, hue: string; begin FileList := TStringList.create; @@ -1341,8 +1335,19 @@ begin t := cp1.NumXForms; for i := 0 to t - 1 do FileList.Add(cp1.xform[i].ToXMLString); -// if cp1.HasFinalXForm then FileList.Add(cp1.finalxform.FinalToXMLString(cp1.finalXformEnabled)); - if cp1.HasFinalXForm then FileList.Add(cp1.xform[t].FinalToXMLString(cp1.finalXformEnabled)); + if cp1.HasFinalXForm then + begin + +{$if false} // new file format - how about this? + FileList.Add(Format(' ', [IfThen(cp1.finalXformEnabled, 1, 0)])); + FileList.Add(' ' + cp1.xform[t].ToXMLString); + Filelist.Add(' '); +{$else} + FileList.Add(cp1.xform[i].FinalToXMLString(cp1.finalXformEnabled)); +{$ifend} + + end; + { Write palette data } if not sheep then begin if compact then // say no to duplicated data! (?) @@ -1728,13 +1733,15 @@ procedure TMainForm.DrawFlame; begin RedrawTimer.Enabled := False; if Assigned(Renderer) then begin + assert(Renderer.Suspended = false); + Renderer.Terminate; Renderer.WaitFor; Renderer.Free; Renderer := nil; end; - assert(Renderer = nil); //... + assert(Renderer = nil); //...? if not Assigned(Renderer) then begin @@ -1769,22 +1776,6 @@ end; { ---Z--- moved to ControlPoint ---Z--- } -{ // unused function, hmmm... - -procedure CP_compute(var cp1: TControlPoint; t1, t0: TTriangle; const i: integer); -begin - solve3(t0.x[0], t0.y[0], t1.x[0], - t0.x[1], t0.y[1], t1.x[1], - t0.x[2], t0.y[2], t1.x[2], - cp1.xform[i].c[0][0], cp1.xform[i].c[1][0], cp1.xform[i].c[2][0]); - - solve3(t0.x[0], t0.y[0], t1.y[0], - t0.x[1], t0.y[1], t1.y[1], - t0.x[2], t0.y[2], t1.y[2], - cp1.xform[i].c[0][1], cp1.xform[i].c[1][1], cp1.xform[i].c[2][1]); -end; -} - function FlameToString(Title: string): string; { Creates a string containing the formated flame parameter set } var @@ -3529,7 +3520,8 @@ begin StopThread; nxform := 0; FinalXformLoaded := false; - Parsecp.cmapindex := -2; // generate pallet from cmapindex and hue (apo 1 and earlier) + activeXformSet:=0; + Parsecp.cmapindex := -2; // generate palette from cmapindex and hue (apo 1 and earlier) ParseCp.symmetry := 0; ParseCP.finalXformEnabled := false; XMLScanner.LoadFromBuffer(params); @@ -3726,87 +3718,6 @@ begin end; end; -procedure TMainForm.mnuPostSheepClick(Sender: TObject); -{ -var - URL: string; - StringList: TStringList; - ResponseStream: TMemoryStream; - MultiPartFormDataStream: TmsMultiPartFormDataStream; -} -begin -// if MainCp.HasNewVariants then begin -// showMessage('The posting of sheep with new variants (exponential, power, cosine and sawtooth) is disabled in this version.'); -// Exit; -// end; - -// if MainCp.FAngle <> 0 then begin -// showMessage('The posting of sheep with are rotated is disabled in this version.'); -// Exit; -// end; -{ - if SheepDialog.ShowModal = mrOK then - begin - DeleteFile('apophysis.log'); - SetCurrentDir(ExtractFilePath(Application.exename)); - StringList := TStringList.Create; - MultiPartFormDataStream := TmsMultiPartFormDataStream.Create; - ResponseStream := TMemoryStream.Create; - try - LogFile.Active := True; - StringList.Text := FlameToXMLSheep(SheepDialog.cp); - if FileExists('sheep.flame') then DeleteFile('sheep.flame'); - StringList.SaveToFile('sheep.flame'); - HTTP.Request.ContentType := MultiPartFormDataStream.RequestContentType; - MultiPartFormDataStream.AddFormField('type', 'upload'); - MultiPartFormDataStream.AddFile('file', 'sheep.flame', 'text/xml'); - MultiPartFormDataStream.AddFormField('nick', SheepDialog.txtNick.text); - MultiPartFormDataStream.AddFormField('url', SheepDialog.txtURL.text); - MultiPartFormDataStream.AddFormField('pw', SheepPW); //SheepPw - // You must make sure you call this method *before* sending the stream - MultiPartFormDataStream.PrepareStreamForDispatch; - MultiPartFormDataStream.Position := 0; - URL := URLEncode(SheepServer + 'cgi/apophysis.cgi'); - try - HTTP.Post(URL, MultiPartFormDataStream, ResponseStream); - except - on E: Exception do - StatusBar.SimpleText := (E.Message); - end; - ResponseStream.SaveToFile('response.log'); - StringList.LoadFromFile('response.log'); - if Trim(StringList.Text) = 'bad password.' then - ShowMessage('Bad Password'); - finally - MultiPartFormDataStream.Free; - ResponseStream.Free; - StringList.Free; - logFile.Active := False; - end; - end; -} -end; - -{ -procedure TMainForm.HTTPRedirect(Sender: TObject; var dest: string; - var NumRedirect: Integer; var Handled: Boolean; - var VMethod: TIdHTTPMethod); -var - URL: string; -begin - URL := SheepServer + 'cgi/' + dest; - ShellExecute(ValidParentForm(Self).Handle, 'open', PChar(URL), - nil, nil, SW_SHOWNORMAL); - Handled := True; -end; - -procedure TMainForm.HTTPStatus(ASender: TObject; const AStatus: TIdStatus; - const AStatusText: string); -begin - StatusBar.SimpleText := AStatusTExt; -end; -} - procedure TMainForm.ListXmlScannerStartTag(Sender: TObject; TagName: string; Attributes: TAttrList); begin @@ -3821,7 +3732,18 @@ var v: string; begin Tokens := TStringList.Create; - try + try + + if TagName='xformset' then + begin + v := Attributes.Value('enabled'); + if v <> '' then ParseCP.finalXformEnabled := (StrToInt(v) <> 0) + else ParseCP.finalXformEnabled := false; + + inc(activeXformSet); + end + else if TagName='flame' then + begin v := Attributes.value('name'); if v <> '' then Parsecp.name := v else Parsecp.name := 'untitled'; v := Attributes.Value('time'); @@ -3893,10 +3815,10 @@ begin v := Attributes.Value('url'); if Trim(v) = '' then v := SheepUrl; Parsecp.URL := v; - - finally - Tokens.free; end; + finally + Tokens.free; + end; end; procedure ParseCompactcolors(cp: TControlPoint; count: integer; in_data: string); @@ -3948,7 +3870,7 @@ begin if (TagName = 'finalxform') and (FinalXformLoaded) then ShowMessage('ERROR: No xforms allowed after FinalXform!') else begin - if (TagName = 'finalxform') then FinalXformLoaded := true; + if (TagName = 'finalxform') or (activeXformSet > 0) then FinalXformLoaded := true; with ParseCP.xform[nXform] do begin Clear; @@ -3960,13 +3882,16 @@ begin if v <> '' then ParseCP.finalXformEnabled := (StrToInt(v) <> 0) else ParseCP.finalXformEnabled := false; end; + + if activexformset > 0 then density := 0; // tmp... + v := Attributes.Value('color'); if v <> '' then color := StrToFloat(v); v := Attributes.Value('symmetry'); if v <> '' then symmetry := StrToFloat(v); v := Attributes.Value('coefs'); GetTokens(v, tokens); - if Tokens.Count < 6 then ShowMessage('Not enough cooeficients...crash?'); + if Tokens.Count < 6 then ShowMessage('Not enough coefficients...crash?'); c[0][0] := StrToFloat(Tokens[0]); c[0][1] := StrToFloat(Tokens[1]); c[1][0] := StrToFloat(Tokens[2]); @@ -3977,7 +3902,7 @@ begin v := Attributes.Value('post'); if v <> '' then begin GetTokens(v, tokens); - if Tokens.Count < 6 then ShowMessage('Not enough post-cooeficients...crash?'); + if Tokens.Count < 6 then ShowMessage('Not enough post-coefficients...crash?'); p[0][0] := StrToFloat(Tokens[0]); p[0][1] := StrToFloat(Tokens[1]); p[1][0] := StrToFloat(Tokens[2]); diff --git a/2.10/Source/Render.pas b/2.10/Source/Render.pas index 87b0f34..93692c5 100644 --- a/2.10/Source/Render.pas +++ b/2.10/Source/Render.pas @@ -1,6 +1,7 @@ { Flame screensaver Copyright (C) 2002 Ronald Hordijk Apophysis Copyright (C) 2001-2004 Mark Townsend + Apophysis Copyright (C) 2005-2006 Ronald Hordijk, Piotr Boris, 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 diff --git a/2.10/Source/Render64.pas b/2.10/Source/Render64.pas index cad4f47..5acbd5d 100644 --- a/2.10/Source/Render64.pas +++ b/2.10/Source/Render64.pas @@ -1,6 +1,7 @@ { Flame screensaver Copyright (C) 2002 Ronald Hordijk Apophysis Copyright (C) 2001-2004 Mark Townsend + Apophysis Copyright (C) 2005-2006 Ronald Hordijk, Piotr Boris, 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 @@ -21,7 +22,7 @@ unit Render64; interface uses - Windows, Graphics, ImageMaker, + Windows, Forms, Graphics, ImageMaker, Render, xform, Controlpoint; type @@ -54,10 +55,19 @@ type procedure CreateColorMap; procedure CreateCamera; - procedure AddPointsToBuckets(const points: TPointsArray); - procedure AddPointsToBucketsAngle(const points: TPointsArray); - procedure SetPixels; + + private + PropTable: array[0..SUB_BATCH_SIZE] of TXform; + finalXform: TXform; + UseFinalXform: boolean; + + procedure Prepare; + procedure IterateBatch; + procedure IterateBatchAngle; + procedure IterateBatchFX; + procedure IterateBatchAngleFX; + public constructor Create; override; destructor Destroy; override; @@ -71,6 +81,8 @@ type implementation +{$define _ASM_} + uses Math, Sysutils; @@ -200,8 +212,12 @@ begin Bucketwidth := oversample * fcp.Width + 2 * max_gutter_width; BucketSize := BucketWidth * BucketHeight; - if high(buckets) <> (BucketSize - 1) then begin + if high(buckets) <> (BucketSize - 1) then + try SetLength(buckets, BucketSize); + except + on EOutOfMemory do + Application.MessageBox('Error: not enough memory for this render!', 'Apophysis', 48) end; // share the buffer with imagemaker @@ -215,130 +231,64 @@ begin CreateCamera; CreateColorMap; - + fcp.Prepare; end; -/////////////////////////////////////////////////////////////////////////////// -procedure TRenderer64.AddPointsToBuckets(const points: TPointsArray); -var - i: integer; - px, py: double; - Bucket: PBucket; - MapColor: PColorMapColor; -begin - for i := SUB_BATCH_SIZE - 1 downto 0 do begin -// if FStop then Exit; - - px := points[i].x - camX0; - if (px < 0) or (px > camW) then continue; - py := points[i].y - camY0; - if (py < 0) or (py > camH) then continue; - - Bucket := @buckets[Round(bws * px) + Round(bhs * py) * BucketWidth]; - MapColor := @ColorMap[Round(points[i].c * 255)]; - - Inc(Bucket.Red, MapColor.Red); - Inc(Bucket.Green, MapColor.Green); - Inc(Bucket.Blue, MapColor.Blue); - Inc(Bucket.Count); - end; -end; - -/////////////////////////////////////////////////////////////////////////////// -procedure TRenderer64.AddPointsToBucketsAngle(const points: TPointsArray); -var - i: integer; - px, py: double; - Bucket: PBucket; - MapColor: PColorMapColor; -begin - for i := SUB_BATCH_SIZE - 1 downto 0 do begin -// if FStop then Exit; - - px := points[i].x * cosa + points[i].y * sina + rcX; - if (px < 0) or (px > camW) then continue; - py := points[i].y * cosa - points[i].x * sina + rcY; - if (py < 0) or (py > camH) then continue; - - Bucket := @buckets[Round(bws * px) + Round(bhs * py) * BucketWidth]; - MapColor := @ColorMap[Round(points[i].c * 255)]; - - Inc(Bucket.Red, MapColor.Red); - Inc(Bucket.Green, MapColor.Green); - Inc(Bucket.Blue, MapColor.Blue); - Inc(Bucket.Count); - end; -end; - /////////////////////////////////////////////////////////////////////////////// procedure TRenderer64.SetPixels; var i: integer; nsamples: Int64; nrbatches: Integer; - points: TPointsArray; - AddPointsProc: procedure (const points: TPointsArray) of object; + //points: TPointsArray; + IterateBatchProc: procedure of object; begin -// if FileExists('c:\temp\flame.txt') then -// Deletefile('c:\temp\flame.txt'); + Prepare; + Randomize; -// AssignFile(F, 'c:\temp\flame.txt'); -// Rewrite(F); - if FCP.FAngle = 0 then - AddPointsProc := AddPointsToBuckets - else - AddPointsProc := AddPointsToBucketsAngle; - - SetLength(Points, SUB_BATCH_SIZE); + if FCP.FAngle = 0 then begin + if UseFinalXform then + IterateBatchProc := IterateBatchFX + else + IterateBatchProc := IterateBatch; + end + else begin + if UseFinalXform then + IterateBatchProc := IterateBatchAngleFX + else + IterateBatchProc := IterateBatchAngle; + end; nsamples := Round(sample_density * bucketSize / (oversample * oversample)); nrbatches := Round(nsamples / (fcp.nbatches * SUB_BATCH_SIZE)); - Randomize; for i := 0 to nrbatches do begin if FStop then Exit; - if ((i and $F) = 0) then + if ((i and $1F) = 0) then if nrbatches > 0 then Progress(i / nrbatches) else Progress(0); - // generate points -{$IFDEF TESTVARIANT} -// if i > 10 then -// break; - fcp.Testiterate(SUB_BATCH_SIZE, points); -{$ELSE} -{ - case Compatibility of - 0: fcp.iterate_Old(SUB_BATCH_SIZE, points); - 1: fcp.iterateXYC(SUB_BATCH_SIZE, points); - end; -} - fcp.IterateXYC(SUB_BATCH_SIZE, points); -{$ENDIF} - -// for j := SUB_BATCH_SIZE - 1 downto 0 do -// Writeln(f, FloatTostr(points[j].x) + #9 + FloatTostr(points[j].y) + #9 + FloatTostr(points[j].c)); - - AddPointsProc(points); + IterateBatchProc; end; -// closefile(f); - Progress(1); end; /////////////////////////////////////////////////////////////////////////////// procedure TRenderer64.Render; begin + if fcp.NumXForms <= 0 then exit; + FStop := False; FImageMaker.SetCP(FCP); FImageMaker.Init; + InitValues; ClearBuffers; @@ -373,6 +323,272 @@ begin FImageMaker.SaveImage(FileName); end; -/////////////////////////////////////////////////////////////////////////////// +//****************************************************************************** + +procedure TRenderer64.Prepare; +var + i, n: Integer; + propsum: double; + LoopValue: double; + j: integer; + TotValue: double; +begin + totValue := 0; + n := fcp.NumXforms; + assert(n > 0); + + finalXform := fcp.xform[n]; + finalXform.Prepare; + useFinalXform := fcp.FinalXformEnabled and fcp.HasFinalXform; + + for i := 0 to n - 1 do begin + fcp.xform[i].Prepare; + totValue := totValue + fcp.xform[i].density; + end; + + LoopValue := 0; + for i := 0 to PROP_TABLE_SIZE-1 do begin + propsum := 0; + j := -1; + repeat + inc(j); + propsum := propsum + fcp.xform[j].density; + until (propsum > LoopValue) or (j = n - 1); + PropTable[i] := fcp.xform[j]; + LoopValue := LoopValue + TotValue / PROP_TABLE_SIZE; + end; +end; + +procedure TRenderer64.IterateBatch; +var + i: integer; + px, py: double; + Bucket: PBucket; + MapColor: PColorMapColor; + + p: TCPPoint; +begin +{$ifndef _ASM_} + p.x := 2 * random - 1; + p.y := 2 * random - 1; + p.c := random; +{$else} +asm + fld1 + call System.@RandExt + fadd st, st + fsub st, st(1) + fstp qword ptr [p.x] + call System.@RandExt + fadd st, st + fsubrp st(1), st + fstp qword ptr [p.y] + call System.@RandExt + fstp qword ptr [p.c] +end; +{$endif} + + try + for i := 0 to FUSE do + PropTable[Random(PROP_TABLE_SIZE)].NextPoint(p); + + for i := 0 to SUB_BATCH_SIZE-1 do begin + PropTable[Random(PROP_TABLE_SIZE)].NextPoint(p); + + px := p.x - camX0; + if (px < 0) or (px > camW) then continue; + py := p.y - camY0; + if (py < 0) or (py > camH) then continue; + + Bucket := @buckets[Round(bws * px) + Round(bhs * py) * BucketWidth]; + MapColor := @ColorMap[Round(p.c * 255)]; + + Inc(Bucket.Red, MapColor.Red); + Inc(Bucket.Green, MapColor.Green); + Inc(Bucket.Blue, MapColor.Blue); + Inc(Bucket.Count); + end; + + except + on EMathError do begin + exit; + end; + end; +end; + +procedure TRenderer64.IterateBatchAngle; +var + i: integer; + px, py: double; + Bucket: PBucket; + MapColor: PColorMapColor; + + p: TCPPoint; +begin +{$ifndef _ASM_} + p.x := 2 * random - 1; + p.y := 2 * random - 1; + p.c := random; +{$else} +asm + fld1 + call System.@RandExt + fadd st, st + fsub st, st(1) + fstp qword ptr [p.x] + call System.@RandExt + fadd st, st + fsubrp st(1), st + fstp qword ptr [p.y] + call System.@RandExt + fstp qword ptr [p.c] +end; +{$endif} + + try + for i := 0 to FUSE do + PropTable[Random(PROP_TABLE_SIZE)].NextPoint(p); + + for i := 0 to SUB_BATCH_SIZE-1 do begin + PropTable[Random(PROP_TABLE_SIZE)].NextPoint(p); + + px := p.x * cosa + p.y * sina + rcX; + if (px < 0) or (px > camW) then continue; + py := p.y * cosa - p.x * sina + rcY; + if (py < 0) or (py > camH) then continue; + + Bucket := @buckets[Round(bws * px) + Round(bhs * py) * BucketWidth]; + MapColor := @ColorMap[Round(p.c * 255)]; + + Inc(Bucket.Red, MapColor.Red); + Inc(Bucket.Green, MapColor.Green); + Inc(Bucket.Blue, MapColor.Blue); + Inc(Bucket.Count); + end; + + except + on EMathError do begin + exit; + end; + end; +end; + + +procedure TRenderer64.IterateBatchFX; +var + i: integer; + px, py: double; + Bucket: PBucket; + MapColor: PColorMapColor; + + p, q: TCPPoint; +begin +{$ifndef _ASM_} + p.x := 2 * random - 1; + p.y := 2 * random - 1; + p.c := random; +{$else} +asm + fld1 + call System.@RandExt + fadd st, st + fsub st, st(1) + fstp qword ptr [p.x] + call System.@RandExt + fadd st, st + fsubrp st(1), st + fstp qword ptr [p.y] + call System.@RandExt + fstp qword ptr [p.c] +end; +{$endif} + + try + for i := 0 to FUSE do + PropTable[Random(PROP_TABLE_SIZE)].NextPoint(p); + + for i := 0 to SUB_BATCH_SIZE-1 do begin + PropTable[Random(PROP_TABLE_SIZE)].NextPoint(p); + finalXform.NextPointTo(p, q); + + px := q.x - camX0; + if (px < 0) or (px > camW) then continue; + py := q.y - camY0; + if (py < 0) or (py > camH) then continue; + + Bucket := @buckets[Round(bws * px) + Round(bhs * py) * BucketWidth]; + MapColor := @ColorMap[Round(q.c * 255)]; + + Inc(Bucket.Red, MapColor.Red); + Inc(Bucket.Green, MapColor.Green); + Inc(Bucket.Blue, MapColor.Blue); + Inc(Bucket.Count); + end; + + except + on EMathError do begin + exit; + end; + end; +end; + +procedure TRenderer64.IterateBatchAngleFX; +var + i: integer; + px, py: double; + Bucket: PBucket; + MapColor: PColorMapColor; + + p, q: TCPPoint; +begin +{$ifndef _ASM_} + p.x := 2 * random - 1; + p.y := 2 * random - 1; + p.c := random; +{$else} +asm + fld1 + call System.@RandExt + fadd st, st + fsub st, st(1) + fstp qword ptr [p.x] + call System.@RandExt + fadd st, st + fsubrp st(1), st + fstp qword ptr [p.y] + call System.@RandExt + fstp qword ptr [p.c] +end; +{$endif} + + try + for i := 0 to FUSE do + PropTable[Random(PROP_TABLE_SIZE)].NextPoint(p); + + for i := 0 to SUB_BATCH_SIZE-1 do begin + PropTable[Random(PROP_TABLE_SIZE)].NextPoint(p); + finalXform.NextPointTo(p, q); + + px := q.x * cosa + q.y * sina + rcX; + if (px < 0) or (px > camW) then continue; + py := q.y * cosa - q.x * sina + rcY; + if (py < 0) or (py > camH) then continue; + + Bucket := @buckets[Round(bws * px) + Round(bhs * py) * BucketWidth]; + MapColor := @ColorMap[Round(q.c * 255)]; + + Inc(Bucket.Red, MapColor.Red); + Inc(Bucket.Green, MapColor.Green); + Inc(Bucket.Blue, MapColor.Blue); + Inc(Bucket.Count); + end; + + except + on EMathError do begin + exit; + end; + end; +end; + end. diff --git a/2.10/Source/Render64MT.pas b/2.10/Source/Render64MT.pas index 8a33cf7..1a70d89 100644 --- a/2.10/Source/Render64MT.pas +++ b/2.10/Source/Render64MT.pas @@ -1,6 +1,7 @@ { Flame screensaver Copyright (C) 2002 Ronald Hordijk Apophysis Copyright (C) 2001-2004 Mark Townsend + Apophysis Copyright (C) 2005-2006 Ronald Hordijk, Piotr Boris, 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 diff --git a/2.10/Source/RenderMM.pas b/2.10/Source/RenderMM.pas index dfc5f20..a07e070 100644 --- a/2.10/Source/RenderMM.pas +++ b/2.10/Source/RenderMM.pas @@ -1,6 +1,7 @@ { Flame screensaver Copyright (C) 2002 Ronald Hordijk Apophysis Copyright (C) 2001-2004 Mark Townsend + Apophysis Copyright (C) 2005-2006 Ronald Hordijk, Piotr Boris, 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 diff --git a/2.10/Source/RenderMM_MT.pas b/2.10/Source/RenderMM_MT.pas index 2b61cfe..8c52152 100644 --- a/2.10/Source/RenderMM_MT.pas +++ b/2.10/Source/RenderMM_MT.pas @@ -1,6 +1,7 @@ { Flame screensaver Copyright (C) 2002 Ronald Hordijk Apophysis Copyright (C) 2001-2004 Mark Townsend + Apophysis Copyright (C) 2005-2006 Ronald Hordijk, Piotr Boris, 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 diff --git a/2.10/Source/RenderThread.pas b/2.10/Source/RenderThread.pas index 4bd6701..71e7f65 100644 --- a/2.10/Source/RenderThread.pas +++ b/2.10/Source/RenderThread.pas @@ -1,6 +1,7 @@ { Flame screensaver Copyright (C) 2002 Ronald Hordijk Apophysis Copyright (C) 2001-2004 Mark Townsend + Apophysis Copyright (C) 2005-2006 Ronald Hordijk, Piotr Boris, 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 diff --git a/2.10/Source/RndFlame.pas b/2.10/Source/RndFlame.pas index 323dfe1..54c0da8 100644 --- a/2.10/Source/RndFlame.pas +++ b/2.10/Source/RndFlame.pas @@ -60,21 +60,6 @@ begin end; end; -/////////////////////////////////////////////////////////////////////////////// -function NumXForms(const cp: TControlPoint): integer; -var - i: integer; -begin - Result := NXFORMS; - for i := 0 to NXFORMS - 1 do begin - if cp.xform[i].density = 0 then - begin - Result := i; - Break; - end; - end; -end; - /////////////////////////////////////////////////////////////////////////////// procedure RandomVariation(cp: TControlPoint); { Randomise variation parameters } @@ -90,7 +75,7 @@ begin VarPossible := VarPossible or Variations[j]; end; - for i := 0 to NumXForms(cp) - 1 do begin + for i := 0 to cp.NumXForms - 1 do begin for j := 0 to NRVAR - 1 do cp.xform[i].vars[j] := 0; @@ -125,7 +110,7 @@ begin if Variation = vRandom then begin RandomVariation(cp); end else - for i := 0 to NumXForms(cp) - 1 do begin + for i := 0 to cp.NumXForms - 1 do begin for j := 0 to NRVAR - 1 do cp.xform[i].vars[j] := 0; cp.xform[i].vars[integer(Variation)] := 1; @@ -222,7 +207,7 @@ procedure EqualizeWeights(var cp: TControlPoint); var t, i: integer; begin - t := NumXForms(cp); + t := cp.NumXForms; for i := 0 to t - 1 do cp.xform[i].density := 1.0 / t; end; @@ -234,12 +219,12 @@ var td: double; begin td := 0.0; - for i := 0 to NumXForms(cp) - 1 do + for i := 0 to cp.NumXForms - 1 do td := td + cp.xform[i].Density; if (td < 0.001) then EqualizeWeights(cp) else - for i := 0 to NumXForms(cp) - 1 do + for i := 0 to cp.NumXForms - 1 do cp.xform[i].Density := cp.xform[i].Density / td; end; diff --git a/2.10/Source/XForm.pas b/2.10/Source/XForm.pas index 0f70552..a6e820e 100644 --- a/2.10/Source/XForm.pas +++ b/2.10/Source/XForm.pas @@ -3,10 +3,7 @@ unit XForm; interface uses - XFormMan, baseVariation; - -type - TCalcMethod = procedure of object; + XFormMan, BaseVariation; type TCPpoint = record @@ -47,8 +44,8 @@ type private FNrFunctions: Integer; - FFunctionList: array of TCalcMethod; - FCalcFunctionList: array[0..64] of TCalcMethod; + FFunctionList: array of TCalcFunction; + FCalcFunctionList: array[0..64] of TCalcFunction; FTx, FTy: double; FPx, FPy: double; @@ -140,7 +137,7 @@ uses SysUtils, Math; const - EPS: double = 1E-6; + EPS: double = 1E-300; procedure SinCos(const Theta: double; var Sin, Cos: double); // to avoid using 'extended' type asm @@ -215,11 +212,13 @@ begin FRegVariations[i].FTY := @FTY; FRegVariations[i].vvar := vars[i + NRLOCVAR]; - FRegVariations[i].prepare; + FRegVariations[i].Prepare; + FRegVariations[i].GetCalcFunction(FFunctionList[NRLOCVAR + i]); end; - CalculateAngle := (vars[5] <> 0.0) or (vars[6] <> 0.0) or (vars[7] <> 0.0) or (vars[8] <> 0.0) or - (vars[12] <> 0.0) or (vars[13] <> 0.0) or (vars[21] <> 0.0) or (vars[22] <> 0.0); + CalculateAngle := (vars[5] <> 0.0) or (vars[6] <> 0.0) or (vars[7] <> 0.0) or + (vars[8] <> 0.0) or (vars[12] <> 0.0) or (vars[13] <> 0.0) or + (vars[21] <> 0.0) or (vars[22] <> 0.0) or (vars[27] <> 0.0); // CalculateLength := False; CalculateSinCos := (vars[9] <> 0.0) or (vars[11] <> 0.0) or (vars[19] <> 0.0) or (vars[21] <> 0.0); @@ -447,7 +446,7 @@ procedure TXForm.Spherical; var r: double; begin - r := vars[2] / (sqr(FTx) + sqr(FTy) + 1E-6); + r := vars[2] / (sqr(FTx) + sqr(FTy) + EPS); FPx := FPx + FTx * r; FPy := FPy + FTy * r; {$else} @@ -1513,9 +1512,9 @@ var r, sinr, cosr: double; begin SinCos(random * 2*pi, sinr, cosr); - r := vars[27]*random; - FPx := FPx + FTx*r*cosr; - FPy := FPy + FTy*r*sinr; + r := vars[27] * random; + FPx := FPx + FTx * r * cosr; + FPy := FPy + FTy * r * sinr; {$else} asm mov edx, [ebx + vars] diff --git a/2.10/Source/formPostProcess.pas b/2.10/Source/formPostProcess.pas index 87746ca..c962f0b 100644 --- a/2.10/Source/formPostProcess.pas +++ b/2.10/Source/formPostProcess.pas @@ -378,7 +378,7 @@ begin if pnlDragged then begin - UpdateFlame; + //UpdateFlame; pnlDragged := False; end; end;