102 class FaceToCellConnectivity
106 : m_face_family(
nullptr)
107 , m_cell_family(
nullptr) {
109 throw FatalErrorException(
"FaceToCellIncrementalConnectivity must be created with face family as source and cell family as target. Exiting.");
110 m_face_family =
dynamic_cast<FaceFamily*
>(source_family);
111 m_cell_family =
dynamic_cast<CellFamily*
>(target_family);
118 ItemInternal* face = m_face_family->itemsInternal()[source_item.localId()];
119 ItemInternal* cell = m_cell_family->itemsInternal()[target_item.localId()];
123 ConnectivityItemVector face_nodes_connectivity(
mesh->itemFamilyNetwork()->getConnectivity(m_face_family,
mesh->nodeFamily(), connectivityName(m_face_family,
mesh->nodeFamily())));
125 ConnectivityItemVector cell_nodes_connectivity(
mesh->itemFamilyNetwork()->getConnectivity(m_cell_family,
mesh->nodeFamily(), connectivityName(m_cell_family,
mesh->nodeFamily())));
131 for (
const auto& node : face_nodes)
132 face_node_uids.
add(node.uniqueId().asInt64());
133 std::set<Int64> face_nodes_set(face_node_uids.
begin(),face_node_uids.
end());
135 for (
Integer local_face_index = 0; local_face_index < cell_type_info->
nbLocalFace() && face_index == -1; ++local_face_index)
137 local_face_node_uids.
clear();
139 if (local_face.
nbNode() != face_nodes.
size())
continue;
140 for (
Integer local_node_index = 0; local_node_index < local_face.
nbNode() ; ++local_node_index) local_face_node_uids.
add(cell_nodes[local_face.
node(local_node_index)].uniqueId().asInt64());
141 std::set<Int64> local_face_nodes_set(local_face_node_uids.
begin(),local_face_node_uids.
end());
142 if (local_face_nodes_set == face_nodes_set) face_index = local_face_index;
144 if (face_index == -1)
throw FatalErrorException(String::format(
"Face {0} does not belong to Cell {1}. Cannot connect. Exiting."));
145 if (
mesh->dimension() == 1)
146 return (face_index == 1);
158 arcaneThrowIfNull(source,
"source",
"Invalid null source item");
159 arcaneThrowIfNull(target,
"target",
"Invalid null target item");
161 ARCANE_UNUSED(source);
162 ARCANE_UNUSED(target);
177class ARCANE_MESH_EXPORT FaceToCellIncrementalItemConnectivity
178:
public FaceToCellConnectivity
179,
public IncrementalItemConnectivity
183 : FaceToCellConnectivity(source_family,target_family)
184 , IncrementalItemConnectivity(source_family,target_family,aname){}
186 virtual ~FaceToCellIncrementalItemConnectivity() {}
196 if (isFrontCell(source_item,target_item))
197 _addFrontCellToFace(m_face_family->itemsInternal()[source_item.localId()],m_cell_family->itemsInternal()[target_item.localId()]);
199 _addBackCellToFace(m_face_family->itemsInternal()[source_item.localId()],m_cell_family->itemsInternal()[target_item.localId()]);
214 _checkValidSourceTargetItems(face,cell);
224 const bool check_orientation =
false;
225 if (check_orientation){
229 " This is most probably due to the fact that the face"
230 " is connected to a reverse cell with a negative volume."
231 " Face={0}. new_cell={1} current_cell={2}",face->
uniqueId().asInt64(),
240 Int32 iback_cell_lid = (nb_cell==1) ? face->cellId(0) : NULL_ITEM_LOCAL_ID;
241 ItemLocalId back_cell_lid(iback_cell_lid);
242 ItemLocalId front_cell_lid(cell->
localId());
243 _setBackAndFrontCells(face,back_cell_lid,front_cell_lid);
248 void _addBackCellToFace(ItemInternal* face, ItemInternal* cell){
249 _checkValidSourceTargetItems(face,cell);
251 Integer nb_cell = nbConnectedItem(ItemLocalId(face));
259 const bool check_orientation =
false;
260 if (check_orientation){
261 if (face->flags() & ItemFlags::II_HasBackCell){
262 ItemInternal* current_cell = m_cell_family->itemsInternal()[connectedItemLocalId(ItemLocalId(face),0)];
264 " This is most probably due to the fact that the face"
265 " is connected to a reverse cell with a negative volume."
266 " Face={0}. new_cell={1} current_cell={2}",face->uniqueId().asInt64(),
267 cell->uniqueId().asInt64(),current_cell->uniqueId().asInt64());
272 ARCANE_FATAL(
"face '{0}' already has two cells",face->uniqueId().asInt64());
274 Int32 ifront_cell_lid = (nb_cell==1) ? face->cellId(0) : NULL_ITEM_LOCAL_ID;
276 ItemLocalId back_cell_lid(cell->localId());
277 ItemLocalId front_cell_lid(ifront_cell_lid);
278 _setBackAndFrontCells(face,back_cell_lid,front_cell_lid);
284 void _setBackAndFrontCells(ItemInternal* face,ItemLocalId back_cell_lid,ItemLocalId front_cell_lid){
285 ItemLocalId face_lid(face->localId());
289 removeConnectedItems(face_lid);
291 if (front_cell_lid==NULL_ITEM_LOCAL_ID){
292 if (back_cell_lid!=NULL_ITEM_LOCAL_ID){
294 IncrementalItemConnectivity::addConnectedItem(face_lid,back_cell_lid);
296 mod_flags = (ItemFlags::II_Boundary | ItemFlags::II_HasBackCell | ItemFlags::II_BackCellIsFirst);
301 else if (back_cell_lid==NULL_ITEM_LOCAL_ID){
303 IncrementalItemConnectivity::addConnectedItem(face_lid,front_cell_lid);
305 mod_flags = (ItemFlags::II_Boundary | ItemFlags::II_HasFrontCell | ItemFlags::II_FrontCellIsFirst);
309 IncrementalItemConnectivity::addConnectedItem(face_lid,back_cell_lid);
310 IncrementalItemConnectivity::addConnectedItem(face_lid,front_cell_lid);
312 mod_flags = (ItemFlags::II_HasFrontCell | ItemFlags::II_HasBackCell | ItemFlags::II_BackCellIsFirst);
314 Int32 face_flags = face->flags();
315 face_flags &= ~ItemFlags::II_InterfaceFlags;
316 face_flags |= mod_flags;
317 face->setFlags(face_flags);
325 _checkValidSourceTargetItems(face,m_cell_family->itemsInternal()[cell_to_remove_lid]);
333 ARCANE_FATAL(
"Can not remove cell lid={0} from face uid={1} with no cell connected",
334 cell_to_remove_lid, face->
uniqueId());
337 Integer nb_cell_after = nb_cell-1;
338 const Int32 null_cell_lid = NULL_ITEM_LOCAL_ID;
343 if (nb_cell_after!=0){
344 Int32 cell0 = face->cellId(0);
345 Int32 cell1 = face->cellId(1);
348 if (cell0==cell_to_remove_lid){