| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372 |
- //
- // This unit is part of the GLScene Engine, http://glscene.org
- //
- unit GLS.ShaderHiddenLine;
- (*
- A shader that renders hidden (back-faced) lines differently from visible
- (front) lines. Polygon offset is used to displace fragments depths a little
- so that there is no z-fighting in rendering the same geometry multiple times.
- *)
- interface
- {$I GLScene.inc}
- uses
- Winapi.OpenGL,
- System.Classes,
-
- OpenGLTokens,
- GLScene,
- GLColor,
- GLMaterial,
- GLBaseClasses,
- GLRenderContextInfo,
- GLState,
- GLContext;
- type
- TGLLineSettings = class(TGLUpdateAbleObject)
- private
- FColor: TGLColor;
- FWidth: Single;
- FPattern: TGLushort;
- FForceMaterial: Boolean;
- procedure SetPattern(const value: TGLushort);
- procedure SetColor(const v: TGLColor);
- procedure SetWidth(const Value: Single);
- procedure SetForceMaterial(v: boolean);
- public
- constructor Create(AOwner: TPersistent); override;
- destructor Destroy; override;
- procedure Apply(var rci: TGLRenderContextInfo);
- procedure UnApply(var rci: TGLRenderContextInfo);
- published
- property Width: Single read FWidth write SetWidth;
- property Color: TGLColor read FColor write SetColor;
- property Pattern: TGLushort read FPattern write SetPattern default $FFFF;
- (* Set ForceMaterial to true to enforce the application of the line settings
- for objects that sets their own color, line width and pattern. *)
- property ForceMaterial: Boolean read FForceMaterial write SetForceMaterial
- default false;
- end;
- TGLHiddenLineShader = class(TGLShader)
- private
- FPassCount: integer;
- FLineSmooth: Boolean;
- FSolid: Boolean;
- FBackGroundColor: TGLColor;
- FFrontLine: TGLLineSettings;
- FBackLine: TGLLineSettings;
- FLighting: Boolean;
- FShadeModel: TGLShadeModel;
- procedure SetlineSmooth(v: boolean);
- procedure SetSolid(v: boolean);
- procedure SetBackgroundColor(AColor: TGLColor);
- procedure SetLighting(v: boolean);
- procedure SetShadeModel(const val: TGLShadeModel);
- protected
- procedure DoApply(var rci: TGLRenderContextInfo; Sender: TObject); override;
- function DoUnApply(var rci: TGLRenderContextInfo): Boolean; override;
- public
- constructor Create(AOwner: TComponent); override;
- destructor Destroy; override;
- published
- property FrontLine: TGLLineSettings read FFrontLine write FFrontLine;
- property BackLine: TGLLineSettings read FBackLine write FBackLine;
- // Line smoothing control
- property LineSmooth: Boolean read FlineSmooth write SetlineSmooth default
- false;
- // Solid controls if you can see through the front-line wireframe.
- property Solid: Boolean read FSolid write SetSolid default false;
- // Color used for solid fill.
- property BackgroundColor: TGLColor read FBackgroundColor write
- SetBackgroundColor;
- // When Solid is True, determines if lighting or background color is used.
- property SurfaceLit: Boolean read FLighting write SetLighting default true;
- // Shade model. Default is "Smooth".
- property ShadeModel: TGLShadeModel read FShadeModel write SetShadeModel
- default smDefault;
- end;
- // ------------------------------------------------------------------
- implementation
- // ------------------------------------------------------------------
- // ------------------
- // ------------------ TGLLineSettings ------------------
- // ------------------
- constructor TGLLineSettings.Create(AOwner: TPersistent);
- begin
- inherited;
- FColor := TGLColor.Create(Self);
- FColor.Initialize(clrGray20);
- FWidth := 2;
- Pattern := $FFFF;
- ForceMaterial := false;
- end;
- destructor TGLLineSettings.Destroy;
- begin
- FColor.Free;
- inherited;
- end;
- procedure TGLLineSettings.SetPattern(const value: TGLushort);
- begin
- if FPattern <> value then
- begin
- FPattern := Value;
- NotifyChange(self);
- end;
- end;
- procedure TGLLineSettings.SetColor(const v: TGLColor);
- begin
- FColor.Color := v.Color;
- NotifyChange(Self);
- end;
- procedure TGLLineSettings.SetWidth(const Value: Single);
- begin
- FWidth := Value;
- NotifyChange(Self);
- end;
- var
- IgnoreMatSave: boolean;
- procedure TGLLineSettings.Apply(var rci: TGLRenderContextInfo);
- begin
- rci.GLStates.LineWidth := Width;
- gl.Color4fv(Color.AsAddress);
- if Pattern <> $FFFF then
- begin
- rci.GLStates.Enable(stLineStipple);
- rci.GLStates.LineStippleFactor := 1;
- rci.GLStates.LineStipplePattern := Pattern;
- end
- else
- rci.GLStates.Disable(stLineStipple);
- if ForceMaterial then
- begin
- IgnoreMatSave := rci.ignoreMaterials;
- rci.ignoreMaterials := true;
- end;
- end;
- procedure TGLLineSettings.UnApply(var rci: TGLRenderContextInfo);
- begin
- if ForceMaterial then
- rci.ignoreMaterials := IgnoreMatSave;
- end;
- procedure TGLLineSettings.SetForceMaterial(v: boolean);
- begin
- if FForceMaterial <> v then
- begin
- FForceMaterial := v;
- NotifyChange(self);
- end;
- end;
- // ------------------
- // ------------------ TGLHiddenLineShader ------------------
- // ------------------
- constructor TGLHiddenLineShader.Create(AOwner: TComponent);
- begin
- inherited;
- FFrontLine := TGLLineSettings.Create(self);
- FBackLine := TGLLineSettings.Create(self);
- FSolid := false;
- FBackgroundColor := TGLColor.Create(Self);
- FBackgroundColor.Initialize(clrBtnFace);
- FLineSmooth := False;
- FLighting := true;
- FShadeModel := smDefault;
- end;
- destructor TGLHiddenLineShader.Destroy;
- begin
- FFrontLine.Free;
- FBackLine.Free;
- FBackgroundColor.Free;
- inherited;
- end;
- procedure TGLHiddenLineShader.DoApply(var rci: TGLRenderContextInfo; Sender:
- TObject);
- begin
- FPassCount := 1;
- if solid then
- with rci.GLStates do
- begin
- // draw filled front faces in first pass
- PolygonMode := pmFill;
- CullFaceMode := cmBack;
- if FLighting then
- begin
- case ShadeModel of
- smDefault, smSmooth: gl.ShadeModel(GL_SMOOTH);
- smFlat: gl.ShadeModel(GL_FLAT);
- end
- end
- else
- begin
- Disable(stLighting);
- gl.Color4fv(FBackgroundColor.AsAddress); // use background color
- end;
- // enable and adjust polygon offset
- Enable(stPolygonOffsetFill);
- end
- else
- with rci.GLStates do
- begin
- Disable(stLighting);
- // draw back lines in first pass
- FBackLine.Apply(rci);
- CullFaceMode := cmFront;
- PolygonMode := pmLines;
- // enable and adjust polygon offset
- Enable(stPolygonOffsetLine);
- end;
- rci.GLStates.SetPolygonOffset(1, 2);
- end;
- function TGLHiddenLineShader.DoUnApply(var rci: TGLRenderContextInfo): Boolean;
- procedure SetLineSmoothBlend;
- begin
- with rci.GLStates do
- begin
- LineStippleFactor := 1;
- LineStipplePattern := $FFFF;
- if LineSmooth then
- begin
- LineSmoothHint := hintNicest;
- Enable(stLineSmooth);
- end
- else
- Disable(stLineSmooth);
- if LineSmooth or (FBackLine.FColor.Alpha < 1)
- or (FFrontLine.FColor.Alpha < 1) then
- begin
- Enable(stBlend);
- SetBlendFunc(bfSrcAlpha, bfOneMinusSrcAlpha);
- end
- else
- Disable(stBlend);
- end;
- end;
- begin
- case FPassCount of
- 1:
- with rci.GLStates do begin
- // draw front line in 2nd pass
- FPassCount := 2;
- FBackLine.UnApply(rci);
- FFrontLine.Apply(rci);
- SetLineSmoothBlend;
- if solid and FLighting then
- Disable(stLighting);
- PolygonMode := pmLines;
- CullFaceMode := cmBack;
- if solid then
- rci.GLStates.Disable(stPolygonOffsetFill)
- else
- rci.GLStates.Disable(stPolygonOffsetLine);
- Result := True;
- end;
- 2:
- begin
- FFrontLine.UnApply(rci);
- rci.GLStates.PolygonMode := pmFill;
- Result := false;
- end;
- else
- Assert(False);
- Result := False;
- end;
- end;
- procedure TGLHiddenLineShader.SetBackgroundColor(AColor: TGLColor);
- begin
- FBackgroundColor.Color := AColor.Color;
- NotifyChange(Self);
- end;
- procedure TGLHiddenLineShader.SetlineSmooth(v: boolean);
- begin
- if FlineSmooth <> v then
- begin
- FlineSmooth := v;
- NotifyChange(self);
- end;
- end;
- procedure TGLHiddenLineShader.SetLighting(v: boolean);
- begin
- if FLighting <> v then
- begin
- FLighting := v;
- NotifyChange(self);
- end;
- end;
- procedure TGLHiddenLineShader.SetSolid(v: boolean);
- begin
- if FSolid <> v then
- begin
- FSolid := v;
- NotifyChange(self);
- end;
- end;
- procedure TGLHiddenLineShader.SetShadeModel(const val: TGLShadeModel);
- begin
- if FShadeModel <> val then
- begin
- FShadeModel := val;
- NotifyChange(Self);
- end;
- end;
- end.
|