From b0afa94d10cabb2709524aaa29d4f490cf3faa58 Mon Sep 17 00:00:00 2001 From: mzernova Date: Tue, 22 Nov 2022 11:30:24 +0000 Subject: [PATCH] 0027848: Visualization - sensitivity of lines is too high Improved objects sorting by depth. The sensitivity calculation for the manipulator has been changed. Now the sensitivity of the elements of the manipulator depends on its size. Also fixed the problem of erroneous selection of unfilled circles (SelectMgr_Frustum::hasCircleOverlap() function). Test case vselect/bugs/bug27848 added. --- src/AIS/AIS_Manipulator.cxx | 102 +++++++++++---------- src/SelectMgr/SelectMgr_Frustum.lxx | 21 ++--- src/SelectMgr/SelectMgr_SortCriterion.hxx | 25 ++++- src/SelectMgr/SelectMgr_ViewerSelector.cxx | 2 +- tests/v3d/dimensions/bug24389 | 2 +- tests/v3d/manipulator/drag_in_2d_view | 4 +- tests/v3d/manipulator/dragg | 2 +- tests/v3d/manipulator/rotate | 10 +- tests/v3d/manipulator/translate | 2 +- tests/vselect/bugs/bug23012 | 2 +- tests/vselect/bugs/bug23539_2 | 2 +- tests/vselect/bugs/bug27848 | 23 +++++ tests/vselect/edge/A10 | 16 ++-- 13 files changed, 128 insertions(+), 85 deletions(-) create mode 100644 tests/vselect/bugs/bug27848 diff --git a/src/AIS/AIS_Manipulator.cxx b/src/AIS/AIS_Manipulator.cxx index a6f739bdb0..4cd43844b8 100644 --- a/src/AIS/AIS_Manipulator.cxx +++ b/src/AIS/AIS_Manipulator.cxx @@ -1116,18 +1116,23 @@ void AIS_Manipulator::ComputeSelection (const Handle(SelectMgr_Selection)& theSe const Standard_Integer theMode) { //Check mode - AIS_ManipulatorMode aMode = (AIS_ManipulatorMode) theMode; + const AIS_ManipulatorMode aMode = (AIS_ManipulatorMode) theMode; if (aMode == AIS_MM_None) { return; } Handle(SelectMgr_EntityOwner) anOwner; - if (aMode == AIS_MM_None) - { - anOwner = new SelectMgr_EntityOwner (this, 5); - } - if (aMode == AIS_MM_Translation || aMode == AIS_MM_None) + // Sensitivity calculation for manipulator parts allows to avoid + // overlapping of sensitive areas when size of manipulator is small. + // Sensitivity is calculated relative to the default size of the manipulator (100.0f). + const Standard_ShortReal aSensitivityCoef = myAxes[0].Size() / 100.0f; + const Standard_Integer aHighSensitivity = Max (Min (RealToInt (aSensitivityCoef * 15), 15), 3); // clamp sensitivity within range [3, 15] + const Standard_Integer aLowSensitivity = Max (Min (RealToInt (aSensitivityCoef * 10), 10), 2); // clamp sensitivity within range [2, 10] + + switch (aMode) + { + case AIS_MM_Translation: { for (Standard_Integer anIt = 0; anIt < 3; ++anIt) { @@ -1136,23 +1141,21 @@ void AIS_Manipulator::ComputeSelection (const Handle(SelectMgr_Selection)& theSe continue; } const Axis& anAxis = myAxes[anIt]; - if (aMode != AIS_MM_None) - { - anOwner = new AIS_ManipulatorOwner (this, anIt, AIS_MM_Translation, 9); - } + anOwner = new AIS_ManipulatorOwner(this, anIt, AIS_MM_Translation, 9); + // define sensitivity by line - Handle(Select3D_SensitiveSegment) aLine = new Select3D_SensitiveSegment (anOwner, gp::Origin(), anAxis.TranslatorTipPosition()); - aLine->SetSensitivityFactor (15); + Handle(Select3D_SensitiveSegment) aLine = new Select3D_SensitiveSegment(anOwner, gp::Origin(), anAxis.TranslatorTipPosition()); + aLine->SetSensitivityFactor (aHighSensitivity); theSelection->Add (aLine); // enlarge sensitivity by triangulation - Handle(Select3D_SensitivePrimitiveArray) aTri = new Select3D_SensitivePrimitiveArray (anOwner); + Handle(Select3D_SensitivePrimitiveArray) aTri = new Select3D_SensitivePrimitiveArray(anOwner); aTri->InitTriangulation (anAxis.TriangleArray()->Attributes(), anAxis.TriangleArray()->Indices(), TopLoc_Location()); theSelection->Add (aTri); } + break; } - - if (aMode == AIS_MM_Rotation || aMode == AIS_MM_None) + case AIS_MM_Rotation: { for (Standard_Integer anIt = 0; anIt < 3; ++anIt) { @@ -1161,22 +1164,20 @@ void AIS_Manipulator::ComputeSelection (const Handle(SelectMgr_Selection)& theSe continue; } const Axis& anAxis = myAxes[anIt]; - if (aMode != AIS_MM_None) - { - anOwner = new AIS_ManipulatorOwner (this, anIt, AIS_MM_Rotation, 9); - } + anOwner = new AIS_ManipulatorOwner(this, anIt, AIS_MM_Rotation, 9); + // define sensitivity by circle - const gp_Circ aGeomCircle (gp_Ax2 (gp::Origin(), anAxis.ReferenceAxis().Direction()), anAxis.RotatorDiskRadius()); - Handle(Select3D_SensitiveCircle) aCircle = new ManipSensCircle (anOwner, aGeomCircle); - aCircle->SetSensitivityFactor (15); - theSelection->Add (aCircle); + const gp_Circ aGeomCircle (gp_Ax2(gp::Origin(), anAxis.ReferenceAxis().Direction()), anAxis.RotatorDiskRadius()); + Handle(Select3D_SensitiveCircle) aCircle = new ManipSensCircle(anOwner, aGeomCircle); + aCircle->SetSensitivityFactor (aLowSensitivity); + theSelection->Add(aCircle); // enlarge sensitivity by triangulation - Handle(Select3D_SensitiveTriangulation) aTri = new ManipSensTriangulation (anOwner, myAxes[anIt].RotatorDisk().Triangulation(), anAxis.ReferenceAxis().Direction()); + Handle(Select3D_SensitiveTriangulation) aTri = new ManipSensTriangulation(anOwner, myAxes[anIt].RotatorDisk().Triangulation(), anAxis.ReferenceAxis().Direction()); theSelection->Add (aTri); } + break; } - - if (aMode == AIS_MM_Scaling || aMode == AIS_MM_None) + case AIS_MM_Scaling: { for (Standard_Integer anIt = 0; anIt < 3; ++anIt) { @@ -1184,21 +1185,19 @@ void AIS_Manipulator::ComputeSelection (const Handle(SelectMgr_Selection)& theSe { continue; } - if (aMode != AIS_MM_None) - { - anOwner = new AIS_ManipulatorOwner (this, anIt, AIS_MM_Scaling, 9); - } + anOwner = new AIS_ManipulatorOwner(this, anIt, AIS_MM_Scaling, 9); + // define sensitivity by point - Handle(Select3D_SensitivePoint) aPnt = new Select3D_SensitivePoint (anOwner, myAxes[anIt].ScalerCubePosition()); - aPnt->SetSensitivityFactor (15); + Handle(Select3D_SensitivePoint) aPnt = new Select3D_SensitivePoint(anOwner, myAxes[anIt].ScalerCubePosition()); + aPnt->SetSensitivityFactor (aHighSensitivity); theSelection->Add (aPnt); // enlarge sensitivity by triangulation - Handle(Select3D_SensitiveTriangulation) aTri = new Select3D_SensitiveTriangulation (anOwner, myAxes[anIt].ScalerCube().Triangulation(), TopLoc_Location(), Standard_True); + Handle(Select3D_SensitiveTriangulation) aTri = new Select3D_SensitiveTriangulation(anOwner, myAxes[anIt].ScalerCube().Triangulation(), TopLoc_Location(), Standard_True); theSelection->Add (aTri); } + break; } - - if (aMode == AIS_MM_TranslationPlane || aMode == AIS_MM_None) + case AIS_MM_TranslationPlane: { for (Standard_Integer anIt = 0; anIt < 3; ++anIt) { @@ -1206,28 +1205,33 @@ void AIS_Manipulator::ComputeSelection (const Handle(SelectMgr_Selection)& theSe { continue; } - if (aMode != AIS_MM_None) - { - anOwner = new AIS_ManipulatorOwner(this, anIt, AIS_MM_TranslationPlane, 9); - } + anOwner = new AIS_ManipulatorOwner(this, anIt, AIS_MM_TranslationPlane, 9); // define sensitivity by two crossed lines - gp_Pnt aP1, aP2; - aP1 = myAxes[((anIt + 1) % 3)].TranslatorTipPosition(); - aP2 = myAxes[((anIt + 2) % 3)].TranslatorTipPosition(); + Standard_Real aSensitivityOffset = ZoomPersistence() ? aHighSensitivity * (0.5 + M_SQRT2) : 0.0; + gp_Pnt aP1 = myAxes[((anIt + 1) % 3)].TranslatorTipPosition().Translated (myAxes[((anIt + 2) % 3)].ReferenceAxis().Direction().XYZ() * aSensitivityOffset); + gp_Pnt aP2 = myAxes[((anIt + 2) % 3)].TranslatorTipPosition().Translated (myAxes[((anIt + 1) % 3)].ReferenceAxis().Direction().XYZ() * aSensitivityOffset); gp_XYZ aMidP = (aP1.XYZ() + aP2.XYZ()) / 2.0; + gp_XYZ anOrig = aMidP.Normalized().Multiplied (aSensitivityOffset); Handle(Select3D_SensitiveSegment) aLine1 = new Select3D_SensitiveSegment(anOwner, aP1, aP2); - aLine1->SetSensitivityFactor(10); - theSelection->Add(aLine1); - Handle(Select3D_SensitiveSegment) aLine2 = new Select3D_SensitiveSegment(anOwner, gp::Origin(), aMidP); - aLine2->SetSensitivityFactor(10); - theSelection->Add(aLine2); + aLine1->SetSensitivityFactor(aLowSensitivity); + theSelection->Add (aLine1); + Handle(Select3D_SensitiveSegment) aLine2 = new Select3D_SensitiveSegment(anOwner, anOrig, aMidP); + aLine2->SetSensitivityFactor (aLowSensitivity); + theSelection->Add (aLine2); // enlarge sensitivity by triangulation - Handle(Select3D_SensitiveTriangulation) aTri = new Select3D_SensitiveTriangulation(anOwner, myAxes[anIt].DraggerSector().Triangulation(), TopLoc_Location(), Standard_True); - theSelection->Add(aTri); + Handle(Select3D_SensitiveTriangulation) aTri = new Select3D_SensitiveTriangulation(anOwner, myAxes[anIt].DraggerSector().Triangulation(), TopLoc_Location(), Standard_True); + theSelection->Add (aTri); } + break; + } + default: + { + anOwner = new SelectMgr_EntityOwner(this, 5); + break; + } } } diff --git a/src/SelectMgr/SelectMgr_Frustum.lxx b/src/SelectMgr/SelectMgr_Frustum.lxx index c01e50b6bf..882e74a37d 100644 --- a/src/SelectMgr/SelectMgr_Frustum.lxx +++ b/src/SelectMgr/SelectMgr_Frustum.lxx @@ -869,32 +869,27 @@ Standard_Boolean SelectMgr_Frustum::hasCircleOverlap (const Standard_Real the const gp_Pnt aCenterProject (aCoefA * aTCenter, aCoefB * aTCenter, aCoefC * aTCenter); - if (isDotInside (aCenterProject, aVertices)) - { - return true; - } - Standard_Boolean isInside = true; + const Standard_Boolean isCenterInside = isDotInside (aCenterProject, aVertices); + + Standard_Boolean isInside = false; for (Standard_Integer anIdx = aVertices.Lower(); anIdx <= aVertices.Upper(); anIdx++) { if (aVertices.Value (anIdx).Distance (aCenterProject) > theRadius) { - isInside = false; + isInside = true; break; } } if (theInside != NULL) { - *theInside = false; + *theInside = isInside && isCenterInside; } - if (!theIsFilled && isInside) - { - return false; - } - - return isInside; + return theIsFilled + ? !isInside || (isCenterInside && isInside) + : isInside && isCenterInside; } //======================================================================= diff --git a/src/SelectMgr/SelectMgr_SortCriterion.hxx b/src/SelectMgr/SelectMgr_SortCriterion.hxx index 6a0d9065d3..183965efc5 100644 --- a/src/SelectMgr/SelectMgr_SortCriterion.hxx +++ b/src/SelectMgr/SelectMgr_SortCriterion.hxx @@ -58,12 +58,33 @@ public: return ZLayerPosition > theOther.ZLayerPosition; } - // closest object is selected unless difference is within tolerance - if (Abs (Depth - theOther.Depth) > (Tolerance + theOther.Tolerance)) + // closest object is selected if their depths are not equal within tolerance + if (Abs (Depth - theOther.Depth) > Tolerance + theOther.Tolerance) { return Depth < theOther.Depth; } + Standard_Real aCos = 1.0; + if (Normal.Modulus() > 0 && theOther.Normal.Modulus() > 0) + { + gp_Dir aNormal (Normal.x(), Normal.y(), Normal.z()); + gp_Dir anOtherNormal (theOther.Normal.x(), theOther.Normal.y(), theOther.Normal.z()); + aCos = Abs (Cos (aNormal.Angle (anOtherNormal))); + } + + Standard_Real aDepth = Depth - Tolerance; + Standard_Real anOtherDepth = theOther.Depth - theOther.Tolerance; + // Comparison depths taking into account tolerances occurs when the surfaces are parallel + // or have the same sensitivity and the angle between them is less than 60 degrees. + if (Abs (aDepth - anOtherDepth) > Precision::Confusion()) + { + if ((aCos > 0.5 && Abs (Tolerance - theOther.Tolerance) < Precision::Confusion()) + || Abs (aCos - 1.0) < Precision::Confusion()) + { + return aDepth < anOtherDepth; + } + } + // if two objects have similar depth, select the one with higher priority if (Priority > theOther.Priority) { diff --git a/src/SelectMgr/SelectMgr_ViewerSelector.cxx b/src/SelectMgr/SelectMgr_ViewerSelector.cxx index 4ad9473157..5f86073410 100644 --- a/src/SelectMgr/SelectMgr_ViewerSelector.cxx +++ b/src/SelectMgr/SelectMgr_ViewerSelector.cxx @@ -284,9 +284,9 @@ void SelectMgr_ViewerSelector::checkOverlap (const Handle(Select3D_SensitiveEnti aCriterion.NbOwnerMatches = aPrevCriterion->NbOwnerMatches; if (theMgr.GetActiveSelectionType() != SelectMgr_SelectionType_Box) { + updatePoint3d (aCriterion, aPickResult, theEntity, theInversedTrsf, theMgr); if (aCriterion.IsCloserDepth (*aPrevCriterion)) { - updatePoint3d (aCriterion, aPickResult, theEntity, theInversedTrsf, theMgr); *aPrevCriterion = aCriterion; } } diff --git a/tests/v3d/dimensions/bug24389 b/tests/v3d/dimensions/bug24389 index 76ccda5ec0..666c1a57d3 100644 --- a/tests/v3d/dimensions/bug24389 +++ b/tests/v3d/dimensions/bug24389 @@ -176,7 +176,7 @@ check_picking $pick_coord $check_coord "diameter dimension (diam1)" check_cross_picking $pick_coord diam1 "diameter dimension (diam1)" # check sensitives "diam2" -set pick_coord { { 221 99 } { 285 99 } } +set pick_coord { { 222 99 } { 285 99 } } set check_coord { 239 99 } check_picking $pick_coord $check_coord "diameter dimension (diam2)" check_cross_picking $pick_coord diam2 "diameter dimension (diam2)" diff --git a/tests/v3d/manipulator/drag_in_2d_view b/tests/v3d/manipulator/drag_in_2d_view index 9eef222bb6..a648763549 100644 --- a/tests/v3d/manipulator/drag_in_2d_view +++ b/tests/v3d/manipulator/drag_in_2d_view @@ -15,8 +15,8 @@ vmanipulator m -attach b vdump $imagedir/${casename}_1.png -set mouse_pick {226 214} -set mouse_drag {306 265} +set mouse_pick {231 207} +set mouse_drag {311 258} # note: mouse events cannot be emulated here, so the original bug cannot be reproduced by this test case vmoveto {*}$mouse_pick diff --git a/tests/v3d/manipulator/dragg b/tests/v3d/manipulator/dragg index 7cb67c7f1e..db20f78729 100644 --- a/tests/v3d/manipulator/dragg +++ b/tests/v3d/manipulator/dragg @@ -61,7 +61,7 @@ vdump $anImage2 # ------------------------------------------- set mouse_pick {316 261} -set mouse_drag {279 286} +set mouse_drag {281 286} vmoveto {*}$mouse_pick vselect {*}$mouse_pick diff --git a/tests/v3d/manipulator/rotate b/tests/v3d/manipulator/rotate index 4b87f97fb6..0d2debeaa8 100644 --- a/tests/v3d/manipulator/rotate +++ b/tests/v3d/manipulator/rotate @@ -47,7 +47,7 @@ vmanipulator m -attach c1 -adjustPosition 1 -adjustSize 0 -enableModes 1 -size 4 vmanipulator m -followRotation 1 -set mouse_pick {200 092} +set mouse_pick {199 092} set mouse_drag {176 142} vmoveto {*}$mouse_pick @@ -65,8 +65,8 @@ vdump $anImage1 vmanipulator m -followRotation 1 -set mouse_pick {173 137} -set mouse_drag {233 144} +set mouse_pick {175 135} +set mouse_drag {232 144} vmoveto {*}$mouse_pick vselect {*}$mouse_pick @@ -113,7 +113,7 @@ vmanipulator m -followRotation 0 # test rotation around y axis (world reference frame) # --------------------------------------------------- -set mouse_pick {205 087} +set mouse_pick {205 088} set mouse_drag {232 127} vmoveto {*}$mouse_pick @@ -129,7 +129,7 @@ vdump $anImage4 # test rotation around z axis (world reference frame) # --------------------------------------------------- -set mouse_pick {228 141} +set mouse_pick {228 142} set mouse_drag {184 143} vmoveto {*}$mouse_pick diff --git a/tests/v3d/manipulator/translate b/tests/v3d/manipulator/translate index ddc0ec2563..69daa33716 100644 --- a/tests/v3d/manipulator/translate +++ b/tests/v3d/manipulator/translate @@ -144,7 +144,7 @@ vdump $anImage4 # ---------------------------------------------------- set mouse_pick {199 164} -set mouse_drag {246 177} +set mouse_drag {246 176} vmoveto {*}$mouse_pick vselect {*}$mouse_pick diff --git a/tests/vselect/bugs/bug23012 b/tests/vselect/bugs/bug23012 index de98fe644f..984c036aa0 100644 --- a/tests/vselect/bugs/bug23012 +++ b/tests/vselect/bugs/bug23012 @@ -11,7 +11,7 @@ set x1 210 set y1 184 set x2 207 -set y2 180 +set y2 182 stepread [locate_data_file OCC23012-Sample_3.stp] a * stepread [locate_data_file OCC23012-Sample_9.stp] b * diff --git a/tests/vselect/bugs/bug23539_2 b/tests/vselect/bugs/bug23539_2 index 05659b71b5..5577a18cae 100644 --- a/tests/vselect/bugs/bug23539_2 +++ b/tests/vselect/bugs/bug23539_2 @@ -16,7 +16,7 @@ vselect 300 200 300 60 400 60 407 150 -xor set NbSelected1 [vnbselected] if { ${NbSelected1} != 13 } { puts "Error : Polygonal shift selection doesn't work properly" } -vselect 350 120 -xor +vselect 350 121 -xor set NbSelected1 [vnbselected] if { ${NbSelected1} != 12 } { puts "Error : (case 2)" } diff --git a/tests/vselect/bugs/bug27848 b/tests/vselect/bugs/bug27848 new file mode 100644 index 0000000000..19e4d5a33a --- /dev/null +++ b/tests/vselect/bugs/bug27848 @@ -0,0 +1,23 @@ +puts "=============" +puts "0027848: Visualization - sensitivity of lines is too high" +puts "=============" + +pload VISUALIZATION + +vinit View1 +vclear + +box b 10 10 0.1 +vdisplay b -dispmode 1 + +vline l 0 0 0 10 10 0 +vdisplay l + +vpoint p 20 20 0 + +vtop +vfit + +vselect 100 305 + +if { [string match "*AIS_Line*" [vstate]] } { puts "Error: AIS_Shape should be detected" } diff --git a/tests/vselect/edge/A10 b/tests/vselect/edge/A10 index f42646f62d..2738b8c05c 100644 --- a/tests/vselect/edge/A10 +++ b/tests/vselect/edge/A10 @@ -39,28 +39,28 @@ if { ![check_highlighting 0 $coords] } { puts "ERROR: incorrect highlighting of edge 10" } -vselnext -if { ![check_highlighting 1 $coords] } { - puts "ERROR: incorrect highlighting of edge 2 after vselnext call" -} vselnext if { ![check_highlighting 2 $coords] } { puts "ERROR: incorrect highlighting of edge 1 after vselnext call" } vselnext +if { ![check_highlighting 1 $coords] } { + puts "ERROR: incorrect highlighting of edge 2 after vselnext call" +} +vselnext if { ![check_highlighting 0 $coords] } { puts "ERROR: incorrect highlighting of edge 10 after vselnext call" } -vselprev -if { ![check_highlighting 2 $coords] } { - puts "ERROR: incorrect highlighting of edge 1 after vselprev call" -} vselprev if { ![check_highlighting 1 $coords] } { puts "ERROR: incorrect highlighting of edge 2 after vselprev call" } vselprev +if { ![check_highlighting 2 $coords] } { + puts "ERROR: incorrect highlighting of edge 1 after vselprev call" +} +vselprev if { ![check_highlighting 0 $coords] } { puts "ERROR: incorrect highlighting of edge 10 after vselprev call" }