// // surf.h // // Copyright (C) 1996 Limit Point Systems, Inc. // // Author: Curtis Janssen // Maintainer: LPS // // This file is part of the SC Toolkit. // // The SC Toolkit is free software; you can redistribute it and/or modify // it under the terms of the GNU Library General Public License as published by // the Free Software Foundation; either version 2, or (at your option) // any later version. // // The SC Toolkit 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 Library General Public License for more details. // // You should have received a copy of the GNU Library General Public License // along with the SC Toolkit; see the file COPYING.LIB. If not, write to // the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. // // The U.S. Government is granted a limited license as per AL 91-7. // #ifndef _math_isosurf_surf_h #define _math_isosurf_surf_h #ifdef __GNUC__ #pragma interface #endif #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include namespace sc { template inline void erase_elements_by_value(C &container, I begin, I end) { for (I i=begin; i!=end; i++) { container.erase(*i); } } class TriangulatedSurface: public DescribedClass { protected: int _verbose; int _debug; int _completed_surface; // sets of objects that make up the surface std::set > _vertices; std::set > _edges; std::set > _triangles; // map objects to an integer index std::map,int> _vertex_to_index; std::map,int> _edge_to_index; std::map,int> _triangle_to_index; // map integer indices to an object std::vector > _index_to_vertex; std::vector > _index_to_edge; std::vector > _index_to_triangle; // mappings between array element numbers int** _triangle_vertex; int** _triangle_edge; int** _edge_vertex; // values for each of the vertices int _have_values; std::vector _values; // what to use to integrate over the surface, by default Ref _integrator; // other integrators, in terms of time & accuracy: // _fast_integrator <= _integrator <= _accurate_interator Ref _fast_integrator; Ref _accurate_integrator; void clear_int_arrays(); void complete_ref_arrays(); void complete_int_arrays(); void recompute_index_maps(); void add_triangle(const Ref&); void add_vertex(const Ref&); void add_edge(const Ref&); // these members must be used to allocate new triangles and edges // since specializations of TriangulatedSurface might need to // override these to produce triangles and edges with interpolation // data. virtual Triangle* newTriangle(const Ref&, const Ref&, const Ref&, int orientation) const; virtual Edge* newEdge(const Ref&,const Ref&) const; // this map of edges to vertices is used to construct the surface std::map,std::set > > _tmp_edges; public: TriangulatedSurface(); TriangulatedSurface(const Ref&); virtual ~TriangulatedSurface(); // control printing int verbose() const { return _verbose; } void verbose(int v) { _verbose = v; } // set up an integrator void set_integrator(const Ref&); void set_fast_integrator(const Ref&); void set_accurate_integrator(const Ref&); virtual Ref integrator(int itri); virtual Ref fast_integrator(int itri); virtual Ref accurate_integrator(int itri); // construct the surface void add_triangle(const Ref&, const Ref&, const Ref&); Ref find_edge(const Ref&, const Ref&); virtual void complete_surface(); // clean up the surface virtual void remove_short_edges(double cutoff_length = 1.0e-6, const Ref &vol=0, double isoval=0.0); virtual void remove_slender_triangles( int remove_slender, double height_cutoff, int remove_small, double area_cutoff, const Ref &vol=0, double isoval=0.0); virtual void fix_orientation(); virtual void clear(); // get information from the object sets int nvertex() const { return _vertices.size(); }; Ref vertex(int i) const { return _index_to_vertex[i]; }; int vertex_index(const Ref &o) { std::map,int>::iterator i = _vertex_to_index.find(o); if (i != _vertex_to_index.end()) return i->second; return -1; } int nedge() const { return _edges.size(); }; Ref edge(int i) const { return _index_to_edge[i]; }; int edge_index(const Ref &o) { std::map,int>::iterator i = _edge_to_index.find(o); if (i != _edge_to_index.end()) return i->second; return -1; } int ntriangle() const { return _triangles.size(); }; Ref triangle(int i) const { return _index_to_triangle[i]; } int triangle_index(const Ref &o) { std::map,int>::iterator i = _triangle_to_index.find(o); if (i != _triangle_to_index.end()) return i->second; return -1; } // information from the index mappings int triangle_vertex(int i,int j) const { return _triangle_vertex[i][j]; }; int triangle_edge(int i,int j) const { return _triangle_edge[i][j]; }; int edge_vertex(int i,int j) const { return _edge_vertex[i][j]; }; // associate values with vertices //void compute_colors(Volume&); void compute_values(Ref&); // properties of the surface virtual double flat_area(); // use flat triangles virtual double flat_volume(); // use flat triangles virtual double area(); virtual double volume(); // output of the surface virtual void print(std::ostream&o=ExEnv::out0()) const; virtual void print_vertices_and_triangles(std::ostream&o=ExEnv::out0()) const; virtual void print_geomview_format(std::ostream&o=ExEnv::out0()) const; virtual void render(const Ref &render); // print information about the topology void topology_info(std::ostream&o=ExEnv::out0()); void topology_info(int nvertex, int nedge, int ntri, std::ostream&o=ExEnv::out0()); }; class TriangulatedSurfaceIntegrator { private: Ref _ts; int _itri; int _irs; double _r; double _s; double _weight; double _surface_element; Ref _current; SCVector3 _dA; Ref (TriangulatedSurface::*_integrator)(int itri); Ref _grp; public: TriangulatedSurfaceIntegrator(); // the surface cannot be changed until this is destroyed TriangulatedSurfaceIntegrator(const Ref&); ~TriangulatedSurfaceIntegrator(); // Objects initialized by these operators are not automatically // updated. This must be done with the update member. // The _grp is not copied. void operator = (const TriangulatedSurfaceIntegrator&); TriangulatedSurfaceIntegrator(const TriangulatedSurfaceIntegrator&i) { operator = (i); } // Return the number of integration points. int n(); // Assign the surface. Don't do this while iterating. void set_surface(const Ref&); // returns the number of the vertex in the current triangle int vertex_number(int i); inline double r() const { return _r; } inline double s() const { return _s; } inline double w() const { return _weight*_surface_element; } double surface_element() const { return _surface_element; } double weight() const { return _weight; } const SCVector3& dA() const { return _dA; } Ref current(); // Tests to see if this point is valid, if it is then // _r, _s, etc are computed and 1 is returned. int update(); // This can be used to loop through unique pairs of points. // The argument should be a TriangulatedSurfaceIntegrator for // the same surface as this. int operator < (TriangulatedSurfaceIntegrator&i) { update(); return _itrii._itri?0:(_irs*_integrator)(_itri)->n(); } void distribute(const Ref &); void use_fast_integrator(); void use_accurate_integrator(); void use_default_integrator(); }; class TriangulatedImplicitSurface: public TriangulatedSurface { private: // The surface is defined as an isosurface of the volume vol_. Ref vol_; double isovalue_; int fix_orientation_; int remove_short_edges_; double short_edge_factor_; int remove_slender_triangles_; double slender_triangle_factor_; int remove_small_triangles_; double small_triangle_factor_; double resolution_; int order_; int inited_; public: TriangulatedImplicitSurface(const Ref&); ~TriangulatedImplicitSurface(); Ref volume_object() const { return vol_; } double isovalue() const { return isovalue_; } void init(); int inited() const { return inited_; } }; } #endif // Local Variables: // mode: c++ // c-file-style: "CLJ" // End: