From 7587cf271aea15fd2708f41fcadb534354afa152 Mon Sep 17 00:00:00 2001
From: Karsten Rink <karsten.rink@ufz.de>
Date: Wed, 19 Feb 2014 17:32:50 +0100
Subject: [PATCH] added methods to each element class that test if an element
 is valid

---
 GeoLib/AnalyticalGeometry.cpp        | 16 ++++++++++++++++
 GeoLib/AnalyticalGeometry.h          | 15 ++++++++++++++-
 MeshLib/Elements/Element.h           |  6 ++++++
 MeshLib/Elements/TemplateHex.h       |  5 +++++
 MeshLib/Elements/TemplateHex.tpp     | 14 ++++++++++++++
 MeshLib/Elements/TemplateLine.h      |  5 +++++
 MeshLib/Elements/TemplatePrism.h     |  5 +++++
 MeshLib/Elements/TemplatePrism.tpp   | 14 ++++++++++++++
 MeshLib/Elements/TemplatePyramid.h   |  5 +++++
 MeshLib/Elements/TemplatePyramid.tpp | 10 ++++++++++
 MeshLib/Elements/TemplateQuad.h      |  4 ++++
 MeshLib/Elements/TemplateQuad.tpp    | 26 ++++++++++++++++++++++++++
 MeshLib/Elements/TemplateTet.h       |  5 +++++
 MeshLib/Elements/TemplateTri.h       |  6 ++++++
 scripts/cmake/Functions.cmake        |  3 ++-
 15 files changed, 137 insertions(+), 2 deletions(-)

diff --git a/GeoLib/AnalyticalGeometry.cpp b/GeoLib/AnalyticalGeometry.cpp
index 0b8cd960aa8..fec6cbbdcfe 100644
--- a/GeoLib/AnalyticalGeometry.cpp
+++ b/GeoLib/AnalyticalGeometry.cpp
@@ -339,4 +339,20 @@ double scalarTriple(GeoLib::Point const& u, GeoLib::Point const& v, GeoLib::Poin
 	return result;
 }
 
+bool dividedByPlane(const GeoLib::Point& a, const GeoLib::Point& b, const GeoLib::Point& c, const GeoLib::Point& d)
+{
+	double abc(0), abd(0);
+	for (unsigned x=0; x<3; ++x)
+	{
+		const unsigned y=(x+1)%3;
+		abc = (b[x] - a[x])*(c[y] - a[y]) - (b[y] - a[y])*(c[x] - a[x]);
+		abd = (b[x] - a[x])*(d[y] - a[y]) - (b[y] - a[y])*(d[x] - a[x]);
+
+		if ((abc>0 && abd<0) || (abc<0 && abd>0))
+			return true;		
+		continue;
+	}
+	return false;
+}
+
 } // end namespace GeoLib
diff --git a/GeoLib/AnalyticalGeometry.h b/GeoLib/AnalyticalGeometry.h
index a991293f509..9623083b768 100644
--- a/GeoLib/AnalyticalGeometry.h
+++ b/GeoLib/AnalyticalGeometry.h
@@ -135,9 +135,22 @@ bool lineSegmentIntersect (const GeoLib::Point& a, const GeoLib::Point& b,
  */
 GeoLib::Point* triangleLineIntersection(GeoLib::Point const& a, GeoLib::Point const& b, GeoLib::Point const& c, GeoLib::Point const& p, GeoLib::Point const& q);
 
-// Calculates the scalar triple (u x v) . w
+/// Calculates the scalar triple (u x v) . w
 double scalarTriple(GeoLib::Point const& u, GeoLib::Point const& v, GeoLib::Point const& w);
 
+/** 
+ * Checks if a und b can be placed on a plane such that c and d lie on different sides of said plane.
+ * (In 2D space this checks if c and d are on different sides of a line through a and b.)
+ * @param a first point on plane
+ * @param b second point on plane
+ * @param c first point to test
+ * @param d second point to test
+ * @return true, if such a plane can be found, false otherwise 
+ */
+ bool dividedByPlane(const GeoLib::Point& a, const GeoLib::Point& b, 
+	 const GeoLib::Point& c, const GeoLib::Point& d);
+
+
 } // end namespace GeoLib
 
 #endif /* ANALYTICAL_GEOMETRY_H_ */
