2010-05-14 21:32:37 +06:00
|
|
|
// Copyright (C) 2007-2010 CEA/DEN, EDF R&D, OPEN CASCADE
|
2009-02-17 10:27:49 +05:00
|
|
|
//
|
|
|
|
// Copyright (C) 2003-2007 OPEN CASCADE, EADS/CCR, LIP6, CEA/DEN,
|
2005-06-07 19:22:20 +06:00
|
|
|
// CEDRAT, EDF R&D, LEG, PRINCIPIA R&D, BUREAU VERITAS
|
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
|
|
// License as published by the Free Software Foundation; either
|
|
|
|
// version 2.1 of the License.
|
|
|
|
//
|
|
|
|
// This library is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
// License along with this library; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
//
|
2009-02-17 10:27:49 +05:00
|
|
|
// See http://www.salome-platform.org/ or email : webmaster.salome@opencascade.com
|
|
|
|
//
|
2010-05-14 21:32:37 +06:00
|
|
|
|
2009-02-17 10:27:49 +05:00
|
|
|
// SMESH SMESHGUI : GUI for SMESH component
|
|
|
|
// File : SMESHGUI_Utils.cxx
|
|
|
|
// Author : Open CASCADE S.A.S.
|
|
|
|
// SMESH includes
|
|
|
|
//
|
2004-12-01 15:48:31 +05:00
|
|
|
#include "SMESHGUI_Utils.h"
|
2008-03-07 12:47:05 +05:00
|
|
|
#include "SMESHGUI.h"
|
2004-12-01 15:48:31 +05:00
|
|
|
|
2009-02-17 10:27:49 +05:00
|
|
|
#include <SMDS_MeshNode.hxx>
|
|
|
|
#include <SMDS_MeshFace.hxx>
|
2005-06-07 19:22:20 +06:00
|
|
|
|
2009-02-17 10:27:49 +05:00
|
|
|
// SALOME GUI includes
|
|
|
|
#include <SUIT_Desktop.h>
|
|
|
|
#include <SUIT_Session.h>
|
|
|
|
#include <SUIT_MessageBox.h>
|
|
|
|
#include <SUIT_ResourceMgr.h>
|
2004-12-01 15:48:31 +05:00
|
|
|
|
2009-02-17 10:27:49 +05:00
|
|
|
#include <LightApp_SelectionMgr.h>
|
|
|
|
#include <SalomeApp_Application.h>
|
|
|
|
#include <SalomeApp_Module.h>
|
|
|
|
#include <SalomeApp_Study.h>
|
2004-12-01 15:48:31 +05:00
|
|
|
|
2009-02-17 10:27:49 +05:00
|
|
|
#include <SALOME_ListIO.hxx>
|
2004-12-01 15:48:31 +05:00
|
|
|
|
2009-02-17 10:27:49 +05:00
|
|
|
// OCC includes
|
|
|
|
#include <gp_XYZ.hxx>
|
|
|
|
#include <TColgp_Array1OfXYZ.hxx>
|
2004-12-01 15:48:31 +05:00
|
|
|
|
2009-02-17 10:27:49 +05:00
|
|
|
namespace SMESH
|
|
|
|
{
|
2005-06-07 19:22:20 +06:00
|
|
|
SUIT_Desktop*
|
|
|
|
GetDesktop(const CAM_Module* theModule)
|
|
|
|
{
|
|
|
|
return theModule->application()->desktop();
|
|
|
|
}
|
|
|
|
|
2005-11-03 13:38:17 +05:00
|
|
|
LightApp_SelectionMgr*
|
2005-06-07 19:22:20 +06:00
|
|
|
GetSelectionMgr(const SalomeApp_Module* theModule)
|
|
|
|
{
|
|
|
|
return theModule->getApp()->selectionMgr();
|
2004-12-01 15:48:31 +05:00
|
|
|
}
|
|
|
|
|
2005-06-08 16:45:19 +06:00
|
|
|
SUIT_ResourceMgr*
|
|
|
|
GetResourceMgr( const SalomeApp_Module* )
|
|
|
|
{
|
|
|
|
return SUIT_Session::session()->resourceMgr();
|
|
|
|
}
|
2004-12-01 15:48:31 +05:00
|
|
|
|
2005-06-07 19:22:20 +06:00
|
|
|
_PTR(Study)
|
|
|
|
GetCStudy(const SalomeApp_Study* theStudy)
|
|
|
|
{
|
|
|
|
return theStudy->studyDS();
|
|
|
|
}
|
|
|
|
|
|
|
|
CORBA::Object_var
|
2005-11-03 13:38:17 +05:00
|
|
|
DataOwnerToObject(const LightApp_DataOwnerPtr& theOwner)
|
2005-06-07 19:22:20 +06:00
|
|
|
{
|
|
|
|
CORBA::Object_var anObj;
|
|
|
|
if(theOwner){
|
|
|
|
const Handle(SALOME_InteractiveObject)& anIO = theOwner->IO();
|
|
|
|
if(!anIO.IsNull()){
|
2009-10-01 12:28:20 +06:00
|
|
|
if(anIO->hasEntry()){
|
|
|
|
_PTR(Study) aStudy = GetActiveStudyDocument();
|
|
|
|
_PTR(SObject) aSObj = aStudy->FindObjectID(anIO->getEntry());
|
|
|
|
anObj = SObjectToObject(aSObj,aStudy);
|
|
|
|
}
|
2005-06-07 19:22:20 +06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return anObj;
|
2004-12-01 15:48:31 +05:00
|
|
|
}
|
|
|
|
|
2005-06-07 19:22:20 +06:00
|
|
|
SUIT_Study* GetActiveStudy()
|
|
|
|
{
|
|
|
|
SUIT_Application* app = SUIT_Session::session()->activeApplication();
|
|
|
|
if (app)
|
|
|
|
return app->activeStudy();
|
|
|
|
else
|
|
|
|
return NULL;
|
2004-12-01 15:48:31 +05:00
|
|
|
}
|
|
|
|
|
2005-06-07 19:22:20 +06:00
|
|
|
SUIT_ViewWindow* GetActiveWindow()
|
|
|
|
{
|
|
|
|
SalomeApp_Application* app = dynamic_cast<SalomeApp_Application*>
|
|
|
|
(SUIT_Session::session()->activeApplication());
|
2005-06-16 17:04:03 +06:00
|
|
|
if (app && app->desktop() )
|
|
|
|
return app->desktop()->activeWindow();
|
2005-06-07 19:22:20 +06:00
|
|
|
else
|
|
|
|
return NULL;
|
|
|
|
}
|
2004-12-01 15:48:31 +05:00
|
|
|
|
2005-06-07 19:22:20 +06:00
|
|
|
_PTR(Study) GetActiveStudyDocument()
|
|
|
|
{
|
|
|
|
SalomeApp_Study* aStudy = dynamic_cast<SalomeApp_Study*>(GetActiveStudy());
|
|
|
|
if (aStudy)
|
|
|
|
return aStudy->studyDS();
|
|
|
|
else
|
|
|
|
return _PTR(Study)();
|
2004-12-01 15:48:31 +05:00
|
|
|
}
|
|
|
|
|
2005-06-07 19:22:20 +06:00
|
|
|
_PTR(SObject) FindSObject (CORBA::Object_ptr theObject)
|
|
|
|
{
|
|
|
|
SalomeApp_Application* app = dynamic_cast<SalomeApp_Application*>
|
|
|
|
(SUIT_Session::session()->activeApplication());
|
|
|
|
if (app && !CORBA::is_nil(theObject)) {
|
|
|
|
if(_PTR(Study) aStudy = GetActiveStudyDocument()){
|
2009-10-01 12:28:20 +06:00
|
|
|
CORBA::String_var anIOR = app->orb()->object_to_string(theObject);
|
|
|
|
if (strcmp(anIOR.in(), "") != 0)
|
|
|
|
return aStudy->FindObjectIOR(anIOR.in());
|
2005-06-07 19:22:20 +06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return _PTR(SObject)();
|
|
|
|
}
|
2004-12-01 15:48:31 +05:00
|
|
|
|
2009-02-17 10:27:49 +05:00
|
|
|
void SetName (_PTR(SObject) theSObject, const QString& theName)
|
2005-06-07 19:22:20 +06:00
|
|
|
{
|
|
|
|
_PTR(Study) aStudy = GetActiveStudyDocument();
|
|
|
|
if (aStudy->GetProperties()->IsLocked())
|
2004-12-01 15:48:31 +05:00
|
|
|
return;
|
2009-02-17 10:27:49 +05:00
|
|
|
SMESHGUI::GetSMESHGen()->SetName(theSObject->GetIOR().c_str(), theName.toLatin1().data());
|
2004-12-01 15:48:31 +05:00
|
|
|
}
|
|
|
|
|
2009-02-17 10:27:49 +05:00
|
|
|
void SetValue (_PTR(SObject) theSObject, const QString& theValue)
|
2005-06-07 19:22:20 +06:00
|
|
|
{
|
|
|
|
_PTR(Study) aStudy = GetActiveStudyDocument();
|
|
|
|
if (aStudy->GetProperties()->IsLocked())
|
2004-12-01 15:48:31 +05:00
|
|
|
return;
|
2005-06-07 19:22:20 +06:00
|
|
|
_PTR(StudyBuilder) aBuilder = aStudy->NewBuilder();
|
|
|
|
_PTR(GenericAttribute) anAttr =
|
|
|
|
aBuilder->FindOrCreateAttribute(theSObject, "AttributeComment");
|
|
|
|
_PTR(AttributeComment) aComment = anAttr;
|
|
|
|
if (aComment)
|
2009-02-17 10:27:49 +05:00
|
|
|
aComment->SetValue(theValue.toLatin1().data());
|
2004-12-01 15:48:31 +05:00
|
|
|
}
|
2006-03-13 20:29:49 +05:00
|
|
|
|
2009-02-17 10:27:49 +05:00
|
|
|
void setFileName (_PTR(SObject) theSObject, const QString& theValue)
|
2006-03-13 20:29:49 +05:00
|
|
|
{
|
|
|
|
_PTR(Study) aStudy = GetActiveStudyDocument();
|
|
|
|
if (aStudy->GetProperties()->IsLocked())
|
|
|
|
return;
|
|
|
|
_PTR(StudyBuilder) aBuilder = aStudy->NewBuilder();
|
|
|
|
_PTR(GenericAttribute) anAttr =
|
|
|
|
aBuilder->FindOrCreateAttribute(theSObject, "AttributeExternalFileDef");
|
|
|
|
_PTR(AttributeExternalFileDef) aFileName = anAttr;
|
|
|
|
if (aFileName)
|
2009-02-17 10:27:49 +05:00
|
|
|
aFileName->SetValue(theValue.toLatin1().data());
|
2006-03-13 20:29:49 +05:00
|
|
|
}
|
|
|
|
|
2009-02-17 10:27:49 +05:00
|
|
|
void setFileType (_PTR(SObject) theSObject, const QString& theValue)
|
2006-03-13 20:29:49 +05:00
|
|
|
{
|
|
|
|
_PTR(Study) aStudy = GetActiveStudyDocument();
|
|
|
|
if (aStudy->GetProperties()->IsLocked())
|
|
|
|
return;
|
|
|
|
_PTR(StudyBuilder) aBuilder = aStudy->NewBuilder();
|
|
|
|
_PTR(GenericAttribute) anAttr =
|
|
|
|
aBuilder->FindOrCreateAttribute(theSObject, "AttributeFileType");
|
|
|
|
_PTR(AttributeFileType) aFileType = anAttr;
|
|
|
|
if (aFileType)
|
2009-02-17 10:27:49 +05:00
|
|
|
aFileType->SetValue(theValue.toLatin1().data());
|
2006-03-13 20:29:49 +05:00
|
|
|
}
|
2004-12-01 15:48:31 +05:00
|
|
|
|
2005-06-07 19:22:20 +06:00
|
|
|
CORBA::Object_var SObjectToObject (_PTR(SObject) theSObject,
|
|
|
|
_PTR(Study) theStudy)
|
2004-12-01 15:48:31 +05:00
|
|
|
{
|
2005-06-07 19:22:20 +06:00
|
|
|
SalomeApp_Application* app = dynamic_cast<SalomeApp_Application*>
|
|
|
|
(SUIT_Session::session()->activeApplication());
|
|
|
|
if (theSObject) {
|
|
|
|
_PTR(GenericAttribute) anAttr;
|
|
|
|
if (theSObject->FindAttribute(anAttr, "AttributeIOR")) {
|
2009-10-01 12:28:20 +06:00
|
|
|
_PTR(AttributeIOR) anIOR = anAttr;
|
|
|
|
CORBA::String_var aVal = anIOR->Value().c_str();
|
2010-05-14 21:32:37 +06:00
|
|
|
// string_to_object() DOC: If the input string is not valid ...
|
|
|
|
// a CORBA::SystemException is thrown.
|
|
|
|
if ( aVal && strlen( aVal ) > 0 )
|
|
|
|
return app->orb()->string_to_object(aVal);
|
2004-12-01 15:48:31 +05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return CORBA::Object::_nil();
|
|
|
|
}
|
|
|
|
|
2005-06-07 19:22:20 +06:00
|
|
|
CORBA::Object_var SObjectToObject (_PTR(SObject) theSObject)
|
|
|
|
{
|
|
|
|
_PTR(Study) aStudy = GetActiveStudyDocument();
|
2004-12-01 15:48:31 +05:00
|
|
|
return SObjectToObject(theSObject,aStudy);
|
|
|
|
}
|
|
|
|
|
2010-11-25 17:44:43 +05:00
|
|
|
_PTR(SObject) ObjectToSObject( CORBA::Object_ptr theObject )
|
|
|
|
{
|
|
|
|
_PTR(SObject) res;
|
|
|
|
SalomeApp_Application* app = dynamic_cast<SalomeApp_Application*>
|
|
|
|
(SUIT_Session::session()->activeApplication());
|
|
|
|
if ( app ) {
|
|
|
|
QString IOR = app->orb()->object_to_string( theObject );
|
|
|
|
SalomeApp_Study* study = dynamic_cast<SalomeApp_Study*>( app->activeStudy() );
|
|
|
|
if ( study && !IOR.isEmpty() )
|
2010-12-24 13:18:34 +05:00
|
|
|
res = study->studyDS()->FindObjectIOR( IOR.toLatin1().constData() );
|
2010-11-25 17:44:43 +05:00
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2005-06-07 19:22:20 +06:00
|
|
|
CORBA::Object_var IObjectToObject (const Handle(SALOME_InteractiveObject)& theIO)
|
|
|
|
{
|
|
|
|
if (!theIO.IsNull()) {
|
|
|
|
if (theIO->hasEntry()) {
|
2009-10-01 12:28:20 +06:00
|
|
|
_PTR(Study) aStudy = GetActiveStudyDocument();
|
|
|
|
_PTR(SObject) anObj = aStudy->FindObjectID(theIO->getEntry());
|
|
|
|
return SObjectToObject(anObj,aStudy);
|
2004-12-01 15:48:31 +05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return CORBA::Object::_nil();
|
|
|
|
}
|
|
|
|
|
2009-02-17 10:27:49 +05:00
|
|
|
CORBA::Object_var IORToObject (const QString& theIOR)
|
2005-06-07 19:22:20 +06:00
|
|
|
{
|
|
|
|
SalomeApp_Application* app = dynamic_cast<SalomeApp_Application*>
|
|
|
|
(SUIT_Session::session()->activeApplication());
|
2009-02-17 10:27:49 +05:00
|
|
|
return app->orb()->string_to_object(theIOR.toLatin1().data());
|
2004-12-01 15:48:31 +05:00
|
|
|
}
|
|
|
|
|
2005-11-03 13:38:17 +05:00
|
|
|
int GetNameOfSelectedIObjects(LightApp_SelectionMgr* theMgr, QString& theName)
|
2004-12-01 15:48:31 +05:00
|
|
|
{
|
2005-06-07 19:22:20 +06:00
|
|
|
if (!theMgr)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
SALOME_ListIO selected;
|
|
|
|
theMgr->selectedObjects(selected);
|
|
|
|
int aNbSel = selected.Extent();
|
2004-12-01 15:48:31 +05:00
|
|
|
if (aNbSel == 1) {
|
2005-06-07 19:22:20 +06:00
|
|
|
Handle(SALOME_InteractiveObject) anIObject = selected.First();
|
2004-12-01 15:48:31 +05:00
|
|
|
theName = anIObject->getName();
|
2005-06-07 19:22:20 +06:00
|
|
|
} else {
|
2004-12-01 15:48:31 +05:00
|
|
|
theName = QObject::tr("SMESH_OBJECTS_SELECTED").arg(aNbSel);
|
|
|
|
}
|
|
|
|
return aNbSel;
|
|
|
|
}
|
|
|
|
|
2005-06-07 19:22:20 +06:00
|
|
|
_PTR(SObject) GetMeshOrSubmesh (_PTR(SObject) theSObject)
|
|
|
|
{
|
2004-12-01 15:48:31 +05:00
|
|
|
GEOM::GEOM_Object_var aShape = SObjectToInterface<GEOM::GEOM_Object>(theSObject);
|
2005-06-07 19:22:20 +06:00
|
|
|
if (!aShape->_is_nil()){ //It s a shape
|
2004-12-01 15:48:31 +05:00
|
|
|
return theSObject->GetFather();
|
|
|
|
}
|
2005-06-07 19:22:20 +06:00
|
|
|
_PTR(SObject) aSObject;
|
|
|
|
if (theSObject->ReferencedObject(aSObject)) {
|
2004-12-01 15:48:31 +05:00
|
|
|
aSObject = theSObject->GetFather();
|
|
|
|
return aSObject->GetFather();
|
|
|
|
}
|
|
|
|
return theSObject->GetFather();
|
|
|
|
}
|
|
|
|
|
2008-03-07 12:47:05 +05:00
|
|
|
void ModifiedMesh (_PTR(SObject) theSObject, bool theIsNotModif, bool isEmptyMesh)
|
2004-12-01 15:48:31 +05:00
|
|
|
{
|
2005-06-07 19:22:20 +06:00
|
|
|
_PTR(Study) aStudy = GetActiveStudyDocument();
|
|
|
|
if (aStudy->GetProperties()->IsLocked())
|
|
|
|
return;
|
2004-12-01 15:48:31 +05:00
|
|
|
|
2005-06-07 19:22:20 +06:00
|
|
|
_PTR(StudyBuilder) aBuilder = aStudy->NewBuilder();
|
|
|
|
_PTR(GenericAttribute) anAttr =
|
2004-12-01 15:48:31 +05:00
|
|
|
aBuilder->FindOrCreateAttribute(theSObject,"AttributePixMap");
|
2005-06-07 19:22:20 +06:00
|
|
|
_PTR(AttributePixMap) aPixmap = anAttr;
|
2008-03-07 12:47:05 +05:00
|
|
|
if (theIsNotModif) {
|
2004-12-01 15:48:31 +05:00
|
|
|
aPixmap->SetPixMap("ICON_SMESH_TREE_MESH");
|
2008-03-07 12:47:05 +05:00
|
|
|
} else if ( isEmptyMesh ) {
|
2004-12-01 15:48:31 +05:00
|
|
|
aPixmap->SetPixMap("ICON_SMESH_TREE_MESH_WARN");
|
2008-03-07 12:47:05 +05:00
|
|
|
} else {
|
|
|
|
aPixmap->SetPixMap("ICON_SMESH_TREE_MESH_PARTIAL");
|
2004-12-01 15:48:31 +05:00
|
|
|
}
|
2005-06-07 19:22:20 +06:00
|
|
|
|
|
|
|
_PTR(ChildIterator) anIter = aStudy->NewChildIterator(theSObject);
|
2004-12-01 15:48:31 +05:00
|
|
|
for (int i = 1; anIter->More(); anIter->Next(), i++) {
|
2005-06-07 19:22:20 +06:00
|
|
|
_PTR(SObject) aSObj = anIter->Value();
|
|
|
|
if (i >= 4) {
|
2009-10-01 12:28:20 +06:00
|
|
|
_PTR(ChildIterator) anIter1 = aStudy->NewChildIterator(aSObj);
|
|
|
|
for ( ; anIter1->More(); anIter1->Next()) {
|
|
|
|
_PTR(SObject) aSObj1 = anIter1->Value();
|
|
|
|
anAttr = aBuilder->FindOrCreateAttribute(aSObj1, "AttributePixMap");
|
|
|
|
aPixmap = anAttr;
|
2008-03-07 12:47:05 +05:00
|
|
|
if (theIsNotModif) {
|
|
|
|
aPixmap->SetPixMap("ICON_SMESH_TREE_MESH");
|
|
|
|
} else if ( isEmptyMesh ) {
|
|
|
|
aPixmap->SetPixMap("ICON_SMESH_TREE_MESH_WARN");
|
|
|
|
} else {
|
|
|
|
aPixmap->SetPixMap("ICON_SMESH_TREE_MESH_PARTIAL");
|
|
|
|
}
|
2009-10-01 12:28:20 +06:00
|
|
|
}
|
2004-12-01 15:48:31 +05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-02-17 10:27:49 +05:00
|
|
|
void ShowHelpFile (const QString& theHelpFileName)
|
2008-03-07 12:47:05 +05:00
|
|
|
{
|
|
|
|
LightApp_Application* app = (LightApp_Application*)(SUIT_Session::session()->activeApplication());
|
|
|
|
if (app) {
|
|
|
|
SMESHGUI* gui = SMESHGUI::GetSMESHGUI();
|
|
|
|
app->onHelpContextModule(gui ? app->moduleName(gui->moduleName()) : QString(""),
|
|
|
|
theHelpFileName);
|
|
|
|
}
|
|
|
|
else {
|
2009-02-17 10:27:49 +05:00
|
|
|
SUIT_MessageBox::warning(0, QObject::tr("WRN_WARNING"),
|
2009-10-01 12:28:20 +06:00
|
|
|
QObject::tr("EXTERNAL_BROWSER_CANNOT_SHOW_PAGE").
|
|
|
|
arg(app->resourceMgr()->stringValue("ExternalBrowser",
|
|
|
|
"application")).
|
|
|
|
arg(theHelpFileName));
|
2009-02-17 10:27:49 +05:00
|
|
|
}
|
2008-03-07 12:47:05 +05:00
|
|
|
}
|
|
|
|
|
2009-02-17 10:27:49 +05:00
|
|
|
//=======================================================================
|
|
|
|
/**
|
|
|
|
Return normale to a given face
|
|
|
|
*/
|
|
|
|
//=======================================================================
|
|
|
|
gp_XYZ getNormale( const SMDS_MeshFace* theFace )
|
|
|
|
{
|
|
|
|
gp_XYZ n;
|
|
|
|
int aNbNode = theFace->NbNodes();
|
|
|
|
TColgp_Array1OfXYZ anArrOfXYZ(1,4);
|
|
|
|
SMDS_ElemIteratorPtr aNodeItr = theFace->nodesIterator();
|
|
|
|
int i = 1;
|
|
|
|
for ( ; aNodeItr->more() && i <= 4; i++ ) {
|
|
|
|
SMDS_MeshNode* aNode = (SMDS_MeshNode*)aNodeItr->next();
|
|
|
|
anArrOfXYZ.SetValue(i, gp_XYZ( aNode->X(), aNode->Y(), aNode->Z() ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
gp_XYZ q1 = anArrOfXYZ.Value(2) - anArrOfXYZ.Value(1);
|
|
|
|
gp_XYZ q2 = anArrOfXYZ.Value(3) - anArrOfXYZ.Value(1);
|
|
|
|
n = q1 ^ q2;
|
|
|
|
if ( aNbNode > 3 ) {
|
|
|
|
gp_XYZ q3 = anArrOfXYZ.Value(4) - anArrOfXYZ.Value(1);
|
|
|
|
n += q2 ^ q3;
|
|
|
|
}
|
|
|
|
double len = n.Modulus();
|
|
|
|
if ( len > 0 )
|
|
|
|
n /= len;
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // end of namespace SMESH
|