TTK
Loading...
Searching...
No Matches
ttkUtils.cpp
Go to the documentation of this file.
1#include <ttkUtils.h>
2
3#include <vtkAbstractArray.h>
4#include <vtkCellArray.h>
5#include <vtkDoubleArray.h>
6#include <vtkFieldData.h>
7#include <vtkIdTypeArray.h>
8#include <vtkNew.h>
9#include <vtkPoints.h>
10#include <vtkPolyData.h>
11#include <vtkSmartPointer.h>
12#include <vtkStringArray.h>
13#include <vtkUnstructuredGrid.h>
14
15#ifdef TTK_ENABLE_MPI_TIME
16#include <mpi.h>
17#endif
18int ttkUtils::replaceVariable(const std::string &iString,
19 vtkFieldData *fieldData,
20 std::string &oString,
21 std::string &errorMsg) {
22 std::string varName = iString;
23 int varIndex = -1;
24 bool varIndexDefined = false;
25
26 // Check if varIndex is specified
27 size_t indexDelimiter0 = iString.find('[');
28 size_t indexDelimiter1 = iString.find(']');
29 if(indexDelimiter0 != std::string::npos
30 && indexDelimiter1 != std::string::npos) {
31 if(indexDelimiter0 > indexDelimiter1
32 || iString.find('[', indexDelimiter0 + 1) != std::string::npos
33 || iString.find('}', indexDelimiter1 + 1) != std::string::npos) {
34 errorMsg = "Invalid Syntax:\n" + iString;
35 return 0;
36 }
37
38 varName = iString.substr(0, indexDelimiter0);
39 varIndex = stoi(iString.substr(
40 indexDelimiter0 + 1, indexDelimiter1 - indexDelimiter0 - 1));
41 varIndexDefined = true;
42 }
43
44 // Search in candidates
45 auto column = fieldData->GetAbstractArray(varName.data());
46 if(column == nullptr) {
47 errorMsg = "FieldData does not contain array '" + varName + "'";
48 return 0;
49 }
50
51 size_t n = column->GetNumberOfTuples();
52 size_t m = column->GetNumberOfComponents();
53 int s = n * m;
54
55 if(!varIndexDefined) {
56 if(s > 0) {
57 oString = column->GetVariantValue(0).ToString();
58 for(int i = 1; i < s; i++)
59 oString += "," + column->GetVariantValue(i).ToString();
60 }
61 } else {
62 if(varIndex < 0 || varIndex >= s) {
63 errorMsg = "Index " + std::to_string(varIndex) + "/" + std::to_string(s)
64 + " for FieldData Array '" + varName + "' out of range";
65 return 0;
66 }
67 oString = column->GetVariantValue(varIndex).ToString();
68 }
69
70 return 1;
71}
72
73int ttkUtils::replaceVariables(const std::string &iString,
74 vtkFieldData *fieldData,
75 std::string &oString,
76 std::string &errorMsg) {
77 oString = iString;
78
79 while(oString.find('{') != std::string::npos
80 && oString.find('}') != std::string::npos) {
81 size_t o = oString.find('{');
82 size_t c = oString.find('}');
83 // {...{....{...}...}..}
84 // | |
85 // o c
86
87 size_t oNext = oString.find('{', o + 1);
88 while(oNext != std::string::npos && oNext < c) {
89 o = oNext;
90 oNext = oString.find('{', o + 1);
91 }
92
93 // {...{....{var}...}..}
94 // | |
95 // o c
96 std::string var = oString.substr(o + 1, c - 1 - o);
97
98 std::string rVar;
99 if(!replaceVariable(var, fieldData, rVar, errorMsg))
100 return 0;
101
102 oString = oString.substr(0, o).append(rVar).append(
103 oString.substr(c + 1, oString.length() - c - 1));
104 }
105
106 if(oString.find('{') != std::string::npos
107 || oString.find('}') != std::string::npos) {
108 errorMsg = "Invalid Syntax:\n" + iString;
109 return 0;
110 }
111
112 return 1;
113}
114
115int ttkUtils::stringListToVector(const std::string &iString,
116 std::vector<std::string> &v) {
117 // ...,...,....,
118 // | |
119 // i j
120 size_t i = 0;
121 size_t j = iString.find(',');
122 while(j != std::string::npos) {
123 v.push_back(iString.substr(i, j - i));
124 i = j + 1;
125 j = iString.find(',', i);
126 }
127 if(iString.length() > i)
128 v.push_back(iString.substr(i, iString.length() - i));
129
130 return 1;
131}
132
133int ttkUtils::stringListToDoubleVector(const std::string &iString,
134 std::vector<double> &v) {
135 std::vector<std::string> stringVector;
136 if(!ttkUtils::stringListToVector(iString, stringVector))
137 return 0;
138
139 size_t n = stringVector.size();
140 v.resize(n);
141 // try {
142 for(size_t i = 0; i < n; i++)
143 v[i] = stod(stringVector[i]);
144 // } catch(std::invalid_argument &e) {
145 // return 0;
146 // }
147
148 return 1;
149}
150
152 ttkUtils::csvToVtkArray(const std::string &line) {
153 size_t firstComma = line.find(',', 0);
154
155 if(firstComma == std::string::npos)
156 return nullptr;
157
158 std::string arrayName = line.substr(0, firstComma);
159
160 std::vector<std::string> valuesAsString;
162 line.substr(firstComma + 1, std::string::npos), valuesAsString);
163 size_t nValues = valuesAsString.size();
164 if(nValues < 1)
165 return nullptr;
166
167 // Check if all elements are numbers
168 bool isNumeric = true;
169
170 std::vector<double> valuesAsDouble(nValues);
171 try {
172 for(size_t i = 0; i < nValues; i++)
173 valuesAsDouble[i] = std::stod(valuesAsString[i]);
174 } catch(const std::invalid_argument &) {
175 isNumeric = false;
176 } catch(const std::out_of_range &) {
177 isNumeric = false;
178 }
179
180 if(isNumeric) {
182 array->SetName(arrayName.data());
183 array->SetNumberOfValues(nValues);
184 for(size_t i = 0; i < nValues; i++)
185 array->SetValue(i, valuesAsDouble[i]);
186 return array;
187 } else {
189 array->SetName(arrayName.data());
190 array->SetNumberOfValues(nValues);
191 for(size_t i = 0; i < nValues; i++)
192 array->SetValue(i, valuesAsString[i]);
193 return array;
194 }
195}
196
198 ttkUtils::csvToDoubleArray(const std::string &line) {
199 size_t firstComma = line.find(',', 0);
200
201 if(firstComma == std::string::npos)
202 return nullptr;
203
204 std::string arrayName = line.substr(0, firstComma);
205 std::string valuesAsString = line.substr(firstComma + 1, std::string::npos);
206
207 std::vector<double> values;
208 ttkUtils::stringListToDoubleVector(valuesAsString, values);
209 size_t n = values.size();
210
212 array->SetName(arrayName.data());
213 array->SetNumberOfComponents(1);
214 array->SetNumberOfTuples(n);
215 auto arrayData = reinterpret_cast<double *>(GetVoidPointer(array));
216 for(size_t i = 0; i < n; i++)
217 arrayData[i] = values[i];
218
219 return array;
220}
221
225void *ttkUtils::GetVoidPointer(vtkDataArray *array, vtkIdType start) {
226 void *outPtr = nullptr;
227 if(array == nullptr)
228 return outPtr;
229
230 switch(array->GetDataType()) {
231 vtkTemplateMacro(
232 auto *aosArray = vtkAOSDataArrayTemplate<VTK_TT>::FastDownCast(array);
233 if(aosArray) { outPtr = aosArray->GetVoidPointer(start); });
234 }
235 return outPtr;
236}
237
238void *ttkUtils::GetVoidPointer(vtkPoints *points, vtkIdType start) {
239 return GetVoidPointer(points->GetData(), start);
240}
241
243 vtkIdType idx) {
244 auto slicedArray
245 = vtkSmartPointer<vtkAbstractArray>::Take(array->NewInstance());
246 slicedArray->SetName(array->GetName());
247 slicedArray->SetNumberOfComponents(array->GetNumberOfComponents());
248 slicedArray->SetNumberOfTuples(1);
249 slicedArray->SetTuple(0, idx, array);
250 return slicedArray;
251}
252
253void *ttkUtils::WriteVoidPointer(vtkDataArray *array,
254 vtkIdType valueIdx,
255 vtkIdType numValues) {
256 void *outPtr = nullptr;
257 switch(array->GetDataType()) {
258 vtkTemplateMacro(auto *aosArray
259 = vtkAOSDataArrayTemplate<VTK_TT>::FastDownCast(array);
260 if(aosArray) {
261 outPtr = aosArray->WriteVoidPointer(valueIdx, numValues);
262 });
263 }
264 return outPtr;
265}
266
267void *ttkUtils::WritePointer(vtkDataArray *array,
268 vtkIdType valueIdx,
269 vtkIdType numValues) {
270 void *outPtr = nullptr;
271 switch(array->GetDataType()) {
272 vtkTemplateMacro(
273 auto *aosArray = vtkAOSDataArrayTemplate<VTK_TT>::FastDownCast(array);
274 if(aosArray) { outPtr = aosArray->WritePointer(valueIdx, numValues); });
275 }
276 return outPtr;
277}
278
279void ttkUtils::SetVoidArray(vtkDataArray *array,
280 void *data,
281 vtkIdType size,
282 int save) {
283 switch(array->GetDataType()) {
284 vtkTemplateMacro(
285 auto *aosArray = vtkAOSDataArrayTemplate<VTK_TT>::FastDownCast(array);
286 if(aosArray) { aosArray->SetVoidArray(data, size, save); } else {
287 std::cerr << "SetVoidArray on incompatible vtkDataArray:" << endl;
288 array->Print(std::cerr);
289 });
290 }
291}
292
293[[deprecated]] void ttkUtils::FillCellArrayFromSingle(vtkIdType const *cells,
294 vtkIdType ncells,
295 vtkCellArray *cellArray) {
296 size_t curPos = 0;
297 vtkNew<vtkIdList> verts;
298 for(vtkIdType cid = 0; cid < ncells; cid++) {
299 const vtkIdType nbVerts = cells[curPos];
300 verts->SetNumberOfIds(nbVerts);
301 curPos++;
302 for(vtkIdType v = 0; v < nbVerts; v++) {
303 verts->SetId(v, cells[curPos]);
304 curPos++;
305 }
306 cellArray->InsertNextCell(verts);
307 }
308}
309
310void ttkUtils::FillCellArrayFromDual(vtkIdType const *cells_co,
311 vtkIdType const *cells_off,
312 vtkIdType ncells,
313 vtkCellArray *cellArray) {
314 size_t curPos = 0;
315 vtkNew<vtkIdList> verts;
316 for(vtkIdType cid = 0; cid < ncells; cid++) {
317 const vtkIdType nbVerts = cells_off[cid + 1] - cells_off[cid];
318 verts->SetNumberOfIds(nbVerts);
319 for(vtkIdType v = 0; v < nbVerts; v++) {
320 verts->SetId(v, cells_co[curPos]);
321 curPos++;
322 }
323 cellArray->InsertNextCell(verts);
324 }
325}
326
327int ttkUtils::CellVertexFromPoints(vtkDataSet *const dataSet,
328 vtkPoints *const points) {
329
330 if(dataSet == nullptr || points == nullptr) {
331 return 0;
332 }
333
334 if(!dataSet->IsA("vtkUnstructuredGrid") && !dataSet->IsA("vtkPolyData")) {
335 return 0;
336 }
337
338 const size_t nPoints = points->GetNumberOfPoints();
339 if(nPoints == 0) {
340 return 0;
341 }
342
343 vtkNew<vtkCellArray> cells{};
344 cells->InsertNextCell(nPoints);
345 for(size_t i = 0; i < nPoints; ++i) {
346 cells->InsertCellPoint(i);
347 }
348
349 if(dataSet->IsA("vtkUnstructuredGrid")) {
350 const auto vtu{vtkUnstructuredGrid::SafeDownCast(dataSet)};
351 if(vtu != nullptr) {
352 vtu->SetPoints(points);
353 vtu->SetCells(VTK_POLY_VERTEX, cells);
354 }
355 } else if(dataSet->IsA("vtkPolyData")) {
356 const auto vtp{vtkPolyData::SafeDownCast(dataSet)};
357 if(vtp != nullptr) {
358 vtp->SetPoints(points);
359 vtp->SetVerts(cells);
360 }
361 }
362
363 return 1;
364}
static int replaceVariables(const std::string &iString, vtkFieldData *fieldData, std::string &oString, std::string &errorMsg)
Definition: ttkUtils.cpp:73
static void * GetVoidPointer(vtkDataArray *array, vtkIdType start=0)
Definition: ttkUtils.cpp:225
static int stringListToDoubleVector(const std::string &iString, std::vector< double > &v)
Definition: ttkUtils.cpp:133
static vtkSmartPointer< vtkAbstractArray > SliceArray(vtkAbstractArray *array, vtkIdType idx)
Definition: ttkUtils.cpp:242
static void FillCellArrayFromSingle(vtkIdType const *cells, vtkIdType ncells, vtkCellArray *cellArray)
Definition: ttkUtils.cpp:293
static void * WritePointer(vtkDataArray *array, vtkIdType start, vtkIdType numValues)
Definition: ttkUtils.cpp:267
static int replaceVariable(const std::string &iString, vtkFieldData *fieldData, std::string &oString, std::string &errorMsg)
Definition: ttkUtils.cpp:18
static int CellVertexFromPoints(vtkDataSet *const dataSet, vtkPoints *const points)
Definition: ttkUtils.cpp:327
static vtkSmartPointer< vtkAbstractArray > csvToVtkArray(const std::string &line)
Definition: ttkUtils.cpp:152
static vtkSmartPointer< vtkDoubleArray > csvToDoubleArray(const std::string &line)
Definition: ttkUtils.cpp:198
static void FillCellArrayFromDual(vtkIdType const *cells_co, vtkIdType const *cells_off, vtkIdType ncells, vtkCellArray *cellArray)
Definition: ttkUtils.cpp:310
static void SetVoidArray(vtkDataArray *array, void *data, vtkIdType size, int save)
Definition: ttkUtils.cpp:279
static void * WriteVoidPointer(vtkDataArray *array, vtkIdType start, vtkIdType numValues)
Definition: ttkUtils.cpp:253
static int stringListToVector(const std::string &iString, std::vector< std::string > &v)
Definition: ttkUtils.cpp:115