From b586500b1e0a0d74472af2846564b636cb7f2e57 Mon Sep 17 00:00:00 2001 From: vpa Date: Thu, 14 May 2015 19:32:05 +0300 Subject: [PATCH] 0026209: Visualization - provide a method to fit view to the specific bounding box Methods V3d_View::FitAll and NIS_View::FitAll3d now take arbitrary bounding box as a parameter; Option -selected added to vfit command to fit only selected entities in Draw; F key press now fits selected objects if any by default. --- src/AIS/AIS_InteractiveContext.cdl | 10 +++ src/AIS/AIS_InteractiveContext.cxx | 65 +++++++++++++++++++ src/AIS/AIS_InteractiveContext_1.cxx | 7 +- src/AIS/AIS_LocalContext.cxx | 15 +++-- src/AIS/AIS_LocalContext.lxx | 2 +- src/AIS/AIS_LocalContext_1.cxx | 42 ++++++------ src/SelectMgr/FILES | 1 + src/SelectMgr/SelectMgr.cdl | 5 +- src/SelectMgr/SelectMgr_IndexedMapOfOwner.hxx | 28 ++++++++ src/SelectMgr/SelectMgr_SelectableObject.cdl | 7 ++ src/SelectMgr/SelectMgr_SelectableObject.cxx | 46 +++++++++++++ src/V3d/V3d_View.cdl | 13 ++++ src/V3d/V3d_View.cxx | 15 ++++- src/ViewerTest/ViewerTest_ViewerCommands.cxx | 55 ++++++++++++++-- tests/bugs/vis/bug26209 | 33 ++++++++++ 15 files changed, 301 insertions(+), 43 deletions(-) create mode 100644 src/SelectMgr/SelectMgr_IndexedMapOfOwner.hxx create mode 100644 tests/bugs/vis/bug26209 diff --git a/src/AIS/AIS_InteractiveContext.cdl b/src/AIS/AIS_InteractiveContext.cdl index 8026b714fc..b6cbb71695 100644 --- a/src/AIS/AIS_InteractiveContext.cdl +++ b/src/AIS/AIS_InteractiveContext.cdl @@ -1942,6 +1942,16 @@ is ClearActiveSensitive(me:mutable;aView:View from V3d) is static; + FitSelected (me : mutable; + theView : View from V3d; + theMargin : Real from Standard = 0.01; + theToUpdate : Boolean from Standard = Standard_True) + is static; + ---Level: Public + ---Purpose: Fits the view correspondingly to the bounds of selected objects. + -- Infinite objects are ignored if infinite state of AIS_InteractiveObject + -- is set to true. + DisplayActiveSensitive(me:mutable; diff --git a/src/AIS/AIS_InteractiveContext.cxx b/src/AIS/AIS_InteractiveContext.cxx index e653481028..ef088df226 100644 --- a/src/AIS/AIS_InteractiveContext.cxx +++ b/src/AIS/AIS_InteractiveContext.cxx @@ -68,6 +68,9 @@ namespace return TCollection_AsciiString ("AIS_CurContext_") + TCollection_AsciiString (Standard_Atomic_Increment (&THE_AIS_INDEX_CUR)); } + + typedef NCollection_DataMap AIS_MapOfObjectOwners; + typedef NCollection_DataMap::Iterator AIS_MapIteratorOfMapOfObjectOwners; } //======================================================================= @@ -2824,3 +2827,65 @@ void AIS_InteractiveContext::Disconnect (const Handle(AIS_InteractiveObject)& th else return; } + +//======================================================================= +//function : FitSelected +//purpose : Fits the view corresponding to the bounds of selected objects +//======================================================================= +void AIS_InteractiveContext::FitSelected (const Handle(V3d_View)& theView, + const Standard_Real theMargin, + const Standard_Boolean theToUpdate) +{ + Standard_CString aSelName = HasOpenedContext() ? + myLocalContexts (myCurLocalIndex)->SelectionName().ToCString() + : myCurrentName.ToCString(); + + Bnd_Box aBndSelected; + + const Handle(AIS_Selection)& aSelection = AIS_Selection::Selection (aSelName); + AIS_MapOfObjectOwners anObjectOwnerMap; + for (aSelection->Init(); aSelection->More(); aSelection->Next()) + { + const Handle(AIS_InteractiveObject)& anObj = + Handle(AIS_InteractiveObject)::DownCast (aSelection->Value()); + if (!anObj.IsNull()) + { + if (anObj->IsInfinite()) + continue; + + Bnd_Box aTmpBnd; + anObj->BoundingBox (aTmpBnd); + aBndSelected.Add (aTmpBnd); + } + else + { + const Handle(SelectMgr_EntityOwner)& anOwner = + Handle(SelectMgr_EntityOwner)::DownCast (aSelection->Value()); + if (anOwner.IsNull()) + continue; + + Handle(SelectMgr_IndexedMapOfOwner) anOwnerMap; + if (!anObjectOwnerMap.Find (anOwner->Selectable(), anOwnerMap)) + { + anOwnerMap = new SelectMgr_IndexedMapOfOwner(); + anObjectOwnerMap.Bind (anOwner->Selectable(), anOwnerMap); + } + + anOwnerMap->Add (anOwner); + } + } + + for (AIS_MapIteratorOfMapOfObjectOwners anIter (anObjectOwnerMap); anIter.More(); anIter.Next()) + { + const Handle(SelectMgr_SelectableObject) anObject = anIter.Key(); + Bnd_Box aTmpBox = anObject->BndBoxOfSelected (anIter.ChangeValue()); + aBndSelected.Add (aTmpBox); + } + + anObjectOwnerMap.Clear(); + + if (aBndSelected.IsVoid()) + return; + + theView->FitAll (aBndSelected, theMargin, theToUpdate); +} diff --git a/src/AIS/AIS_InteractiveContext_1.cxx b/src/AIS/AIS_InteractiveContext_1.cxx index d82f767c21..cbbacf2d37 100644 --- a/src/AIS/AIS_InteractiveContext_1.cxx +++ b/src/AIS/AIS_InteractiveContext_1.cxx @@ -1079,7 +1079,7 @@ Handle(SelectMgr_EntityOwner) AIS_InteractiveContext::SelectedOwner() const //function : EntityOwners //purpose : //======================================================================= -void AIS_InteractiveContext::EntityOwners(SelectMgr_IndexedMapOfOwner& theOwners, +void AIS_InteractiveContext::EntityOwners(Handle(SelectMgr_IndexedMapOfOwner)& theOwners, const Handle(AIS_InteractiveObject)& theIObj, const Standard_Integer theMode) const { @@ -1092,6 +1092,9 @@ void AIS_InteractiveContext::EntityOwners(SelectMgr_IndexedMapOfOwner& theOwners else aModes.Append( theMode ); + if (theOwners.IsNull()) + theOwners = new SelectMgr_IndexedMapOfOwner(); + TColStd_ListIteratorOfListOfInteger anItr( aModes ); for (; anItr.More(); anItr.Next() ) { @@ -1110,7 +1113,7 @@ void AIS_InteractiveContext::EntityOwners(SelectMgr_IndexedMapOfOwner& theOwners Handle(SelectMgr_EntityOwner) aOwner = Handle(SelectMgr_EntityOwner)::DownCast(aEntity->OwnerId()); if ( !aOwner.IsNull() ) - theOwners.Add( aOwner ); + theOwners->Add( aOwner ); } } } diff --git a/src/AIS/AIS_LocalContext.cxx b/src/AIS/AIS_LocalContext.cxx index 259cf016f6..a072d0d9e1 100644 --- a/src/AIS/AIS_LocalContext.cxx +++ b/src/AIS/AIS_LocalContext.cxx @@ -74,6 +74,7 @@ mySM(aCtx->SelectionManager()), myMainVS(aCtx->MainSelector()), myFilters(new SelectMgr_OrFilter()), myAutoHilight(Standard_True), +myMapOfOwner (new SelectMgr_IndexedMapOfOwner()), mylastindex(0), mylastgood(0), myCurDetected(0), @@ -606,7 +607,7 @@ void AIS_LocalContext::Terminate (const Standard_Boolean theToUpdate) { ClearDetected(); Clear(); - myMapOfOwner.Clear(); + myMapOfOwner->Clear(); mylastindex=0; // clear the selector... @@ -1050,16 +1051,16 @@ HasFilters(const TopAbs_ShapeEnum aType) const void AIS_LocalContext::ClearDetected() { - for(Standard_Integer I=1;I<=myMapOfOwner.Extent();I++) + for(Standard_Integer I=1;I<=myMapOfOwner->Extent();I++) { - if(!myMapOfOwner(I).IsNull()) + if(!myMapOfOwner->FindKey (I).IsNull()) { - if(myMapOfOwner(I)->IsHilighted(myMainPM)) - myMapOfOwner(I)->Unhilight(myMainPM); + if(myMapOfOwner->FindKey (I)->IsHilighted(myMainPM)) + myMapOfOwner->FindKey (I)->Unhilight(myMainPM); else { const Handle(SelectMgr_SelectableObject)& SO = - myMapOfOwner.FindKey(I)->Selectable(); + myMapOfOwner->FindKey (I)->Selectable(); if(myActiveObjects.IsBound(SO)) { const Handle(AIS_LocalStatus)& Att = myActiveObjects(SO); @@ -1068,7 +1069,7 @@ void AIS_LocalContext::ClearDetected() Att->DisplayMode()==-1 && Att->SelectionModes().IsEmpty()) { - myMapOfOwner(I)->Clear(myMainPM); + myMapOfOwner->FindKey (I)->Clear(myMainPM); } } } diff --git a/src/AIS/AIS_LocalContext.lxx b/src/AIS/AIS_LocalContext.lxx index aebb0905be..c72cff74bf 100644 --- a/src/AIS/AIS_LocalContext.lxx +++ b/src/AIS/AIS_LocalContext.lxx @@ -61,4 +61,4 @@ inline Standard_Boolean AIS_LocalContext::HasNextDetected() const {return myCurDetected0 && indx<=myMapOfOwner.Extent());} +{ return (indx>0 && indx<=myMapOfOwner->Extent());} diff --git a/src/AIS/AIS_LocalContext_1.cxx b/src/AIS/AIS_LocalContext_1.cxx index 04c094d896..7a405b6039 100644 --- a/src/AIS/AIS_LocalContext_1.cxx +++ b/src/AIS/AIS_LocalContext_1.cxx @@ -95,10 +95,10 @@ AIS_StatusOfDetection AIS_LocalContext::MoveTo (const Standard_Integer theXpix, // result of courses.. if (aDetectedNb == 0 || myDetectedSeq.IsEmpty()) { - if (mylastindex != 0 && mylastindex <= myMapOfOwner.Extent()) + if (mylastindex != 0 && mylastindex <= myMapOfOwner->Extent()) { myMainPM->ClearImmediateDraw(); - Unhilight (myMapOfOwner (mylastindex), theView); + Unhilight (myMapOfOwner->FindKey (mylastindex), theView); if (theToRedrawImmediate) { theView->RedrawImmediate(); @@ -148,7 +148,7 @@ AIS_StatusOfPick AIS_LocalContext::Select (const Standard_Boolean toUpdateViewer return (AIS_Selection::Extent() == 0) ? AIS_SOP_NothingSelected : AIS_SOP_Removed; } - const Handle(SelectMgr_EntityOwner)& anOwner = myMapOfOwner (aDetIndex); + const Handle(SelectMgr_EntityOwner)& anOwner = myMapOfOwner->FindKey (aDetIndex); ClearSelected (Standard_False); @@ -314,7 +314,7 @@ AIS_StatusOfPick AIS_LocalContext::ShiftSelect (const Standard_Boolean toUpdateV { AIS_Selection::SetCurrentSelection (mySelName.ToCString()); Standard_Integer aSelNum = AIS_Selection::Extent(); - const Handle(SelectMgr_EntityOwner)& anOwner = myMapOfOwner (aDetIndex); + const Handle(SelectMgr_EntityOwner)& anOwner = myMapOfOwner->FindKey (aDetIndex); Standard_Boolean toSelect = anOwner->IsSelected() ? Standard_False : Standard_True; AIS_Selection::Select (anOwner); anOwner->SetSelected (toSelect); @@ -940,9 +940,9 @@ void AIS_LocalContext::ClearOutdatedSelection (const Handle(AIS_InteractiveObjec // 4. AIS_LocalContext - myMapOfOwner : remove entity owners from myMapOfOwner SelectMgr_IndexedMapOfOwner anOwnersToKeep; - for (Standard_Integer anIdx = 1; anIdx <= myMapOfOwner.Extent(); anIdx++) + for (Standard_Integer anIdx = 1; anIdx <= myMapOfOwner->Extent(); anIdx++) { - Handle(SelectMgr_EntityOwner) anOwner = myMapOfOwner (anIdx); + Handle(SelectMgr_EntityOwner) anOwner = myMapOfOwner->FindKey (anIdx); if (anOwner.IsNull()) { continue; @@ -960,9 +960,9 @@ void AIS_LocalContext::ClearOutdatedSelection (const Handle(AIS_InteractiveObjec } } } - myMapOfOwner.Clear(); - myMapOfOwner.Assign (anOwnersToKeep); - mylastindex = myMapOfOwner.FindIndex (aLastPicked); + myMapOfOwner->Clear(); + myMapOfOwner->Assign (anOwnersToKeep); + mylastindex = myMapOfOwner->FindIndex (aLastPicked); if (!IsValidIndex (mylastindex)) { myMainPM->ClearImmediateDraw(); @@ -1159,9 +1159,9 @@ void AIS_LocalContext::manageDetected (const Handle(SelectMgr_EntityOwner)& theP // //======================================================================================================= - const Standard_Integer aNewIndex = myMapOfOwner.Contains (thePickOwner) - ? myMapOfOwner.FindIndex (thePickOwner) - : myMapOfOwner.Add (thePickOwner); + const Standard_Integer aNewIndex = myMapOfOwner->Contains (thePickOwner) + ? myMapOfOwner->FindIndex (thePickOwner) + : myMapOfOwner->Add (thePickOwner); // For the advanced mesh selection mode the owner indices comparison // is not effective because in that case only one owner manage the @@ -1173,9 +1173,9 @@ void AIS_LocalContext::manageDetected (const Handle(SelectMgr_EntityOwner)& theP { myMainPM->ClearImmediateDraw(); if (mylastindex != 0 - && mylastindex <= myMapOfOwner.Extent()) + && mylastindex <= myMapOfOwner->Extent()) { - const Handle(SelectMgr_EntityOwner)& aLastOwner = myMapOfOwner (mylastindex); + const Handle(SelectMgr_EntityOwner)& aLastOwner = myMapOfOwner->FindKey (mylastindex); Unhilight (aLastOwner, theView); } @@ -1222,7 +1222,7 @@ AIS_LocalContext::DetectedShape() const static TopoDS_Shape bidsh; if(mylastindex != 0) { - Handle(StdSelect_BRepOwner) BROwnr = Handle(StdSelect_BRepOwner)::DownCast(myMapOfOwner(mylastindex)); + Handle(StdSelect_BRepOwner) BROwnr = Handle(StdSelect_BRepOwner)::DownCast(myMapOfOwner->FindKey (mylastindex)); if(BROwnr.IsNull()) return bidsh; return BROwnr->Shape(); } @@ -1239,7 +1239,7 @@ AIS_LocalContext::DetectedInteractive() const { Handle(AIS_InteractiveObject) Iobj; if(IsValidIndex(mylastindex)){ - Handle(SelectMgr_SelectableObject) SO = myMapOfOwner.FindKey(mylastindex)->Selectable(); + Handle(SelectMgr_SelectableObject) SO = myMapOfOwner->FindKey(mylastindex)->Selectable(); Iobj = *((Handle(AIS_InteractiveObject)*) &SO); } return Iobj; @@ -1252,7 +1252,7 @@ Handle(SelectMgr_EntityOwner) AIS_LocalContext::DetectedOwner() const { Handle(SelectMgr_EntityOwner) bid; if(!IsValidIndex(mylastindex)) return bid; - return myMapOfOwner.FindKey(mylastindex); + return myMapOfOwner->FindKey(mylastindex); } @@ -1263,7 +1263,7 @@ Handle(SelectMgr_EntityOwner) AIS_LocalContext::DetectedOwner() const Standard_Boolean AIS_LocalContext::ComesFromDecomposition(const Standard_Integer PickedIndex) const { - const Handle(SelectMgr_EntityOwner)& OWN = myMapOfOwner.FindKey(PickedIndex); + const Handle(SelectMgr_EntityOwner)& OWN = myMapOfOwner->FindKey(PickedIndex); Handle(SelectMgr_SelectableObject) aSel = OWN->Selectable(); if (myActiveObjects.IsBound (aSel)) { // debug of jmi const Handle(AIS_LocalStatus)& Stat = myActiveObjects(aSel); @@ -1300,7 +1300,7 @@ void AIS_LocalContext::ClearSensitive(const Handle(V3d_View)& aviou) Standard_Boolean AIS_LocalContext::IsShape(const Standard_Integer Index) const { - if(Handle(StdSelect_BRepOwner)::DownCast(myMapOfOwner.FindKey(Index)).IsNull()) + if(Handle(StdSelect_BRepOwner)::DownCast(myMapOfOwner->FindKey(Index)).IsNull()) return Standard_False; return ComesFromDecomposition(Index); @@ -1383,12 +1383,12 @@ Standard_Boolean AIS_LocalContext::UnhilightLastDetected (const Handle(V3d_View) } myMainPM->BeginImmediateDraw(); - const Handle(SelectMgr_EntityOwner)& anOwner = myMapOfOwner (mylastindex); + const Handle(SelectMgr_EntityOwner)& anOwner = myMapOfOwner->FindKey (mylastindex); const Standard_Integer aHilightMode = anOwner->HasSelectable() ? GetHiMod (Handle(AIS_InteractiveObject)::DownCast (anOwner->Selectable())) : 0; - myMapOfOwner (mylastindex)->Unhilight (myMainPM, aHilightMode); + myMapOfOwner->FindKey (mylastindex)->Unhilight (myMainPM, aHilightMode); myMainPM->EndImmediateDraw (theView); mylastindex = 0; return Standard_True; diff --git a/src/SelectMgr/FILES b/src/SelectMgr/FILES index 9bf32bac46..a8e5cfdedc 100755 --- a/src/SelectMgr/FILES +++ b/src/SelectMgr/FILES @@ -1,6 +1,7 @@ SelectMgr_CompareResults.hxx SelectMgr_FrustumBuilder.hxx SelectMgr_FrustumBuilder.cxx +SelectMgr_IndexedMapOfOwner.hxx SelectMgr_SelectableObjectSet.hxx SelectMgr_SelectableObjectSet.cxx SelectMgr_BaseFrustum.hxx diff --git a/src/SelectMgr/SelectMgr.cdl b/src/SelectMgr/SelectMgr.cdl index 954487e457..278acdb8be 100644 --- a/src/SelectMgr/SelectMgr.cdl +++ b/src/SelectMgr/SelectMgr.cdl @@ -236,9 +236,6 @@ is class SequenceOfOwner instantiates Sequence from TCollection (EntityOwner from SelectMgr); - class IndexedMapOfOwner instantiates IndexedMap from TCollection - (EntityOwner from SelectMgr,MapTransientHasher from TColStd); - class SequenceOfSelector instantiates Sequence from TCollection (ViewerSelector from SelectMgr); @@ -256,7 +253,7 @@ is pointer SOPtr to SelectableObject from SelectMgr; imported CompareResults; - + imported transient class IndexedMapOfOwner; imported SelectableObjectSet; imported FrustumBuilder; imported BaseFrustum; diff --git a/src/SelectMgr/SelectMgr_IndexedMapOfOwner.hxx b/src/SelectMgr/SelectMgr_IndexedMapOfOwner.hxx new file mode 100644 index 0000000000..1ae2dbc153 --- /dev/null +++ b/src/SelectMgr/SelectMgr_IndexedMapOfOwner.hxx @@ -0,0 +1,28 @@ +// Created on: 2015-05-14 +// Created by: Varvara POSKONINA +// Copyright (c) 2005-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 _SelectMgr_IndexedMapOfOwner_HeaderFile +#define _SelectMgr_IndexedMapOfOwner_HeaderFile + +#include +#include + +class SelectMgr_EntityOwner; +class Handle(SelectMgr_EntityOwner); + +typedef NCollection_IndexedMap SelectMgr_IndexedMapOfOwner; +typedef NCollection_Handle > Handle(SelectMgr_IndexedMapOfOwner); + +#endif // _SelectMgr_IndexedMapOfOwner_HeaderFile diff --git a/src/SelectMgr/SelectMgr_SelectableObject.cdl b/src/SelectMgr/SelectMgr_SelectableObject.cdl index 6231ea3b32..5e1394bfb1 100644 --- a/src/SelectMgr/SelectMgr_SelectableObject.cdl +++ b/src/SelectMgr/SelectMgr_SelectableObject.cdl @@ -30,6 +30,7 @@ deferred class SelectableObject from SelectMgr inherits PresentableObject from uses Box from Bnd, + IndexedMapOfOwner from SelectMgr, SelectionManager from SelectMgr, Selection from SelectMgr, SequenceOfSelection from SelectMgr, @@ -239,6 +240,12 @@ is ---C++: return const& ---Purpose: Returns common entity owner if the object is an assembly + BndBoxOfSelected (me : mutable; + theOwners : out IndexedMapOfOwner from SelectMgr) + returns Box from Bnd; + ---Purpose: Returns a bounding box of sensitive entities with the owners given + -- if they are a part of activated selection + fields myselections : SequenceOfSelection is protected; diff --git a/src/SelectMgr/SelectMgr_SelectableObject.cxx b/src/SelectMgr/SelectMgr_SelectableObject.cxx index 3985350d93..e59069c047 100644 --- a/src/SelectMgr/SelectMgr_SelectableObject.cxx +++ b/src/SelectMgr/SelectMgr_SelectableObject.cxx @@ -580,3 +580,49 @@ const Handle(SelectMgr_EntityOwner)& SelectMgr_SelectableObject::GetAssemblyOwne { return myAssemblyOwner; } + +//======================================================================= +//function : BndBoxOfSelected +//purpose : Returns a bounding box of sensitive entities with the owners given +// if they are a part of activated selection +//======================================================================= +Bnd_Box SelectMgr_SelectableObject::BndBoxOfSelected (Handle(SelectMgr_IndexedMapOfOwner)& theOwners) +{ + Bnd_Box aBnd; + + if (theOwners->IsEmpty()) + return aBnd; + + for (Init(); More(); Next()) + { + const Handle(SelectMgr_Selection)& aSel = CurrentSelection(); + if (aSel->GetSelectionState() != SelectMgr_SOS_Activated) + continue; + + for (aSel->Init(); aSel->More(); aSel->Next()) + { + const Handle(SelectMgr_EntityOwner) anOwner = + Handle(SelectMgr_EntityOwner)::DownCast (aSel->Sensitive()->BaseSensitive()->OwnerId()); + if (theOwners->Contains (anOwner)) + { + Select3D_BndBox3d aBox = aSel->Sensitive()->BaseSensitive()->BoundingBox(); + Bnd_Box aTmpBnd; + aTmpBnd.Update (aBox.CornerMin().x(), aBox.CornerMin().y(), aBox.CornerMin().z(), + aBox.CornerMax().x(), aBox.CornerMax().y(), aBox.CornerMax().z()); + aBnd.Add (aTmpBnd); + + Standard_Integer anOwnerIdx = theOwners->FindIndex (anOwner); + if (theOwners->Size() != anOwnerIdx) + { + theOwners->Swap (anOwnerIdx, theOwners->Size()); + } + theOwners->RemoveLast(); + + if (theOwners->IsEmpty()) + return aBnd; + } + } + } + + return aBnd; +} diff --git a/src/V3d/V3d_View.cdl b/src/V3d/V3d_View.cdl index e9d71c157c..2aba4de34e 100644 --- a/src/V3d/V3d_View.cdl +++ b/src/V3d/V3d_View.cdl @@ -796,6 +796,19 @@ is -- @param theMargin [in] the margin coefficient for view borders. -- @param theToUpdate [in] flag to perform view update. + FitAll (me : mutable; + theBox : Box from Bnd; + theMargin : Coefficient = 0.01; + theToUpdate : Boolean from Standard = Standard_True); + ---Level: Public + ---Purpose: Adjust view parameters to fit the displayed scene, respecting height / width ratio + -- according to the custom bounding box given. + -- Throws program error exception if margin coefficient is < 0 or >= 1. + -- Updates the view. + -- @param theBox [in] the custom bounding box to fit. + -- @param theMargin [in] the margin coefficient for view borders. + -- @param theToUpdate [in] flag to perform view update. + DepthFitAll( me : mutable ; Aspect : Coefficient = 0.01; Margin : Coefficient = 0.01 ); ---Level: Public diff --git a/src/V3d/V3d_View.cxx b/src/V3d/V3d_View.cxx index d93a45cf7e..132caf8171 100644 --- a/src/V3d/V3d_View.cxx +++ b/src/V3d/V3d_View.cxx @@ -1451,16 +1451,25 @@ void V3d_View::SetAxialScale( const Standard_Real Sx, const Standard_Real Sy, co //function : FitAll //purpose : //============================================================================= -void V3d_View::FitAll (const Standard_Real theMargin, const Standard_Boolean theToUpdate) +void V3d_View::FitAll (const Quantity_Coefficient theMargin, const Standard_Boolean theToUpdate) { - Standard_ASSERT_RAISE (theMargin >= 0.0 && theMargin < 1.0, "Invalid margin coefficient"); + FitAll (MyView->MinMaxValues(), theMargin, theToUpdate); +} + +//============================================================================= +//function : FitAll +//purpose : +//============================================================================= +void V3d_View::FitAll (const Bnd_Box& theBox, const Quantity_Coefficient theMargin, const Standard_Boolean theToUpdate) +{ + Standard_ASSERT_RAISE(theMargin >= 0.0 && theMargin < 1.0, "Invalid margin coefficient"); if (MyView->NumberOfDisplayedStructures() == 0) { return; } - if (!FitMinMax (myCamera, MyView->MinMaxValues(), theMargin, 10.0 * Precision::Confusion())) + if (!FitMinMax (myCamera, theBox, theMargin, 10.0 * Precision::Confusion())) { return; } diff --git a/src/ViewerTest/ViewerTest_ViewerCommands.cxx b/src/ViewerTest/ViewerTest_ViewerCommands.cxx index 844efb7f14..864d3eaf8c 100644 --- a/src/ViewerTest/ViewerTest_ViewerCommands.cxx +++ b/src/ViewerTest/ViewerTest_ViewerCommands.cxx @@ -1345,8 +1345,15 @@ void VT_ProcessKeyPress (const char* buf_ret) } else if (!strcasecmp (buf_ret, "F")) { - // FitAll - aView->FitAll(); + if (ViewerTest::GetAISContext()->NbSelected() > 0) + { + ViewerTest::GetAISContext()->FitSelected (aView); + } + else + { + // FitAll + aView->FitAll(); + } } else if (!strcasecmp (buf_ret, "H")) { @@ -2458,7 +2465,6 @@ static void OSWindowSetup() } - //============================================================================== //function : VFit @@ -2466,11 +2472,49 @@ static void OSWindowSetup() //Draw arg : No args //============================================================================== -static int VFit(Draw_Interpretor& , Standard_Integer , const char** ) +static int VFit (Draw_Interpretor& /*theDi*/, Standard_Integer theArgc, const char** theArgv) { + if (theArgc > 2) + { + std::cout << "Wrong number of arguments! Use: vfit [-selected]" << std::endl; + } + const Handle(V3d_View) aView = ViewerTest::CurrentView(); +<<<<<<< .mine + Handle(NIS_View) V = Handle(NIS_View)::DownCast (aView); + + if (theArgc == 2) + { + TCollection_AsciiString anArg (theArgv[1]); + anArg.LowerCase(); + if (anArg == "-selected") + { + ViewerTest::GetAISContext()->FitSelected (V.IsNull() ? aView : V); + return 0; + } + } + + if (V.IsNull() == Standard_False) { + V->FitAll3d(); + } else if (aView.IsNull() == Standard_False) { +======= if (!aView.IsNull()) { + + + + + + + + + + + + + + +>>>>>>> .theirs aView->FitAll(); } return 0; @@ -8413,7 +8457,8 @@ void ViewerTest::ViewerCommands(Draw_Interpretor& theCommands) "vpick : vpick X Y Z [shape subshape] ( all variables as string )", VPick,group); theCommands.Add("vfit" , - "vfit or : vfit", + "vfit or [-selected]" + "\n\t\t: [-selected] fits the scene according to bounding box of currently selected objects", __FILE__,VFit,group); theCommands.Add ("vfitarea", "vfitarea x1 y1 x2 y2" diff --git a/tests/bugs/vis/bug26209 b/tests/bugs/vis/bug26209 new file mode 100644 index 0000000000..f907030b24 --- /dev/null +++ b/tests/bugs/vis/bug26209 @@ -0,0 +1,33 @@ +puts "============" +puts "CR26209" +puts "============" +puts "" + +########################################################################################## +puts "Visualization - provide a method to fit view to the specific bounding box" +########################################################################################## + +pload VISUALIZATION MODELING + +box b 1 1 1 +vinit View1 +vdisplay b +vfit +vselmode b 2 1 +vselmode b 1 1 + +vselect 330 334 +vfit -selected +checkcolor 330 334 0 0 0 +checkcolor 330 131 0.8 0.8 0.8 + +vselect 0 0 +vfit + +vselect 29 104 +vselect 204 2 1 +vfit -selected +checkcolor 29 104 0 0 0 +checkcolor 2 317 0.8 0.8 0.8 + +vdump ${imagedir}/${casename}.png