| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773 | //// The graphics engine GLXEngine. The unit of GXScene for Delphi//unit GXS.Mesh;(*  Raw Mesh support.  This unit is for simple meshes and legacy support, VectorFileObjects  implements more efficient (though more complex) mesh tools.*)interface{$I Stage.Defines.inc}uses  Winapi.OpenGL,  Winapi.OpenGLext,  System.Classes,  System.SysUtils,  Stage.OpenGL4, // wglAllocateMemoryNV  Stage.VectorTypes,  Stage.Strings,  Stage.VectorGeometry,  GXS.BaseClasses,  GXS.Color,  GXS.State,  GXS.Context,  GXS.Scene,  GXS.RenderContextInfo;type  TMeshMode = (mmTriangleStrip, mmTriangleFan, mmTriangles, mmQuadStrip,    mmQuads, mmPolygon);  TVertexMode = (vmV, vmVN, vmVNC, vmVNCT, vmVNT, vmVT);const  cMeshModeToGLEnum: array[Low(TMeshMode)..High(TMeshMode)    ] of GLEnum = (GL_TRIANGLE_STRIP, GL_TRIANGLE_FAN, GL_TRIANGLES,    GL_QUAD_STRIP, GL_QUADS, GL_POLYGON);  cVertexModeToGLEnum: array[Low(TVertexMode)..High(TVertexMode)    ] of GLEnum = (GL_V3F, GL_N3F_V3F, GL_C4F_N3F_V3F, GL_T2F_C4F_N3F_V3F,    GL_T2F_N3F_V3F, GL_T2F_V3F);type  TgxVertexData = packed record    textCoord: TTexPoint;    color: TVector4f;    normal: TAffineVector;    coord: TVertex;  end;  PgxVertexData = ^TgxVertexData;  TgxVertexDataArray = array[0..(MAXINT shr 6)] of TgxVertexData;  PVKVertexDataArray = ^TgxVertexDataArray;  (* Stores an interlaced vertex list for direct use in OpenGL.    Locking (hardware passthrough) is supported, see "Locked" property for details. *)  TgxVertexList = class(TgxUpdateAbleObject)  private    FValues: PVKVertexDataArray;    FCount: Integer;    FCapacity, FGrowth: Integer;    FLockedOldValues: PVKVertexDataArray;  protected    procedure SetCapacity(const val: Integer);    procedure SetGrowth(const val: Integer);    procedure Grow;    procedure SetVertices(index: Integer; const val: TgxVertexData);    function GetVertices(index: Integer): TgxVertexData;    procedure SetVertexCoord(index: Integer; const val: TAffineVector);    function GetVertexCoord(index: Integer): TAffineVector;    procedure SetVertexNormal(index: Integer; const val: TAffineVector);    function GetVertexNormal(index: Integer): TAffineVector;    procedure SetVertexTexCoord(index: Integer; const val: TTexPoint);    function GetVertexTexCoord(index: Integer): TTexPoint;    procedure SetVertexColor(index: Integer; const val: TVector4f);    function GetVertexColor(index: Integer): TVector4f;    function GetFirstEntry: PGLFloat;    function GetFirstColor: PGLFloat;    function GetFirstNormal: PGLFloat;    function GetFirstVertex: PGLFloat;    function GetFirstTexPoint: PGLFloat;    function GetLocked: Boolean;    procedure SetLocked(val: Boolean);  public    constructor Create(AOwner: TPersistent); override;    destructor Destroy; override;    function CreateInterpolatedCoords(list2: TgxVertexList; lerpFactor: Single): TgxVertexList;    // Adds a vertex to the list, fastest method.    procedure AddVertex(const vertexData: TgxVertexData); overload;    // Adds a vertex to the list, fastest method for adding a triangle.    procedure AddVertex3(const vd1, vd2, vd3: TgxVertexData); overload;    { Adds a vertex to the list.      Use the NullVector, NullHmgVector or NullTexPoint constants for      params you don't want to set. }    procedure AddVertex(const aVertex: TVertex; const aNormal: TAffineVector;      const aColor: TgxColorVector; const aTexPoint: TTexPoint); overload;    // Adds a vertex to the list, no texturing version.    procedure AddVertex(const vertex: TVertex; const normal: TAffineVector;      const color: TgxColorVector); overload;    // Adds a vertex to the list, no texturing, not color version.    procedure AddVertex(const vertex: TVertex;      const normal: TAffineVector); overload;    // Duplicates the vertex of given index and adds it at the end of the list.    procedure DuplicateVertex(index: Integer);    procedure Assign(Source: TPersistent); override;    procedure Clear;    property Vertices[index: Integer]: TgxVertexData read GetVertices    write SetVertices; default;    property VertexCoord[index: Integer]: TAffineVector read GetVertexCoord    write SetVertexCoord;    property VertexNormal[index: Integer]: TAffineVector read GetVertexNormal    write SetVertexNormal;    property VertexTexCoord[index: Integer]: TTexPoint read GetVertexTexCoord    write SetVertexTexCoord;    property VertexColor[index: Integer]: TVector4f read GetVertexColor    write SetVertexColor;    property Count: Integer read FCount;    { Capacity of the list (nb of vertex).      Use this to allocate memory quickly before calling AddVertex. }    property Capacity: Integer read FCapacity write SetCapacity;    { Vertex capacity that will be added each time the list needs to grow.      default value is 256 (chunks of approx 13 kb). }    property Growth: Integer read FGrowth write SetGrowth;    { Calculates the sum of all vertex coords }    function SumVertexCoords: TAffineVector;    { Calculates the extents of the vertice coords. }    procedure GetExtents(var min, max: TAffineVector);    { Normalizes all normals. }    procedure NormalizeNormals;    { Translate all coords by given vector }    procedure Translate(const v: TAffineVector);    procedure DefineOpenGLArrays;    property FirstColor: PGLFloat read GetFirstColor;    property FirstEntry: PGLFloat read GetFirstEntry;    property FirstNormal: PGLFloat read GetFirstNormal;    property FirstVertex: PGLFloat read GetFirstVertex;    property FirstTexPoint: PGLFloat read GetFirstTexPoint;    { Locking state of the vertex list.      You can "Lock" a list to increase rendering performance on some      OpenGL implementations (NVidia's). A Locked list size shouldn't be      changed and calculations should be avoided.      Performance can only be gained from a lock for osDirectDraw object,      ie. meshes that are updated for each frame (the default build list      mode is faster on static meshes).      Be aware that the "Locked" state enforcement is not very strict      to avoid performance hits, and GXScene may not always notify you      that you're doing things you shouldn't on a locked list! }    property Locked: Boolean read GetLocked write SetLocked;    procedure EnterLockSection;    procedure LeaveLockSection;  end;  // TgxMesh  //  { Basic mesh object.     Each mesh holds a set of vertices and a Mode value defines how they make    up the mesh (triangles, strips...) }  TgxMesh = class(TgxSceneObject)  private        FVertices: TgxVertexList;    FMode: TMeshMode;    FVertexMode: TVertexMode;    FAxisAlignedDimensionsCache: TVector4f;  protected        procedure SetMode(AValue: TMeshMode);    procedure SetVertices(AValue: TgxVertexList);    procedure SetVertexMode(AValue: TVertexMode);    procedure VerticesChanged(Sender: TObject);  public        constructor Create(AOwner: TComponent); override;    destructor Destroy; override;    procedure Assign(Source: TPersistent); override;    procedure BuildList(var rci: TgxRenderContextInfo); override;    procedure CalcNormals(Frontface: TgxFaceWinding);    property Vertices: TgxVertexList read FVertices write SetVertices;    function AxisAlignedDimensionsUnscaled: TVector4f; override;    procedure StructureChanged; override;  published        property Mode: TMeshMode read FMode write SetMode;    property VertexMode: TVertexMode read FVertexMode write SetVertexMode      default vmVNCT;  end;// ------------------------------------------------------------------implementation// ------------------------------------------------------------------// ----------------- TgxVertexList ------------------------------------------------constructor TgxVertexList.Create(AOwner: TPersistent);begin  inherited;  FValues := nil;  FCount := 0;  FCapacity := 0;  FGrowth := 256;end;destructor TgxVertexList.Destroy;begin  Locked := False;  FreeMem(FValues);  inherited;end;function TgxVertexList.CreateInterpolatedCoords(list2: TgxVertexList;  lerpFactor: Single): TgxVertexList;var  i: Integer;begin  Assert(Count = list2.Count);  Result := TgxVertexList.Create(nil);  Result.Capacity := Count;  Move(FValues[0], Result.FValues[0], Count * SizeOf(TgxVertexData));  // interpolate vertices  for i := 0 to Count - 1 do    VectorLerp(FValues^[i].coord, list2.FValues^[i].coord, lerpFactor,      Result.FValues^[i].coord);end;procedure TgxVertexList.SetCapacity(const val: Integer);begin  Assert(not Locked, 'Cannot change locked list capacity !');  FCapacity := val;  if FCapacity < FCount then    FCapacity := FCount;  ReallocMem(FValues, FCapacity * SizeOf(TgxVertexData));end;procedure TgxVertexList.SetGrowth(const val: Integer);begin  if val > 16 then    FGrowth := val  else    FGrowth := 16;end;procedure TgxVertexList.Grow;begin  Assert(not Locked, 'Cannot add to a locked list !');  FCapacity := FCapacity + FGrowth;  ReallocMem(FValues, FCapacity * SizeOf(TgxVertexData));end;function TgxVertexList.GetFirstColor: PGLFloat;begin  Result := @(FValues^[0].color);end;// GetFirstEntry//function TgxVertexList.GetFirstEntry: PGLFloat;begin  Result := Pointer(FValues);end;function TgxVertexList.GetFirstNormal: PGLFloat;begin  Result := @(FValues^[0].normal);end;function TgxVertexList.GetFirstVertex: PGLFloat;begin  Result := @(FValues^[0].coord);end;function TgxVertexList.GetFirstTexPoint: PGLFloat;begin  Result := @(FValues^[0].textCoord);end;function TgxVertexList.GetLocked: Boolean;begin  Result := Assigned(FLockedOldValues);end;procedure TgxVertexList.SetLocked(val: Boolean);var  size: Integer;begin  if val <> Locked then  begin    if (CurrentContext <> nil) then    begin      size := FCount * SizeOf(TgxVertexData);      if val then      begin        // Lock        FLockedOldValues := FValues;        {$IFDEF MSWINDOWS}        FValues := wglAllocateMemoryNV(size, 0, 0, 0.5);        {$ENDIF}        {$IFDEF LINUX}        FValues := glxAllocateMemoryNV(size, 0, 0, 0.5);        {$ENDIF}        if FValues = nil then        begin          FValues := FLockedOldValues;          FLockedOldValues := nil;        end        else          Move(FLockedOldValues^, FValues^, size);      end      else      begin        // Unlock        wglFreeMemoryNV(nil);        FValues := FLockedOldValues;        FLockedOldValues := nil;      end;    end;  end;end;procedure TgxVertexList.EnterLockSection;begin  if Locked then  begin    glVertexArrayRangeNV(FCount * SizeOf(TgxVertexData), FValues);    glEnableClientState(GL_VERTEX_ARRAY_RANGE_NV);  end;end;procedure TgxVertexList.LeaveLockSection;begin  if Locked then  begin    glDisableClientState(GL_VERTEX_ARRAY_RANGE_NV);    glFlushVertexArrayRangeNV;  end;end;procedure TgxVertexList.SetVertices(index: Integer; const val: TgxVertexData);begin  Assert(Cardinal(index) < Cardinal(Count));  FValues^[index] := val;  NotifyChange(Self);end;function TgxVertexList.GetVertices(index: Integer): TgxVertexData;begin  Assert(Cardinal(index) < Cardinal(Count));  Result := FValues^[index];end;procedure TgxVertexList.SetVertexCoord(index: Integer; const val: TAffineVector);begin  FValues^[index].coord := val;  NotifyChange(Self);end;function TgxVertexList.GetVertexCoord(index: Integer): TAffineVector;begin  Result := FValues^[index].coord;end;procedure TgxVertexList.SetVertexNormal(index: Integer; const val: TAffineVector);begin  FValues^[index].normal := val;  NotifyChange(Self);end;function TgxVertexList.GetVertexNormal(index: Integer): TAffineVector;begin  Result := FValues^[index].normal;end;procedure TgxVertexList.SetVertexTexCoord(index: Integer; const val: TTexPoint);begin  FValues^[index].textCoord := val;  NotifyChange(Self);end;function TgxVertexList.GetVertexTexCoord(index: Integer): TTexPoint;begin  Result := FValues^[index].textCoord;end;procedure TgxVertexList.SetVertexColor(index: Integer; const val: TVector4f);begin  FValues^[index].color := val;  NotifyChange(Self);end;function TgxVertexList.GetVertexColor(index: Integer): TVector4f;begin  Result := FValues^[index].color;end;procedure TgxVertexList.AddVertex(const vertexData: TgxVertexData);begin  if FCount = FCapacity then    Grow;  FValues^[FCount] := vertexData;  Inc(FCount);  NotifyChange(Self);end;procedure TgxVertexList.AddVertex3(const vd1, vd2, vd3: TgxVertexData);begin  // extend memory space  if FCount + 2 >= FCapacity then    Grow;  // calculate destination address for new vertex data  FValues^[FCount] := vd1;  FValues^[FCount + 1] := vd2;  FValues^[FCount + 2] := vd3;  Inc(FCount, 3);  NotifyChange(Self);end;procedure TgxVertexList.AddVertex(const aVertex: TVertex;  const aNormal: TAffineVector; const aColor: TgxColorVector;  const aTexPoint: TTexPoint);begin  if FCount = FCapacity then    Grow;  // calculate destination address for new vertex data  with FValues^[FCount] do  begin    textCoord := aTexPoint;    color := aColor;    normal := aNormal;    coord := aVertex;  end;  Inc(FCount);  NotifyChange(Self);end;procedure TgxVertexList.AddVertex(const vertex: TVertex;  const normal: TAffineVector; const color: TgxColorVector);begin  AddVertex(vertex, normal, color, NullTexPoint);end;procedure TgxVertexList.AddVertex(const vertex: TVertex;  const normal: TAffineVector);begin  AddVertex(vertex, normal, clrBlack, NullTexPoint);end;procedure TgxVertexList.DuplicateVertex(index: Integer);begin  Assert(Cardinal(index) < Cardinal(Count));  if FCount = FCapacity then    Grow;  FValues[FCount] := FValues[index];  Inc(FCount);  NotifyChange(Self);end;procedure TgxVertexList.Clear;begin  Assert(not Locked, 'Cannot clear a locked list !');  FreeMem(FValues);  FCount := 0;  FCapacity := 0;  FValues := nil;  NotifyChange(Self);end;function TgxVertexList.SumVertexCoords: TAffineVector;var  i: Integer;begin  Result := NullVector;  for i := 0 to Count - 1 do    AddVector(Result, FValues^[i].coord);end;procedure TgxVertexList.GetExtents(var min, max: TAffineVector);var  i, k: Integer;  f: Single;const  cBigValue: Single = 1E50;  cSmallValue: Single = -1E50;begin  SetVector(min, cBigValue, cBigValue, cBigValue);  SetVector(max, cSmallValue, cSmallValue, cSmallValue);  for i := 0 to Count - 1 do  begin    with FValues^[i] do      for k := 0 to 2 do      begin        f := coord.V[k];        if f < min.V[k] then          min.V[k] := f;        if f > max.V[k] then          max.V[k] := f;      end;  end;end;procedure TgxVertexList.NormalizeNormals;var  i: Integer;begin  for i := 0 to Count - 1 do    NormalizeVector(FValues^[i].coord);end;procedure TgxVertexList.Translate(const v: TAffineVector);var  i: Integer;begin  for i := 0 to Count - 1 do    AddVector(FValues^[i].coord, v);end;procedure TgxVertexList.DefineOpenGLArrays;begin  glEnableClientState(GL_VERTEX_ARRAY);  glVertexPointer(3, GL_FLOAT, SizeOf(TgxVertexData) - SizeOf(TAffineVector),    FirstVertex);  glEnableClientState(GL_NORMAL_ARRAY);  glNormalPointer(GL_FLOAT, SizeOf(TgxVertexData) - SizeOf(TAffineVector),    FirstNormal);  glEnableClientState(GL_TEXTURE_COORD_ARRAY);  glTexCoordPointer(2, GL_FLOAT, SizeOf(TgxVertexData) - SizeOf(TTexPoint),    FirstTexPoint);end;procedure TgxVertexList.Assign(Source: TPersistent);begin  if Assigned(Source) and (Source is TgxVertexList) then  begin    FCount := TgxVertexList(Source).FCount;    FCapacity := FCount;    ReallocMem(FValues, FCount * SizeOf(TgxVertexData));    Move(TgxVertexList(Source).FValues^, FValues^, FCount * SizeOf(TgxVertexData));  end  else    inherited Assign(Source);end;// ----------------- TgxMesh ------------------------------------------------------constructor TgxMesh.Create(AOwner: TComponent);begin  inherited Create(AOwner);  // ObjectStyle:=ObjectStyle+[osDirectDraw];  FVertices := TgxVertexList.Create(Self);  FVertices.AddVertex(XVector, ZVector, NullHmgVector, NullTexPoint);  FVertices.AddVertex(YVector, ZVector, NullHmgVector, NullTexPoint);  FVertices.AddVertex(ZVector, ZVector, NullHmgVector, NullTexPoint);  FVertices.OnNotifyChange := VerticesChanged;  FAxisAlignedDimensionsCache.X := -1;  FVertexMode := vmVNCT;  // should change this later to default to vmVN. But need toend; // change GLMeshPropform so that it greys out unused vertex infodestructor TgxMesh.Destroy;begin  FVertices.Free;  inherited Destroy;end;procedure TgxMesh.VerticesChanged(Sender: TObject);begin  StructureChanged;end;procedure TgxMesh.BuildList(var rci: TgxRenderContextInfo);var  VertexCount: Longint;begin  inherited;  if osDirectDraw in ObjectStyle then    FVertices.EnterLockSection;  case FVertexMode of    vmV:      glInterleavedArrays(GL_V3F, SizeOf(TgxVertexData), FVertices.FirstVertex);    vmVN:      glInterleavedArrays(GL_N3F_V3F, SizeOf(TgxVertexData),        FVertices.FirstNormal);    vmVNC:      glInterleavedArrays(GL_C4F_N3F_V3F, SizeOf(TgxVertexData),        FVertices.FirstColor);    vmVNT, vmVNCT:      glInterleavedArrays(GL_T2F_C4F_N3F_V3F, 0, FVertices.FirstEntry);    vmVT:      glInterleavedArrays(GL_T2F_V3F, 0, FVertices.FirstEntry);  else    Assert(False, strInterleaveNotSupported);  end;  if FVertexMode in [vmVNC, vmVNCT] then  begin    rci.gxStates.Enable(stColorMaterial);    glColorMaterial(GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE);    rci.gxStates.SetMaterialColors(cmFront, clrBlack, clrGray20, clrGray80,      clrBlack, 0);    rci.gxStates.SetMaterialColors(cmBack, clrBlack, clrGray20, clrGray80,      clrBlack, 0);  end;  VertexCount := FVertices.Count;  case FMode of    mmTriangleStrip:      glDrawArrays(GL_TRIANGLE_STRIP, 0, VertexCount);    mmTriangleFan:      glDrawArrays(GL_TRIANGLE_FAN, 0, VertexCount);    mmTriangles:      glDrawArrays(GL_TRIANGLES, 0, VertexCount);    mmQuadStrip:      glDrawArrays(GL_QUAD_STRIP, 0, VertexCount);    mmQuads:      glDrawArrays(GL_QUADS, 0, VertexCount);    mmPolygon:      glDrawArrays(GL_POLYGON, 0, VertexCount);  else    Assert(False);  end;  if osDirectDraw in ObjectStyle then    FVertices.LeaveLockSection;end;procedure TgxMesh.SetMode(AValue: TMeshMode);begin  if AValue <> FMode then  begin    FMode := AValue;    StructureChanged;  end;end;procedure TgxMesh.SetVertices(AValue: TgxVertexList);begin  if AValue <> FVertices then  begin    FVertices.Assign(AValue);    StructureChanged;  end;end;procedure TgxMesh.SetVertexMode(AValue: TVertexMode);begin  if AValue <> FVertexMode then  begin    FVertexMode := AValue;    StructureChanged;  end;end;procedure TgxMesh.CalcNormals(Frontface: TgxFaceWinding);var  vn: TAffineFltVector;  i, j: Integer;begin  case FMode of    mmTriangleStrip:      with Vertices do        for i := 0 to Count - 3 do        begin          if (Frontface = fwCounterClockWise) xor ((i and 1) = 0) then            vn := CalcPlaneNormal(FValues^[i + 0].coord, FValues^[i + 1].coord,              FValues^[i + 2].coord)          else            vn := CalcPlaneNormal(FValues^[i + 2].coord, FValues^[i + 1].coord,              FValues^[i + 0].coord);          FValues^[i].normal := vn;        end;    mmTriangles:      with Vertices do        for i := 0 to ((Count - 3) div 3) do        begin          j := i * 3;          if Frontface = fwCounterClockWise then            vn := CalcPlaneNormal(FValues^[j + 0].coord, FValues^[j + 1].coord,              FValues^[j + 2].coord)          else            vn := CalcPlaneNormal(FValues^[j + 2].coord, FValues^[j + 1].coord,              FValues^[j + 0].coord);          FValues^[j + 0].normal := vn;          FValues^[j + 1].normal := vn;          FValues^[j + 2].normal := vn;        end;    mmQuads:      with Vertices do        for i := 0 to ((Count - 4) div 4) do        begin          j := i * 4;          if Frontface = fwCounterClockWise then            vn := CalcPlaneNormal(FValues^[j + 0].coord, FValues^[j + 1].coord,              FValues^[j + 2].coord)          else            vn := CalcPlaneNormal(FValues^[j + 2].coord, FValues^[j + 1].coord,              FValues^[j + 0].coord);          FValues^[j + 0].normal := vn;          FValues^[j + 1].normal := vn;          FValues^[j + 2].normal := vn;          FValues^[j + 3].normal := vn;        end;    mmPolygon:      with Vertices do        if Count > 2 then        begin          if Frontface = fwCounterClockWise then            vn := CalcPlaneNormal(FValues^[0].coord, FValues^[1].coord,              FValues^[2].coord)          else            vn := CalcPlaneNormal(FValues^[2].coord, FValues^[1].coord,              FValues^[0].coord);          for i := 0 to Count - 1 do            FValues^[i].normal := vn;        end;  else    Assert(False);  end;  StructureChanged;end;procedure TgxMesh.Assign(Source: TPersistent);begin  if Assigned(Source) and (Source is TgxMesh) then  begin    FVertices.Assign(TgxMesh(Source).Vertices);    FMode := TgxMesh(Source).FMode;    FVertexMode := TgxMesh(Source).FVertexMode;  end  else    inherited Assign(Source);end;function TgxMesh.AxisAlignedDimensionsUnscaled: TVector4f;var  dMin, dMax: TAffineVector;begin  if FAxisAlignedDimensionsCache.X < 0 then  begin    Vertices.GetExtents(dMin, dMax);    FAxisAlignedDimensionsCache.X := MaxFloat(Abs(dMin.X), Abs(dMax.X));    FAxisAlignedDimensionsCache.Y := MaxFloat(Abs(dMin.Y), Abs(dMax.Y));    FAxisAlignedDimensionsCache.Z := MaxFloat(Abs(dMin.Z), Abs(dMax.Z));  end;  SetVector(Result, FAxisAlignedDimensionsCache);end;procedure TgxMesh.StructureChanged;begin  FAxisAlignedDimensionsCache.X := -1;  inherited;end;// ------------------------------------------------------------------initialization// ------------------------------------------------------------------    RegisterClasses([TgxMesh]);end.
 |