Salome HOME
1615842b6dce8d4504489b985a2a1c302aeb8a17
[modules/med.git] / src / MEDCoupling / MEDCouplingMesh.cxx
1 // Copyright (C) 2007-2013  CEA/DEN, EDF R&D
2 //
3 // This library is free software; you can redistribute it and/or
4 // modify it under the terms of the GNU Lesser General Public
5 // License as published by the Free Software Foundation; either
6 // version 2.1 of the License.
7 //
8 // This library is distributed in the hope that it will be useful,
9 // but WITHOUT ANY WARRANTY; without even the implied warranty of
10 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
11 // Lesser General Public License for more details.
12 //
13 // You should have received a copy of the GNU Lesser General Public
14 // License along with this library; if not, write to the Free Software
15 // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA
16 //
17 // See http://www.salome-platform.org/ or email : webmaster.salome@opencascade.com
18 //
19 // Author : Anthony Geay (CEA/DEN)
20
21 #include "MEDCouplingMesh.hxx"
22 #include "MEDCouplingUMesh.hxx"
23 #include "MEDCouplingMemArray.hxx"
24 #include "MEDCouplingFieldDouble.hxx"
25 #include "MEDCouplingFieldDiscretization.hxx"
26 #include "MEDCouplingAutoRefCountObjectPtr.hxx"
27
28 #include <set>
29 #include <cmath>
30 #include <sstream>
31 #include <fstream>
32 #include <iterator>
33
34 using namespace ParaMEDMEM;
35
36 MEDCouplingMesh::MEDCouplingMesh():_time(0.),_iteration(-1),_order(-1)
37 {
38 }
39
40 MEDCouplingMesh::MEDCouplingMesh(const MEDCouplingMesh& other):_name(other._name),_description(other._description),
41                                                                _time(other._time),_iteration(other._iteration),
42                                                                _order(other._order),_time_unit(other._time_unit)
43 {
44 }
45
46 std::size_t MEDCouplingMesh::getHeapMemorySize() const
47 {
48   return _name.capacity()+_description.capacity()+_time_unit.capacity();
49 }
50
51 /*!
52  * This method is only for ParaMEDMEM in ParaFIELD constructor.
53  */
54 bool MEDCouplingMesh::isStructured() const
55 {
56   return getType()==CARTESIAN;
57 }
58
59 bool MEDCouplingMesh::isEqualIfNotWhy(const MEDCouplingMesh *other, double prec, std::string& reason) const throw(INTERP_KERNEL::Exception)
60 {
61   if(!other)
62     throw INTERP_KERNEL::Exception("MEDCouplingMesh::isEqualIfNotWhy : other instance is NULL !");
63   std::ostringstream oss; oss.precision(15);
64   if(_name!=other->_name)
65     {
66       oss << "Mesh names differ : this name = \"" << _name << "\" and other name = \"" << other->_name << "\" !";
67       reason=oss.str();
68       return false;
69     }
70   if(_description!=other->_description)
71     {
72       oss << "Mesh descriptions differ : this description = \"" << _description << "\" and other description = \"" << other->_description << "\" !";
73       reason=oss.str();
74       return false;
75     }
76   if(_iteration!=other->_iteration)
77     {
78       oss << "Mesh iterations differ : this iteration = \"" << _iteration << "\" and other iteration = \"" << other->_iteration << "\" !";
79       reason=oss.str();
80       return false;
81     }
82   if(_order!=other->_order)
83     {
84       oss << "Mesh orders differ : this order = \"" << _order << "\" and other order = \"" << other->_order << "\" !";
85       reason=oss.str();
86       return false;
87     }
88   if(_time_unit!=other->_time_unit)
89     {
90       oss << "Mesh time units differ : this time unit = \"" << _time_unit << "\" and other time unit = \"" << other->_time_unit << "\" !";
91       reason=oss.str();
92       return false;
93     }
94   if(fabs(_time-other->_time)>=1e-12)
95     {
96       oss << "Mesh times differ : this time = \"" << _time << "\" and other time = \"" << other->_time << "\" !";
97       reason=oss.str();
98       return false;
99     }
100   return true;
101 }
102
103 /*!
104  * Checks if \a this and another MEDCouplingMesh are fully equal.
105  *  \param [in] other - an instance of MEDCouplingMesh to compare with \a this one.
106  *  \param [in] prec - precision value used to compare node coordinates.
107  *  \return bool - \c true if the two meshes are equal, \c false else.
108  */
109 bool MEDCouplingMesh::isEqual(const MEDCouplingMesh *other, double prec) const throw(INTERP_KERNEL::Exception)
110 {
111   std::string tmp;
112   return isEqualIfNotWhy(other,prec,tmp);
113 }
114
115 /*!
116  * This method checks geo equivalence between two meshes : \a this and \a other.
117  * If no exception is thrown \a this and \a other are geometrically equivalent regarding \a levOfCheck level.
118  * This method is typically used to change the mesh of a field "safely" depending the \a levOfCheck level considered.
119  * 
120  * In case of success cell \c other[i] is equal to the cell \c this[cellCor[i]].
121  * In case of success node \c other->getCoords()[i] is equal to the node \c this->getCoords()[nodeCor[i]].
122  *
123  * If \a cellCor is null (or Py_None) it means that for all #i cell in \a other is equal to cell # i in \a this.
124  *
125  * If \a nodeCor is null (or Py_None) it means that for all #i node in \a other is equal to node # i in \a this.
126  *
127  * So null (or Py_None) returned in \a cellCor and/or \a nodeCor means identity array. This is for optimization reason to avoid to build useless arrays
128  * for some \a levOfCheck (for example 0).
129  *
130  * **Warning a not null output does not mean that it is not identity !**
131  *
132  * \param [in] other - the mesh to be compared with \a this.
133  * \param [in] levOfCheck - input that specifies the level of check specified. The possible values are listed below.
134  * \param [in] prec - input that specifies precision for double float data used for comparison in meshes.
135  * \param [out] cellCor - output array not always informed (depending \a levOfCheck param) that gives the corresponding array for cells from \a other to \a this.
136  * \param [out] nodeCor - output array not always informed (depending \a levOfCheck param) that gives the corresponding array for nodes from \a other to \a this.
137  *
138  * Possible values for levOfCheck :
139  *   - 0 for strict equality. This is the strongest level. \a cellCor and \a nodeCor params are never informed.
140  *   - 10,11,12 (10+x) for less strict equality. Two meshes are compared geometrically. In case of success \a cellCor and \a nodeCor are informed. Warning ! These equivalences are CPU/Mem costly. The 3 values correspond respectively to policy used for cell comparison (see MEDCouplingUMesh::zipConnectivityTraducer to have more details)
141  *   - 20,21,22 (20+x), for less strict equality. Two meshes are compared geometrically. The difference with the previous version is that nodes(coordinates) are expected to be the same between this and other. In case of success \a cellCor is informed. Warning ! These equivalences are CPU/Mem costly. The 3 values correspond respectively to policy used for cell comparison (see MEDCouplingUMesh::zipConnectivityTraducer to have more details)
142  *   - 1 for fast 'equality'. This is a lazy level. Just number of cells and number of nodes are considered here and 3 cells (begin,middle,end)
143  *   - 2 for deep 'equality' as 0 option except that no control is done on all strings in mesh.
144  *
145  * So the most strict level of check is 0 (equality). The least strict is 12. If the level of check 12 throws, the 2 meshes \a this and \a other are not similar enough
146  * to be compared. An interpolation using MEDCouplingRemapper class should be then used.
147  */
148 void MEDCouplingMesh::checkGeoEquivalWith(const MEDCouplingMesh *other, int levOfCheck, double prec,
149                                           DataArrayInt *&cellCor, DataArrayInt *&nodeCor) const throw(INTERP_KERNEL::Exception)
150 {
151   cellCor=0;
152   nodeCor=0;
153   if(this==other)
154     return ;
155   switch(levOfCheck)
156     {
157     case 0:
158       {
159         if(!isEqual(other,prec))
160           throw INTERP_KERNEL::Exception("checkGeoFitWith : Meshes are not equal !");
161         return ;
162       }
163     case 10:
164     case 11:
165     case 12:
166       {
167         checkDeepEquivalWith(other,levOfCheck-10,prec,cellCor,nodeCor);
168         return ;
169       }
170     case 20:
171     case 21:
172     case 22:
173       {
174         checkDeepEquivalOnSameNodesWith(other,levOfCheck-20,prec,cellCor);
175         return ;
176       }
177     case 1:
178       {
179         checkFastEquivalWith(other,prec);
180         return;
181       }
182     case 2:
183       {
184         if(!isEqualWithoutConsideringStr(other,prec))
185           throw INTERP_KERNEL::Exception("checkGeoFitWith : Meshes are not equal without considering strings !");
186         return ;
187       }
188     default:
189       throw INTERP_KERNEL::Exception("checkGeoFitWith : Invalid levOfCheck specified ! Value must be in 0,1,2,10,11 or 12.");
190     }
191 }
192
193 /*!
194  * Finds cells whose all nodes are in a given array of node ids.
195  *  \param [in] partBg - the array of node ids.
196  *  \param [in] partEnd - end of \a partBg, i.e. a pointer to a (last+1)-th element
197  *          of \a partBg.
198  *  \return DataArrayInt * - a new instance of DataArrayInt holding ids of found
199  *          cells. The caller is to delete this array using decrRef() as it is no
200  *          more needed.
201  */
202 DataArrayInt *MEDCouplingMesh::getCellIdsFullyIncludedInNodeIds(const int *partBg, const int *partEnd) const
203 {
204   std::vector<int> crest;
205   std::set<int> p(partBg,partEnd);
206   int nbOfCells=getNumberOfCells();
207   for(int i=0;i<nbOfCells;i++)
208     {
209       std::vector<int> conn;
210       getNodeIdsOfCell(i,conn);
211       bool cont=true;
212       for(std::vector<int>::const_iterator iter=conn.begin();iter!=conn.end() && cont;iter++)
213         if(p.find(*iter)==p.end())
214           cont=false;
215       if(cont)
216         crest.push_back(i);
217     }
218   DataArrayInt *ret=DataArrayInt::New();
219   ret->alloc((int)crest.size(),1);
220   std::copy(crest.begin(),crest.end(),ret->getPointer());
221   return ret;
222 }
223
224 /*!
225  * This method checks fastly that \a this and \a other are equal. All common checks are done here.
226  */
227 void MEDCouplingMesh::checkFastEquivalWith(const MEDCouplingMesh *other, double prec) const throw(INTERP_KERNEL::Exception)
228 {
229   if(!other)
230     throw INTERP_KERNEL::Exception("MEDCouplingMesh::checkFastEquivalWith : input mesh is null !");
231   if(getMeshDimension()!=other->getMeshDimension())
232     throw INTERP_KERNEL::Exception("checkFastEquivalWith : Mesh dimensions are not equal !");
233   if(getSpaceDimension()!=other->getSpaceDimension())
234     throw INTERP_KERNEL::Exception("checkFastEquivalWith : Space dimensions are not equal !");
235   if(getNumberOfCells()!=other->getNumberOfCells())
236     throw INTERP_KERNEL::Exception("checkFastEquivalWith : number of cells are not equal !");
237 }
238
239 /*!
240  * This method is very poor and looks only if \a this and \a other are candidate for merge of fields lying repectively on them.
241  */
242 bool MEDCouplingMesh::areCompatibleForMerge(const MEDCouplingMesh *other) const
243 {
244   if(!other)
245     throw INTERP_KERNEL::Exception("MEDCouplingMesh::areCompatibleForMerge : input mesh is null !");
246   if(getMeshDimension()!=other->getMeshDimension())
247     return false;
248   if(getSpaceDimension()!=other->getSpaceDimension())
249     return false;
250   return true;
251 }
252
253 /*!
254  * This method is equivalent to MEDCouplingMesh::buildPart method except that here the cell ids are specified using slice \a beginCellIds \a endCellIds and \a stepCellIds.
255  *
256  * \sa MEDCouplingMesh::buildPart
257  */
258 MEDCouplingMesh *MEDCouplingMesh::buildPartRange(int beginCellIds, int endCellIds, int stepCellIds) const throw(INTERP_KERNEL::Exception)
259 {
260   MEDCouplingAutoRefCountObjectPtr<DataArrayInt> cellIds=DataArrayInt::Range(beginCellIds,endCellIds,stepCellIds);
261   return buildPart(cellIds->begin(),cellIds->end());
262 }
263
264 /*!
265  * This method is equivalent to MEDCouplingMesh::buildPartAndReduceNodes method except that here the cell ids are specified using slice \a beginCellIds \a endCellIds and \a stepCellIds.
266  *
267  * \sa MEDCouplingMesh::buildPartAndReduceNodes
268  */
269 MEDCouplingMesh *MEDCouplingMesh::buildPartRangeAndReduceNodes(int beginCellIds, int endCellIds, int stepCellIds, int& beginOut, int& endOut, int& stepOut, DataArrayInt*& arr) const throw(INTERP_KERNEL::Exception)
270 {
271   MEDCouplingAutoRefCountObjectPtr<DataArrayInt> cellIds=DataArrayInt::Range(beginCellIds,endCellIds,stepCellIds);
272   return buildPartAndReduceNodes(cellIds->begin(),cellIds->end(),arr);
273 }
274
275 /*!
276  * This method builds a field lying on \a this with 'nbOfComp' components.
277  * 'func' is a pointer that points to a function that takes 2 arrays in parameter and returns a boolean.
278  * The first array is a in-param of size this->getSpaceDimension and the second an out param of size 'nbOfComp'.
279  * The return field will have type specified by 't'. 't' is also used to determine where values of field will be
280  * evaluate.
281  * Contrary to other fillFromAnalytic methods this method requests a C++ function pointer as input.
282  * The 'func' is a callback that takes as first parameter an input array of size 'this->getSpaceDimension()',
283  * the second parameter is a pointer on a valid zone of size at least equal to 'nbOfComp' values. And too finish
284  * the returned value is a boolean that is equal to False in case of invalid evaluation (log(0) for example...)
285  * 
286  * \param t type of field returned and specifies where the evaluation of func will be done.
287  * \param nbOfComp number of components of returned field.
288  * \param func pointer to a function that should return false if the evaluation failed. (division by 0. for example)
289  * \return field with counter = 1.
290  */
291 MEDCouplingFieldDouble *MEDCouplingMesh::fillFromAnalytic(TypeOfField t, int nbOfComp, FunctionToEvaluate func) const
292 {
293   MEDCouplingAutoRefCountObjectPtr<MEDCouplingFieldDouble> ret=MEDCouplingFieldDouble::New(t,ONE_TIME);
294   ret->setMesh(this);
295   ret->fillFromAnalytic(nbOfComp,func);
296   ret->synchronizeTimeWithSupport();
297   return ret.retn();
298 }
299
300 /*!
301  * This method copyies all tiny strings from other (name and components name).
302  * @throw if other and this have not same mesh type.
303  */
304 void MEDCouplingMesh::copyTinyStringsFrom(const MEDCouplingMesh *other) throw(INTERP_KERNEL::Exception)
305 {
306   if(!other)
307     throw INTERP_KERNEL::Exception("MEDCouplingMesh::copyTinyStringsFrom : input mesh is null !");
308   _name=other->_name;
309   _description=other->_description;
310   _time_unit=other->_time_unit;
311 }
312
313 /*!
314  * This method copies all attributes that are \b NOT arrays in this.
315  * All tiny attributes not usefully for state of \a this are ignored.
316  */
317 void MEDCouplingMesh::copyTinyInfoFrom(const MEDCouplingMesh *other) throw(INTERP_KERNEL::Exception)
318 {
319   copyTinyStringsFrom(other);
320   _time=other->_time;
321   _iteration=other->_iteration;
322   _order=other->_order;
323 }
324
325 /*!
326  * \anchor mcmesh_fillFromAnalytic
327  * Creates a new MEDCouplingFieldDouble of a given type, one time, with given number of
328  * components, lying on \a this mesh, with contents got by applying a specified
329  * function to coordinates of field location points (defined by the given field type).
330  * For example, if \a t == ParaMEDMEM::ON_CELLS, the function is applied to cell
331  * barycenters.<br>
332  * For more info on supported expressions that can be used in the function, see \ref
333  * MEDCouplingArrayApplyFuncExpr. The function can include arbitrary named variables
334  * (e.g. "x","y" or "va44") to refer to components of point coordinates. Names of
335  * variables are sorted in \b alphabetical \b order to associate a variable name with a
336  * component. For example, in the expression "2*x+z", "x" stands for the component #0
337  * and "z" stands for the component #1 (\b not #2)!<br>
338  * In a general case, a value resulting from the function evaluation is assigned to all
339  * components of the field. But there is a possibility to have its own expression for
340  * each component within one function. For this purpose, there are predefined variable
341  * names (IVec, JVec, KVec, LVec etc) each dedicated to a certain component (IVec, to
342  * the component #0 etc). A factor of such a variable is added to the
343  * corresponding component only.<br>
344  * For example, \a nbOfComp == 4, \a this->getSpaceDimension() == 3, coordinates of a
345  * point are (1.,3.,7.), then
346  *   - "2*x + z"               produces (5.,5.,5.,5.)
347  *   - "2*x + 0*y + z"         produces (9.,9.,9.,9.)
348  *   - "2*x*IVec + (x+z)*LVec" produces (2.,0.,0.,4.)
349  *   - "2*y*IVec + z*KVec + x" produces (7.,1.,1.,4.)
350  *
351  *  \param [in] t - the field type. It defines, apart from other things, points to
352  *         coordinates of which the function is applied to get field values.
353  *  \param [in] nbOfComp - the number of components in the result field.
354  *  \param [in] func - a string defining the expression which is evaluated to get
355  *         field values.
356  *  \return MEDCouplingFieldDouble * - a new instance of MEDCouplingFieldDouble. The
357  *         caller is to delete this field using decrRef() as it is no more needed. 
358  *  \throw If the nodal connectivity of cells is not defined.
359  *  \throw If computing \a func fails.
360  *
361  *  \ref cpp_mcmesh_fillFromAnalytic "Here is a C++ example".<br>
362  *  \ref  py_mcmesh_fillFromAnalytic "Here is a Python example".
363  */
364 MEDCouplingFieldDouble *MEDCouplingMesh::fillFromAnalytic(TypeOfField t, int nbOfComp, const char *func) const
365 {
366   MEDCouplingAutoRefCountObjectPtr<MEDCouplingFieldDouble> ret=MEDCouplingFieldDouble::New(t,ONE_TIME);
367   ret->setMesh(this);
368   ret->fillFromAnalytic(nbOfComp,func);
369   ret->synchronizeTimeWithSupport();
370   return ret.retn();
371 }
372
373 /*!
374  * Creates a new MEDCouplingFieldDouble of a given type, one time, with given number of
375  * components, lying on \a this mesh, with contents got by applying a specified
376  * function to coordinates of field location points (defined by the given field type).
377  * For example, if \a t == ParaMEDMEM::ON_CELLS, the function is applied to cell
378  * barycenters. This method differs from
379  * \ref MEDCouplingMesh::fillFromAnalytic(TypeOfField t, int nbOfComp, const char *func) const "fillFromAnalytic()"
380  * by the way how variable
381  * names, used in the function, are associated with components of coordinates of field
382  * location points; here, a variable name corresponding to a component is retrieved from
383  * a corresponding node coordinates array (where it is set via
384  * DataArrayDouble::setInfoOnComponent()).<br>
385  * For more info on supported expressions that can be used in the function, see \ref
386  * MEDCouplingArrayApplyFuncExpr. <br> 
387  * In a general case, a value resulting from the function evaluation is assigned to all
388  * components of a field value. But there is a possibility to have its own expression for
389  * each component within one function. For this purpose, there are predefined variable
390  * names (IVec, JVec, KVec, LVec etc) each dedicated to a certain component (IVec, to
391  * the component #0 etc). A factor of such a variable is added to the
392  * corresponding component only.<br>
393  * For example, \a nbOfComp == 4, \a this->getSpaceDimension() == 3, names of
394  * spatial components are "x", "y" and "z", coordinates of a
395  * point are (1.,3.,7.), then
396  *   - "2*x + z"               produces (9.,9.,9.,9.)
397  *   - "2*x*IVec + (x+z)*LVec" produces (2.,0.,0.,8.)
398  *   - "2*y*IVec + z*KVec + x" produces (7.,1.,1.,8.)
399  *
400  *  \param [in] t - the field type. It defines, apart from other things, the points to
401  *         coordinates of which the function is applied to get field values.
402  *  \param [in] nbOfComp - the number of components in the result field.
403  *  \param [in] func - a string defining the expression which is evaluated to get
404  *         field values.
405  *  \return MEDCouplingFieldDouble * - a new instance of MEDCouplingFieldDouble. The
406  *         caller is to delete this field using decrRef() as it is no more needed. 
407  *  \throw If the node coordinates are not defined.
408  *  \throw If the nodal connectivity of cells is not defined.
409  *  \throw If computing \a func fails.
410  *
411  *  \ref cpp_mcmesh_fillFromAnalytic2 "Here is a C++ example".<br>
412  *  \ref  py_mcmesh_fillFromAnalytic2 "Here is a Python example".
413  */
414 MEDCouplingFieldDouble *MEDCouplingMesh::fillFromAnalytic2(TypeOfField t, int nbOfComp, const char *func) const
415 {
416   MEDCouplingAutoRefCountObjectPtr<MEDCouplingFieldDouble> ret=MEDCouplingFieldDouble::New(t,ONE_TIME);
417   ret->setMesh(this);
418   ret->fillFromAnalytic2(nbOfComp,func);
419   ret->synchronizeTimeWithSupport();
420   return ret.retn();
421 }
422
423 /*!
424  * Creates a new MEDCouplingFieldDouble of a given type, one time, with given number of
425  * components, lying on \a this mesh, with contents got by applying a specified
426  * function to coordinates of field location points (defined by the given field type).
427  * For example, if \a t == ParaMEDMEM::ON_CELLS, the function is applied to cell
428  * barycenters. This method differs from \ref  \ref mcmesh_fillFromAnalytic
429  * "fillFromAnalytic()" by the way how variable
430  * names, used in the function, are associated with components of coordinates of field
431  * location points; here, a component index of a variable is defined by a
432  * rank of the variable within the input array \a varsOrder.<br>
433  * For more info on supported expressions that can be used in the function, see \ref
434  * MEDCouplingArrayApplyFuncExpr.
435  * In a general case, a value resulting from the function evaluation is assigned to all
436  * components of the field. But there is a possibility to have its own expression for
437  * each component within one function. For this purpose, there are predefined variable
438  * names (IVec, JVec, KVec, LVec etc) each dedicated to a certain component (IVec, to
439  * the component #0 etc). A factor of such a variable is added to the
440  * corresponding component only.<br>
441  * For example, \a nbOfComp == 4, \a this->getSpaceDimension() == 3, names of
442  * spatial components are given in \a varsOrder: ["x", "y","z"], coordinates of a
443  * point are (1.,3.,7.), then
444  *   - "2*x + z"               produces (9.,9.,9.,9.)
445  *   - "2*x*IVec + (x+z)*LVec" produces (2.,0.,0.,8.)
446  *   - "2*y*IVec + z*KVec + x" produces (7.,1.,1.,8.)
447  *
448  *  \param [in] t - the field type. It defines, apart from other things, the points to
449  *         coordinates of which the function is applied to get field values.
450  *  \param [in] nbOfComp - the number of components in the result field.
451  *  \param [in] varsOrder - the vector defining names of variables used to refer to
452  *         components of coordinates of field location points. A variable named
453  *         varsOrder[0] refers to the component #0 etc.
454  *  \param [in] func - a string defining the expression which is evaluated to get
455  *         field values.
456  *  \return MEDCouplingFieldDouble * - a new instance of MEDCouplingFieldDouble. The
457  *         caller is to delete this field using decrRef() as it is no more needed. 
458  *  \throw If the node coordinates are not defined.
459  *  \throw If the nodal connectivity of cells is not defined.
460  *  \throw If computing \a func fails.
461  *
462  *  \ref cpp_mcmesh_fillFromAnalytic3 "Here is a C++ example".<br>
463  *  \ref  py_mcmesh_fillFromAnalytic3 "Here is a Python example".
464  */
465 MEDCouplingFieldDouble *MEDCouplingMesh::fillFromAnalytic3(TypeOfField t, int nbOfComp, const std::vector<std::string>& varsOrder, const char *func) const
466 {
467   MEDCouplingAutoRefCountObjectPtr<MEDCouplingFieldDouble> ret=MEDCouplingFieldDouble::New(t,ONE_TIME);
468   ret->setMesh(this);
469   ret->fillFromAnalytic3(nbOfComp,varsOrder,func);
470   ret->synchronizeTimeWithSupport();
471   return ret.retn();
472 }
473
474 /*!
475  * Creates a new MEDCouplingMesh by concatenating two given meshes, if possible.
476  * Cells and nodes of
477  * the first mesh precede cells and nodes of the second mesh within the result mesh.
478  * The meshes must be of the same mesh type, else, an exception is thrown. The method
479  * MergeMeshes(), accepting a vector of input meshes, has no such a limitation.
480  *  \param [in] mesh1 - the first mesh.
481  *  \param [in] mesh2 - the second mesh.
482  *  \return MEDCouplingMesh * - the result mesh. It is a new instance of
483  *          MEDCouplingMesh. The caller is to delete this mesh using decrRef() as it
484  *          is no more needed.
485  *  \throw If the meshes are of different mesh type.
486  */
487 MEDCouplingMesh *MEDCouplingMesh::MergeMeshes(const MEDCouplingMesh *mesh1, const MEDCouplingMesh *mesh2) throw(INTERP_KERNEL::Exception)
488 {
489   if(!mesh1)
490     throw INTERP_KERNEL::Exception("MEDCouplingMesh::MergeMeshes : first parameter is an empty mesh !");
491   if(!mesh2)
492     throw INTERP_KERNEL::Exception("MEDCouplingMesh::MergeMeshes : second parameter is an empty mesh !");
493   return mesh1->mergeMyselfWith(mesh2);
494 }
495
496 /*!
497  * Creates a new MEDCouplingMesh by concatenating all given meshes, if possible.
498  * Cells and nodes of
499  * the *i*-th mesh precede cells and nodes of the (*i*+1)-th mesh within the result mesh.
500  * This method performs a systematic conversion to unstructured meshes before
501  * performing aggregation contrary to the other MergeMeshes()
502  * with two parameters that works only on the same type of meshes. So here it is possible
503  * to mix different type of meshes. 
504  *  \param [in] meshes - a vector of meshes to concatenate.
505  *  \return MEDCouplingMesh * - the result mesh. It is a new instance of
506  *          MEDCouplingUMesh. The caller is to delete this mesh using decrRef() as it
507  *          is no more needed.
508  *  \throw If \a meshes.size() == 0.
509  *  \throw If \a size[ *i* ] == NULL.
510  *  \throw If the coordinates is not set in none of the meshes.
511  *  \throw If \a meshes[ *i* ]->getMeshDimension() < 0.
512  *  \throw If the \a meshes are of different dimension (getMeshDimension()).
513  */
514 MEDCouplingMesh *MEDCouplingMesh::MergeMeshes(std::vector<const MEDCouplingMesh *>& meshes) throw(INTERP_KERNEL::Exception)
515 {
516   std::vector< MEDCouplingAutoRefCountObjectPtr<MEDCouplingUMesh> > ms1(meshes.size());
517   std::vector< const MEDCouplingUMesh * > ms2(meshes.size());
518   for(std::size_t i=0;i<meshes.size();i++)
519     {
520       if(meshes[i])
521         {
522           MEDCouplingUMesh *cur=meshes[i]->buildUnstructured();
523           ms1[i]=cur;  ms2[i]=cur;
524         }
525       else
526         {
527           std::ostringstream oss; oss << "MEDCouplingMesh::MergeMeshes(std::vector<const MEDCouplingMesh *>& meshes) : mesh at pos #" << i << " of input vector of size " << meshes.size() << " is empty !";
528           throw INTERP_KERNEL::Exception(oss.str().c_str());
529         }
530     }
531   return MEDCouplingUMesh::MergeUMeshes(ms2);
532 }
533
534 /*!
535  * For example if \a type is INTERP_KERNEL::NORM_TRI3 , INTERP_KERNEL::NORM_POLYGON is returned.
536  * If \a type is INTERP_KERNEL::NORM_HEXA8 , INTERP_KERNEL::NORM_POLYHED is returned.
537  * 
538  * \param [in] type the geometric type for which the corresponding dynamic type, is asked.
539  * \return the corresponding dynamic type, able to store the input \a type.
540  * 
541  * \throw if type is equal to \c INTERP_KERNEL::NORM_ERROR or to an unexisting geometric type.
542  */
543 INTERP_KERNEL::NormalizedCellType MEDCouplingMesh::GetCorrespondingPolyType(INTERP_KERNEL::NormalizedCellType type) throw(INTERP_KERNEL::Exception)
544 {
545   const INTERP_KERNEL::CellModel& cm=INTERP_KERNEL::CellModel::GetCellModel(type);
546   return cm.getCorrespondingPolyType();
547 }
548
549 /*!
550  * \param [in] type the geometric type for which the number of nodes consituting it, is asked.
551  * \return number of nodes consituting the input geometric type \a type.
552  * 
553  * \throw if type is dynamic as \c INTERP_KERNEL::NORM_POLYHED , \c INTERP_KERNEL::NORM_POLYGON , \c INTERP_KERNEL::NORM_QPOLYG
554  * \throw if type is equal to \c INTERP_KERNEL::NORM_ERROR or to an unexisting geometric type.
555  */
556 int MEDCouplingMesh::GetNumberOfNodesOfGeometricType(INTERP_KERNEL::NormalizedCellType type) throw(INTERP_KERNEL::Exception)
557 {
558   const INTERP_KERNEL::CellModel& cm=INTERP_KERNEL::CellModel::GetCellModel(type);
559   if(cm.isDynamic())
560     throw INTERP_KERNEL::Exception("MEDCouplingMesh::GetNumberOfNodesOfGeometricType : the input geometric type is dynamic ! Impossible to return a fixed number of nodes constituting it !");
561   return (int) cm.getNumberOfNodes();
562 }
563
564 /*!
565  * \param [in] type the geometric type for which the status static/dynamic is asked.
566  * \return true for static geometric type, false for dynamic geometric type.
567  * 
568  * \throw if type is equal to \c INTERP_KERNEL::NORM_ERROR or to an unexisting geometric type.
569  */
570 bool MEDCouplingMesh::IsStaticGeometricType(INTERP_KERNEL::NormalizedCellType type) throw(INTERP_KERNEL::Exception)
571 {
572   const INTERP_KERNEL::CellModel& cm=INTERP_KERNEL::CellModel::GetCellModel(type);
573   return !cm.isDynamic();
574 }
575
576 bool MEDCouplingMesh::IsLinearGeometricType(INTERP_KERNEL::NormalizedCellType type) throw(INTERP_KERNEL::Exception)
577 {
578   const INTERP_KERNEL::CellModel& cm=INTERP_KERNEL::CellModel::GetCellModel(type);
579   return !cm.isQuadratic();
580 }
581
582 /*!
583  * \param [in] type the geometric type for which the dimension is asked.
584  * \return the dimension associated to the input geometric type \a type.
585  * 
586  * \throw if type is equal to \c INTERP_KERNEL::NORM_ERROR or to an unexisting geometric type.
587  */
588 int MEDCouplingMesh::GetDimensionOfGeometricType(INTERP_KERNEL::NormalizedCellType type) throw(INTERP_KERNEL::Exception)
589 {
590   const INTERP_KERNEL::CellModel& cm=INTERP_KERNEL::CellModel::GetCellModel(type);
591   return (int) cm.getDimension();
592 }
593
594 /*!
595  * \param [in] type the geometric type for which the representation is asked.
596  * \return the string representation corresponding to the input geometric type \a type.
597  * 
598  * \throw if type is equal to \c INTERP_KERNEL::NORM_ERROR or to an unexisting geometric type.
599  */
600 const char *MEDCouplingMesh::GetReprOfGeometricType(INTERP_KERNEL::NormalizedCellType type) throw(INTERP_KERNEL::Exception)
601 {
602   const INTERP_KERNEL::CellModel& cm=INTERP_KERNEL::CellModel::GetCellModel(type);
603   return cm.getRepr();
604 }
605
606 /*!
607  * Finds cells in contact with a ball (i.e. a point with precision).
608  * \warning This method is suitable if the caller intends to evaluate only one
609  *          point, for more points getCellsContainingPoints() is recommended as it is
610  *          faster. 
611  *  \param [in] pos - array of coordinates of the ball central point.
612  *  \param [in] eps - ball radius.
613  *  \param [in,out] elts - vector returning ids of the found cells. It is cleared
614  *         before inserting ids.
615  *
616  *  \ref cpp_mcumesh_getCellsContainingPoint "Here is a C++ example".<br>
617  *  \ref  py_mcumesh_getCellsContainingPoint "Here is a Python example".
618  */
619 void MEDCouplingMesh::getCellsContainingPoint(const double *pos, double eps, std::vector<int>& elts) const
620 {
621   int ret=getCellContainingPoint(pos,eps);
622   elts.push_back(ret);
623 }
624
625 /*!
626  * Finds cells in contact with several balls (i.e. points with precision).
627  * This method is an extension of getCellContainingPoint() and
628  * getCellsContainingPoint() for the case of multiple points.
629  *  \param [in] pos - an array of coordinates of points in full interlace mode :
630  *         X0,Y0,Z0,X1,Y1,Z1,... Size of the array must be \a
631  *         this->getSpaceDimension() * \a nbOfPoints 
632  *  \param [in] nbOfPoints - number of points to locate within \a this mesh.
633  *  \param [in] eps - radius of balls (i.e. the precision).
634  *  \param [out] elts - vector returning ids of found cells.
635  *  \param [out] eltsIndex - an array, of length \a nbOfPoints + 1,
636  *         dividing cell ids in \a elts into groups each referring to one
637  *         point. Its every element (except the last one) is an index pointing to the
638  *         first id of a group of cells. For example cells in contact with the *i*-th
639  *         point are described by following range of indices:
640  *         [ \a eltsIndex[ *i* ], \a eltsIndex[ *i*+1 ] ) and the cell ids are
641  *         \a elts[ \a eltsIndex[ *i* ]], \a elts[ \a eltsIndex[ *i* ] + 1 ], ...
642  *         Number of cells in contact with the *i*-th point is
643  *         \a eltsIndex[ *i*+1 ] - \a eltsIndex[ *i* ].
644  *
645  *  \ref cpp_mcumesh_getCellsContainingPoints "Here is a C++ example".<br>
646  *  \ref  py_mcumesh_getCellsContainingPoints "Here is a Python example".
647  */
648 void MEDCouplingMesh::getCellsContainingPoints(const double *pos, int nbOfPoints, double eps, MEDCouplingAutoRefCountObjectPtr<DataArrayInt>& elts, MEDCouplingAutoRefCountObjectPtr<DataArrayInt>& eltsIndex) const
649 {
650   eltsIndex=DataArrayInt::New(); elts=DataArrayInt::New(); eltsIndex->alloc(nbOfPoints+1,1); eltsIndex->setIJ(0,0,0); elts->alloc(0,1);
651   int *eltsIndexPtr(eltsIndex->getPointer());
652   int spaceDim(getSpaceDimension());
653   const double *work(pos);
654   for(int i=0;i<nbOfPoints;i++,work+=spaceDim)
655     {
656       int ret=getCellContainingPoint(work,eps);
657       if(ret>=0)
658         {
659           elts->pushBackSilent(ret);
660           eltsIndexPtr[i+1]=eltsIndexPtr[i]+1;
661         }
662       else
663         eltsIndexPtr[i+1]=eltsIndexPtr[i];
664     }
665 }
666
667 /*!
668  * Writes \a this mesh into a VTK format file named as specified.
669  *  \param [in] fileName - the name of the file to write in.
670  *  \throw If \a fileName is not a writable file.
671  */
672 void MEDCouplingMesh::writeVTK(const char *fileName, bool isBinary) const throw(INTERP_KERNEL::Exception)
673 {
674   std::string cda,pda;
675   MEDCouplingAutoRefCountObjectPtr<DataArrayByte> byteArr;
676   if(isBinary)
677     { byteArr=DataArrayByte::New(); byteArr->alloc(0,1); }
678   writeVTKAdvanced(fileName,cda,pda,byteArr);
679 }
680
681 void MEDCouplingMesh::writeVTKAdvanced(const char *fileName, const std::string& cda, const std::string& pda, DataArrayByte *byteData) const throw(INTERP_KERNEL::Exception)
682 {
683   std::ofstream ofs(fileName);
684   ofs << "<VTKFile type=\""  << getVTKDataSetType() << "\" version=\"0.1\" byte_order=\"" << MEDCouplingByteOrderStr() << "\">\n";
685   writeVTKLL(ofs,cda,pda,byteData);
686   if(byteData)
687     {
688       ofs << "<AppendedData encoding=\"raw\">\n_1234";
689       ofs << std::flush; ofs.close();
690       std::ofstream ofs2(fileName,std::ios_base::binary | std::ios_base::app);
691       ofs2.write(byteData->begin(),byteData->getNbOfElems()); ofs2 << std::flush; ofs2.close();
692       std::ofstream ofs3(fileName,std::ios_base::app); ofs3 << "\n</AppendedData>\n</VTKFile>\n"; ofs3.close();
693     }
694   else
695     {
696       ofs << "</VTKFile>\n";
697       ofs.close();
698     }
699 }