diff --git a/MeshLib/Elements/Element.h b/MeshLib/Elements/Element.h
index a609ff86fb0..170b503077f 100644
--- a/MeshLib/Elements/Element.h
+++ b/MeshLib/Elements/Element.h
@@ -16,6 +16,7 @@
 #define ELEMENT_H_
 
 #include <vector>
+#include <limits>
 #include "MeshEnums.h"
 #include "Mesh.h"
 
@@ -132,6 +133,11 @@ public:
 	 */
 	unsigned getValue() const { return _value; }
 
+	/**
+	 * Tests if the element is geometrically valid, i.e. convex with volume > 0.
+	 */
+	virtual bool isValid() const = 0;
+
 	/**
 	 * Set the index value for external information.
 	 * @param value an unsigned value for linking with external information
diff --git a/MeshLib/Elements/TemplateHex.h b/MeshLib/Elements/TemplateHex.h
index b98f5b072dd..007882a7533 100644
--- a/MeshLib/Elements/TemplateHex.h
+++ b/MeshLib/Elements/TemplateHex.h
@@ -98,6 +98,11 @@ public:
 	/// Returns true if these two indices form an edge and false otherwise
 	bool isEdge(unsigned i, unsigned j) const;
 
+	/**
+	 * Tests if the element is geometrically valid, i.e. convex with volume > 0.
+	 */
+	virtual bool isValid() const;
+
 	/**
 	 * Method clone is inherited from class Element. It makes a deep copy of the Hex instance.
 	 * @return an exact copy of the object
diff --git a/MeshLib/Elements/TemplateHex.tpp b/MeshLib/Elements/TemplateHex.tpp
index 92d2075a676..f2e7538d8c2 100644
--- a/MeshLib/Elements/TemplateHex.tpp
+++ b/MeshLib/Elements/TemplateHex.tpp
@@ -155,6 +155,20 @@ unsigned TemplateHex<NNODES,CELLHEXTYPE>::identifyFace(Node* nodes[3]) const
 	return std::numeric_limits<unsigned>::max();
 }
 
+template <unsigned NNODES, CellType CELLHEXTYPE>
+bool TemplateHex<NNODES,CELLHEXTYPE>::isValid() const
+{
+	for (unsigned i=0; i<6; ++i)
+	{
+		const MeshLib::Quad* quad (dynamic_cast<const MeshLib::Quad*>(this->getFace(i)));
+		const bool quad_is_valid (quad->isValid());
+		delete quad;
+		if (!quad_is_valid)
+			return false;
+	}
+	return true;
+}
+
 template <unsigned NNODES, CellType CELLHEXTYPE>
 Element* TemplateHex<NNODES,CELLHEXTYPE>::reviseElement() const
 {
diff --git a/MeshLib/Elements/TemplateLine.h b/MeshLib/Elements/TemplateLine.h
index 0b462c1d45b..73095891d86 100644
--- a/MeshLib/Elements/TemplateLine.h
+++ b/MeshLib/Elements/TemplateLine.h
@@ -72,6 +72,11 @@ public:
 	 */
 	virtual CellType getCellType() const { return CELLLINETYPE; }
 
