diff --git a/src/AIS/AIS_ViewController.cxx b/src/AIS/AIS_ViewController.cxx index 479972103b..0f7b063173 100644 --- a/src/AIS/AIS_ViewController.cxx +++ b/src/AIS/AIS_ViewController.cxx @@ -2266,6 +2266,12 @@ void AIS_ViewController::handleCameraActions (const Handle(AIS_InteractiveContex myGL.Orientation.ToFitAll = false; } + if (theView->Viewer()->Grid()->IsActive() + && theView->Viewer()->GridEcho()) + { + theView->Viewer()->Grid()->Update(); + } + if (myGL.IsNewGesture) { if (myAnchorPointPrs1->HasInteractiveContext()) diff --git a/src/Aspect/Aspect_Grid.cxx b/src/Aspect/Aspect_Grid.cxx index eed3460bd8..2bd2aa1d6c 100644 --- a/src/Aspect/Aspect_Grid.cxx +++ b/src/Aspect/Aspect_Grid.cxx @@ -16,10 +16,10 @@ IMPLEMENT_STANDARD_RTTIEXT(Aspect_Grid,Standard_Transient) Aspect_Grid::Aspect_Grid (const Standard_Real theXOrigin, - const Standard_Real theYOrigin, - const Standard_Real theAngle, - const Quantity_Color& theColor, - const Quantity_Color& theTenthColor) + const Standard_Real theYOrigin, + const Standard_Real theAngle, + const Quantity_Color& theColor, + const Quantity_Color& theTenthColor) : myRotationAngle (theAngle), myXOrigin (theXOrigin), myYOrigin (theYOrigin), @@ -60,7 +60,7 @@ void Aspect_Grid::Rotate (const Standard_Real theAngle) } void Aspect_Grid::Translate (const Standard_Real theDx, - const Standard_Real theDy) + const Standard_Real theDy) { myXOrigin += theDx; myYOrigin += theDy; @@ -69,7 +69,7 @@ void Aspect_Grid::Translate (const Standard_Real theDx, } void Aspect_Grid::SetColors (const Quantity_Color& theColor, - const Quantity_Color& theTenthColor) + const Quantity_Color& theTenthColor) { myColor = theColor; myTenthColor = theTenthColor; @@ -77,16 +77,16 @@ void Aspect_Grid::SetColors (const Quantity_Color& theColor, } void Aspect_Grid::Colors (Quantity_Color& theColor, - Quantity_Color& theTenthColor) const + Quantity_Color& theTenthColor) const { theColor = myColor; theTenthColor = myTenthColor; } void Aspect_Grid::Hit (const Standard_Real theX, - const Standard_Real theY, - Standard_Real& theGridX, - Standard_Real& theGridY) const + const Standard_Real theY, + Standard_Real& theGridX, + Standard_Real& theGridY) const { if (myIsActive) { diff --git a/src/Aspect/Aspect_Grid.hxx b/src/Aspect/Aspect_Grid.hxx index cce1715371..ab8e76eab1 100644 --- a/src/Aspect/Aspect_Grid.hxx +++ b/src/Aspect/Aspect_Grid.hxx @@ -86,6 +86,9 @@ public: //! Display the grid at screen. Standard_EXPORT virtual void Display() = 0; + + //! Update grid presentation + Standard_EXPORT virtual void Update() = 0; //! Erase the grid from screen. Standard_EXPORT virtual void Erase() const = 0; diff --git a/src/Aspect/Aspect_GridParams.hxx b/src/Aspect/Aspect_GridParams.hxx new file mode 100644 index 0000000000..de3068f59d --- /dev/null +++ b/src/Aspect/Aspect_GridParams.hxx @@ -0,0 +1,67 @@ +// Created on: 1995-03-02 +// Created by: Jean-Louis Frenkel +// Copyright (c) 1995-1999 Matra Datavision +// Copyright (c) 1999-2014 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _Aspect_GridParams_HeaderFile +#define _Aspect_GridParams_HeaderFile + +#include +#include + +class Aspect_GridParams +{ +public: + + //! Constructor. + Aspect_GridParams() + : myColor (0.8, 0.8, 0.8, Quantity_TOC_RGB), + myPosition (0.0, 0.0, 0.0), + myScale (0.01), + myLineThickness (0.01), + myIsBackground (false), + myIsDrawAxis (true), + myIsInfinity (false) + {} + + //! Getters + Standard_Boolean IsBackground() const { return myIsBackground; } + Standard_Boolean IsDrawAxis() const { return myIsDrawAxis; } + const Quantity_Color& Color() const { return myColor; } + Standard_Real Scale() const { return myScale; } + Standard_Boolean IsInfinity() const { return myIsInfinity; } + const gp_Pnt& Position() const { return myPosition; } + Standard_Real LineThickness() const { return myLineThickness; } + + //! Setters + void SetIsBackground (Standard_Boolean theValue) { myIsBackground = theValue; } + void SetIsDrawAxis (Standard_Boolean theValue) { myIsDrawAxis = theValue; } + void SetColor (const Quantity_Color& theColor) { myColor = theColor; } + void SetScale (Standard_Real theScale) { myScale = theScale; } + void SetIsInfinity (Standard_Boolean theValue) { myIsInfinity = theValue; } + void SetPosition (const gp_Pnt& thePosition) { myPosition = thePosition; } + void SetLineThickness (Standard_Real theThickness) { myLineThickness = theThickness; } + +private: + + Quantity_Color myColor; + gp_Pnt myPosition; + Standard_Real myScale; + Standard_Real myLineThickness; + Standard_Boolean myIsBackground; + Standard_Boolean myIsDrawAxis; + Standard_Boolean myIsInfinity; +}; + +#endif // _Aspect_GridParams_HeaderFile diff --git a/src/Aspect/Aspect_RectangularGrid.cxx b/src/Aspect/Aspect_RectangularGrid.cxx index ad8712fafd..a6263ec5b0 100644 --- a/src/Aspect/Aspect_RectangularGrid.cxx +++ b/src/Aspect/Aspect_RectangularGrid.cxx @@ -27,8 +27,11 @@ Aspect_RectangularGrid::Aspect_RectangularGrid( const Standard_Real aFirstAngle, const Standard_Real aSecondAngle, const Standard_Real aRotationAngle) -:Aspect_Grid(anXOrigin,anYOrigin,aRotationAngle),myXStep(aXStep),myYStep(aYStep),myFirstAngle(aFirstAngle),mySecondAngle(aSecondAngle) - +: Aspect_Grid (anXOrigin,anYOrigin,aRotationAngle), + myXStep (aXStep), + myYStep (aYStep), + myFirstAngle (aFirstAngle), + mySecondAngle (aSecondAngle) { Standard_NumericError_Raise_if(!CheckAngle (aFirstAngle,mySecondAngle), "networks are parallel"); diff --git a/src/Aspect/FILES b/src/Aspect/FILES index b1c7fa3f98..7152c78790 100755 --- a/src/Aspect/FILES +++ b/src/Aspect/FILES @@ -25,6 +25,7 @@ Aspect_GraphicsLibrary.hxx Aspect_Grid.cxx Aspect_Grid.hxx Aspect_GridDrawMode.hxx +Aspect_GridParams.hxx Aspect_GridType.hxx Aspect_NeutralWindow.cxx Aspect_NeutralWindow.hxx diff --git a/src/Graphic3d/Graphic3d_CView.hxx b/src/Graphic3d/Graphic3d_CView.hxx index abebcebc61..e8f6e95502 100644 --- a/src/Graphic3d/Graphic3d_CView.hxx +++ b/src/Graphic3d/Graphic3d_CView.hxx @@ -14,6 +14,7 @@ #ifndef _Graphic3d_CView_HeaderFile #define _Graphic3d_CView_HeaderFile +#include #include #include #include @@ -547,6 +548,12 @@ public: //! @name obsolete Graduated Trihedron functionality //! Erases Graduated Trihedron. virtual void GraduatedTrihedronErase() {} + //! Displays Grid. + virtual void GridDisplay (const Aspect_GridParams& theGridParams) { (void)theGridParams; } + + //! Erases Grid. + virtual void GridErase() {} + //! Sets minimum and maximum points of scene bounding box for Graduated Trihedron stored in graphic view object. //! @param theMin [in] the minimum point of scene. //! @param theMax [in] the maximum point of scene. diff --git a/src/Graphic3d/Graphic3d_ShaderManager.cxx b/src/Graphic3d/Graphic3d_ShaderManager.cxx index 8650a554df..dc6db5f0bd 100644 --- a/src/Graphic3d/Graphic3d_ShaderManager.cxx +++ b/src/Graphic3d/Graphic3d_ShaderManager.cxx @@ -1,4 +1,4 @@ -// Copyright (c) 2013-2021 OPEN CASCADE SAS +// Copyright (c) 2013-2021 OPEN CASCADE SAS // // This file is part of Open CASCADE Technology software library. // @@ -2183,3 +2183,110 @@ Handle(Graphic3d_ShaderProgram) Graphic3d_ShaderManager::getColoredQuadProgram() return aProgSrc; } + +Handle(Graphic3d_ShaderProgram) Graphic3d_ShaderManager::getGridProgram() const +{ + Handle(Graphic3d_ShaderProgram) aProgSrc = new Graphic3d_ShaderProgram(); + + Graphic3d_ShaderObject::ShaderVariableList aUniforms, aStageInOuts; + aStageInOuts.Append (Graphic3d_ShaderObject::ShaderVariable ("vec3 NearPoint", Graphic3d_TOS_VERTEX | Graphic3d_TOS_FRAGMENT)); + aStageInOuts.Append (Graphic3d_ShaderObject::ShaderVariable ("vec3 FarPoint", Graphic3d_TOS_VERTEX | Graphic3d_TOS_FRAGMENT)); + aStageInOuts.Append (Graphic3d_ShaderObject::ShaderVariable ("mat4 MVP", Graphic3d_TOS_VERTEX | Graphic3d_TOS_FRAGMENT)); + aUniforms.Append (Graphic3d_ShaderObject::ShaderVariable ("float uZNear", Graphic3d_TOS_FRAGMENT)); + aUniforms.Append (Graphic3d_ShaderObject::ShaderVariable ("float uZFar", Graphic3d_TOS_FRAGMENT)); + aUniforms.Append (Graphic3d_ShaderObject::ShaderVariable ("float uScale", Graphic3d_TOS_FRAGMENT)); + aUniforms.Append (Graphic3d_ShaderObject::ShaderVariable ("float uThickness", Graphic3d_TOS_FRAGMENT)); + aUniforms.Append (Graphic3d_ShaderObject::ShaderVariable ("bool uIsDrawAxis", Graphic3d_TOS_FRAGMENT)); + aUniforms.Append (Graphic3d_ShaderObject::ShaderVariable ("bool uIsBackground", Graphic3d_TOS_FRAGMENT)); + aUniforms.Append (Graphic3d_ShaderObject::ShaderVariable ("vec3 uColor", Graphic3d_TOS_FRAGMENT)); + + TCollection_AsciiString aSrcVert = TCollection_AsciiString() + + EOL"vec3 gridPlane[6] = vec3[] (vec3( 1.0, 1.0, 0.0), vec3(-1.0, -1.0, 0.0), vec3(-1.0, 1.0, 0.0)," + EOL" vec3(-1.0, -1.0, 0.0), vec3( 1.0, 1.0, 0.0), vec3( 1.0, -1.0, 0.0));" + + EOL"vec3 UnprojectPoint (float aX, float anY, float aZ)" + EOL"{" + EOL" vec4 anUnprojPnt = occModelWorldMatrixInverse * occWorldViewMatrixInverse * occProjectionMatrixInverse * vec4 (aX, anY, aZ, 1.0);" + EOL" return anUnprojPnt.xyz / anUnprojPnt.w;" + EOL"}" + + EOL"void main()" + EOL"{" + EOL" vec3 aVertex = gridPlane[gl_VertexID];" + EOL" NearPoint = UnprojectPoint (aVertex.x, aVertex.y, 0.0);" + EOL" FarPoint = UnprojectPoint (aVertex.x, aVertex.y, 1.0);" + EOL" MVP = occProjectionMatrix * occWorldViewMatrix * occModelWorldMatrix;" + EOL" gl_Position = vec4 (aVertex, 1.0);" + EOL"}"; + + TCollection_AsciiString aSrcFrag = TCollection_AsciiString() + + EOL"vec4 grid (vec3 theFragPos3D, vec3 theColor, float theScale, bool theIsDrawAxis, float theThickness)" + EOL"{" + EOL" vec2 aCoord = theFragPos3D.xy * theScale;" + + EOL" vec2 aDerivative = max (fwidth (aCoord), vec2 (theThickness));" + EOL" vec2 aGrid = abs (fract (aCoord - 0.5) - 0.5) / aDerivative;" + EOL" float aLine = min (aGrid.x, aGrid.y);" + EOL" float aMinY = min (aDerivative.y, 1.0);" + EOL" float aMinX = min (aDerivative.x, 1.0);" + + EOL" vec4 aColor = vec4 (theColor, round (1.0 - min (aLine, 1.0)));" + EOL" if (uIsDrawAxis && theIsDrawAxis)" + EOL" {" + EOL" bool isYAxis = abs(aCoord.x) < aMinX;" + EOL" bool isXAxis = abs(aCoord.y) < aMinY;" + EOL" if (isXAxis && isYAxis) { aColor.xyz = vec3 (0.0, 0.0, 1.0); }" + EOL" else if (isXAxis) { aColor.xyz = vec3 (1.0, 0.0, 0.0); }" + EOL" else if (isYAxis) { aColor.xyz = vec3 (0.0, 1.0, 0.0); }" + EOL" }" + EOL" return aColor;" + EOL" }" + + EOL"float computeDepth (vec3 thePos)" + EOL"{" + EOL" vec4 aClipSpacePos = MVP * vec4 (thePos, 1.0);" + EOL" return (aClipSpacePos.z / aClipSpacePos.w);" + EOL"}" + + EOL"float computeLinearDepth (vec3 thePos)" + EOL"{" + EOL" float aClipSpaceDepth = computeDepth (thePos) * 2.0 - 1.0;" + EOL" float aLinearDepth = (2.0 * uZNear * uZFar) / (uZFar + uZNear - aClipSpaceDepth * (uZFar - uZNear));" + EOL" return aLinearDepth / uZFar;" + EOL"}" + + EOL"void main()" + EOL"{" + EOL" float aParam = -NearPoint.z / (FarPoint.z - NearPoint.z);" + EOL" vec3 aFragPos3D = NearPoint + aParam * (FarPoint - NearPoint);" + EOL" float aLinearDepth = computeLinearDepth (aFragPos3D);" + + EOL" vec4 aBigGridColor = grid (aFragPos3D, uColor, uScale, true, uThickness);" + EOL" vec4 aColor = aBigGridColor.a == 0.0" + EOL" ? grid (aFragPos3D, 0.25 * uColor, uScale * 10.0, false, uThickness)" + EOL" : aBigGridColor;" + EOL" float aDepth = computeDepth (aFragPos3D);" + EOL" float aFar = gl_DepthRange.far;" + EOL" float aNear = gl_DepthRange.near;" + EOL" aDepth = ((aFar - aNear) * aDepth + aNear + aFar) * 0.5;" + EOL" if (aColor.a == 0.0 || (-1.0 < aLinearDepth && aLinearDepth < 0.0))" + EOL" {" + EOL" discard;" + EOL" };" + + EOL" if (aLinearDepth < 0.0)" + EOL" {" + EOL" float aFading = max (0.0, 0.5 * log (abs (aLinearDepth)));" + EOL" aColor.a *= aFading;" + EOL" }" + + EOL" gl_FragDepth = uIsBackground ? aFar - 0.001 : min (aDepth, aFar - 0.001);" + EOL" occFragColor = aColor;" + EOL"}"; + + defaultGlslVersion (aProgSrc, "grid", 0); + aProgSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (aSrcVert, Graphic3d_TOS_VERTEX, aUniforms, aStageInOuts)); + aProgSrc->AttachShader (Graphic3d_ShaderObject::CreateFromSource (aSrcFrag, Graphic3d_TOS_FRAGMENT, aUniforms, aStageInOuts)); + + return aProgSrc; +} diff --git a/src/Graphic3d/Graphic3d_ShaderManager.hxx b/src/Graphic3d/Graphic3d_ShaderManager.hxx index 023b574260..1f2df5040f 100644 --- a/src/Graphic3d/Graphic3d_ShaderManager.hxx +++ b/src/Graphic3d/Graphic3d_ShaderManager.hxx @@ -148,6 +148,9 @@ protected: //! Generates shader program to render correctly colored quad. Standard_EXPORT Handle(Graphic3d_ShaderProgram) getColoredQuadProgram() const; + //! Generates shader program to render grid. + Standard_EXPORT Handle(Graphic3d_ShaderProgram) getGridProgram() const; + //! Prepare GLSL source for IBL generation used in PBR pipeline. Standard_EXPORT Handle(Graphic3d_ShaderProgram) getPBREnvBakingProgram (Standard_Integer theIndex) const; diff --git a/src/OpenGl/OpenGl_ShaderManager.cxx b/src/OpenGl/OpenGl_ShaderManager.cxx index eff6bf9f39..de9725674e 100644 --- a/src/OpenGl/OpenGl_ShaderManager.cxx +++ b/src/OpenGl/OpenGl_ShaderManager.cxx @@ -1361,6 +1361,27 @@ Standard_Boolean OpenGl_ShaderManager::preparePBREnvBakingProgram (Standard_Inte return Standard_True; } +// ======================================================================= +// function : prepareGridProgram +// purpose : +// ======================================================================= +Standard_Boolean OpenGl_ShaderManager::prepareGridProgram() +{ + Handle(Graphic3d_ShaderProgram) aProgramSrc = getGridProgram(); + + TCollection_AsciiString aKey; + if (!Create (aProgramSrc, aKey, myGridProgram)) + { + myGridProgram = new OpenGl_ShaderProgram(); // just mark as invalid + return Standard_False; + } + + myContext->BindProgram (myGridProgram); + myContext->BindProgram (Handle(OpenGl_ShaderProgram)()); + + return Standard_True; +} + // ======================================================================= // function : GetBgCubeMapProgram // purpose : diff --git a/src/OpenGl/OpenGl_ShaderManager.hxx b/src/OpenGl/OpenGl_ShaderManager.hxx index f8dc9da70b..f3f0210d96 100644 --- a/src/OpenGl/OpenGl_ShaderManager.hxx +++ b/src/OpenGl/OpenGl_ShaderManager.hxx @@ -225,6 +225,16 @@ public: return myContext->BindProgram (myPBREnvBakingProgram[theIndex]); } + //! Bind program for grid visualisation. + Standard_Boolean BindGridProgram() + { + if (myGridProgram.IsNull()) + { + prepareGridProgram(); + } + return bindProgramWithState (myGridProgram, Graphic3d_TypeOfShadingModel_Unlit); + } + //! Generates shader program to render environment cubemap as background. Standard_EXPORT const Handle(Graphic3d_ShaderProgram)& GetBgCubeMapProgram(); @@ -709,6 +719,9 @@ protected: Standard_Boolean IsPbrAllowed() const { return myShadingModel == Graphic3d_TypeOfShadingModel_Pbr || myShadingModel == Graphic3d_TypeOfShadingModel_PbrFacet; } + //! Generate standard GLSL program for grid. + Standard_EXPORT Standard_Boolean prepareGridProgram(); + protected: //! Packed properties of light source @@ -776,6 +789,7 @@ protected: Handle(Graphic3d_ShaderProgram) myBgCubeMapProgram; //!< program for background cubemap rendering Handle(Graphic3d_ShaderProgram) myBgSkydomeProgram; //!< program for background cubemap rendering Handle(Graphic3d_ShaderProgram) myColoredQuadProgram; //!< program for correct quad rendering + Handle(OpenGl_ShaderProgram) myGridProgram; //!< program for grid rendering Handle(OpenGl_ShaderProgram) myStereoPrograms[Graphic3d_StereoMode_NB]; //!< standard stereo programs diff --git a/src/OpenGl/OpenGl_View.cxx b/src/OpenGl/OpenGl_View.cxx index e1010f47df..e69ed71dfb 100644 --- a/src/OpenGl/OpenGl_View.cxx +++ b/src/OpenGl/OpenGl_View.cxx @@ -108,6 +108,7 @@ OpenGl_View::OpenGl_View (const Handle(Graphic3d_StructureManager)& theMgr, myCaps (theCaps), myWasRedrawnGL (Standard_False), myToShowGradTrihedron (false), + myToShowGrid (false), myStateCounter (theCounter), myCurrLightSourceState (theCounter->Increment()), myLightsRevision (0), @@ -502,6 +503,25 @@ void OpenGl_View::GraduatedTrihedronErase() myToShowGradTrihedron = false; } +// ======================================================================= +// function : GridDisplay +// purpose : +// ======================================================================= +void OpenGl_View::GridDisplay (const Aspect_GridParams& theGridParams) +{ + myGridParams = theGridParams; + myToShowGrid = true; +} + +// ======================================================================= +// function : GridErase +// purpose : +// ======================================================================= +void OpenGl_View::GridErase() +{ + myToShowGrid = false; +} + // ======================================================================= // function : GraduatedTrihedronMinMaxValues // purpose : @@ -2498,8 +2518,14 @@ void OpenGl_View::render (Graphic3d_Camera::Projection theProjection, aContext->core11fwd->glDisable (GL_LIGHTING); } + // ==================================== + // Step 3: Redraw grid + // ==================================== + + renderGrid(); + // ================================= - // Step 3: Redraw main plane + // Step 4: Redraw main plane // ================================= // if the view is scaled normal vectors are scaled to unit @@ -2560,7 +2586,7 @@ void OpenGl_View::render (Graphic3d_Camera::Projection theProjection, myWorkspace->SetEnvironmentTexture (Handle(OpenGl_TextureSet)()); // =============================== - // Step 4: Trihedron + // Step 5: Trihedron // =============================== // Resetting GL parameters according to the default aspects @@ -2590,6 +2616,82 @@ void OpenGl_View::render (Graphic3d_Camera::Projection theProjection, } } +// ======================================================================= +// function : renderGrid +// purpose : +// ======================================================================= +void OpenGl_View::renderGrid() +{ + if (!myToShowGrid) + { + return; + } + + const Handle(OpenGl_Context)& aContext = myWorkspace->GetGlContext(); + + Bnd_Box aBnd = MinMaxValues (Standard_True); + aBnd.Add (myGridParams.Position()); + aContext->Camera()->ZFitAll (1.0, aBnd, aBnd); + + const Standard_Real aZNear = aContext->Camera()->ZNear(); + const Standard_Real aZFar = aContext->Camera()->ZFar(); + const Graphic3d_Camera::Projection aProjectionType = aContext->Camera()->ProjectionType(); + + aContext->Camera()->SetZRange (aZNear, Max (aZNear * 1.001, aZFar)); + if (myGridParams.IsBackground()) + { + aContext->Camera()->SetProjectionType (Graphic3d_Camera::Projection_Orthographic); + } + + aContext->ProjectionState.Push(); + aContext->ProjectionState.SetCurrent (aContext->Camera()->ProjectionMatrixF()); + aContext->ApplyProjectionMatrix(); + + OpenGl_Mat4 aWorldViewState = myGridParams.IsBackground() + ? OpenGl_Mat4() + : aContext->WorldViewState.Current(); + OpenGl_Mat4 aMat; + const gp_Pnt& aPosition = myGridParams.Position(); + aMat.SetColumn (3, Graphic3d_Vec4 ((float)aPosition.X(), (float)aPosition.Y(), (float)aPosition.Z(), 1.0)); + + aContext->WorldViewState.Push(); + aContext->WorldViewState.SetCurrent (aWorldViewState * aMat); + aContext->ApplyWorldViewMatrix(); + + aContext->core11fwd->glEnable (GL_DEPTH_TEST); + aContext->core11fwd->glDepthFunc (GL_LESS); + aContext->core11fwd->glEnable (GL_BLEND); + aContext->core11fwd->glBlendFunc (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); + + const Standard_Real aCameraScale = aContext->Camera()->Scale(); + Standard_Real aScale = myGridParams.IsInfinity() + ? 10.0 / pow (10.0, floor (log10 (Max (aCameraScale, 1.0))) + 1.0) + : myGridParams.Scale(); + + if (aContext->ShaderManager()->BindGridProgram()) + { + const Handle(OpenGl_ShaderProgram)& aProg = aContext->ActiveProgram(); + aProg->SetUniform (aContext, "uZNear", GLfloat (aContext->Camera()->ZNear())); + aProg->SetUniform (aContext, "uZFar", GLfloat (aContext->Camera()->ZFar())); + aProg->SetUniform (aContext, "uScale", GLfloat (aScale)); + aProg->SetUniform (aContext, "uThickness", GLfloat (myGridParams.LineThickness())); + aProg->SetUniform (aContext, "uColor", OpenGl_Vec3 (myGridParams.Color().Rgb())); + aProg->SetUniform (aContext, "uIsDrawAxis", GLboolean (myGridParams.IsDrawAxis())); + aProg->SetUniform (aContext, "uIsBackground", GLboolean (myGridParams.IsBackground())); + + aContext->core11fwd->glDrawArrays (GL_TRIANGLES, 0, 6); + aContext->BindProgram (NULL); + } + + aContext->Camera()->SetZRange (aZNear, aZFar); + aContext->Camera()->SetProjectionType (aProjectionType); + + aContext->WorldViewState.Pop(); + aContext->ProjectionState.Pop(); + aContext->ApplyWorldViewMatrix(); + aContext->ApplyProjectionMatrix(); +} + // ======================================================================= // function : InvalidateBVHData // purpose : diff --git a/src/OpenGl/OpenGl_View.hxx b/src/OpenGl/OpenGl_View.hxx index 766fccd9f1..9af36bee24 100644 --- a/src/OpenGl/OpenGl_View.hxx +++ b/src/OpenGl/OpenGl_View.hxx @@ -306,6 +306,12 @@ public: //! @name obsolete Graduated Trihedron functionality //! Erases Graduated Trihedron. Standard_EXPORT virtual void GraduatedTrihedronErase() Standard_OVERRIDE; + //! Displays Grid. + Standard_EXPORT virtual void GridDisplay (const Aspect_GridParams& theGridParams) Standard_OVERRIDE; + + //! Erases Grid. + Standard_EXPORT virtual void GridErase() Standard_OVERRIDE; + //! Sets minimum and maximum points of scene bounding box for Graduated Trihedron stored in graphic view object. //! @param theMin [in] the minimum point of scene. //! @param theMax [in] the maximum point of scene. @@ -396,6 +402,9 @@ protected: //! @name Rendering of GL graphics (with prepared drawing buffer). OpenGl_FrameBuffer* theOitAccumFbo, const Standard_Boolean theToDrawImmediate); + //! Renders grid + Standard_EXPORT void renderGrid(); + //! Renders trihedron. void renderTrihedron (const Handle(OpenGl_Workspace) &theWorkspace); @@ -449,7 +458,9 @@ protected: gp_XYZ myLocalOrigin; Handle(OpenGl_FrameBuffer) myFBO; Standard_Boolean myToShowGradTrihedron; + Standard_Boolean myToShowGrid; Graphic3d_GraduatedTrihedron myGTrihedronData; + Aspect_GridParams myGridParams; Handle(Graphic3d_LightSet) myNoShadingLight; Handle(Graphic3d_LightSet) myLights; diff --git a/src/V3d/V3d_CircularGrid.cxx b/src/V3d/V3d_CircularGrid.cxx index ca4a7c911d..a812fd9896 100644 --- a/src/V3d/V3d_CircularGrid.cxx +++ b/src/V3d/V3d_CircularGrid.cxx @@ -106,6 +106,11 @@ void V3d_CircularGrid::Display () UpdateDisplay(); } +void V3d_CircularGrid::Update() +{ + // +} + void V3d_CircularGrid::Erase () const { myStructure->Erase (); diff --git a/src/V3d/V3d_CircularGrid.hxx b/src/V3d/V3d_CircularGrid.hxx index f7278ef9c8..674ad7e2ff 100644 --- a/src/V3d/V3d_CircularGrid.hxx +++ b/src/V3d/V3d_CircularGrid.hxx @@ -38,15 +38,17 @@ public: Standard_EXPORT virtual ~V3d_CircularGrid(); Standard_EXPORT void SetColors (const Quantity_Color& aColor, const Quantity_Color& aTenthColor) Standard_OVERRIDE; - + Standard_EXPORT void Display() Standard_OVERRIDE; - + + Standard_EXPORT void Update() Standard_OVERRIDE; + Standard_EXPORT void Erase() const Standard_OVERRIDE; - + Standard_EXPORT Standard_Boolean IsDisplayed() const Standard_OVERRIDE; - + Standard_EXPORT void GraphicValues (Standard_Real& Radius, Standard_Real& OffSet) const; - + Standard_EXPORT void SetGraphicValues (const Standard_Real Radius, const Standard_Real OffSet); //! Dumps the content of me into the stream diff --git a/src/V3d/V3d_RectangularGrid.cxx b/src/V3d/V3d_RectangularGrid.cxx index 35b1d2b257..fb13f15d88 100644 --- a/src/V3d/V3d_RectangularGrid.cxx +++ b/src/V3d/V3d_RectangularGrid.cxx @@ -106,6 +106,12 @@ void V3d_RectangularGrid::Display () UpdateDisplay(); } +void V3d_RectangularGrid::Update() +{ + SetXStep (XStep() + 1); + UpdateDisplay(); +} + void V3d_RectangularGrid::Erase () const { myStructure->Erase (); diff --git a/src/V3d/V3d_RectangularGrid.hxx b/src/V3d/V3d_RectangularGrid.hxx index e8ea5241e7..35e5922c39 100644 --- a/src/V3d/V3d_RectangularGrid.hxx +++ b/src/V3d/V3d_RectangularGrid.hxx @@ -40,6 +40,8 @@ public: Standard_EXPORT virtual void Display() Standard_OVERRIDE; + Standard_EXPORT virtual void Update() Standard_OVERRIDE; + Standard_EXPORT virtual void Erase() const Standard_OVERRIDE; Standard_EXPORT virtual Standard_Boolean IsDisplayed() const Standard_OVERRIDE; diff --git a/src/V3d/V3d_View.cxx b/src/V3d/V3d_View.cxx index adafb827f4..b44fe23ae7 100644 --- a/src/V3d/V3d_View.cxx +++ b/src/V3d/V3d_View.cxx @@ -3797,6 +3797,24 @@ void V3d_View::GraduatedTrihedronErase() myView->GraduatedTrihedronErase(); } +//============================================================================= +//function : GridDisplay +//purpose : +//============================================================================= +void V3d_View::GridDisplay (const Aspect_GridParams& theGridParams) +{ + myView->GridDisplay (theGridParams); +} + +//============================================================================= +//function : GridErase +//purpose : +//============================================================================= +void V3d_View::GridErase() +{ + myView->GridErase(); +} + // ======================================================================= // function : DumpJson // purpose : diff --git a/src/V3d/V3d_View.hxx b/src/V3d/V3d_View.hxx index 93804995a4..c7dc8c870e 100644 --- a/src/V3d/V3d_View.hxx +++ b/src/V3d/V3d_View.hxx @@ -318,6 +318,12 @@ public: //! Erases a graduated trihedron from the view. Standard_EXPORT void GraduatedTrihedronErase(); + //! Displays a grid. + Standard_EXPORT void GridDisplay (const Aspect_GridParams& theGridParams); + + //! Erases a grid from the view. + Standard_EXPORT void GridErase(); + //! modify the Projection of the view perpendicularly to //! the privileged plane of the viewer. Standard_EXPORT void SetFront(); diff --git a/src/ViewerTest/ViewerTest_ViewerCommands.cxx b/src/ViewerTest/ViewerTest_ViewerCommands.cxx index 630ac437f4..bf595f6975 100644 --- a/src/ViewerTest/ViewerTest_ViewerCommands.cxx +++ b/src/ViewerTest/ViewerTest_ViewerCommands.cxx @@ -5392,6 +5392,116 @@ static int VGrid (Draw_Interpretor& /*theDI*/, return 0; } +//============================================================================== +//function : VInfGrid +//purpose : +//============================================================================== +static int VInfGrid (Draw_Interpretor& /*theDI*/, + Standard_Integer theArgNb, + const char** theArgVec) +{ + Handle(V3d_View) aView = ViewerTest::CurrentView(); + Handle(V3d_Viewer) aViewer = ViewerTest::GetViewerFromContext(); + if (aView.IsNull() || aViewer.IsNull()) + { + Message::SendFail("Error: no active viewer"); + return 1; + } + + Aspect_GridParams aGridParams; + Standard_Boolean toDisplay = true; + ViewerTest_AutoUpdater anUpdateTool (ViewerTest::GetAISContext(), aView); + for (Standard_Integer anArgIter = 1; anArgIter < theArgNb; ++anArgIter) + { + TCollection_AsciiString anArg (theArgVec[anArgIter]); + anArg.LowerCase(); + if (anUpdateTool.parseRedrawMode(theArgVec[anArgIter])) + { + continue; + } + else if (anArgIter < theArgNb && anArg == "-background") + { + aGridParams.SetIsBackground (Standard_True); + } + else if (anArgIter + 1 < theArgNb && anArg == "-drawaxis") + { + Standard_Integer aVal = Draw::Atoi (theArgVec[++anArgIter]); + if (aVal == 0) + { + aGridParams.SetIsDrawAxis (Standard_False); + } + else if (aVal == 1) + { + aGridParams.SetIsDrawAxis (Standard_True); + } + else + { + Message::SendFail() << "Syntax error at '" << anArg << " " << aVal <<"'"; + return 1; + } + } + else if (anArgIter + 3 < theArgNb && (anArg == "-color")) + { + Quantity_Color aColor; + aColor.SetValues (Draw::Atof (theArgVec[anArgIter + 1]), Draw::Atof (theArgVec[anArgIter + 2]), Draw::Atof (theArgVec[anArgIter + 3]), Quantity_TOC_RGB); + aGridParams.SetColor (aColor); + anArgIter += 3; + } + else if (anArgIter + 3 < theArgNb && anArg == "-origin") + { + gp_Pnt aPoint; + aPoint.SetXYZ (gp_XYZ (Draw::Atof (theArgVec[anArgIter + 1]), Draw::Atof (theArgVec[anArgIter + 2]), Draw::Atof (theArgVec[anArgIter + 3]))); + aGridParams.SetPosition (aPoint); + anArgIter += 3; + } + else if (anArgIter + 1 < theArgNb && (anArg == "-inf")) + { + Standard_Integer aVal = Draw::Atoi(theArgVec[++anArgIter]); + if (aVal == 0) + { + aGridParams.SetIsInfinity (Standard_False); + } + else if (aVal == 1) + { + aGridParams.SetIsInfinity (Standard_True); + } + else + { + Message::SendFail() << "Syntax error at '" << anArg << " " << aVal << "'"; + return 1; + } + } + else if (anArgIter + 1 < theArgNb && (anArg == "-scale")) + { + aGridParams.SetScale (Draw::Atof(theArgVec[++anArgIter])); + } + else if (anArgIter + 1 < theArgNb && anArg == "-linethickness") + { + aGridParams.SetLineThickness (Draw::Atof (theArgVec[++anArgIter])); + } + else if (anArgIter >= theArgNb && anArg == "off") + { + toDisplay = Standard_False; + } + else + { + Message::SendFail() << "Syntax error at '" << anArg << "'"; + return 1; + } + } + + if (toDisplay) + { + ViewerTest::CurrentView()->GridDisplay (aGridParams); + } + else + { + ViewerTest::CurrentView()->GridErase(); + } + + return 0; +} + //============================================================================== //function : VPriviledgedPlane //purpose : @@ -14425,6 +14535,11 @@ vgrid [off] [-type {rect|circ}] [-mode {line|point}] [-origin X Y] [-rotAngle An [-step StepRadius NbDivisions] [-radius Radius] )" /* [vgrid] */); + addCmd("vinfgrid", VInfGrid, /* [vinfgrid] */ R"( +vinfgrid [off] [-background] [-drawAxis {0|1}] [-color R G B] [-origin X Y Z] + [-inf {0|1}] [-scale value] [-lineThickness value] +)" /* [vinfgrid] */); + addCmd ("vpriviledgedplane", VPriviledgedPlane, /* [vpriviledgedplane] */ R"( vpriviledgedplane [Ox Oy Oz Nx Ny Nz [Xx Xy Xz]] Sets or prints viewer's priviledged plane geometry: diff --git a/tests/v3d/grid/ortho b/tests/v3d/grid/ortho new file mode 100644 index 0000000000..58bae04cc2 --- /dev/null +++ b/tests/v3d/grid/ortho @@ -0,0 +1,23 @@ +puts "==================================" +puts "Grid in ortographic projection" +puts "==================================" + +pload MODELING VISUALIZATION + +vclear +vinit View1 +vaxo + +box b 1 2 3 +vdisplay b -dispMode 1 + +vcamera -ortho + +vfit +vzoom 0.1 + +vinfgrid +vdump $imagedir/${casename}_0.png + +vinfgrid -background -drawAxis 0 -color 0 0 1 -origin 1 1 1 -inf 1 -lineThickness 0.05 +vdump $imagedir/${casename}_1.png diff --git a/tests/v3d/grid/persp b/tests/v3d/grid/persp new file mode 100644 index 0000000000..0eb24ad5e3 --- /dev/null +++ b/tests/v3d/grid/persp @@ -0,0 +1,23 @@ +puts "==================================" +puts "Grid in perspective projection" +puts "==================================" + +pload MODELING VISUALIZATION + +vclear +vinit View1 +vaxo + +box b 1 2 3 +vdisplay b -dispMode 1 + +vcamera -persp + +vfit +vzoom 0.1 + +vinfgrid +vdump $imagedir/${casename}_0.png + +vinfgrid -background -drawAxis 0 -color 0 0 1 -origin 1 1 1 -inf 1 -lineThickness 0.05 +vdump $imagedir/${casename}_1.png