1 /*---------------------------------------------------------------------------*\
3 \\ / F ield | OpenFOAM: The Open Source CFD Toolbox
5 \\ / A nd | Copyright (C) 2011 OpenFOAM Foundation
7 -------------------------------------------------------------------------------
9 This file is part of OpenFOAM.
11 OpenFOAM is free software: you can redistribute it and/or modify it
12 under the terms of the GNU General Public License as published by
13 the Free Software Foundation, either version 3 of the License, or
14 (at your option) any later version.
16 OpenFOAM is distributed in the hope that it will be useful, but WITHOUT
17 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
18 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
21 You should have received a copy of the GNU General Public License
22 along with OpenFOAM. If not, see <http://www.gnu.org/licenses/>.
24 \*---------------------------------------------------------------------------*/
26 #include "NASsurfaceFormat.H"
28 #include "IStringStream.H"
30 // * * * * * * * * * * * * * * * * Constructors * * * * * * * * * * * * * * //
33 Foam::fileFormats::NASsurfaceFormat<Face>::NASsurfaceFormat
35 const fileName& filename
42 // * * * * * * * * * * * * * * * Member Functions * * * * * * * * * * * * * //
45 bool Foam::fileFormats::NASsurfaceFormat<Face>::read
47 const fileName& filename
50 const bool mustTriangulate = this->isTri();
53 IFstream is(filename);
58 "fileFormats::NASsurfaceFormat::read(const fileName&)"
60 << "Cannot read file " << filename
64 // Nastran index of points
65 DynamicList<label> pointId;
66 DynamicList<point> dynPoints;
67 DynamicList<Face> dynFaces;
68 DynamicList<label> dynZones;
69 DynamicList<label> dynSizes;
72 // assume the types are not intermixed
73 // leave faces that didn't have a group in 0
77 // Name for face group
80 // Ansa tags. Denoted by $ANSA_NAME.
81 // These will appear just before the first use of a type.
82 // We read them and store the PSHELL types which are used to name
85 word ansaType, ansaName;
87 // A single warning per unrecognized command
88 HashSet<word> unhandledCmd;
96 if (line.substr(0, 10) == "$ANSA_NAME")
98 string::size_type sem0 = line.find (';', 0);
99 string::size_type sem1 = line.find (';', sem0+1);
100 string::size_type sem2 = line.find (';', sem1+1);
105 && sem1 != string::npos
106 && sem2 != string::npos
111 IStringStream(line.substr(sem0+1, sem1-sem0-1))()
113 ansaType = line.substr(sem1+1, sem2-sem1-1);
117 if (rawName[rawName.size()-1] == '\r')
119 rawName = rawName.substr(1, rawName.size()-2);
123 rawName = rawName.substr(1, rawName.size()-1);
126 string::stripInvalid<word>(rawName);
129 // Info<< "ANSA tag for NastranID:" << ansaId
130 // << " of type " << ansaType
131 // << " name " << ansaName << endl;
136 // Hypermesh extension
137 // $HMNAME COMP 1"partName"
140 line.substr(0, 12) == "$HMNAME COMP"
141 && line.find ('"') != string::npos
144 label groupId = readLabel
146 IStringStream(line.substr(16, 16))()
149 IStringStream lineStream(line.substr(32));
152 lineStream >> rawName;
153 string::stripInvalid<word>(rawName);
155 word groupName(rawName);
156 nameLookup.insert(groupId, groupName);
158 // Info<< "group " << groupId << " => " << groupName << endl;
162 // Skip empty or comment
163 if (line.empty() || line[0] == '$')
168 // Check if character 72 is continuation
169 if (line.size() > 72 && line[72] == '+')
171 line = line.substr(0, 72);
178 if (buf.size() > 72 && buf[72] == '+')
180 line += buf.substr(8, 64);
184 line += buf.substr(8, buf.size()-8);
192 IStringStream lineStream(line);
200 label groupId = readLabel(IStringStream(line.substr(16,8))());
201 fTri[0] = readLabel(IStringStream(line.substr(24,8))());
202 fTri[1] = readLabel(IStringStream(line.substr(32,8))());
203 fTri[2] = readLabel(IStringStream(line.substr(40,8))());
205 // Convert groupID into zoneId
206 Map<label>::const_iterator fnd = lookup.find(groupId);
207 if (fnd != lookup.end())
211 // pshell types are intermixed
218 zoneI = dynSizes.size();
219 lookup.insert(groupId, zoneI);
221 // Info<< "zone" << zoneI << " => group " << groupId <<endl;
224 dynFaces.append(fTri);
225 dynZones.append(zoneI);
228 else if (cmd == "CQUAD4")
231 labelUList& f = static_cast<labelUList&>(fQuad);
233 label groupId = readLabel(IStringStream(line.substr(16,8))());
234 fQuad[0] = readLabel(IStringStream(line.substr(24,8))());
235 fQuad[1] = readLabel(IStringStream(line.substr(32,8))());
236 fQuad[2] = readLabel(IStringStream(line.substr(40,8))());
237 fQuad[3] = readLabel(IStringStream(line.substr(48,8))());
239 // Convert groupID into zoneId
240 Map<label>::const_iterator fnd = lookup.find(groupId);
241 if (fnd != lookup.end())
245 // pshell types are intermixed
252 zoneI = dynSizes.size();
253 lookup.insert(groupId, zoneI);
255 // Info<< "zone" << zoneI << " => group " << groupId <<endl;
261 dynFaces.append(triFace(f[0], f[1], f[2]));
262 dynFaces.append(triFace(f[0], f[2], f[3]));
263 dynZones.append(zoneI);
264 dynZones.append(zoneI);
265 dynSizes[zoneI] += 2;
269 dynFaces.append(Face(f));
270 dynZones.append(zoneI);
274 else if (cmd == "GRID")
276 label index = readLabel(IStringStream(line.substr(8,8))());
277 scalar x = parseNASCoord(line.substr(24, 8));
278 scalar y = parseNASCoord(line.substr(32, 8));
279 scalar z = parseNASCoord(line.substr(40, 8));
281 pointId.append(index);
282 dynPoints.append(point(x, y, z));
284 else if (cmd == "GRID*")
286 // Long format is on two lines with '*' continuation symbol
287 // on start of second line.
288 // Typical line (spaces compacted)
289 // GRID* 126 0 -5.55999875E+02 -5.68730474E+02
292 label index = readLabel(IStringStream(line.substr(8,16))());
293 scalar x = parseNASCoord(line.substr(40, 16));
294 scalar y = parseNASCoord(line.substr(56, 16));
301 "fileFormats::NASsurfaceFormat::read(const fileName&)"
303 << "Expected continuation symbol '*' when reading GRID*"
304 << " (double precision coordinate) format" << nl
305 << "Read:" << line << nl
306 << "File:" << is.name() << " line:" << is.lineNumber()
309 scalar z = parseNASCoord(line.substr(8, 16));
311 pointId.append(index);
312 dynPoints.append(point(x, y, z));
314 else if (cmd == "PSHELL")
316 // pshell type for zone names with the Ansa extension
317 label groupId = readLabel(IStringStream(line.substr(8,8))());
319 if (groupId == ansaId && ansaType == "PSHELL")
321 nameLookup.insert(ansaId, ansaName);
322 // Info<< "group " << groupId << " => " << ansaName << endl;
325 else if (unhandledCmd.insert(cmd))
327 Info<< "Unhandled Nastran command " << line << nl
328 << "File:" << is.name() << " line:" << is.lineNumber()
333 // Info<< "Read faces:" << dynFaces.size()
334 // << " points:" << dynPoints.size()
337 // transfer to normal lists
338 this->storedPoints().transfer(dynPoints);
343 // Build inverse mapping (NASTRAN pointId -> index)
344 Map<label> mapPointId(2*pointId.size());
347 mapPointId.insert(pointId[i], i);
354 Face& f = dynFaces[i];
357 f[fp] = mapPointId[f[fp]];
360 pointId.clearStorage();
364 // create default zone names, or from ANSA/Hypermesh information
365 List<word> names(dynSizes.size());
366 forAllConstIter(Map<label>, lookup, iter)
368 const label zoneI = iter();
369 const label groupI = iter.key();
371 Map<word>::const_iterator fnd = nameLookup.find(groupI);
372 if (fnd != nameLookup.end())
374 names[zoneI] = fnd();
378 names[zoneI] = word("zone") + ::Foam::name(zoneI);
382 this->sortFacesAndStore(dynFaces.xfer(), dynZones.xfer(), sorted);
384 // add zones, culling empty ones
385 this->addZones(dynSizes, names, true);
391 // ************************************************************************* //