{ 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 Global; interface uses SysUtils, Classes, SyncObjs, Controls, Graphics, Math, cmap, ControlPoint, Xform; type EFormatInvalid = class(Exception); { Weight manipulation } { Triangle transformations } function triangle_area(t: TTriangle): double; function transform_affine(const t: TTriangle; const Triangles: TTriangles): boolean; function line_dist(x, y, x1, y1, x2, y2: double): double; function dist(x1, y1, x2, y2: double): double; { Parsing functions } function GetVal(token: string): string; function ReplaceTabs(str: string): string; { Palette and gradient functions } //function GetGradient(FileName, Entry: string): string; { Misc } function det(a, b, c, d: double): double; function solve3(x1, x2, x1h, y1, y2, y1h, z1, z2, z1h: double; var a, b, e: double): double; const APP_NAME: string = 'Apophysis 2.0'; prefilter_white: integer = 1024; eps: double = 1E-10; White_level = 200; clyellow1 = TColor($17FCFF); clplum2 = TColor($ECA9E6); clSlateGray = TColor($837365); FT_BMP = 1; FT_PNG = 2; FT_JPG = 3; const crEditArrow = 20; crEditMove = 21; crEditRotate = 22; crEditScale = 23; var MainSeed: integer; MainTriangles: TTriangles; Transforms: integer; // Count of Tranforms EnableFinalXform: boolean; AppPath: string; // Path of applicatio file OpenFile: string; // Name of currently open file CanDrawOnResize: boolean; PreserveWeights: boolean; { UPR Options } UPRSampleDensity: integer; UPRFilterRadius: double; UPROversample: integer; UPRAdjustDensity: boolean; UPRColoringIdent: string; UPRColoringFile: string; UPRFormulaIdent: string; UPRFormulaFile: string; UPRWidth: Integer; UPRHeight: Integer; ImageFolder: string; UPRPath: string; // Name and folder of last UPR file cmap_index: integer; // Index to current gradient Variation: TVariation; // Current variation NumTries, TryLength: integer; // Settings for smooth palette SmoothPaletteFile: string; { Editor } UseFlameBackground, UseTransformColors: boolean; HelpersEnabled: boolean; EditorBkgColor, ReferenceTriangleColor: integer; GridColor1, GridColor2, HelpersColor: integer; ExtEditEnabled, TransformAxisLock: boolean; DoubleClickSetVars: boolean; ShowAllXforms: boolean; { Display } defSampleDensity, defPreviewDensity: Double; defGamma, defBrightness, defVibrancy, defFilterRadius: Double; defOversample: integer; { Render } renderDensity, renderFilterRadius: double; renderOversample, renderWidth, renderHeight: integer; renderBitsPerSample: integer; renderPath: string; JPEGQuality: integer; renderFileFormat: integer; InternalBitsPerSample: integer; NrTreads: Integer; UseNrThreads: integer; PNGTransparency: integer; ShowTransparency: boolean; MainPreviewScale: double; ExtendMainPreview: boolean; { Defaults } ConfirmDelete: boolean; // Flag confirmation of entry deletion OldPaletteFormat: boolean; SavePath, SmoothPalettePath: string; RandomPrefix, RandomDate: string; RandomIndex: integer; FlameFile, GradientFile, GradientEntry, FlameEntry: string; ParamFolder: string; prevLowQuality, prevMediumQuality, prevHighQuality: double; defSmoothPaletteFile: string; BrowserPath: string; // Stored path of browser open dialog EditPrevQual, MutatePrevQual, AdjustPrevQual: Integer; randMinTransforms, randMaxTransforms: integer; mutantMinTransforms, mutantMaxTransforms: integer; KeepBackground: boolean; randGradient: Integer; randGradientFile: string; defFlameFile: string; PlaySoundOnRenderComplete: boolean; RenderCompleteSoundFile: string; SaveIncompleteRenders: boolean; ShowRenderStats: boolean; SymmetryType: integer; SymmetryOrder: integer; SymmetryNVars: integer; Variations: array[0..63] of boolean; VariationOptions: int64; MainForm_RotationMode: integer; PreserveQuality: boolean; { For random gradients } MinNodes, MaxNodes, MinHue, MaxHue, MinSat, MaxSat, MinLum, MaxLum: integer; ReferenceMode: integer;//FixedReference: boolean; BatchSize: Integer; Compatibility: integer; //0 = original, 1 = Drave's Favorites: TStringList; Script: string; ScriptPath: string; SheepServer, SheepNick, SheepURL, SheepPW, flam3Path: string; ExportBatches, ExportOversample, ExportWidth, ExportHeight, ExportFileFormat: Integer; ExportFilter, ExportDensity: Double; ExportEstimator, ExportEstimatorMin, ExportEstimatorCurve: double; ExportJitters: integer; ExportGammaTreshold: double; OpenFileType: TFileType; // ResizeOnLoad: Boolean; ShowProgress: Boolean; defLibrary: string; LimitVibrancy: Boolean; DefaultPalette: TColorMap; function Round6(x: double): double; implementation { IFS } function det(a, b, c, d: double): double; begin Result := (a * d - b * c); end; function Round6(x: double): double; // Really ugly, but it works begin // --Z-- this is ridiculous: // Result := StrToFloat(Format('%.6f', [x])); // and yes, this is REALLY ugly :-\ Result := RoundTo(x, -6); end; function solve3(x1, x2, x1h, y1, y2, y1h, z1, z2, z1h: double; var a, b, e: double): double; var det1: double; begin det1 := x1 * det(y2, 1.0, z2, 1.0) - x2 * det(y1, 1.0, z1, 1.0) + 1 * det(y1, y2, z1, z2); if (det1 = 0.0) then begin Result := det1; EXIT; end else begin a := (x1h * det(y2, 1.0, z2, 1.0) - x2 * det(y1h, 1.0, z1h, 1.0) + 1 * det(y1h, y2, z1h, z2)) / det1; b := (x1 * det(y1h, 1.0, z1h, 1.0) - x1h * det(y1, 1.0, z1, 1.0) + 1 * det(y1, y1h, z1, z1h)) / det1; e := (x1 * det(y2, y1h, z2, z1h) - x2 * det(y1, y1h, z1, z1h) + x1h * det(y1, y2, z1, z2)) / det1; a := Round6(a); b := Round6(b); e := Round6(e); Result := det1; end; end; function dist(x1, y1, x2, y2: double): double; //var // d2: double; begin (* { From FDesign source { float pt_pt_distance(float x1, float y1, float x2, float y2) } d2 := (x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2); if (d2 = 0.0) then begin Result := 0.0; exit; end else Result := sqrt(d2); *) // --Z-- This is just amazing... :-\ // Someone needed an 'FDesign source' - to compute distance between two points??!? Result := Hypot(x2-x1, y2-y1); end; function line_dist(x, y, x1, y1, x2, y2: double): double; var a, b, e, c: double; begin if ((x = x1) and (y = y1)) then a := 0.0 else a := sqrt((x - x1) * (x - x1) + (y - y1) * (y - y1)); if ((x = x2) and (y = y2)) then b := 0.0 else b := sqrt((x - x2) * (x - x2) + (y - y2) * (y - y2)); if ((x1 = x2) and (y1 = y2)) then e := 0.0 else e := sqrt((x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2)); if ((a * a + e * e) < (b * b)) then Result := a else if ((b * b + e * e) < (a * a)) then Result := b else if (e <> 0.0) then begin c := (b * b - a * a - e * e) / (-2 * e); if ((a * a - c * c) < 0.0) then Result := 0.0 else Result := sqrt(a * a - c * c); end else Result := a; end; function transform_affine(const t: TTriangle; const Triangles: TTriangles): boolean; var ra, rb, rc, a, b, c: double; begin Result := True; ra := dist(Triangles[-1].y[0], Triangles[-1].x[0], Triangles[-1].y[1], Triangles[-1].x[1]); rb := dist(Triangles[-1].y[1], Triangles[-1].x[1], Triangles[-1].y[2], Triangles[-1].x[2]); rc := dist(Triangles[-1].y[2], Triangles[-1].x[2], Triangles[-1].y[0], Triangles[-1].x[0]); a := dist(t.y[0], t.x[0], t.y[1], t.x[1]); b := dist(t.y[1], t.x[1], t.y[2], t.x[2]); c := dist(t.y[2], t.x[2], t.y[0], t.x[0]); if (a > ra) then Result := False else if (b > rb) then Result := False else if (c > rc) then Result := False else if ((a = ra) and (b = rb) and (c = rc)) then Result := False; end; function triangle_area(t: TTriangle): double; var base, height: double; begin try base := dist(t.x[0], t.y[0], t.x[1], t.y[1]); height := line_dist(t.x[2], t.y[2], t.x[1], t.y[1], t.x[0], t.y[0]); if (base < 1.0) then Result := height else if (height < 1.0) then Result := base else Result := 0.5 * base * height; except on E: EMathError do Result := 0; end; end; { Parse } function GetVal(token: string): string; var p: integer; begin p := Pos('=', token); Delete(Token, 1, p); Result := Token; end; function ReplaceTabs(str: string): string; {Changes tab characters in a string to spaces} var i: integer; begin for i := 1 to Length(str) do begin if str[i] = #9 then begin Delete(str, i, 1); Insert(#32, str, i); end; end; Result := str; end; (* { Palette and gradient functions } function RGBToColor(Pal: TMapPalette; index: integer): Tcolor; begin { Converts the RGB values from a palette index to the TColor type ... could maybe change it to SHLs } Result := (Pal.Blue[index] * 65536) + (Pal.Green[index] * 256) + Pal.Red[index]; end; procedure rgb2hsv(const rgb: array of double; out hsv: array of double); var maxval, minval: double; del: double; begin Maxval := Max(rgb[0], Max(rgb[1], rgb[2])); Minval := Min(rgb[0], Min(rgb[1], rgb[2])); hsv[2] := maxval; // v if (Maxval > 0) and (maxval <> minval) then begin del := maxval - minval; hsv[1] := del / Maxval; //s hsv[0] := 0; if (rgb[0] > rgb[1]) and (rgb[0] > rgb[2]) then begin hsv[0] := (rgb[1] - rgb[2]) / del; end else if (rgb[1] > rgb[2]) then begin hsv[0] := 2 + (rgb[2] - rgb[0]) / del; end else begin hsv[0] := 4 + (rgb[0] - rgb[1]) / del; end; if hsv[0] < 0 then hsv[0] := hsv[0] + 6; end else begin hsv[0] := 0; hsv[1] := 0; end; end; procedure hsv2rgb(const hsv: array of double; out rgb: array of double); var j: integer; f, p, q, t, v: double; begin j := floor(hsv[0]); f := hsv[0] - j; v := hsv[2]; p := hsv[2] * (1 - hsv[1]); q := hsv[2] * (1 - hsv[1] * f); t := hsv[2] * (1 - hsv[1] * (1 - f)); case j of 0: begin rgb[0] := v; rgb[1] := t; rgb[2] := p; end; 1: begin rgb[0] := q; rgb[1] := v; rgb[2] := p; end; 2: begin rgb[0] := p; rgb[1] := v; rgb[2] := t; end; 3: begin rgb[0] := p; rgb[1] := q; rgb[2] := v; end; 4: begin rgb[0] := t; rgb[1] := p; rgb[2] := v; end; 5: begin rgb[0] := v; rgb[1] := p; rgb[2] := t; end; end; end; function GetGradient(FileName, Entry: string): string; var FileStrings: TStringList; GradStrings: TStringList; i: integer; begin FileStrings := TStringList.Create; GradStrings := TStringList.Create; try try FileStrings.LoadFromFile(FileName); for i := 0 to FileStrings.count - 1 do if Pos(Entry + ' ', Trim(FileStrings[i])) = 1 then break; GradStrings.Add(FileStrings[i]); repeat inc(i); GradStrings.Add(FileStrings[i]); until Pos('}', FileStrings[i]) <> 0; GetGradient := GradStrings.Text; except on exception do Result := ''; end; finally GradStrings.Free; FileStrings.Free; end; end; *) end.