+	/**
+	 * Tests if the element is geometrically valid, i.e. convex with volume > 0.
+	 */
+	virtual bool isValid() const { return (this->_length) > std::numeric_limits<double>::epsilon(); }
+
 	/**
 	 * Method clone is inherited from class Element. It makes a deep copy of the TemplateLine instance.
 	 * @return an exact copy of the object
diff --git a/MeshLib/Elements/TemplatePrism.h b/MeshLib/Elements/TemplatePrism.h
index eb14f544dc2..343a32e4fcf 100644
--- a/MeshLib/Elements/TemplatePrism.h
+++ b/MeshLib/Elements/TemplatePrism.h
@@ -96,6 +96,11 @@ public:
 	/// Returns true if these two indeces form an edge and false otherwise
 	bool isEdge(unsigned i, unsigned j) const;
 
+	/**
+	 * Tests if the element is geometrically valid, i.e. convex with volume > 0.
+	 */
+	virtual bool isValid() const;
+
 	/**
 	 * Method clone is inherited from class Element. It makes a deep copy of the
 	 * Hex instance employing the copy constructor of class TemplatePrism.
diff --git a/MeshLib/Elements/TemplatePrism.tpp b/MeshLib/Elements/TemplatePrism.tpp
index 328e31a8314..dafee3a33ec 100644
--- a/MeshLib/Elements/TemplatePrism.tpp
+++ b/MeshLib/Elements/TemplatePrism.tpp
@@ -164,6 +164,20 @@ unsigned TemplatePrism<NNODES,CELLPRISMTYPE>::identifyFace(Node* nodes[3]) const
 	return std::numeric_limits<unsigned>::max();
 }
 
+template <unsigned NNODES, CellType CELLPRISMTYPE>
+bool TemplatePrism<NNODES,CELLPRISMTYPE>::isValid() const
+{
+	for (unsigned i=1; i<4; ++i)
+	{
+		const MeshLib::Quad* quad (dynamic_cast<const MeshLib::Quad*>(this->getFace(i)));
+		const bool quad_is_valid (quad->isValid());
+		delete quad;
+		if (!quad_is_valid)
+			return false;
+	}
+	return true;
+}
+
 template <unsigned NNODES, CellType CELLPRISMTYPE>
 Element* TemplatePrism<NNODES,CELLPRISMTYPE>::reviseElement() const
 {
diff --git a/MeshLib/Elements/TemplatePyramid.h b/MeshLib/Elements/TemplatePyramid.h
index e38d6e8c764..802bab5d460 100644
--- a/MeshLib/Elements/TemplatePyramid.h
+++ b/MeshLib/Elements/TemplatePyramid.h
@@ -94,6 +94,11 @@ public:
 	/// Returns true if these two indeces form an edge and false otherwise
 	bool isEdge(unsigned i, unsigned j) const;
 
+	/**
+	 * Tests if the element is geometrically valid, i.e. convex with volume > 0.
+	 */
+	virtual bool isValid() const;
+
 	/**
 	 * Method clone is inherited from class Element. It makes a deep copy of the
 	 * TemplatePyramid instance employing the copy constructor of class TemplatePyramid.
diff --git a/MeshLib/Elements/TemplatePyramid.tpp b/MeshLib/Elements/TemplatePyramid.tpp
index 8efa44a025c..0c136901dfc 100644
--- a/MeshLib/Elements/TemplatePyramid.tpp
+++ b/MeshLib/Elements/TemplatePyramid.tpp
@@ -166,6 +166,16 @@ unsigned TemplatePyramid<NNODES,CELLPYRAMIDTYPE>::identifyFace(Node* nodes[3]) c
 	return std::numeric_limits<unsigned>::max();
 }
 
+template <unsigned NNODES, CellType CELLPYRAMIDTYPE>
+bool TemplatePyramid<NNODES,CELLPYRAMIDTYPE>::isValid() const
+{
+	const MeshLib::Quad* base (dynamic_cast<const MeshLib::Quad*>(this->getFace(4)));
+	const bool base_is_valid (base->isValid());
+	delete base;
+	return base_is_valid;
+}
+
+
 template <unsigned NNODES, CellType CELLPYRAMIDTYPE>
 Element* TemplatePyramid<NNODES,CELLPYRAMIDTYPE>::reviseElement() const
 {
diff --git a/MeshLib/Elements/TemplateQuad.h b/MeshLib/Elements/TemplateQuad.h
index d58db41d8ba..0a083f06117 100644
--- a/MeshLib/Elements/TemplateQuad.h
+++ b/MeshLib/Elements/TemplateQuad.h
@@ -97,6 +97,10 @@ public:
 	 */
 	virtual bool isPntInside(GeoLib::Point const& pnt, double eps = std::numeric_limits<double>::epsilon()) const;
 
