MOAB: Mesh Oriented datABase  (version 5.4.1)
MeshTopoUtil.cpp
Go to the documentation of this file.
00001 /**
00002  * MOAB, a Mesh-Oriented datABase, is a software component for creating,
00003  * storing and accessing finite element mesh data.
00004  *
00005  * Copyright 2004 Sandia Corporation.  Under the terms of Contract
00006  * DE-AC04-94AL85000 with Sandia Corporation, the U.S. Government
00007  * retains certain rights in this software.
00008  *
00009  * This library is free software; you can redistribute it and/or
00010  * modify it under the terms of the GNU Lesser General Public
00011  * License as published by the Free Software Foundation; either
00012  * version 2.1 of the License, or (at your option) any later version.
00013  *
00014  */
00015 
00016 #ifdef WIN32
00017 #pragma warning( disable : 4786 )
00018 #endif
00019 
00020 #include "moab/MeshTopoUtil.hpp"
00021 #include "moab/Range.hpp"
00022 #include "Internals.hpp"
00023 #include "moab/Interface.hpp"
00024 #include "moab/CN.hpp"
00025 
00026 #include <cassert>
00027 
00028 #define RR                                        \
00029     {                                             \
00030         if( MB_SUCCESS != result ) return result; \
00031     }
00032 
00033 namespace moab
00034 {
00035 
00036 //! generate all the AEntities bounding the vertices
00037 ErrorCode MeshTopoUtil::construct_aentities( const Range& vertices )
00038 {
00039     Range out_range;
00040     ErrorCode result;
00041     result = mbImpl->get_adjacencies( vertices, 1, true, out_range, Interface::UNION );
00042     if( MB_SUCCESS != result ) return result;
00043     out_range.clear();
00044     result = mbImpl->get_adjacencies( vertices, 2, true, out_range, Interface::UNION );
00045     if( MB_SUCCESS != result ) return result;
00046     out_range.clear();
00047     result = mbImpl->get_adjacencies( vertices, 3, true, out_range, Interface::UNION );
00048 
00049     return result;
00050 }
00051 
00052 //! given an entity, get its average position (avg vertex locations)
00053 ErrorCode MeshTopoUtil::get_average_position( Range& entities, double* avg_position )
00054 {
00055     std::vector< EntityHandle > ent_vec;
00056     std::copy( entities.begin(), entities.end(), std::back_inserter( ent_vec ) );
00057     return get_average_position( &ent_vec[0], ent_vec.size(), avg_position );
00058 }
00059 
00060 //! given an entity, get its average position (avg vertex locations)
00061 ErrorCode MeshTopoUtil::get_average_position( const EntityHandle* entities,
00062                                               const int num_entities,
00063                                               double* avg_position )
00064 {
00065     double dum_pos[3];
00066     avg_position[0] = avg_position[1] = avg_position[2] = 0.0;
00067 
00068     Range connect;
00069     ErrorCode result = mbImpl->get_adjacencies( entities, num_entities, 0, false, connect, Interface::UNION );
00070     if( MB_SUCCESS != result ) return result;
00071 
00072     if( connect.empty() ) return MB_FAILURE;
00073 
00074     for( Range::iterator rit = connect.begin(); rit != connect.end(); ++rit )
00075     {
00076         result = mbImpl->get_coords( &( *rit ), 1, dum_pos );
00077         if( MB_SUCCESS != result ) return result;
00078         avg_position[0] += dum_pos[0];
00079         avg_position[1] += dum_pos[1];
00080         avg_position[2] += dum_pos[2];
00081     }
00082     avg_position[0] /= (double)connect.size();
00083     avg_position[1] /= (double)connect.size();
00084     avg_position[2] /= (double)connect.size();
00085 
00086     return MB_SUCCESS;
00087 }
00088 
00089 //! given an entity, get its average position (avg vertex locations)
00090 ErrorCode MeshTopoUtil::get_average_position( const EntityHandle entity, double* avg_position )
00091 {
00092     const EntityHandle* connect = NULL;
00093     int num_connect             = 0;
00094     if( MBVERTEX == mbImpl->type_from_handle( entity ) ) return mbImpl->get_coords( &entity, 1, avg_position );
00095 
00096     ErrorCode result = mbImpl->get_connectivity( entity, connect, num_connect );
00097     if( MB_SUCCESS != result ) return result;
00098 
00099     return get_average_position( connect, num_connect, avg_position );
00100 }
00101 
00102 // given an entity, find the entities of next higher dimension around
00103 // that entity, ordered by connection through next higher dimension entities;
00104 // if any of the star entities is in only one entity of next higher dimension,
00105 // on_boundary is returned true
00106 ErrorCode MeshTopoUtil::star_entities( const EntityHandle star_center,
00107                                        std::vector< EntityHandle >& star_ents,
00108                                        bool& bdy_entity,
00109                                        const EntityHandle starting_star_entity,
00110                                        std::vector< EntityHandle >* star_entities_dp2,
00111                                        Range* star_candidates_dp2 )
00112 {
00113     // now start the traversal
00114     bdy_entity               = false;
00115     EntityHandle last_entity = starting_star_entity, last_dp2 = 0, next_entity, next_dp2;
00116     std::vector< EntityHandle > star_dp2;
00117     ErrorCode result;
00118     int center_dim = mbImpl->dimension_from_handle( star_center );
00119 
00120     Range tmp_candidates_dp2;
00121     if( NULL != star_candidates_dp2 )
00122         tmp_candidates_dp2 = *star_candidates_dp2;
00123     else
00124     {
00125         result = mbImpl->get_adjacencies( &star_center, 1, center_dim + 2, false, tmp_candidates_dp2 );
00126         if( MB_SUCCESS != result ) return result;
00127     }
00128 
00129     do
00130     {
00131         // get the next star entity
00132         result = star_next_entity( star_center, last_entity, last_dp2, &tmp_candidates_dp2, next_entity, next_dp2 );
00133         if( MB_SUCCESS != result ) return result;
00134 
00135         // special case: if starting_star_entity isn't connected to any entities of next
00136         // higher dimension, it's the only entity in the star; put it on the list and return
00137         if( star_ents.empty() && next_entity == 0 && next_dp2 == 0 )
00138         {
00139             star_ents.push_back( last_entity );
00140             bdy_entity = true;
00141             return MB_SUCCESS;
00142         }
00143 
00144         // if we're at a bdy and bdy_entity hasn't been set yet, we're at the
00145         // first bdy; reverse the lists and start traversing in the other direction; but,
00146         // pop the last star entity off the list and find it again, so that we properly
00147         // check for next_dp2
00148         if( 0 == next_dp2 && !bdy_entity )
00149         {
00150             star_ents.push_back( next_entity );
00151             bdy_entity = true;
00152             std::reverse( star_ents.begin(), star_ents.end() );
00153             star_ents.pop_back();
00154             last_entity = star_ents.back();
00155             if( !star_dp2.empty() )
00156             {
00157                 std::reverse( star_dp2.begin(), star_dp2.end() );
00158                 last_dp2 = star_dp2.back();
00159             }
00160         }
00161         // else if we're not on the bdy and next_entity is already in star, that means
00162         // we've come all the way around; don't put next_entity on list again, and
00163         // zero out last_dp2 to terminate while loop
00164         else if( !bdy_entity && std::find( star_ents.begin(), star_ents.end(), next_entity ) != star_ents.end() &&
00165                  ( std::find( star_dp2.begin(), star_dp2.end(), next_dp2 ) != star_dp2.end() || !next_dp2 ) )
00166         {
00167             last_dp2 = 0;
00168         }
00169 
00170         // else, just assign last entities seen and go on to next iteration
00171         else
00172         {
00173             if( std::find( star_ents.begin(), star_ents.end(), next_entity ) == star_ents.end() )
00174                 star_ents.push_back( next_entity );
00175             if( 0 != next_dp2 )
00176             {
00177                 star_dp2.push_back( next_dp2 );
00178                 tmp_candidates_dp2.erase( next_dp2 );
00179             }
00180             last_entity = next_entity;
00181             last_dp2    = next_dp2;
00182         }
00183     } while( 0 != last_dp2 );
00184 
00185     // copy over the star_dp2 list, if requested
00186     if( NULL != star_entities_dp2 ) ( *star_entities_dp2 ).swap( star_dp2 );
00187 
00188     return MB_SUCCESS;
00189 }
00190 
00191 ErrorCode MeshTopoUtil::star_next_entity( const EntityHandle star_center,
00192                                           const EntityHandle last_entity,
00193                                           const EntityHandle last_dp1,
00194                                           Range* star_candidates_dp1,
00195                                           EntityHandle& next_entity,
00196                                           EntityHandle& next_dp1 )
00197 {
00198     // given a star_center, a last_entity (whose dimension should be 1 greater than center)
00199     // and last_dp1 (dimension 2 higher than center), returns the next star entity across
00200     // last_dp1, and the next dp1 entity sharing next_entity; if star_candidates is non-empty,
00201     // star must come from those
00202     Range from_ents, to_ents;
00203     from_ents.insert( star_center );
00204     if( 0 != last_dp1 ) from_ents.insert( last_dp1 );
00205 
00206     int dim = mbImpl->dimension_from_handle( star_center );
00207 
00208     ErrorCode result = mbImpl->get_adjacencies( from_ents, dim + 1, true, to_ents );
00209     if( MB_SUCCESS != result ) return result;
00210 
00211     // remove last_entity from result, and should only have 1 left, if any
00212     if( 0 != last_entity ) to_ents.erase( last_entity );
00213 
00214     // if no last_dp1, contents of to_ents should share dp1-dimensional entity with last_entity
00215     if( 0 != last_entity && 0 == last_dp1 )
00216     {
00217         Range tmp_to_ents;
00218         for( Range::iterator rit = to_ents.begin(); rit != to_ents.end(); ++rit )
00219         {
00220             if( 0 != common_entity( last_entity, *rit, dim + 2 ) ) tmp_to_ents.insert( *rit );
00221         }
00222         to_ents = tmp_to_ents;
00223     }
00224 
00225     if( 0 == last_dp1 && to_ents.size() > 1 && NULL != star_candidates_dp1 && !star_candidates_dp1->empty() )
00226     {
00227         // if we have a choice of to_ents and no previous dp1 and there are dp1 candidates,
00228         // the one we choose needs to be adjacent to one of the candidates
00229         result = mbImpl->get_adjacencies( *star_candidates_dp1, dim + 1, true, from_ents, Interface::UNION );
00230         if( MB_SUCCESS != result ) return result;
00231         to_ents = intersect( to_ents, from_ents );
00232     }
00233 
00234     if( !to_ents.empty() )
00235         next_entity = *to_ents.begin();
00236     else
00237     {
00238         next_entity = 0;
00239         next_dp1    = 0;
00240         return MB_SUCCESS;
00241     }
00242 
00243     // get next_dp1
00244     if( 0 != star_candidates_dp1 )
00245         to_ents = *star_candidates_dp1;
00246     else
00247         to_ents.clear();
00248 
00249     result = mbImpl->get_adjacencies( &next_entity, 1, dim + 2, true, to_ents );
00250     if( MB_SUCCESS != result ) return result;
00251 
00252     // can't be last one
00253     if( 0 != last_dp1 ) to_ents.erase( last_dp1 );
00254 
00255     if( !to_ents.empty() ) next_dp1 = *to_ents.begin();
00256 
00257     // could be zero, means we're at bdy
00258     else
00259         next_dp1 = 0;
00260 
00261     return MB_SUCCESS;
00262 }
00263 
00264 ErrorCode MeshTopoUtil::star_entities_nonmanifold( const EntityHandle star_entity,
00265                                                    std::vector< std::vector< EntityHandle > >& stars,
00266                                                    std::vector< bool >* bdy_flags,
00267                                                    std::vector< std::vector< EntityHandle > >* dp2_stars )
00268 {
00269     // Get a series of (d+1)-dimensional stars around a d-dimensional entity, such that
00270     // each star is on a (d+2)-manifold containing the d-dimensional entity; each star
00271     // is either open or closed, and also defines a (d+2)-star whose entities are bounded by
00272     // (d+1)-entities on the star and on the (d+2)-manifold
00273     //
00274     // Algorithm:
00275     // get the (d+2)-manifold entities; for d=1 / d+2=3, just assume all connected elements, since
00276     //   we don't do 4d yet
00277     // get intersection of (d+1)-entities adjacent to star entity and union of (d+1)-entities
00278     //   adjacent to (d+2)-manifold entities; these will be the entities in the star
00279     // while (d+1)-entities
00280     //   remove (d+1)-entity from (d+1)-entities
00281     //   get the (d+1)-star and (d+2)-star around that (d+1)-entity (using star_entities)
00282     //   save that star to the star list, and the bdy flag and (d+2)-star if requested
00283     //   remove (d+2)-entities from the (d+2)-manifold entities
00284     //   remove (d+1)-entities from the (d+1)-entities
00285     // (end while)
00286 
00287     int this_dim = mbImpl->dimension_from_handle( star_entity );
00288     if( 3 <= this_dim || 0 > this_dim ) return MB_FAILURE;
00289 
00290     // get the (d+2)-manifold entities; for d=1 / d+2=3, just assume all connected elements, since
00291     //   we don't do 4d yet
00292     Range dp2_manifold;
00293     ErrorCode result = get_manifold( star_entity, this_dim + 2, dp2_manifold );
00294     if( MB_SUCCESS != result ) return result;
00295 
00296     // get intersection of (d+1)-entities adjacent to star and union of (d+1)-entities
00297     //   adjacent to (d+2)-manifold entities; also add manifold (d+1)-entities, to catch
00298     //   any not connected to (d+2)-entities
00299     Range dp1_manifold;
00300     result = mbImpl->get_adjacencies( dp2_manifold, this_dim + 1, false, dp1_manifold, Interface::UNION );
00301     if( MB_SUCCESS != result ) return result;
00302 
00303     result = mbImpl->get_adjacencies( &star_entity, 1, this_dim + 1, false, dp1_manifold );
00304     if( MB_SUCCESS != result ) return result;
00305 
00306     result = get_manifold( star_entity, this_dim + 1, dp1_manifold );
00307     if( MB_SUCCESS != result ) return result;
00308 
00309     // while (d+1)-entities
00310     while( !dp1_manifold.empty() )
00311     {
00312 
00313         //   get (d+1)-entity from (d+1)-entities (don't remove it until after star,
00314         //     since the star entities must come from dp1_manifold)
00315         EntityHandle this_ent = *dp1_manifold.begin();
00316 
00317         //   get the (d+1)-star and (d+2)-star around that (d+1)-entity (using star_entities)
00318         std::vector< EntityHandle > this_star_dp1, this_star_dp2;
00319         bool on_bdy;
00320         result = star_entities( star_entity, this_star_dp1, on_bdy, this_ent, &this_star_dp2, &dp2_manifold );
00321         if( MB_SUCCESS != result ) return result;
00322 
00323         // if there's no star entities, it must mean this_ent isn't bounded by any dp2
00324         // entities (wasn't put into star in star_entities 'cuz we're passing in non-null
00325         // dp2_manifold above); put it in
00326         if( this_star_dp1.empty() )
00327         {
00328             Range dum_range;
00329             result = mbImpl->get_adjacencies( &this_ent, 1, this_dim + 2, false, dum_range );
00330             if( MB_SUCCESS != result ) return result;
00331             if( dum_range.empty() ) this_star_dp1.push_back( this_ent );
00332         }
00333 
00334         // now we can remove it
00335         dp1_manifold.erase( dp1_manifold.begin() );
00336 
00337         //   save that star to the star list, and the bdy flag and (d+2)-star if requested
00338         if( !this_star_dp1.empty() )
00339         {
00340             stars.push_back( this_star_dp1 );
00341             if( NULL != bdy_flags ) bdy_flags->push_back( on_bdy );
00342             if( NULL != dp2_stars ) dp2_stars->push_back( this_star_dp2 );
00343         }
00344 
00345         //   remove (d+2)-entities from the (d+2)-manifold entities
00346         for( std::vector< EntityHandle >::iterator vit = this_star_dp2.begin(); vit != this_star_dp2.end(); ++vit )
00347             dp2_manifold.erase( *vit );
00348 
00349         //   remove (d+1)-entities from the (d+1)-entities
00350         for( std::vector< EntityHandle >::iterator vit = this_star_dp1.begin(); vit != this_star_dp1.end(); ++vit )
00351             dp1_manifold.erase( *vit );
00352 
00353         // (end while)
00354     }
00355 
00356     // check for leftover dp2 manifold entities, these should be in one of the
00357     // stars
00358     if( !dp2_manifold.empty() )
00359     {
00360         for( Range::iterator rit = dp2_manifold.begin(); rit != dp2_manifold.end(); ++rit )
00361         {
00362         }
00363     }
00364 
00365     return MB_SUCCESS;
00366 }
00367 
00368 //! get (target_dim)-dimensional manifold entities connected to star_entity; that is,
00369 //! the entities with <= 1 connected (target_dim+2)-dimensional adjacent entities;
00370 //! for target_dim=3, just return all of them
00371 //! just insert into the list, w/o clearing manifold list first
00372 ErrorCode MeshTopoUtil::get_manifold( const EntityHandle star_entity, const int target_dim, Range& manifold )
00373 {
00374     // get all the entities of target dimension connected to star
00375     Range tmp_range;
00376     ErrorCode result = mbImpl->get_adjacencies( &star_entity, 1, target_dim, false, tmp_range );
00377     if( MB_SUCCESS != result ) return result;
00378 
00379     // now save the ones which are (target_dim+1)-dimensional manifold;
00380     // for target_dim=3, just return whole range, since we don't do 4d
00381     if( target_dim == 3 )
00382     {
00383         manifold.merge( tmp_range );
00384         return MB_SUCCESS;
00385     }
00386 
00387     for( Range::iterator rit = tmp_range.begin(); rit != tmp_range.end(); ++rit )
00388     {
00389         Range dum_range;
00390         // get (target_dim+1)-dimensional entities
00391         result = mbImpl->get_adjacencies( &( *rit ), 1, target_dim + 1, false, dum_range );
00392         if( MB_SUCCESS != result ) return result;
00393 
00394         // if there are only 1 or zero, add to manifold list
00395         if( 1 >= dum_range.size() ) manifold.insert( *rit );
00396     }
00397 
00398     return MB_SUCCESS;
00399 }
00400 
00401 //! get "bridge" or "2nd order" adjacencies, going through dimension bridge_dim
00402 ErrorCode MeshTopoUtil::get_bridge_adjacencies( Range& from_entities,
00403                                                 int bridge_dim,
00404                                                 int to_dim,
00405                                                 Range& to_ents,
00406                                                 int num_layers )
00407 {
00408     Range bridge_ents, accum_layers, new_toents( from_entities );
00409     ErrorCode result;
00410     if( 0 == num_layers || from_entities.empty() ) return MB_FAILURE;
00411 
00412     // for each layer, get bridge-adj entities and accumulate
00413     for( int nl = 0; nl < num_layers; nl++ )
00414     {
00415         Range new_bridges;
00416         // get bridge ents
00417         result = mbImpl->get_adjacencies( new_toents, bridge_dim, true, new_bridges, Interface::UNION );
00418         if( MB_SUCCESS != result ) return result;
00419 
00420         // get to_dim adjacencies, merge into to_ents
00421         Range new_layer;
00422         if( -1 == to_dim )
00423         {
00424             result = mbImpl->get_adjacencies( new_bridges, 3, false, new_layer, Interface::UNION );
00425             if( MB_SUCCESS != result ) return result;
00426             for( int d = 2; d >= 1; d-- )
00427             {
00428                 result = mbImpl->get_adjacencies( to_ents, d, true, new_layer, Interface::UNION );
00429                 if( MB_SUCCESS != result ) return result;
00430             }
00431         }
00432         else
00433         {
00434             result = mbImpl->get_adjacencies( new_bridges, to_dim, false, new_layer, Interface::UNION );
00435             if( MB_SUCCESS != result ) return result;
00436         }
00437 
00438         // subtract last_toents to get new_toents
00439         accum_layers.merge( new_layer );
00440         if( nl < num_layers - 1 ) new_toents = subtract( new_layer, new_toents );
00441     }
00442     to_ents.merge( accum_layers );
00443 
00444     return MB_SUCCESS;
00445 }
00446 
00447 //! get "bridge" or "2nd order" adjacencies, going through dimension bridge_dim
00448 ErrorCode MeshTopoUtil::get_bridge_adjacencies( const EntityHandle from_entity,
00449                                                 const int bridge_dim,
00450                                                 const int to_dim,
00451                                                 Range& to_adjs )
00452 {
00453     // get pointer to connectivity for this entity
00454     const EntityHandle* connect;
00455     int num_connect;
00456     ErrorCode result     = MB_SUCCESS;
00457     EntityType from_type = TYPE_FROM_HANDLE( from_entity );
00458     if( from_type == MBVERTEX )
00459     {
00460         connect     = &from_entity;
00461         num_connect = 1;
00462     }
00463     else
00464     {
00465         result = mbImpl->get_connectivity( from_entity, connect, num_connect );
00466         if( MB_SUCCESS != result ) return result;
00467     }
00468 
00469     if( from_type >= MBENTITYSET ) return MB_FAILURE;
00470 
00471     int from_dim = CN::Dimension( from_type );
00472 
00473     Range to_ents;
00474 
00475     if( bridge_dim < from_dim )
00476     {
00477         // looping over each sub-entity of dimension bridge_dim...
00478         if( MBPOLYGON == from_type )
00479         {
00480             for( int i = 0; i < num_connect; i++ )
00481             {
00482                 // loop over edges, and get the vertices
00483                 EntityHandle verts_on_edge[2] = { connect[i], connect[( i + 1 ) % num_connect] };
00484                 to_ents.clear();
00485                 ErrorCode tmp_result =
00486                     mbImpl->get_adjacencies( verts_on_edge, 2, to_dim, false, to_ents, Interface::INTERSECT );
00487                 if( MB_SUCCESS != tmp_result ) result = tmp_result;
00488                 to_adjs.merge( to_ents );
00489             }
00490         }
00491         else
00492         {
00493             EntityHandle bridge_verts[MAX_SUB_ENTITIES];
00494             int bridge_indices[MAX_SUB_ENTITIES];
00495             for( int i = 0; i < CN::NumSubEntities( from_type, bridge_dim ); i++ )
00496             {
00497 
00498                 // get the vertices making up this sub-entity
00499                 int num_bridge_verts = CN::VerticesPerEntity( CN::SubEntityType( from_type, bridge_dim, i ) );
00500                 assert( num_bridge_verts >= 0 && num_bridge_verts <= MAX_SUB_ENTITIES );
00501                 CN::SubEntityVertexIndices( from_type, bridge_dim, i, bridge_indices );
00502                 for( int j = 0; j < num_bridge_verts; ++j )
00503                 {
00504                     if( bridge_indices[j] >= 0 && bridge_indices[j] < num_connect )
00505                         bridge_verts[j] = connect[bridge_indices[j]];
00506                     else
00507                         bridge_verts[j] = 0;
00508                 }
00509                 // CN::SubEntityConn(connect, from_type, bridge_dim, i, &bridge_verts[0],
00510                 // num_bridge_verts);
00511 
00512                 // get the to_dim entities adjacent
00513                 to_ents.clear();
00514                 ErrorCode tmp_result = mbImpl->get_adjacencies( bridge_verts, num_bridge_verts, to_dim, false, to_ents,
00515                                                                 Interface::INTERSECT );
00516                 if( MB_SUCCESS != tmp_result ) result = tmp_result;
00517 
00518                 to_adjs.merge( to_ents );
00519             }
00520         }
00521     }
00522 
00523     // now get the direct ones too, or only in the case where we're
00524     // going to higher dimension for bridge
00525     Range bridge_ents, tmp_ents;
00526     tmp_ents.insert( from_entity );
00527     ErrorCode tmp_result = mbImpl->get_adjacencies( tmp_ents, bridge_dim, false, bridge_ents, Interface::UNION );
00528     if( MB_SUCCESS != tmp_result ) return tmp_result;
00529 
00530     tmp_result = mbImpl->get_adjacencies( bridge_ents, to_dim, false, to_adjs, Interface::UNION );
00531     if( MB_SUCCESS != tmp_result ) return tmp_result;
00532 
00533     // if to_dimension is same as that of from_entity, make sure from_entity isn't
00534     // in list
00535     if( to_dim == from_dim ) to_adjs.erase( from_entity );
00536 
00537     return result;
00538 }
00539 
00540 //! return a common entity of the specified dimension, or 0 if there isn't one
00541 EntityHandle MeshTopoUtil::common_entity( const EntityHandle ent1, const EntityHandle ent2, const int dim )
00542 {
00543     Range tmp_range, tmp_range2;
00544     tmp_range.insert( ent1 );
00545     tmp_range.insert( ent2 );
00546     ErrorCode result = mbImpl->get_adjacencies( tmp_range, dim, false, tmp_range2 );
00547     if( MB_SUCCESS != result || tmp_range2.empty() )
00548         return 0;
00549     else
00550         return *tmp_range2.begin();
00551 }
00552 
00553 //! return the opposite side entity given a parent and bounding entity.
00554 //! This function is only defined for certain types of parent/child types;
00555 //! See CN.hpp::OppositeSide for details.
00556 //!
00557 //! \param parent The parent element
00558 //! \param child The child element
00559 //! \param opposite_element The index of the opposite element
00560 ErrorCode MeshTopoUtil::opposite_entity( const EntityHandle parent,
00561                                          const EntityHandle child,
00562                                          EntityHandle& opposite_element )
00563 {
00564     // get the side no.
00565     int side_no, offset, sense;
00566     ErrorCode result = mbImpl->side_number( parent, child, side_no, offset, sense );
00567     if( MB_SUCCESS != result ) return result;
00568 
00569     // get the child index from CN
00570     int opposite_index, opposite_dim;
00571     int status = CN::OppositeSide( mbImpl->type_from_handle( parent ), side_no, mbImpl->dimension_from_handle( child ),
00572                                    opposite_index, opposite_dim );
00573     if( 0 != status ) return MB_FAILURE;
00574 
00575     // now get the side element from MOAB
00576     result = mbImpl->side_element( parent, opposite_dim, opposite_index, opposite_element );
00577     if( MB_SUCCESS != result ) return result;
00578 
00579     return MB_SUCCESS;
00580 }
00581 
00582 ErrorCode MeshTopoUtil::split_entities_manifold( Range& entities, Range& new_entities, Range* fill_entities )
00583 {
00584     Range tmp_range, *tmp_ptr_fill_entity;
00585     if( NULL != fill_entities )
00586         tmp_ptr_fill_entity = &tmp_range;
00587     else
00588         tmp_ptr_fill_entity = NULL;
00589 
00590     for( Range::iterator rit = entities.begin(); rit != entities.end(); ++rit )
00591     {
00592         EntityHandle new_entity;
00593         if( NULL != tmp_ptr_fill_entity ) tmp_ptr_fill_entity->clear();
00594 
00595         EntityHandle this_ent = *rit;
00596         ErrorCode result      = split_entities_manifold( &this_ent, 1, &new_entity, tmp_ptr_fill_entity );
00597         if( MB_SUCCESS != result ) return result;
00598 
00599         new_entities.insert( new_entity );
00600         if( NULL != fill_entities ) fill_entities->merge( *tmp_ptr_fill_entity );
00601     }
00602 
00603     return MB_SUCCESS;
00604 }
00605 
00606 ErrorCode MeshTopoUtil::split_entities_manifold( EntityHandle* entities,
00607                                                  const int num_entities,
00608                                                  EntityHandle* new_entities,
00609                                                  Range* fill_entities,
00610                                                  EntityHandle* gowith_ents )
00611 {
00612     // split entities by duplicating them; splitting manifold means that there is at
00613     // most two higher-dimension entities bounded by a given entity; after split, the
00614     // new entity bounds one and the original entity bounds the other
00615 
00616 #define ITERATE_RANGE( range, it ) for( Range::iterator it = ( range ).begin(); ( it ) != ( range ).end(); ++( it ) )
00617 #define GET_CONNECT_DECL( ent, connect, num_connect )                                     \
00618     const EntityHandle* connect = NULL;                                                   \
00619     int num_connect             = 0;                                                      \
00620     {                                                                                     \
00621         ErrorCode connect_result = mbImpl->get_connectivity( ent, connect, num_connect ); \
00622         if( MB_SUCCESS != connect_result ) return connect_result;                         \
00623     }
00624 #define GET_CONNECT( ent, connect, num_connect )                                          \
00625     {                                                                                     \
00626         ErrorCode connect_result = mbImpl->get_connectivity( ent, connect, num_connect ); \
00627         if( MB_SUCCESS != connect_result ) return connect_result;                         \
00628     }
00629 #define TC                         \
00630     if( MB_SUCCESS != tmp_result ) \
00631     {                              \
00632         result = tmp_result;       \
00633         continue;                  \
00634     }
00635 
00636     ErrorCode result = MB_SUCCESS;
00637     for( int i = 0; i < num_entities; i++ )
00638     {
00639         ErrorCode tmp_result;
00640 
00641         // get original higher-dimensional bounding entities
00642         Range up_adjs[4];
00643         // can only do a split_manifold if there are at most 2 entities of each
00644         // higher dimension; otherwise it's a split non-manifold
00645         bool valid_up_adjs = true;
00646         for( int dim = 1; dim <= 3; dim++ )
00647         {
00648             tmp_result = mbImpl->get_adjacencies( entities + i, 1, dim, false, up_adjs[dim] );
00649             TC;
00650             if( dim > CN::Dimension( TYPE_FROM_HANDLE( entities[i] ) ) && up_adjs[dim].size() > 2 )
00651             {
00652                 valid_up_adjs = false;
00653                 break;
00654             }
00655         }
00656         if( !valid_up_adjs ) return MB_FAILURE;
00657 
00658         // ok to split; create the new entity, with connectivity of the original
00659         GET_CONNECT_DECL( entities[i], connect, num_connect );
00660         EntityHandle new_entity;
00661         result = mbImpl->create_element( mbImpl->type_from_handle( entities[i] ), connect, num_connect, new_entity );
00662         TC;
00663 
00664         // by definition, new entity and original will be equivalent; need to add explicit
00665         // adjs to distinguish them; don't need to check if there's already one there,
00666         // 'cuz add_adjacency does that for us
00667         for( int dim = 1; dim <= 3; dim++ )
00668         {
00669             if( up_adjs[dim].empty() || dim == CN::Dimension( TYPE_FROM_HANDLE( entities[i] ) ) ) continue;
00670 
00671             if( dim < CN::Dimension( TYPE_FROM_HANDLE( entities[i] ) ) )
00672             {
00673                 // adjacencies from other entities to this one; if any of those are equivalent
00674                 // entities, need to make explicit adjacency to new entity too
00675                 for( Range::iterator rit = up_adjs[dim].begin(); rit != up_adjs[dim].end(); ++rit )
00676                 {
00677                     if( equivalent_entities( *rit ) ) result = mbImpl->add_adjacencies( *rit, &new_entity, 1, false );
00678                 }
00679             }
00680             else
00681             {
00682 
00683                 // get the two up-elements
00684                 EntityHandle up_elem1 = *( up_adjs[dim].begin() ),
00685                              up_elem2 = ( up_adjs[dim].size() > 1 ? *( up_adjs[dim].rbegin() ) : 0 );
00686 
00687                 // if two, and a gowith entity was input, make sure the new entity goes with
00688                 // that one
00689                 if( gowith_ents && up_elem2 && gowith_ents[i] != up_elem1 && gowith_ents[i] == up_elem2 )
00690                 {
00691                     EntityHandle tmp_elem = up_elem1;
00692                     up_elem1              = up_elem2;
00693                     up_elem2              = tmp_elem;
00694                 }
00695 
00696                 mbImpl->remove_adjacencies( entities[i], &up_elem1, 1 );
00697                 // (ok if there's an error, that just means there wasn't an explicit adj)
00698 
00699                 tmp_result = mbImpl->add_adjacencies( new_entity, &up_elem1, 1, false );
00700                 TC;
00701                 if( !up_elem2 ) continue;
00702 
00703                 // add adj to other up_adj
00704                 tmp_result = mbImpl->add_adjacencies( entities[i], &up_elem2, 1, false );
00705                 TC;
00706             }
00707         }
00708 
00709         // if we're asked to build a next-higher-dimension object, do so
00710         EntityHandle fill_entity = 0;
00711         EntityHandle tmp_ents[2];
00712         if( NULL != fill_entities )
00713         {
00714             // how to do this depends on dimension
00715             switch( CN::Dimension( TYPE_FROM_HANDLE( entities[i] ) ) )
00716             {
00717                 case 0:
00718                     tmp_ents[0] = entities[i];
00719                     tmp_ents[1] = new_entity;
00720                     tmp_result  = mbImpl->create_element( MBEDGE, tmp_ents, 2, fill_entity );
00721                     TC;
00722                     break;
00723                 case 1:
00724                     tmp_result = mbImpl->create_element( MBPOLYGON, connect, 2, fill_entity );
00725                     TC;
00726                     // need to create explicit adj in this case
00727                     tmp_result = mbImpl->add_adjacencies( entities[i], &fill_entity, 1, false );
00728                     TC;
00729                     tmp_result = mbImpl->add_adjacencies( new_entity, &fill_entity, 1, false );
00730                     TC;
00731                     break;
00732                 case 2:
00733                     tmp_ents[0] = entities[i];
00734                     tmp_ents[1] = new_entity;
00735                     tmp_result  = mbImpl->create_element( MBPOLYHEDRON, tmp_ents, 2, fill_entity );
00736                     TC;
00737                     break;
00738             }
00739             if( 0 == fill_entity )
00740             {
00741                 result = MB_FAILURE;
00742                 continue;
00743             }
00744             fill_entities->insert( fill_entity );
00745         }
00746 
00747         new_entities[i] = new_entity;
00748 
00749     }  // end for over input entities
00750 
00751     return result;
00752 }
00753 
00754 ErrorCode MeshTopoUtil::split_entity_nonmanifold( EntityHandle split_ent,
00755                                                   Range& old_adjs,
00756                                                   Range& new_adjs,
00757                                                   EntityHandle& new_entity )
00758 {
00759     // split an entity into two entities; new entity gets explicit adj to new_adjs,
00760     // old to old_adjs
00761 
00762     // make new entities and add adjacencies
00763     // create the new entity
00764     EntityType split_type = mbImpl->type_from_handle( split_ent );
00765 
00766     ErrorCode result;
00767     if( MBVERTEX == split_type )
00768     {
00769         double coords[3];
00770         result = mbImpl->get_coords( &split_ent, 1, coords );RR;
00771         result = mbImpl->create_vertex( coords, new_entity );RR;
00772     }
00773     else
00774     {
00775         const EntityHandle* connect;
00776         int num_connect;
00777         result = mbImpl->get_connectivity( split_ent, connect, num_connect );RR;
00778         result = mbImpl->create_element( split_type, connect, num_connect, new_entity );RR;
00779 
00780         // remove any explicit adjacencies between new_adjs and split entity
00781         for( Range::iterator rit = new_adjs.begin(); rit != new_adjs.end(); ++rit )
00782             mbImpl->remove_adjacencies( split_ent, &( *rit ), 1 );
00783     }
00784 
00785     if( MBVERTEX != split_type )
00786     {
00787         //  add adj's between new_adjs & new entity, old_adjs & split_entity
00788         for( Range::iterator rit = new_adjs.begin(); rit != new_adjs.end(); ++rit )
00789             mbImpl->add_adjacencies( new_entity, &( *rit ), 1, true );
00790         for( Range::iterator rit = old_adjs.begin(); rit != old_adjs.end(); ++rit )
00791             mbImpl->add_adjacencies( split_ent, &( *rit ), 1, true );
00792     }
00793     else if( split_ent != new_entity )
00794     {
00795         // in addition to explicit adjs, need to check if vertex is part of any
00796         // other entities, and check those entities against ents in old and new adjs
00797         Range other_adjs;
00798         for( int i = 1; i < 4; i++ )
00799         {
00800             result = mbImpl->get_adjacencies( &split_ent, 1, i, false, other_adjs, Interface::UNION );RR;
00801         }
00802         other_adjs = subtract( other_adjs, old_adjs );
00803         other_adjs = subtract( other_adjs, new_adjs );
00804         for( Range::iterator rit1 = other_adjs.begin(); rit1 != other_adjs.end(); ++rit1 )
00805         {
00806             // find an adjacent lower-dimensional entity in old_ or new_ adjs
00807             bool found = false;
00808             for( Range::iterator rit2 = old_adjs.begin(); rit2 != old_adjs.end(); ++rit2 )
00809             {
00810                 if( mbImpl->dimension_from_handle( *rit1 ) != mbImpl->dimension_from_handle( *rit2 ) &&
00811                     common_entity( *rit1, *rit2, mbImpl->dimension_from_handle( *rit1 ) ) )
00812                 {
00813                     found = true;
00814                     old_adjs.insert( *rit1 );
00815                     break;
00816                 }
00817             }
00818             if( found ) continue;
00819             for( Range::iterator rit2 = new_adjs.begin(); rit2 != new_adjs.end(); ++rit2 )
00820             {
00821                 if( mbImpl->dimension_from_handle( *rit1 ) != mbImpl->dimension_from_handle( *rit2 ) &&
00822                     common_entity( *rit1, *rit2, mbImpl->dimension_from_handle( *rit1 ) ) )
00823                 {
00824                     found = true;
00825                     new_adjs.insert( *rit1 );
00826                     break;
00827                 }
00828             }
00829             if( !found ) return MB_FAILURE;
00830         }
00831 
00832         // instead of adjs replace in connectivity
00833         std::vector< EntityHandle > connect;
00834         for( Range::iterator rit = new_adjs.begin(); rit != new_adjs.end(); ++rit )
00835         {
00836             connect.clear();
00837             result = mbImpl->get_connectivity( &( *rit ), 1, connect );RR;
00838             std::replace( connect.begin(), connect.end(), split_ent, new_entity );
00839             result = mbImpl->set_connectivity( *rit, &connect[0], connect.size() );RR;
00840         }
00841     }
00842 
00843     return result;
00844 
00845     /*
00846 
00847     Commented out for now, because I decided to do a different implementation
00848     for the sake of brevity.  However, I still think this function is the right
00849     way to do it, if I ever get the time.  Sigh.
00850 
00851         // split entity d, producing entity nd; generates various new entities,
00852         // see algorithm description in notes from 2/25/05
00853       const EntityHandle split_types = {MBEDGE, MBPOLYGON, MBPOLYHEDRON};
00854       ErrorCode result = MB_SUCCESS;
00855       const int dim = CN::Dimension(TYPE_FROM_HANDLE(d));
00856       MeshTopoUtil mtu(this);
00857 
00858         // get all (d+2)-, (d+1)-cells connected to d
00859       Range dp2s, dp1s, dp1s_manif, dp2s_manif;
00860       result = get_adjacencies(&d, 1, dim+2, false, dp2s); RR;
00861       result = get_adjacencies(&d, 1, dim+1, false, dp1s); RR;
00862 
00863         // also get (d+1)-cells connected to d which are manifold
00864       get_manifold_dp1s(d, dp1s_manif);
00865       get_manifold_dp2s(d, dp2s_manif);
00866 
00867         // make new cell nd, then ndp1
00868       result = copy_entity(d, nd); RR;
00869       EntityHandle tmp_connect[] = {d, nd};
00870       EntityHandle ndp1;
00871       result = create_element(split_types[dim],
00872                               tmp_connect, 2, ndp1); RR;
00873 
00874         // modify (d+2)-cells, depending on what type they are
00875       ITERATE_RANGE(dp2s, dp2) {
00876           // first, get number of connected manifold (d+1)-entities
00877         Range tmp_range, tmp_range2(dp1s_manif);
00878         tmp_range.insert(*dp2);
00879         tmp_range.insert(d);
00880         tmp_result = get_adjacencies(tmp_range, 1, false, tmp_range2); TC;
00881         EntityHandle ndp2;
00882 
00883           // a. manif (d+1)-cells is zero
00884         if (tmp_range2.empty()) {
00885             // construct new (d+1)-cell
00886           EntityHandle ndp1a;
00887           EntityHandle tmp_result = create_element(split_types[dim],
00888                                                      tmp_connect, 2, ndp1a); TC;
00889             // now make new (d+2)-cell
00890           EntityHandle tmp_connect2[] = {ndp1, ndp1a};
00891           tmp_result = create_element(split_types[dim+1],
00892                                       tmp_connect2, 2, ndp2); TC;
00893             // need to add explicit adjacencies, since by definition ndp1, ndp1a will be equivalent
00894           tmp_result = add_adjacencies(ndp1a, &dp2, 1, false); TC;
00895           tmp_result = add_adjacencies(ndp1a, &ndp2, 1, false); TC;
00896           tmp_result = add_adjacencies(ndp1, &ndp2, 1, false); TC;
00897 
00898             // now insert nd into connectivity of dp2, right after d if dim < 1
00899           std::vector<EntityHandle> connect;
00900           tmp_result = get_connectivity(&dp2, 1, connect); TC;
00901           if (dim < 1) {
00902             std::vector<EntityHandle>::iterator vit = std::find(connect.begin(), connect.end(), d);
00903             if (vit == connect.end()) {
00904               result = MB_FAILURE;
00905               continue;
00906             }
00907             connect.insert(vit, nd);
00908           }
00909           else
00910             connect.push_back(nd);
00911           tmp_result = set_connectivity(dp2, connect); TC;
00912 
00913             // if dim < 1, need to add explicit adj from ndp2 to higher-dim ents, since it'll
00914             // be equiv to other dp2 entities
00915           if (dim < 1) {
00916             Range tmp_dp3s;
00917             tmp_result = get_adjacencies(&dp2, 1, dim+3, false, tmp_dp3s); TC;
00918             tmp_result = add_adjacencies(ndp2, tmp_dp3s, false); TC;
00919           }
00920         } // end if (tmp_range2.empty())
00921 
00922           // b. single manifold (d+1)-cell, which isn't adjacent to manifold (d+2)-cell
00923         else if (tmp_range2.size() == 1) {
00924             // b1. check validity, and skip if not valid
00925 
00926             // only change if not dp1-adjacent to manifold dp2cell; check that...
00927           Range tmp_adjs(dp2s_manif);
00928           tmp_result = get_adjacencies(&(*tmp_range2.begin()), 1, dim+2, false, tmp_adjs); TC;
00929           if (!tmp_adjs.empty()) continue;
00930 
00931           EntityHandle dp1 = *tmp_range2.begin();
00932 
00933             // b2. make new (d+1)- and (d+2)-cell next to dp2
00934 
00935             // get the (d+2)-cell on the other side of dp1
00936           tmp_result = get_adjacencies(&dp1, 1, dim+2, false, tmp_adjs); TC;
00937           EntityHandle odp2 = *tmp_adjs.begin();
00938           if (odp2 == dp2) odp2 = *tmp_adjs.rbegin();
00939 
00940             // get od, the d-cell on dp1_manif which isn't d
00941           tmp_result = get_adjacencies(&dp1_manif, 1, dim, false, tmp_adjs); TC;
00942           tmp_adjs.erase(d);
00943           if (tmp_adjs.size() != 1) {
00944             result = MB_FAILURE;
00945             continue;
00946           }
00947           EntityHandle od = *tmp_adjs.begin();
00948 
00949             // make a new (d+1)-cell from od and nd
00950           tmp_adjs.insert(nd);
00951           tmp_result = create_element(split_types[1], tmp_adjs, ndp1a); TC;
00952 
00953             // construct new (d+2)-cell from dp1, ndp1, ndp1a
00954           tmp_adjs.clear();
00955           tmp_adjs.insert(dp1); tmp_adjs.insert(ndp1); tmp_adjs.insert(ndp1a);
00956           tmp_result = create_element(split_types[2], tmp_adjs, ndp2); TC;
00957 
00958             // b3. replace d, dp1 in connect/adjs of odp2
00959           std::vector<EntityHandle> connect;
00960           tmp_result = get_connectivity(&odp2, 1, connect); TC;
00961           if (dim == 0) {
00962             *(std::find(connect.begin(), connect.end(), d)) = nd;
00963             remove_adjacency(dp1, odp2);
00964 
00965 
00966 
00967             // if dp1 was explicitly adj to odp2, remove it
00968           remove_adjacency
00969 
00970     ...
00971 
00972     */
00973 }
00974 
00975 //! return whether entity is equivalent to any other of same type and same vertices;
00976 //! if equivalent entity is found, it's returned in equiv_ents and return value is true,
00977 //! false otherwise.
00978 bool MeshTopoUtil::equivalent_entities( const EntityHandle entity, Range* equiv_ents )
00979 {
00980     const EntityHandle* connect = NULL;
00981     int num_connect             = 0;
00982     ErrorCode result            = mbImpl->get_connectivity( entity, connect, num_connect );
00983     if( MB_SUCCESS != result ) return false;
00984 
00985     Range dum;
00986     result = mbImpl->get_adjacencies( connect, num_connect, mbImpl->dimension_from_handle( entity ), false, dum );
00987     dum.erase( entity );
00988 
00989     if( NULL != equiv_ents )
00990     {
00991         equiv_ents->swap( dum );
00992     }
00993 
00994     if( !dum.empty() )
00995         return true;
00996     else
00997         return false;
00998 }
00999 
01000 }  // namespace moab
 All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Defines