+	/**
+	 * Tests if the element is geometrically valid, i.e. convex with volume > 0.
+	 */
+	virtual bool isValid() const;
 
 	/**
 	 * Method clone is inherited from class Element. It makes a deep copy of the TemplateQuad instance.
diff --git a/MeshLib/Elements/TemplateQuad.tpp b/MeshLib/Elements/TemplateQuad.tpp
index f632083310d..8f8fe18782c 100644
--- a/MeshLib/Elements/TemplateQuad.tpp
+++ b/MeshLib/Elements/TemplateQuad.tpp
@@ -119,6 +119,32 @@ unsigned TemplateQuad<NNODES,CELLQUADTYPE>::identifyFace(Node* nodes[3]) const
 	return std::numeric_limits<unsigned>::max();
 }
 
+template <unsigned NNODES, CellType CELLQUADTYPE>
+bool TemplateQuad<NNODES,CELLQUADTYPE>::isValid() const
+{
+	if (this->_area <= std::numeric_limits<double>::epsilon())
+		return false;
+
+	const GeoLib::Point AB((*_nodes[1])[0]-(*_nodes[0])[0], (*_nodes[1])[1]-(*_nodes[0])[1], (*_nodes[1])[2]-(*_nodes[0])[2]);
+	const GeoLib::Point AC((*_nodes[2])[0]-(*_nodes[0])[0], (*_nodes[2])[1]-(*_nodes[0])[1], (*_nodes[2])[2]-(*_nodes[0])[2]);
+	const GeoLib::Point AD((*_nodes[3])[0]-(*_nodes[0])[0], (*_nodes[3])[1]-(*_nodes[0])[1], (*_nodes[3])[2]-(*_nodes[0])[2]);
+
+	// check if all points lie on the same plane
+	double squared_scalar_triple = pow(GeoLib::scalarTriple(AC, AD, AB), 2);
+	double normalisation_factor  = (AB[0]*AB[0]+AB[1]*AB[1]+AB[2]*AB[2]) * 
+			                        (AC[0]*AC[0]+AC[1]*AC[1]+AC[2]*AC[2]) * 
+									(AD[0]*AD[0]+AD[1]*AD[1]+AD[2]*AD[2]);
+
+	if (squared_scalar_triple/normalisation_factor < std::numeric_limits<double>::epsilon())
+	{
+		// check if quad is convex
+		if (GeoLib::dividedByPlane(*_nodes[0], *_nodes[2], *_nodes[1], *_nodes[3]) &&
+			GeoLib::dividedByPlane(*_nodes[1], *_nodes[3], *_nodes[0], *_nodes[2]))
+			return true;
+	}
+	return false;
+}
+
 template <unsigned NNODES, CellType CELLQUADTYPE>
 Element* TemplateQuad<NNODES,CELLQUADTYPE>::reviseElement() const
 {
diff --git a/MeshLib/Elements/TemplateTet.h b/MeshLib/Elements/TemplateTet.h
index 72bc27b8961..9abc25724ce 100644
--- a/MeshLib/Elements/TemplateTet.h
+++ b/MeshLib/Elements/TemplateTet.h
@@ -93,6 +93,11 @@ public:
 	/// Returns true if these two indeces form an edge and false otherwise
 	bool isEdge(unsigned i, unsigned j) const;
 
+	/**
+	 * Tests if the element is geometrically valid, i.e. convex with volume > 0.
+	 */
+	virtual bool isValid() const { return (this->_volume > std::numeric_limits<double>::epsilon()); }
+
 	/**
 	 * Method clone is inherited from class Element. It makes a deep copy of the TemplateTet instance.
 	 * @return an exact copy of the object
diff --git a/MeshLib/Elements/TemplateTri.h b/MeshLib/Elements/TemplateTri.h
index a6acc86c3d5..4541b9a6209 100644
--- a/MeshLib/Elements/TemplateTri.h
+++ b/MeshLib/Elements/TemplateTri.h
@@ -94,6 +94,12 @@ public:
 	 */
 	virtual bool isPntInside(GeoLib::Point const& pnt, double eps = std::numeric_limits<double>::epsilon()) const;
 
+	/**
+	 * Tests if the element is geometrically valid, i.e. convex with volume > 0.
+	 */
+	virtual bool isValid() const { return (this->_area > std::numeric_limits<double>::epsilon()); }
+
+
 	/**
 	 * Method clone is inherited from class Element. It makes a deep copy of the TemplateTri instance.
 	 * @return an exact copy of the object
diff --git a/scripts/cmake/Functions.cmake b/scripts/cmake/Functions.cmake
index 27b48eeac7e..8a5161e64c9 100644
--- a/scripts/cmake/Functions.cmake
+++ b/scripts/cmake/Functions.cmake
@@ -32,7 +32,8 @@ MACRO(GET_SOURCE_FILES SOURCE_FILES)
 	GET_CURRENT_SOURCE_SUBDIRECTORY(DIRECTORY)
 	SOURCE_GROUP( "${DIRECTORY}${DIR}" FILES
 		${GET_SOURCE_FILES_HEADERS}
-		${GET_SOURCE_FILES_SOURCES})
+		${GET_SOURCE_FILES_SOURCES}
+		${GET_SOURCE_FILES_TEMPLATES})
 
 ENDMACRO()
 
-- 
GitLab