Line data Source code
1 : /******************************************************************************
2 : *
3 : * Project: DXF Translator
4 : * Purpose: Implements OGRDXFLayer class.
5 : * Author: Frank Warmerdam, warmerdam@pobox.com
6 : *
7 : ******************************************************************************
8 : * Copyright (c) 2009, Frank Warmerdam <warmerdam@pobox.com>
9 : * Copyright (c) 2011-2013, Even Rouault <even dot rouault at spatialys.com>
10 : * Copyright (c) 2017-2020, Alan Thomas <alant@outlook.com.au>
11 : *
12 : * SPDX-License-Identifier: MIT
13 : ****************************************************************************/
14 :
15 : #include "ogr_dxf.h"
16 : #include "cpl_conv.h"
17 : #include "ogrdxf_polyline_smooth.h"
18 : #include "ogr_api.h"
19 :
20 : #include <cmath>
21 : #include <algorithm>
22 : #include <limits>
23 : #include <stdexcept>
24 : #include <memory>
25 :
26 : /************************************************************************/
27 : /* push() */
28 : /************************************************************************/
29 :
30 1736 : void OGRDXFFeatureQueue::push(OGRDXFFeature *poFeature)
31 : {
32 1736 : apoFeatures.push(poFeature);
33 1736 : }
34 :
35 : /************************************************************************/
36 : /* pop() */
37 : /************************************************************************/
38 :
39 1736 : void OGRDXFFeatureQueue::pop()
40 : {
41 1736 : CPLAssert(!apoFeatures.empty());
42 1736 : apoFeatures.pop();
43 1736 : }
44 :
45 : /************************************************************************/
46 : /* OGRDXFLayer() */
47 : /************************************************************************/
48 :
49 201 : OGRDXFLayer::OGRDXFLayer(OGRDXFDataSource *poDSIn)
50 201 : : poDS(poDSIn), poFeatureDefn(new OGRFeatureDefn("entities")), iNextFID(0)
51 : {
52 201 : poFeatureDefn->Reference();
53 :
54 201 : int nModes = ODFM_None;
55 201 : if (!poDS->InlineBlocks())
56 24 : nModes |= ODFM_IncludeBlockFields;
57 201 : if (poDS->ShouldIncludeRawCodeValues())
58 1 : nModes |= ODFM_IncludeRawCodeValues;
59 201 : if (poDS->In3DExtensibleMode())
60 1 : nModes |= ODFM_Include3DModeFields;
61 201 : OGRDXFDataSource::AddStandardFields(poFeatureDefn, nModes);
62 :
63 201 : SetDescription(poFeatureDefn->GetName());
64 201 : }
65 :
66 : /************************************************************************/
67 : /* ~OGRDXFLayer() */
68 : /************************************************************************/
69 :
70 382 : OGRDXFLayer::~OGRDXFLayer()
71 :
72 : {
73 201 : ClearPendingFeatures();
74 201 : if (m_nFeaturesRead > 0 && poFeatureDefn != nullptr)
75 : {
76 70 : CPLDebug("DXF", "%d features read on layer '%s'.", (int)m_nFeaturesRead,
77 70 : poFeatureDefn->GetName());
78 : }
79 :
80 201 : if (poFeatureDefn)
81 201 : poFeatureDefn->Release();
82 382 : }
83 :
84 : /************************************************************************/
85 : /* ClearPendingFeatures() */
86 : /************************************************************************/
87 :
88 435 : void OGRDXFLayer::ClearPendingFeatures()
89 :
90 : {
91 435 : while (!apoPendingFeatures.empty())
92 : {
93 55 : OGRDXFFeature *poFeature = apoPendingFeatures.front();
94 55 : apoPendingFeatures.pop();
95 55 : delete poFeature;
96 : }
97 380 : }
98 :
99 : /************************************************************************/
100 : /* ResetReading() */
101 : /************************************************************************/
102 :
103 179 : void OGRDXFLayer::ResetReading()
104 :
105 : {
106 179 : iNextFID = 0;
107 179 : ClearPendingFeatures();
108 179 : m_oInsertState.m_nRowCount = 0;
109 179 : m_oInsertState.m_nColumnCount = 0;
110 179 : poDS->RestartEntities();
111 179 : }
112 :
113 : /************************************************************************/
114 : /* TranslateGenericProperty() */
115 : /* */
116 : /* Try and convert entity properties handled similarly for most */
117 : /* or all entity types. */
118 : /************************************************************************/
119 :
120 6555 : void OGRDXFLayer::TranslateGenericProperty(OGRDXFFeature *poFeature, int nCode,
121 : char *pszValue)
122 :
123 : {
124 6555 : switch (nCode)
125 : {
126 698 : case 8:
127 698 : poFeature->SetField("Layer", TextRecode(pszValue));
128 698 : break;
129 :
130 1324 : case 100:
131 : {
132 2648 : CPLString osSubClass = poFeature->GetFieldAsString("SubClasses");
133 1324 : if (!osSubClass.empty())
134 717 : osSubClass += ":";
135 1324 : osSubClass += pszValue;
136 1324 : poFeature->SetField("SubClasses", osSubClass.c_str());
137 : }
138 1324 : break;
139 :
140 38 : case 101:
141 : // Embedded objects mark the end of meaningful DXF data
142 : // See
143 : // http://docs.autodesk.com/ACDMAC/2016/ENU/ObjectARX_Dev_Guide/files/GUID-C953866F-A335-4FFD-AE8C-256A76065552.htm
144 : {
145 : char szLineBuf[257];
146 : // Eat the rest of this entity
147 38 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) >
148 : 0)
149 : {
150 : }
151 :
152 2 : if (nCode < 0)
153 : {
154 : // Let the entity reader function discover this error for
155 : // itself
156 0 : return;
157 : }
158 :
159 2 : CPLAssert(nCode == 0);
160 2 : poDS->UnreadValue();
161 : }
162 2 : break;
163 :
164 5 : case 60:
165 5 : if (atoi(pszValue))
166 5 : poFeature->oStyleProperties["Hidden"] = "1";
167 5 : break;
168 :
169 12 : case 67:
170 12 : if (atoi(pszValue))
171 12 : poFeature->SetField("PaperSpace", 1);
172 12 : break;
173 :
174 318 : case 62:
175 318 : poFeature->oStyleProperties["Color"] = pszValue;
176 318 : break;
177 :
178 12 : case 420:
179 12 : poFeature->oStyleProperties["TrueColor"] = pszValue;
180 12 : break;
181 :
182 256 : case 6:
183 256 : poFeature->SetField("Linetype", TextRecode(pszValue));
184 256 : break;
185 :
186 7 : case 48:
187 7 : poFeature->oStyleProperties["LinetypeScale"] = pszValue;
188 7 : break;
189 :
190 202 : case 370:
191 : case 39:
192 202 : poFeature->oStyleProperties["LineWeight"] = pszValue;
193 202 : break;
194 :
195 689 : case 5:
196 689 : poFeature->SetField("EntityHandle", pszValue);
197 689 : break;
198 :
199 : // OCS vector.
200 145 : case 210:
201 145 : poFeature->oOCS.dfX = CPLAtof(pszValue);
202 145 : break;
203 :
204 145 : case 220:
205 145 : poFeature->oOCS.dfY = CPLAtof(pszValue);
206 145 : break;
207 :
208 145 : case 230:
209 145 : poFeature->oOCS.dfZ = CPLAtof(pszValue);
210 145 : break;
211 :
212 2595 : default:
213 2595 : if (poDS->ShouldIncludeRawCodeValues())
214 : {
215 : char **papszRawCodeValues =
216 479 : poFeature->GetFieldAsStringList("RawCodeValues");
217 :
218 479 : papszRawCodeValues = CSLDuplicate(papszRawCodeValues);
219 :
220 479 : papszRawCodeValues = CSLAddString(
221 : papszRawCodeValues,
222 479 : CPLString()
223 958 : .Printf("%d %s", nCode, TextRecode(pszValue).c_str())
224 : .c_str());
225 :
226 479 : poFeature->SetField("RawCodeValues", papszRawCodeValues);
227 :
228 479 : CSLDestroy(papszRawCodeValues);
229 : }
230 2595 : break;
231 : }
232 : }
233 :
234 : /************************************************************************/
235 : /* PrepareFeatureStyle() */
236 : /* */
237 : /* - poBlockFeature: If this is not NULL, style properties on */
238 : /* poFeature with ByBlock values will be replaced with the */
239 : /* corresponding property from poBlockFeature. If this */
240 : /* parameter is supplied it is assumed that poFeature is a */
241 : /* clone, not an "original" feature object. */
242 : /************************************************************************/
243 :
244 568 : void OGRDXFLayer::PrepareFeatureStyle(
245 : OGRDXFFeature *const poFeature,
246 : OGRDXFFeature *const poBlockFeature /* = NULL */)
247 :
248 : {
249 568 : const char *pszStyleString = poFeature->GetStyleString();
250 :
251 568 : if (pszStyleString && STARTS_WITH_CI(pszStyleString, "BRUSH("))
252 : {
253 51 : PrepareBrushStyle(poFeature, poBlockFeature);
254 : }
255 517 : else if (pszStyleString && STARTS_WITH_CI(pszStyleString, "LABEL("))
256 : {
257 : // Find the new color of this feature, and replace it into
258 : // the style string
259 128 : const CPLString osNewColor = poFeature->GetColor(poDS, poBlockFeature);
260 :
261 128 : CPLString osNewStyle = pszStyleString;
262 64 : const size_t nColorStartPos = osNewStyle.rfind(",c:");
263 64 : if (nColorStartPos != std::string::npos)
264 : {
265 : const size_t nColorEndPos =
266 64 : osNewStyle.find_first_of(",)", nColorStartPos + 3);
267 :
268 64 : if (nColorEndPos != std::string::npos)
269 : {
270 : osNewStyle.replace(nColorStartPos + 3,
271 64 : nColorEndPos - (nColorStartPos + 3),
272 64 : osNewColor);
273 64 : poFeature->SetStyleString(osNewStyle);
274 : }
275 64 : }
276 : }
277 : else
278 : {
279 453 : PrepareLineStyle(poFeature, poBlockFeature);
280 : }
281 568 : }
282 :
283 : /************************************************************************/
284 : /* PrepareBrushStyle() */
285 : /************************************************************************/
286 :
287 147 : void OGRDXFLayer::PrepareBrushStyle(
288 : OGRDXFFeature *const poFeature,
289 : OGRDXFFeature *const poBlockFeature /* = NULL */)
290 :
291 : {
292 294 : CPLString osStyle = "BRUSH(fc:";
293 147 : osStyle += poFeature->GetColor(poDS, poBlockFeature);
294 147 : osStyle += ")";
295 :
296 147 : poFeature->SetStyleString(osStyle);
297 147 : }
298 :
299 : /************************************************************************/
300 : /* PrepareLineStyle() */
301 : /************************************************************************/
302 :
303 930 : void OGRDXFLayer::PrepareLineStyle(
304 : OGRDXFFeature *const poFeature,
305 : OGRDXFFeature *const poBlockFeature /* = NULL */)
306 :
307 : {
308 1860 : const CPLString osLayer = poFeature->GetFieldAsString("Layer");
309 :
310 : /* -------------------------------------------------------------------- */
311 : /* Get line weight if available. */
312 : /* -------------------------------------------------------------------- */
313 930 : double dfWeight = 0.0;
314 1860 : CPLString osWeight = "-1";
315 :
316 930 : if (poFeature->oStyleProperties.count("LineWeight") > 0)
317 268 : osWeight = poFeature->oStyleProperties["LineWeight"];
318 :
319 : // Use ByBlock lineweight?
320 930 : if (CPLAtof(osWeight) == -2 && poBlockFeature)
321 : {
322 97 : if (poBlockFeature->oStyleProperties.count("LineWeight") > 0)
323 : {
324 : // Inherit lineweight from the owning block
325 14 : osWeight = poBlockFeature->oStyleProperties["LineWeight"];
326 :
327 : // Use the inherited lineweight if we regenerate the style
328 : // string again during block insertion
329 14 : poFeature->oStyleProperties["LineWeight"] = osWeight;
330 : }
331 : else
332 : {
333 : // If the owning block has no explicit lineweight,
334 : // assume ByLayer
335 83 : osWeight = "-1";
336 : }
337 : }
338 :
339 : // Use layer lineweight?
340 930 : if (CPLAtof(osWeight) == -1)
341 : {
342 881 : osWeight = poDS->LookupLayerProperty(osLayer, "LineWeight");
343 : }
344 :
345 : // Will be zero in the case of an invalid value
346 930 : dfWeight = CPLAtof(osWeight) / 100.0;
347 :
348 : /* -------------------------------------------------------------------- */
349 : /* Do we have a dash/dot line style? */
350 : /* -------------------------------------------------------------------- */
351 930 : const char *pszLinetype = poFeature->GetFieldAsString("Linetype");
352 :
353 : // Use ByBlock line style?
354 930 : if (pszLinetype && EQUAL(pszLinetype, "ByBlock") && poBlockFeature)
355 : {
356 27 : pszLinetype = poBlockFeature->GetFieldAsString("Linetype");
357 :
358 : // Use the inherited line style if we regenerate the style string
359 : // again during block insertion
360 27 : if (pszLinetype)
361 27 : poFeature->SetField("Linetype", pszLinetype);
362 : }
363 :
364 : // Use layer line style?
365 930 : if (pszLinetype && EQUAL(pszLinetype, ""))
366 : {
367 655 : pszLinetype = poDS->LookupLayerProperty(osLayer, "Linetype");
368 : }
369 :
370 1860 : const std::vector<double> oLineType = poDS->LookupLineType(pszLinetype);
371 :
372 : // Linetype scale is not inherited from the block feature
373 930 : double dfLineTypeScale = CPLAtof(poDS->GetVariable("$LTSCALE", "1.0"));
374 930 : if (poFeature->oStyleProperties.count("LinetypeScale") > 0)
375 16 : dfLineTypeScale *=
376 16 : CPLAtof(poFeature->oStyleProperties["LinetypeScale"]);
377 :
378 1860 : CPLString osPattern;
379 1020 : for (std::vector<double>::const_iterator oIt = oLineType.begin();
380 1020 : oIt != oLineType.end(); ++oIt)
381 : {
382 : // this is the format specifier %g followed by a literal 'g'
383 : osPattern +=
384 90 : CPLString().Printf("%.11gg ", fabs(*oIt) * dfLineTypeScale);
385 : }
386 :
387 930 : if (osPattern.length() > 0)
388 45 : osPattern.erase(osPattern.end() - 1);
389 :
390 : /* -------------------------------------------------------------------- */
391 : /* Format the style string. */
392 : /* -------------------------------------------------------------------- */
393 :
394 1860 : CPLString osStyle = "PEN(c:";
395 930 : osStyle += poFeature->GetColor(poDS, poBlockFeature);
396 :
397 930 : if (dfWeight > 0.0)
398 : {
399 : char szBuffer[64];
400 13 : CPLsnprintf(szBuffer, sizeof(szBuffer), "%.2g", dfWeight);
401 13 : osStyle += CPLString().Printf(",w:%sg", szBuffer);
402 : }
403 :
404 930 : if (osPattern != "")
405 : {
406 45 : osStyle += ",p:\"";
407 45 : osStyle += osPattern;
408 45 : osStyle += "\"";
409 : }
410 :
411 930 : osStyle += ")";
412 :
413 930 : poFeature->SetStyleString(osStyle);
414 930 : }
415 :
416 : /************************************************************************/
417 : /* TextRecode() */
418 : /************************************************************************/
419 :
420 1463 : CPLString OGRDXFLayer::TextRecode(const char *pszInput)
421 :
422 : {
423 2926 : return CPLString(pszInput).Recode(poDS->GetEncoding(), CPL_ENC_UTF8);
424 : }
425 :
426 : /************************************************************************/
427 : /* TextUnescape() */
428 : /* */
429 : /* Unexcape DXF style escape sequences such as \P for newline */
430 : /* and \~ for space, and do the recoding to UTF8. */
431 : /************************************************************************/
432 :
433 148 : CPLString OGRDXFLayer::TextUnescape(const char *pszInput, bool bIsMText)
434 :
435 : {
436 148 : if (poDS->ShouldTranslateEscapes())
437 147 : return ACTextUnescape(pszInput, poDS->GetEncoding(), bIsMText);
438 :
439 1 : return TextRecode(pszInput);
440 : }
441 :
442 : /************************************************************************/
443 : /* TranslateMTEXT() */
444 : /************************************************************************/
445 :
446 40 : OGRDXFFeature *OGRDXFLayer::TranslateMTEXT()
447 :
448 : {
449 : char szLineBuf[512];
450 40 : int nCode = 0;
451 80 : auto poFeature = std::make_unique<OGRDXFFeature>(poFeatureDefn);
452 40 : double dfX = 0.0;
453 40 : double dfY = 0.0;
454 40 : double dfZ = 0.0;
455 40 : double dfAngle = 0.0;
456 40 : double dfHeight = 0.0;
457 40 : double dfXDirection = 0.0;
458 40 : double dfYDirection = 0.0;
459 40 : bool bHaveZ = false;
460 40 : int nAttachmentPoint = -1;
461 80 : CPLString osText;
462 80 : CPLString osStyleName = "STANDARD";
463 :
464 802 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
465 : {
466 762 : switch (nCode)
467 : {
468 40 : case 10:
469 40 : dfX = CPLAtof(szLineBuf);
470 40 : break;
471 :
472 40 : case 20:
473 40 : dfY = CPLAtof(szLineBuf);
474 40 : break;
475 :
476 34 : case 30:
477 34 : dfZ = CPLAtof(szLineBuf);
478 34 : bHaveZ = true;
479 34 : break;
480 :
481 40 : case 40:
482 40 : dfHeight = CPLAtof(szLineBuf);
483 40 : break;
484 :
485 37 : case 71:
486 37 : nAttachmentPoint = atoi(szLineBuf);
487 37 : break;
488 :
489 0 : case 11:
490 0 : dfXDirection = CPLAtof(szLineBuf);
491 0 : break;
492 :
493 0 : case 21:
494 0 : dfYDirection = CPLAtof(szLineBuf);
495 0 : dfAngle = atan2(dfYDirection, dfXDirection) * 180.0 / M_PI;
496 0 : break;
497 :
498 42 : case 1:
499 : case 3:
500 42 : osText += TextUnescape(szLineBuf, true);
501 42 : break;
502 :
503 29 : case 50:
504 29 : dfAngle = CPLAtof(szLineBuf);
505 29 : break;
506 :
507 29 : case 7:
508 29 : osStyleName = TextRecode(szLineBuf);
509 29 : break;
510 :
511 471 : default:
512 471 : TranslateGenericProperty(poFeature.get(), nCode, szLineBuf);
513 471 : break;
514 : }
515 : }
516 40 : if (nCode < 0)
517 : {
518 0 : DXF_LAYER_READER_ERROR();
519 0 : return nullptr;
520 : }
521 :
522 40 : if (nCode == 0)
523 40 : poDS->UnreadValue();
524 :
525 40 : OGRPoint *poGeom = nullptr;
526 40 : if (bHaveZ)
527 34 : poGeom = new OGRPoint(dfX, dfY, dfZ);
528 : else
529 6 : poGeom = new OGRPoint(dfX, dfY);
530 :
531 : /* We do NOT apply the OCS for MTEXT. See
532 : * https://trac.osgeo.org/gdal/ticket/7049 */
533 : /* ApplyOCSTransformer( poGeom ); */
534 :
535 40 : poFeature->SetGeometryDirectly(poGeom);
536 :
537 : /* -------------------------------------------------------------------- */
538 : /* Apply text after stripping off any extra terminating newline. */
539 : /* -------------------------------------------------------------------- */
540 40 : if (!osText.empty() && osText.back() == '\n')
541 0 : osText.pop_back();
542 :
543 40 : poFeature->SetField("Text", osText);
544 :
545 : /* -------------------------------------------------------------------- */
546 : /* We need to escape double quotes with backslashes before they */
547 : /* can be inserted in the style string. */
548 : /* -------------------------------------------------------------------- */
549 40 : if (strchr(osText, '"') != nullptr)
550 : {
551 10 : std::string osEscaped;
552 :
553 230 : for (size_t iC = 0; iC < osText.size(); iC++)
554 : {
555 220 : if (osText[iC] == '"')
556 20 : osEscaped += "\\\"";
557 : else
558 200 : osEscaped += osText[iC];
559 : }
560 10 : osText = std::move(osEscaped);
561 : }
562 :
563 : /* -------------------------------------------------------------------- */
564 : /* Prepare style string. */
565 : /* -------------------------------------------------------------------- */
566 80 : CPLString osStyle;
567 : char szBuffer[64];
568 :
569 : // Font name
570 40 : osStyle.Printf("LABEL(f:\"");
571 :
572 : // Preserve legacy behavior of specifying "Arial" as a default font name.
573 40 : osStyle += poDS->LookupTextStyleProperty(osStyleName, "Font", "Arial");
574 :
575 40 : osStyle += "\"";
576 :
577 : // Bold, italic
578 40 : if (EQUAL(poDS->LookupTextStyleProperty(osStyleName, "Bold", "0"), "1"))
579 : {
580 4 : osStyle += ",bo:1";
581 : }
582 40 : if (EQUAL(poDS->LookupTextStyleProperty(osStyleName, "Italic", "0"), "1"))
583 : {
584 1 : osStyle += ",it:1";
585 : }
586 :
587 : // Text string itself
588 40 : osStyle += ",t:\"";
589 40 : osStyle += osText;
590 40 : osStyle += "\"";
591 :
592 40 : if (dfAngle != 0.0)
593 : {
594 19 : CPLsnprintf(szBuffer, sizeof(szBuffer), "%.3g", dfAngle);
595 19 : osStyle += CPLString().Printf(",a:%s", szBuffer);
596 : }
597 :
598 40 : if (dfHeight != 0.0)
599 : {
600 40 : CPLsnprintf(szBuffer, sizeof(szBuffer), "%.3g", dfHeight);
601 40 : osStyle += CPLString().Printf(",s:%sg", szBuffer);
602 : }
603 :
604 : const char *pszWidthFactor =
605 40 : poDS->LookupTextStyleProperty(osStyleName, "Width", "1");
606 40 : if (pszWidthFactor && CPLAtof(pszWidthFactor) != 1.0)
607 : {
608 4 : CPLsnprintf(szBuffer, sizeof(szBuffer), "%.4g",
609 4 : CPLAtof(pszWidthFactor) * 100.0);
610 4 : osStyle += CPLString().Printf(",w:%s", szBuffer);
611 : }
612 :
613 40 : if (nAttachmentPoint >= 0 && nAttachmentPoint <= 9)
614 : {
615 : const static int anAttachmentMap[10] = {-1, 7, 8, 9, 4, 5, 6, 1, 2, 3};
616 :
617 : osStyle +=
618 37 : CPLString().Printf(",p:%d", anAttachmentMap[nAttachmentPoint]);
619 : }
620 :
621 : // Color
622 40 : osStyle += ",c:";
623 40 : osStyle += poFeature->GetColor(poDS);
624 :
625 40 : osStyle += ")";
626 :
627 40 : poFeature->SetStyleString(osStyle);
628 :
629 40 : return poFeature.release();
630 : }
631 :
632 : /************************************************************************/
633 : /* TranslateTEXT() */
634 : /* */
635 : /* This function translates TEXT and ATTRIB entities, as well as */
636 : /* ATTDEF entities when we are not inlining blocks. */
637 : /************************************************************************/
638 :
639 63 : OGRDXFFeature *OGRDXFLayer::TranslateTEXT(const bool bIsAttribOrAttdef)
640 :
641 : {
642 : char szLineBuf[257];
643 63 : int nCode = 0;
644 126 : auto poFeature = std::make_unique<OGRDXFFeature>(poFeatureDefn);
645 :
646 63 : double dfX = 0.0;
647 63 : double dfY = 0.0;
648 63 : double dfZ = 0.0;
649 63 : bool bHaveZ = false;
650 :
651 63 : double dfAngle = 0.0;
652 63 : double dfHeight = 0.0;
653 63 : double dfWidthFactor = 1.0;
654 63 : bool bHasAlignmentPoint = false;
655 63 : double dfAlignmentPointX = 0.0;
656 63 : double dfAlignmentPointY = 0.0;
657 :
658 126 : CPLString osText;
659 126 : CPLString osStyleName = "STANDARD";
660 :
661 63 : int nAnchorPosition = 1;
662 63 : int nHorizontalAlignment = 0;
663 63 : int nVerticalAlignment = 0;
664 :
665 958 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
666 : {
667 895 : switch (nCode)
668 : {
669 62 : case 10:
670 62 : dfX = CPLAtof(szLineBuf);
671 62 : break;
672 :
673 62 : case 20:
674 62 : dfY = CPLAtof(szLineBuf);
675 62 : break;
676 :
677 25 : case 11:
678 25 : dfAlignmentPointX = CPLAtof(szLineBuf);
679 25 : break;
680 :
681 25 : case 21:
682 25 : dfAlignmentPointY = CPLAtof(szLineBuf);
683 25 : bHasAlignmentPoint = true;
684 25 : break;
685 :
686 62 : case 30:
687 62 : dfZ = CPLAtof(szLineBuf);
688 62 : bHaveZ = true;
689 62 : break;
690 :
691 62 : case 40:
692 62 : dfHeight = CPLAtof(szLineBuf);
693 62 : break;
694 :
695 12 : case 41:
696 12 : dfWidthFactor = CPLAtof(szLineBuf);
697 12 : break;
698 :
699 62 : case 1:
700 62 : osText += TextUnescape(szLineBuf, false);
701 62 : break;
702 :
703 11 : case 50:
704 11 : dfAngle = CPLAtof(szLineBuf);
705 11 : break;
706 :
707 25 : case 72:
708 25 : nHorizontalAlignment = atoi(szLineBuf);
709 25 : break;
710 :
711 0 : case 73:
712 0 : if (!bIsAttribOrAttdef)
713 0 : nVerticalAlignment = atoi(szLineBuf);
714 0 : break;
715 :
716 8 : case 74:
717 8 : if (bIsAttribOrAttdef)
718 8 : nVerticalAlignment = atoi(szLineBuf);
719 8 : break;
720 :
721 0 : case 7:
722 0 : osStyleName = TextRecode(szLineBuf);
723 0 : break;
724 :
725 : // 2 and 70 are for ATTRIB and ATTDEF entities only
726 37 : case 2:
727 37 : if (bIsAttribOrAttdef)
728 : {
729 : // Attribute tags are not supposed to contain spaces (but
730 : // sometimes they do)
731 37 : while (char *pchSpace = strchr(szLineBuf, ' '))
732 0 : *pchSpace = '_';
733 :
734 37 : poFeature->osAttributeTag = szLineBuf;
735 : }
736 37 : break;
737 :
738 37 : case 70:
739 37 : if (bIsAttribOrAttdef)
740 : {
741 : // When the LSB is set, this ATTRIB is "invisible"
742 37 : if (atoi(szLineBuf) & 1)
743 0 : poFeature->oStyleProperties["Hidden"] = "1";
744 : // If the next bit is set, this ATTDEF is to be preserved
745 : // and treated as constant TEXT
746 37 : else if (atoi(szLineBuf) & 2)
747 2 : poFeature->osAttributeTag.Clear();
748 : }
749 37 : break;
750 :
751 405 : default:
752 405 : TranslateGenericProperty(poFeature.get(), nCode, szLineBuf);
753 405 : break;
754 : }
755 : }
756 63 : if (nCode < 0)
757 : {
758 0 : DXF_LAYER_READER_ERROR();
759 0 : return nullptr;
760 : }
761 :
762 63 : if (nCode == 0)
763 63 : poDS->UnreadValue();
764 :
765 63 : OGRPoint *poGeom = nullptr;
766 63 : if (bHaveZ)
767 62 : poGeom = new OGRPoint(dfX, dfY, dfZ);
768 : else
769 1 : poGeom = new OGRPoint(dfX, dfY);
770 63 : poFeature->ApplyOCSTransformer(poGeom);
771 63 : poFeature->SetGeometryDirectly(poGeom);
772 :
773 : /* -------------------------------------------------------------------- */
774 : /* Determine anchor position. */
775 : /* -------------------------------------------------------------------- */
776 63 : if (nHorizontalAlignment > 0 || nVerticalAlignment > 0)
777 : {
778 25 : switch (nVerticalAlignment)
779 : {
780 0 : case 1: // bottom
781 0 : nAnchorPosition = 10;
782 0 : break;
783 :
784 2 : case 2: // middle
785 2 : nAnchorPosition = 4;
786 2 : break;
787 :
788 6 : case 3: // top
789 6 : nAnchorPosition = 7;
790 6 : break;
791 :
792 17 : default:
793 : // Handle "Middle" alignment approximately (this is rather like
794 : // MTEXT alignment in that it uses the actual height of the text
795 : // string to position the text, and thus requires knowledge of
796 : // text metrics)
797 17 : if (nHorizontalAlignment == 4)
798 1 : nAnchorPosition = 5;
799 17 : break;
800 : }
801 25 : if (nHorizontalAlignment < 3)
802 24 : nAnchorPosition += nHorizontalAlignment;
803 : // TODO other alignment options
804 : }
805 :
806 63 : poFeature->SetField("Text", osText);
807 :
808 : /* -------------------------------------------------------------------- */
809 : /* We need to escape double quotes with backslashes before they */
810 : /* can be inserted in the style string. */
811 : /* -------------------------------------------------------------------- */
812 63 : if (strchr(osText, '"') != nullptr)
813 : {
814 0 : CPLString osEscaped;
815 :
816 0 : for (size_t iC = 0; iC < osText.size(); iC++)
817 : {
818 0 : if (osText[iC] == '"')
819 0 : osEscaped += "\\\"";
820 : else
821 0 : osEscaped += osText[iC];
822 : }
823 0 : osText = std::move(osEscaped);
824 : }
825 :
826 : /* -------------------------------------------------------------------- */
827 : /* Prepare style string. */
828 : /* -------------------------------------------------------------------- */
829 126 : CPLString osStyle;
830 : char szBuffer[64];
831 :
832 : // Font name
833 63 : osStyle.Printf("LABEL(f:\"");
834 :
835 : // Preserve legacy behavior of specifying "Arial" as a default font name.
836 63 : osStyle += poDS->LookupTextStyleProperty(osStyleName, "Font", "Arial");
837 :
838 63 : osStyle += "\"";
839 :
840 : // Bold, italic
841 63 : if (EQUAL(poDS->LookupTextStyleProperty(osStyleName, "Bold", "0"), "1"))
842 : {
843 6 : osStyle += ",bo:1";
844 : }
845 63 : if (EQUAL(poDS->LookupTextStyleProperty(osStyleName, "Italic", "0"), "1"))
846 : {
847 6 : osStyle += ",it:1";
848 : }
849 :
850 : // Text string itself
851 63 : osStyle += ",t:\"";
852 63 : osStyle += osText;
853 63 : osStyle += "\"";
854 :
855 : // Other attributes
856 63 : osStyle += CPLString().Printf(",p:%d", nAnchorPosition);
857 :
858 63 : if (dfAngle != 0.0)
859 : {
860 11 : CPLsnprintf(szBuffer, sizeof(szBuffer), "%.3g", dfAngle);
861 11 : osStyle += CPLString().Printf(",a:%s", szBuffer);
862 : }
863 :
864 63 : if (dfHeight != 0.0)
865 : {
866 62 : CPLsnprintf(szBuffer, sizeof(szBuffer), "%.3g", dfHeight);
867 62 : osStyle += CPLString().Printf(",s:%sg", szBuffer);
868 : }
869 :
870 63 : if (dfWidthFactor != 1.0)
871 : {
872 12 : CPLsnprintf(szBuffer, sizeof(szBuffer), "%.4g", dfWidthFactor * 100.0);
873 12 : osStyle += CPLString().Printf(",w:%s", szBuffer);
874 : }
875 :
876 63 : if (bHasAlignmentPoint && dfAlignmentPointX != dfX)
877 : {
878 23 : CPLsnprintf(szBuffer, sizeof(szBuffer), "%.6g",
879 : dfAlignmentPointX - dfX);
880 23 : osStyle += CPLString().Printf(",dx:%sg", szBuffer);
881 : }
882 :
883 63 : if (bHasAlignmentPoint && dfAlignmentPointY != dfY)
884 : {
885 14 : CPLsnprintf(szBuffer, sizeof(szBuffer), "%.6g",
886 : dfAlignmentPointY - dfY);
887 14 : osStyle += CPLString().Printf(",dy:%sg", szBuffer);
888 : }
889 :
890 : // Color
891 63 : osStyle += ",c:";
892 63 : osStyle += poFeature->GetColor(poDS);
893 :
894 63 : osStyle += ")";
895 :
896 63 : poFeature->SetStyleString(osStyle);
897 :
898 63 : return poFeature.release();
899 : }
900 :
901 : /************************************************************************/
902 : /* TranslatePOINT() */
903 : /************************************************************************/
904 :
905 33 : OGRDXFFeature *OGRDXFLayer::TranslatePOINT()
906 :
907 : {
908 : char szLineBuf[257];
909 33 : int nCode = 0;
910 66 : auto poFeature = std::make_unique<OGRDXFFeature>(poFeatureDefn);
911 33 : double dfX = 0.0;
912 33 : double dfY = 0.0;
913 33 : double dfZ = 0.0;
914 33 : bool bHaveZ = false;
915 :
916 349 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
917 : {
918 316 : switch (nCode)
919 : {
920 33 : case 10:
921 33 : dfX = CPLAtof(szLineBuf);
922 33 : break;
923 :
924 33 : case 20:
925 33 : dfY = CPLAtof(szLineBuf);
926 33 : break;
927 :
928 31 : case 30:
929 31 : dfZ = CPLAtof(szLineBuf);
930 31 : bHaveZ = true;
931 31 : break;
932 :
933 219 : default:
934 219 : TranslateGenericProperty(poFeature.get(), nCode, szLineBuf);
935 219 : break;
936 : }
937 : }
938 33 : if (nCode < 0)
939 : {
940 0 : DXF_LAYER_READER_ERROR();
941 0 : return nullptr;
942 : }
943 :
944 33 : if (nCode == 0)
945 33 : poDS->UnreadValue();
946 :
947 33 : OGRPoint *poGeom = nullptr;
948 33 : if (bHaveZ)
949 31 : poGeom = new OGRPoint(dfX, dfY, dfZ);
950 : else
951 2 : poGeom = new OGRPoint(dfX, dfY);
952 :
953 33 : poFeature->SetGeometryDirectly(poGeom);
954 :
955 : // Set style pen color
956 33 : PrepareLineStyle(poFeature.get());
957 :
958 33 : return poFeature.release();
959 : }
960 :
961 : /************************************************************************/
962 : /* TranslateLINE() */
963 : /************************************************************************/
964 :
965 163 : OGRDXFFeature *OGRDXFLayer::TranslateLINE()
966 :
967 : {
968 : char szLineBuf[257];
969 163 : int nCode = 0;
970 326 : auto poFeature = std::make_unique<OGRDXFFeature>(poFeatureDefn);
971 163 : double dfX1 = 0.0;
972 163 : double dfY1 = 0.0;
973 163 : double dfZ1 = 0.0;
974 163 : double dfX2 = 0.0;
975 163 : double dfY2 = 0.0;
976 163 : double dfZ2 = 0.0;
977 163 : bool bHaveZ = false;
978 :
979 : /* -------------------------------------------------------------------- */
980 : /* Process values. */
981 : /* -------------------------------------------------------------------- */
982 2209 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
983 : {
984 2046 : switch (nCode)
985 : {
986 163 : case 10:
987 163 : dfX1 = CPLAtof(szLineBuf);
988 163 : break;
989 :
990 163 : case 11:
991 163 : dfX2 = CPLAtof(szLineBuf);
992 163 : break;
993 :
994 163 : case 20:
995 163 : dfY1 = CPLAtof(szLineBuf);
996 163 : break;
997 :
998 163 : case 21:
999 163 : dfY2 = CPLAtof(szLineBuf);
1000 163 : break;
1001 :
1002 163 : case 30:
1003 163 : dfZ1 = CPLAtof(szLineBuf);
1004 163 : bHaveZ = true;
1005 163 : break;
1006 :
1007 163 : case 31:
1008 163 : dfZ2 = CPLAtof(szLineBuf);
1009 163 : bHaveZ = true;
1010 163 : break;
1011 :
1012 1068 : default:
1013 1068 : TranslateGenericProperty(poFeature.get(), nCode, szLineBuf);
1014 1068 : break;
1015 : }
1016 : }
1017 163 : if (nCode < 0)
1018 : {
1019 0 : DXF_LAYER_READER_ERROR();
1020 0 : return nullptr;
1021 : }
1022 :
1023 163 : if (nCode == 0)
1024 163 : poDS->UnreadValue();
1025 :
1026 : /* -------------------------------------------------------------------- */
1027 : /* Create geometry */
1028 : /* -------------------------------------------------------------------- */
1029 326 : auto poLS = std::make_unique<OGRLineString>();
1030 163 : if (bHaveZ)
1031 : {
1032 163 : poLS->addPoint(dfX1, dfY1, dfZ1);
1033 163 : poLS->addPoint(dfX2, dfY2, dfZ2);
1034 : }
1035 : else
1036 : {
1037 0 : poLS->addPoint(dfX1, dfY1);
1038 0 : poLS->addPoint(dfX2, dfY2);
1039 : }
1040 :
1041 163 : poFeature->SetGeometryDirectly(poLS.release());
1042 :
1043 163 : PrepareLineStyle(poFeature.get());
1044 :
1045 163 : return poFeature.release();
1046 : }
1047 :
1048 : /************************************************************************/
1049 : /* TranslateLWPOLYLINE() */
1050 : /************************************************************************/
1051 37 : OGRDXFFeature *OGRDXFLayer::TranslateLWPOLYLINE()
1052 :
1053 : {
1054 : // Collect vertices and attributes into a smooth polyline.
1055 : // If there are no bulges, then we are a straight-line polyline.
1056 : // Single-vertex polylines become points.
1057 : // Group code 30 (vertex Z) is not part of this entity.
1058 : char szLineBuf[257];
1059 37 : int nCode = 0;
1060 37 : int nPolylineFlag = 0;
1061 :
1062 74 : auto poFeature = std::make_unique<OGRDXFFeature>(poFeatureDefn);
1063 37 : double dfX = 0.0;
1064 37 : double dfY = 0.0;
1065 37 : double dfZ = 0.0;
1066 37 : bool bHaveX = false;
1067 37 : bool bHaveY = false;
1068 :
1069 37 : int nNumVertices = 1; // use 1 based index
1070 37 : int npolyarcVertexCount = 1;
1071 37 : double dfBulge = 0.0;
1072 74 : DXFSmoothPolyline smoothPolyline;
1073 :
1074 37 : smoothPolyline.setCoordinateDimension(2);
1075 :
1076 : /* -------------------------------------------------------------------- */
1077 : /* Collect information from the LWPOLYLINE object itself. */
1078 : /* -------------------------------------------------------------------- */
1079 675 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
1080 : {
1081 638 : if (npolyarcVertexCount > nNumVertices)
1082 : {
1083 0 : CPLError(CE_Failure, CPLE_AppDefined,
1084 : "Too many vertices found in LWPOLYLINE.");
1085 0 : return nullptr;
1086 : }
1087 :
1088 638 : switch (nCode)
1089 : {
1090 8 : case 38:
1091 : // Constant elevation.
1092 8 : dfZ = CPLAtof(szLineBuf);
1093 8 : smoothPolyline.setCoordinateDimension(3);
1094 8 : break;
1095 :
1096 37 : case 90:
1097 37 : nNumVertices = atoi(szLineBuf);
1098 37 : break;
1099 :
1100 37 : case 70:
1101 37 : nPolylineFlag = atoi(szLineBuf);
1102 37 : break;
1103 :
1104 128 : case 10:
1105 128 : if (bHaveX && bHaveY)
1106 : {
1107 91 : smoothPolyline.AddPoint(dfX, dfY, dfZ, dfBulge);
1108 91 : npolyarcVertexCount++;
1109 91 : dfBulge = 0.0;
1110 91 : bHaveY = false;
1111 : }
1112 128 : dfX = CPLAtof(szLineBuf);
1113 128 : bHaveX = true;
1114 128 : break;
1115 :
1116 128 : case 20:
1117 128 : if (bHaveX && bHaveY)
1118 : {
1119 0 : smoothPolyline.AddPoint(dfX, dfY, dfZ, dfBulge);
1120 0 : npolyarcVertexCount++;
1121 0 : dfBulge = 0.0;
1122 0 : bHaveX = false;
1123 : }
1124 128 : dfY = CPLAtof(szLineBuf);
1125 128 : bHaveY = true;
1126 128 : break;
1127 :
1128 14 : case 42:
1129 14 : dfBulge = CPLAtof(szLineBuf);
1130 14 : break;
1131 :
1132 286 : default:
1133 286 : TranslateGenericProperty(poFeature.get(), nCode, szLineBuf);
1134 286 : break;
1135 : }
1136 : }
1137 37 : if (nCode < 0)
1138 : {
1139 0 : DXF_LAYER_READER_ERROR();
1140 0 : return nullptr;
1141 : }
1142 :
1143 37 : if (nCode == 0)
1144 37 : poDS->UnreadValue();
1145 :
1146 37 : if (bHaveX && bHaveY)
1147 37 : smoothPolyline.AddPoint(dfX, dfY, dfZ, dfBulge);
1148 :
1149 37 : if (smoothPolyline.IsEmpty())
1150 : {
1151 0 : return nullptr;
1152 : }
1153 :
1154 : /* -------------------------------------------------------------------- */
1155 : /* Close polyline if necessary. */
1156 : /* -------------------------------------------------------------------- */
1157 37 : const bool bIsClosed = (nPolylineFlag & 0x01) != 0;
1158 37 : if (bIsClosed)
1159 15 : smoothPolyline.Close();
1160 :
1161 37 : const bool bAsPolygon = bIsClosed && poDS->ClosedLineAsPolygon();
1162 :
1163 37 : smoothPolyline.SetUseMaxGapWhenTessellatingArcs(poDS->InlineBlocks());
1164 : auto poGeom =
1165 74 : std::unique_ptr<OGRGeometry>(smoothPolyline.Tessellate(bAsPolygon));
1166 37 : poFeature->ApplyOCSTransformer(poGeom.get());
1167 37 : poFeature->SetGeometryDirectly(poGeom.release());
1168 :
1169 37 : PrepareLineStyle(poFeature.get());
1170 :
1171 37 : return poFeature.release();
1172 : }
1173 :
1174 : /************************************************************************/
1175 : /* SafeAbs() */
1176 : /************************************************************************/
1177 :
1178 24 : static inline int SafeAbs(int x)
1179 : {
1180 24 : if (x == std::numeric_limits<int>::min())
1181 0 : return std::numeric_limits<int>::max();
1182 24 : return abs(x);
1183 : }
1184 :
1185 : /************************************************************************/
1186 : /* TranslatePOLYLINE() */
1187 : /* */
1188 : /* We also capture the following vertices. */
1189 : /************************************************************************/
1190 :
1191 18 : OGRDXFFeature *OGRDXFLayer::TranslatePOLYLINE()
1192 :
1193 : {
1194 : char szLineBuf[257];
1195 18 : int nCode = 0;
1196 18 : int nPolylineFlag = 0;
1197 36 : auto poFeature = std::make_unique<OGRDXFFeature>(poFeatureDefn);
1198 :
1199 : /* -------------------------------------------------------------------- */
1200 : /* Collect information from the POLYLINE object itself. */
1201 : /* -------------------------------------------------------------------- */
1202 208 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
1203 : {
1204 190 : switch (nCode)
1205 : {
1206 18 : case 70:
1207 18 : nPolylineFlag = atoi(szLineBuf);
1208 18 : break;
1209 :
1210 172 : default:
1211 172 : TranslateGenericProperty(poFeature.get(), nCode, szLineBuf);
1212 172 : break;
1213 : }
1214 : }
1215 18 : if (nCode < 0)
1216 : {
1217 0 : DXF_LAYER_READER_ERROR();
1218 0 : return nullptr;
1219 : }
1220 :
1221 18 : if ((nPolylineFlag & 16) != 0)
1222 : {
1223 0 : CPLDebug("DXF", "Polygon mesh not supported.");
1224 0 : return nullptr;
1225 : }
1226 :
1227 : /* -------------------------------------------------------------------- */
1228 : /* Collect vertices as a smooth polyline. */
1229 : /* -------------------------------------------------------------------- */
1230 18 : double dfX = 0.0;
1231 18 : double dfY = 0.0;
1232 18 : double dfZ = 0.0;
1233 18 : double dfBulge = 0.0;
1234 18 : int nVertexFlag = 0;
1235 36 : DXFSmoothPolyline smoothPolyline;
1236 18 : unsigned int vertexIndex71 = 0;
1237 18 : unsigned int vertexIndex72 = 0;
1238 18 : unsigned int vertexIndex73 = 0;
1239 18 : unsigned int vertexIndex74 = 0;
1240 36 : std::vector<OGRPoint> aoPoints;
1241 36 : auto poPS = std::make_unique<OGRPolyhedralSurface>();
1242 :
1243 18 : smoothPolyline.setCoordinateDimension(2);
1244 :
1245 93 : while (nCode == 0 && !EQUAL(szLineBuf, "SEQEND"))
1246 : {
1247 : // Eat non-vertex objects.
1248 75 : if (!EQUAL(szLineBuf, "VERTEX"))
1249 : {
1250 0 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
1251 : {
1252 : }
1253 0 : if (nCode < 0)
1254 : {
1255 0 : DXF_LAYER_READER_ERROR();
1256 0 : return nullptr;
1257 : }
1258 :
1259 0 : continue;
1260 : }
1261 :
1262 : // process a Vertex
1263 874 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
1264 : {
1265 799 : switch (nCode)
1266 : {
1267 75 : case 10:
1268 75 : dfX = CPLAtof(szLineBuf);
1269 75 : break;
1270 :
1271 75 : case 20:
1272 75 : dfY = CPLAtof(szLineBuf);
1273 75 : break;
1274 :
1275 75 : case 30:
1276 75 : dfZ = CPLAtof(szLineBuf);
1277 75 : smoothPolyline.setCoordinateDimension(3);
1278 75 : break;
1279 :
1280 4 : case 42:
1281 4 : dfBulge = CPLAtof(szLineBuf);
1282 4 : break;
1283 :
1284 64 : case 70:
1285 64 : nVertexFlag = atoi(szLineBuf);
1286 64 : break;
1287 :
1288 6 : case 71:
1289 : // See comment below about negative values for 71, 72, 73,
1290 : // 74
1291 6 : vertexIndex71 = SafeAbs(atoi(szLineBuf));
1292 6 : break;
1293 :
1294 6 : case 72:
1295 6 : vertexIndex72 = SafeAbs(atoi(szLineBuf));
1296 6 : break;
1297 :
1298 6 : case 73:
1299 6 : vertexIndex73 = SafeAbs(atoi(szLineBuf));
1300 6 : break;
1301 :
1302 6 : case 74:
1303 6 : vertexIndex74 = SafeAbs(atoi(szLineBuf));
1304 6 : break;
1305 :
1306 482 : default:
1307 482 : break;
1308 : }
1309 : }
1310 :
1311 75 : if (((nVertexFlag & 64) != 0) && ((nVertexFlag & 128) != 0))
1312 : {
1313 : // add the point to the list of points
1314 : try
1315 : {
1316 8 : aoPoints.emplace_back(dfX, dfY, dfZ);
1317 : }
1318 0 : catch (const std::exception &e)
1319 : {
1320 0 : CPLError(CE_Failure, CPLE_OutOfMemory, "%s", e.what());
1321 0 : return nullptr;
1322 : }
1323 : }
1324 :
1325 : // Note - If any index out of vertexIndex71, vertexIndex72,
1326 : // vertexIndex73 or vertexIndex74 is negative, it means that the line
1327 : // starting from that vertex is invisible. However, it still needs to be
1328 : // constructed as part of the resultant polyhedral surface; there is no
1329 : // way to specify the visibility of individual edges in a polyhedral
1330 : // surface at present
1331 :
1332 75 : if (nVertexFlag == 128)
1333 : {
1334 : // create a polygon and add it to the Polyhedral Surface
1335 12 : auto poLR = std::make_unique<OGRLinearRing>();
1336 6 : int iPoint = 0;
1337 6 : int startPoint = -1;
1338 6 : poLR->set3D(TRUE);
1339 6 : if (vertexIndex71 != 0 && vertexIndex71 <= aoPoints.size())
1340 : {
1341 : // if (startPoint == -1)
1342 6 : startPoint = vertexIndex71 - 1;
1343 6 : poLR->setPoint(iPoint, &aoPoints[vertexIndex71 - 1]);
1344 6 : iPoint++;
1345 6 : vertexIndex71 = 0;
1346 : }
1347 6 : if (vertexIndex72 != 0 && vertexIndex72 <= aoPoints.size())
1348 : {
1349 6 : if (startPoint == -1)
1350 0 : startPoint = vertexIndex72 - 1;
1351 6 : poLR->setPoint(iPoint, &aoPoints[vertexIndex72 - 1]);
1352 6 : iPoint++;
1353 6 : vertexIndex72 = 0;
1354 : }
1355 6 : if (vertexIndex73 != 0 && vertexIndex73 <= aoPoints.size())
1356 : {
1357 6 : if (startPoint == -1)
1358 0 : startPoint = vertexIndex73 - 1;
1359 6 : poLR->setPoint(iPoint, &aoPoints[vertexIndex73 - 1]);
1360 6 : iPoint++;
1361 6 : vertexIndex73 = 0;
1362 : }
1363 6 : if (vertexIndex74 != 0 && vertexIndex74 <= aoPoints.size())
1364 : {
1365 6 : if (startPoint == -1)
1366 0 : startPoint = vertexIndex74 - 1;
1367 6 : poLR->setPoint(iPoint, &aoPoints[vertexIndex74 - 1]);
1368 6 : iPoint++;
1369 6 : vertexIndex74 = 0;
1370 : }
1371 6 : if (startPoint >= 0)
1372 : {
1373 : // complete the ring
1374 6 : poLR->setPoint(iPoint, &aoPoints[startPoint]);
1375 :
1376 6 : OGRPolygon *poPolygon = new OGRPolygon();
1377 6 : poPolygon->addRingDirectly(poLR.release());
1378 :
1379 6 : poPS->addGeometryDirectly(poPolygon);
1380 : }
1381 : }
1382 :
1383 75 : if (nCode < 0)
1384 : {
1385 0 : DXF_LAYER_READER_ERROR();
1386 0 : return nullptr;
1387 : }
1388 :
1389 : // Ignore Spline frame control points ( see #4683 )
1390 75 : if ((nVertexFlag & 16) == 0)
1391 75 : smoothPolyline.AddPoint(dfX, dfY, dfZ, dfBulge);
1392 75 : dfBulge = 0.0;
1393 : }
1394 :
1395 18 : if (smoothPolyline.IsEmpty())
1396 : {
1397 0 : return nullptr;
1398 : }
1399 :
1400 18 : if (poPS->getNumGeometries() > 0)
1401 : {
1402 1 : poFeature->SetGeometryDirectly(poPS.release());
1403 1 : PrepareBrushStyle(poFeature.get());
1404 1 : return poFeature.release();
1405 : }
1406 :
1407 : /* -------------------------------------------------------------------- */
1408 : /* Close polyline if necessary. */
1409 : /* -------------------------------------------------------------------- */
1410 17 : const bool bIsClosed = (nPolylineFlag & 0x01) != 0;
1411 17 : if (bIsClosed)
1412 10 : smoothPolyline.Close();
1413 :
1414 17 : const bool bAsPolygon = bIsClosed && poDS->ClosedLineAsPolygon();
1415 :
1416 17 : smoothPolyline.SetUseMaxGapWhenTessellatingArcs(poDS->InlineBlocks());
1417 17 : OGRGeometry *poGeom = smoothPolyline.Tessellate(bAsPolygon);
1418 :
1419 17 : if ((nPolylineFlag & 8) == 0)
1420 2 : poFeature->ApplyOCSTransformer(poGeom);
1421 17 : poFeature->SetGeometryDirectly(poGeom);
1422 :
1423 17 : PrepareLineStyle(poFeature.get());
1424 :
1425 17 : return poFeature.release();
1426 : }
1427 :
1428 : /************************************************************************/
1429 : /* TranslateMLINE() */
1430 : /************************************************************************/
1431 :
1432 3 : OGRDXFFeature *OGRDXFLayer::TranslateMLINE()
1433 :
1434 : {
1435 : char szLineBuf[257];
1436 3 : int nCode = 0;
1437 :
1438 6 : auto poFeature = std::make_unique<OGRDXFFeature>(poFeatureDefn);
1439 :
1440 3 : bool bIsClosed = false;
1441 3 : int nNumVertices = 0;
1442 3 : int nNumElements = 0;
1443 :
1444 57 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0 &&
1445 : nCode != 11)
1446 : {
1447 54 : switch (nCode)
1448 : {
1449 3 : case 71:
1450 3 : bIsClosed = (atoi(szLineBuf) & 2) == 2;
1451 3 : break;
1452 :
1453 3 : case 72:
1454 3 : nNumVertices = atoi(szLineBuf);
1455 3 : break;
1456 :
1457 3 : case 73:
1458 3 : nNumElements = atoi(szLineBuf);
1459 : // No-one should ever need more than 1000 elements!
1460 3 : if (nNumElements <= 0 || nNumElements > 1000)
1461 : {
1462 0 : CPLDebug("DXF", "Invalid number of MLINE elements (73): %s",
1463 : szLineBuf);
1464 0 : DXF_LAYER_READER_ERROR();
1465 0 : return nullptr;
1466 : }
1467 3 : break;
1468 :
1469 45 : default:
1470 45 : TranslateGenericProperty(poFeature.get(), nCode, szLineBuf);
1471 45 : break;
1472 : }
1473 : }
1474 3 : if (nCode < 0)
1475 : {
1476 0 : DXF_LAYER_READER_ERROR();
1477 0 : return nullptr;
1478 : }
1479 :
1480 3 : if (nCode == 0 || nCode == 11)
1481 3 : poDS->UnreadValue();
1482 :
1483 : /* -------------------------------------------------------------------- */
1484 : /* Read in the position and parameters for each vertex, and */
1485 : /* translate these values into line geometries. */
1486 : /* -------------------------------------------------------------------- */
1487 :
1488 6 : auto poMLS = std::make_unique<OGRMultiLineString>();
1489 6 : std::vector<std::unique_ptr<OGRLineString>> apoCurrentLines(nNumElements);
1490 :
1491 : // For use when bIsClosed is true
1492 6 : std::vector<DXFTriple> aoInitialVertices(nNumElements);
1493 :
1494 : #define EXPECT_CODE(code) \
1495 : if (poDS->ReadValue(szLineBuf, sizeof(szLineBuf)) != (code)) \
1496 : { \
1497 : DXF_LAYER_READER_ERROR(); \
1498 : return nullptr; \
1499 : }
1500 :
1501 13 : for (int iVertex = 0; iVertex < nNumVertices; iVertex++)
1502 : {
1503 10 : EXPECT_CODE(11);
1504 10 : const double dfVertexX = CPLAtof(szLineBuf);
1505 10 : EXPECT_CODE(21);
1506 10 : const double dfVertexY = CPLAtof(szLineBuf);
1507 10 : EXPECT_CODE(31);
1508 10 : const double dfVertexZ = CPLAtof(szLineBuf);
1509 :
1510 10 : EXPECT_CODE(12);
1511 10 : const double dfSegmentDirectionX = CPLAtof(szLineBuf);
1512 10 : EXPECT_CODE(22);
1513 10 : const double dfSegmentDirectionY = CPLAtof(szLineBuf);
1514 10 : EXPECT_CODE(32);
1515 10 : const double dfSegmentDirectionZ = CPLAtof(szLineBuf);
1516 :
1517 10 : EXPECT_CODE(13);
1518 10 : const double dfMiterDirectionX = CPLAtof(szLineBuf);
1519 10 : EXPECT_CODE(23);
1520 10 : const double dfMiterDirectionY = CPLAtof(szLineBuf);
1521 10 : EXPECT_CODE(33);
1522 10 : const double dfMiterDirectionZ = CPLAtof(szLineBuf);
1523 :
1524 34 : for (int iElement = 0; iElement < nNumElements; iElement++)
1525 : {
1526 24 : double dfStartSegmentX = 0.0;
1527 24 : double dfStartSegmentY = 0.0;
1528 24 : double dfStartSegmentZ = 0.0;
1529 :
1530 24 : EXPECT_CODE(74);
1531 24 : const int nNumParameters = atoi(szLineBuf);
1532 :
1533 : // The first parameter is special: it is a distance along the
1534 : // miter vector from the initial vertex to the start of the
1535 : // element line.
1536 24 : if (nNumParameters > 0)
1537 : {
1538 24 : EXPECT_CODE(41);
1539 24 : const double dfDistance = CPLAtof(szLineBuf);
1540 :
1541 24 : dfStartSegmentX = dfVertexX + dfMiterDirectionX * dfDistance;
1542 24 : dfStartSegmentY = dfVertexY + dfMiterDirectionY * dfDistance;
1543 24 : dfStartSegmentZ = dfVertexZ + dfMiterDirectionZ * dfDistance;
1544 :
1545 24 : if (bIsClosed && iVertex == 0)
1546 : {
1547 3 : aoInitialVertices[iElement] = DXFTriple(
1548 : dfStartSegmentX, dfStartSegmentY, dfStartSegmentZ);
1549 : }
1550 :
1551 : // If we have an unfinished line for this element, we need
1552 : // to close it off.
1553 24 : if (apoCurrentLines[iElement])
1554 : {
1555 16 : apoCurrentLines[iElement]->addPoint(
1556 : dfStartSegmentX, dfStartSegmentY, dfStartSegmentZ);
1557 32 : poMLS->addGeometryDirectly(
1558 16 : apoCurrentLines[iElement].release());
1559 : }
1560 : }
1561 :
1562 : // Parameters with an odd index give pen-up distances (breaks),
1563 : // while even indexes are pen-down distances (line segments).
1564 61 : for (int iParameter = 1; iParameter < nNumParameters; iParameter++)
1565 : {
1566 37 : EXPECT_CODE(41);
1567 37 : const double dfDistance = CPLAtof(szLineBuf);
1568 :
1569 37 : const double dfCurrentX =
1570 37 : dfStartSegmentX + dfSegmentDirectionX * dfDistance;
1571 37 : const double dfCurrentY =
1572 37 : dfStartSegmentY + dfSegmentDirectionY * dfDistance;
1573 37 : const double dfCurrentZ =
1574 37 : dfStartSegmentZ + dfSegmentDirectionZ * dfDistance;
1575 :
1576 37 : if (iParameter % 2 == 0)
1577 : {
1578 : // The dfCurrent(X,Y,Z) point is the end of a line segment
1579 7 : CPLAssert(apoCurrentLines[iElement]);
1580 7 : apoCurrentLines[iElement]->addPoint(dfCurrentX, dfCurrentY,
1581 : dfCurrentZ);
1582 14 : poMLS->addGeometryDirectly(
1583 7 : apoCurrentLines[iElement].release());
1584 : }
1585 : else
1586 : {
1587 : // The dfCurrent(X,Y,Z) point is the end of a break
1588 30 : apoCurrentLines[iElement] =
1589 60 : std::make_unique<OGRLineString>();
1590 30 : apoCurrentLines[iElement]->addPoint(dfCurrentX, dfCurrentY,
1591 : dfCurrentZ);
1592 : }
1593 : }
1594 :
1595 24 : EXPECT_CODE(75);
1596 24 : const int nNumAreaFillParams = atoi(szLineBuf);
1597 :
1598 24 : for (int iParameter = 0; iParameter < nNumAreaFillParams;
1599 : iParameter++)
1600 : {
1601 0 : EXPECT_CODE(42);
1602 : }
1603 : }
1604 : }
1605 :
1606 : #undef EXPECT_CODE
1607 :
1608 : // Close the MLINE if required.
1609 3 : if (bIsClosed)
1610 : {
1611 4 : for (int iElement = 0; iElement < nNumElements; iElement++)
1612 : {
1613 3 : if (apoCurrentLines[iElement])
1614 : {
1615 6 : apoCurrentLines[iElement]->addPoint(
1616 3 : aoInitialVertices[iElement].dfX,
1617 3 : aoInitialVertices[iElement].dfY,
1618 3 : aoInitialVertices[iElement].dfZ);
1619 3 : poMLS->addGeometryDirectly(apoCurrentLines[iElement].release());
1620 : }
1621 : }
1622 : }
1623 :
1624 : // Apparently extrusions are ignored for MLINE entities.
1625 : // poFeature->ApplyOCSTransformer( poMLS );
1626 3 : poFeature->SetGeometryDirectly(poMLS.release());
1627 :
1628 3 : PrepareLineStyle(poFeature.get());
1629 :
1630 3 : return poFeature.release();
1631 : }
1632 :
1633 : /************************************************************************/
1634 : /* TranslateCIRCLE() */
1635 : /************************************************************************/
1636 :
1637 23 : OGRDXFFeature *OGRDXFLayer::TranslateCIRCLE()
1638 :
1639 : {
1640 : char szLineBuf[257];
1641 23 : int nCode = 0;
1642 46 : auto poFeature = std::make_unique<OGRDXFFeature>(poFeatureDefn);
1643 23 : double dfX1 = 0.0;
1644 23 : double dfY1 = 0.0;
1645 23 : double dfZ1 = 0.0;
1646 23 : double dfRadius = 0.0;
1647 23 : double dfThickness = 0.0;
1648 23 : bool bHaveZ = false;
1649 :
1650 : /* -------------------------------------------------------------------- */
1651 : /* Process values. */
1652 : /* -------------------------------------------------------------------- */
1653 236 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
1654 : {
1655 213 : switch (nCode)
1656 : {
1657 23 : case 10:
1658 23 : dfX1 = CPLAtof(szLineBuf);
1659 23 : break;
1660 :
1661 23 : case 20:
1662 23 : dfY1 = CPLAtof(szLineBuf);
1663 23 : break;
1664 :
1665 23 : case 30:
1666 23 : dfZ1 = CPLAtof(szLineBuf);
1667 23 : bHaveZ = true;
1668 23 : break;
1669 :
1670 1 : case 39:
1671 1 : dfThickness = CPLAtof(szLineBuf);
1672 1 : break;
1673 :
1674 23 : case 40:
1675 23 : dfRadius = CPLAtof(szLineBuf);
1676 23 : break;
1677 :
1678 120 : default:
1679 120 : TranslateGenericProperty(poFeature.get(), nCode, szLineBuf);
1680 120 : break;
1681 : }
1682 : }
1683 23 : if (nCode < 0)
1684 : {
1685 0 : DXF_LAYER_READER_ERROR();
1686 0 : return nullptr;
1687 : }
1688 :
1689 23 : if (nCode == 0)
1690 23 : poDS->UnreadValue();
1691 :
1692 : /* -------------------------------------------------------------------- */
1693 : /* Create geometry */
1694 : /* -------------------------------------------------------------------- */
1695 : auto poCircle = std::unique_ptr<OGRLineString>(
1696 : OGRGeometryFactory::approximateArcAngles(dfX1, dfY1, dfZ1, dfRadius,
1697 : dfRadius, 0.0, 0.0, 360.0, 0.0,
1698 23 : poDS->InlineBlocks())
1699 46 : ->toLineString());
1700 :
1701 23 : const int nPoints = poCircle->getNumPoints();
1702 :
1703 : // If dfThickness is nonzero, we need to extrude a cylinder of height
1704 : // dfThickness in the Z axis.
1705 23 : if (dfThickness != 0.0 && nPoints > 1)
1706 : {
1707 1 : OGRPolyhedralSurface *poSurface = new OGRPolyhedralSurface();
1708 :
1709 : // Add the bottom base as a polygon
1710 1 : OGRLinearRing *poRing1 = new OGRLinearRing();
1711 1 : poRing1->addSubLineString(poCircle.get());
1712 :
1713 1 : OGRPolygon *poBase1 = new OGRPolygon();
1714 1 : poBase1->addRingDirectly(poRing1);
1715 1 : poSurface->addGeometryDirectly(poBase1);
1716 :
1717 : // Create and add the top base
1718 1 : OGRLinearRing *poRing2 = poRing1->clone();
1719 :
1720 1 : OGRDXFInsertTransformer oTransformer;
1721 1 : oTransformer.dfZOffset = dfThickness;
1722 1 : poRing2->transform(&oTransformer);
1723 :
1724 1 : OGRPolygon *poBase2 = new OGRPolygon();
1725 1 : poBase2->addRingDirectly(poRing2);
1726 1 : poSurface->addGeometryDirectly(poBase2);
1727 :
1728 : // Add the side of the cylinder as two "semicylindrical" polygons
1729 2 : auto poRect = std::make_unique<OGRLinearRing>();
1730 2 : OGRPoint oPoint;
1731 :
1732 47 : for (int iPoint = nPoints / 2; iPoint >= 0; iPoint--)
1733 : {
1734 46 : poRing1->getPoint(iPoint, &oPoint);
1735 46 : poRect->addPoint(&oPoint);
1736 : }
1737 47 : for (int iPoint = 0; iPoint <= nPoints / 2; iPoint++)
1738 : {
1739 46 : poRing2->getPoint(iPoint, &oPoint);
1740 46 : poRect->addPoint(&oPoint);
1741 : }
1742 :
1743 1 : poRect->closeRings();
1744 :
1745 1 : OGRPolygon *poRectPolygon = new OGRPolygon();
1746 1 : poRectPolygon->addRingDirectly(poRect.release());
1747 1 : poSurface->addGeometryDirectly(poRectPolygon);
1748 :
1749 1 : poRect = std::make_unique<OGRLinearRing>();
1750 :
1751 47 : for (int iPoint = nPoints - 1; iPoint >= nPoints / 2; iPoint--)
1752 : {
1753 46 : poRing1->getPoint(iPoint, &oPoint);
1754 46 : poRect->addPoint(&oPoint);
1755 : }
1756 47 : for (int iPoint = nPoints / 2; iPoint < nPoints; iPoint++)
1757 : {
1758 46 : poRing2->getPoint(iPoint, &oPoint);
1759 46 : poRect->addPoint(&oPoint);
1760 : }
1761 :
1762 1 : poRect->closeRings();
1763 :
1764 1 : poRectPolygon = new OGRPolygon();
1765 1 : poRectPolygon->addRingDirectly(poRect.release());
1766 1 : poSurface->addGeometryDirectly(poRectPolygon);
1767 :
1768 : // That's your cylinder, folks
1769 1 : poFeature->ApplyOCSTransformer(poSurface);
1770 2 : poFeature->SetGeometryDirectly(poSurface);
1771 : }
1772 : else
1773 : {
1774 22 : if (!bHaveZ)
1775 0 : poCircle->flattenTo2D();
1776 :
1777 22 : poFeature->ApplyOCSTransformer(poCircle.get());
1778 22 : poFeature->SetGeometryDirectly(poCircle.release());
1779 : }
1780 :
1781 23 : PrepareLineStyle(poFeature.get());
1782 :
1783 23 : return poFeature.release();
1784 : }
1785 :
1786 : /************************************************************************/
1787 : /* TranslateELLIPSE() */
1788 : /************************************************************************/
1789 :
1790 29 : OGRDXFFeature *OGRDXFLayer::TranslateELLIPSE()
1791 :
1792 : {
1793 : char szLineBuf[257];
1794 29 : int nCode = 0;
1795 58 : auto poFeature = std::make_unique<OGRDXFFeature>(poFeatureDefn);
1796 29 : double dfX1 = 0.0;
1797 29 : double dfY1 = 0.0;
1798 29 : double dfZ1 = 0.0;
1799 29 : double dfRatio = 0.0;
1800 29 : double dfStartAngle = 0.0;
1801 29 : double dfEndAngle = 360.0;
1802 29 : double dfAxisX = 0.0;
1803 29 : double dfAxisY = 0.0;
1804 29 : double dfAxisZ = 0.0;
1805 29 : bool bHaveZ = false;
1806 29 : bool bApplyOCSTransform = false;
1807 :
1808 : /* -------------------------------------------------------------------- */
1809 : /* Process values. */
1810 : /* -------------------------------------------------------------------- */
1811 555 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
1812 : {
1813 526 : switch (nCode)
1814 : {
1815 29 : case 10:
1816 29 : dfX1 = CPLAtof(szLineBuf);
1817 29 : break;
1818 :
1819 29 : case 20:
1820 29 : dfY1 = CPLAtof(szLineBuf);
1821 29 : break;
1822 :
1823 29 : case 30:
1824 29 : dfZ1 = CPLAtof(szLineBuf);
1825 29 : bHaveZ = true;
1826 29 : break;
1827 :
1828 29 : case 11:
1829 29 : dfAxisX = CPLAtof(szLineBuf);
1830 29 : break;
1831 :
1832 29 : case 21:
1833 29 : dfAxisY = CPLAtof(szLineBuf);
1834 29 : break;
1835 :
1836 29 : case 31:
1837 29 : dfAxisZ = CPLAtof(szLineBuf);
1838 29 : break;
1839 :
1840 29 : case 40:
1841 29 : dfRatio = CPLAtof(szLineBuf);
1842 29 : break;
1843 :
1844 29 : case 41:
1845 : // These *seem* to always be in radians regardless of $AUNITS
1846 29 : dfEndAngle = -1 * CPLAtof(szLineBuf) * 180.0 / M_PI;
1847 29 : break;
1848 :
1849 29 : case 42:
1850 : // These *seem* to always be in radians regardless of $AUNITS
1851 29 : dfStartAngle = -1 * CPLAtof(szLineBuf) * 180.0 / M_PI;
1852 29 : break;
1853 :
1854 265 : default:
1855 265 : TranslateGenericProperty(poFeature.get(), nCode, szLineBuf);
1856 265 : break;
1857 : }
1858 : }
1859 29 : if (nCode < 0)
1860 : {
1861 0 : DXF_LAYER_READER_ERROR();
1862 0 : return nullptr;
1863 : }
1864 :
1865 29 : if (nCode == 0)
1866 29 : poDS->UnreadValue();
1867 :
1868 : /* -------------------------------------------------------------------- */
1869 : /* Setup coordinate system */
1870 : /* -------------------------------------------------------------------- */
1871 : double adfN[3];
1872 29 : poFeature->oOCS.ToArray(adfN);
1873 :
1874 29 : if ((adfN[0] == 0.0 && adfN[1] == 0.0 && adfN[2] == 1.0) == false)
1875 : {
1876 14 : OGRDXFOCSTransformer oTransformer(adfN, true);
1877 :
1878 7 : bApplyOCSTransform = true;
1879 :
1880 7 : double *x = &dfX1;
1881 7 : double *y = &dfY1;
1882 7 : double *z = &dfZ1;
1883 7 : oTransformer.InverseTransform(1, x, y, z);
1884 :
1885 7 : x = &dfAxisX;
1886 7 : y = &dfAxisY;
1887 7 : z = &dfAxisZ;
1888 7 : oTransformer.InverseTransform(1, x, y, z);
1889 : }
1890 :
1891 : /* -------------------------------------------------------------------- */
1892 : /* Compute primary and secondary axis lengths, and the angle of */
1893 : /* rotation for the ellipse. */
1894 : /* -------------------------------------------------------------------- */
1895 : double dfPrimaryRadius =
1896 29 : sqrt(dfAxisX * dfAxisX + dfAxisY * dfAxisY + dfAxisZ * dfAxisZ);
1897 :
1898 29 : double dfSecondaryRadius = dfRatio * dfPrimaryRadius;
1899 :
1900 29 : double dfRotation = -1 * atan2(dfAxisY, dfAxisX) * 180 / M_PI;
1901 :
1902 : /* -------------------------------------------------------------------- */
1903 : /* Create geometry */
1904 : /* -------------------------------------------------------------------- */
1905 29 : if (dfStartAngle > dfEndAngle)
1906 0 : dfEndAngle += 360.0;
1907 :
1908 29 : if (fabs(dfEndAngle - dfStartAngle) <= 361.0)
1909 : {
1910 : // Only honor OGR_DXF_MAX_GAP if this geometry isn't at risk of
1911 : // being enlarged or shrunk as part of a block insertion.
1912 : auto poEllipse = std::unique_ptr<OGRGeometry>(
1913 : OGRGeometryFactory::approximateArcAngles(
1914 : dfX1, dfY1, dfZ1, dfPrimaryRadius, dfSecondaryRadius,
1915 : dfRotation, dfStartAngle, dfEndAngle, 0.0,
1916 58 : poDS->InlineBlocks()));
1917 :
1918 29 : if (!bHaveZ)
1919 0 : poEllipse->flattenTo2D();
1920 :
1921 29 : if (bApplyOCSTransform == true)
1922 7 : poFeature->ApplyOCSTransformer(poEllipse.get());
1923 29 : poFeature->SetGeometryDirectly(poEllipse.release());
1924 : }
1925 : else
1926 : {
1927 : // TODO: emit error ?
1928 : }
1929 :
1930 29 : PrepareLineStyle(poFeature.get());
1931 :
1932 29 : return poFeature.release();
1933 : }
1934 :
1935 : /************************************************************************/
1936 : /* TranslateARC() */
1937 : /************************************************************************/
1938 :
1939 14 : OGRDXFFeature *OGRDXFLayer::TranslateARC()
1940 :
1941 : {
1942 : char szLineBuf[257];
1943 14 : int nCode = 0;
1944 28 : auto poFeature = std::make_unique<OGRDXFFeature>(poFeatureDefn);
1945 14 : double dfX1 = 0.0;
1946 14 : double dfY1 = 0.0;
1947 14 : double dfZ1 = 0.0;
1948 14 : double dfRadius = 0.0;
1949 14 : double dfStartAngle = 0.0;
1950 14 : double dfEndAngle = 360.0;
1951 14 : bool bHaveZ = false;
1952 :
1953 : /* -------------------------------------------------------------------- */
1954 : /* Process values. */
1955 : /* -------------------------------------------------------------------- */
1956 202 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
1957 : {
1958 188 : switch (nCode)
1959 : {
1960 14 : case 10:
1961 14 : dfX1 = CPLAtof(szLineBuf);
1962 14 : break;
1963 :
1964 14 : case 20:
1965 14 : dfY1 = CPLAtof(szLineBuf);
1966 14 : break;
1967 :
1968 14 : case 30:
1969 14 : dfZ1 = CPLAtof(szLineBuf);
1970 14 : bHaveZ = true;
1971 14 : break;
1972 :
1973 14 : case 40:
1974 14 : dfRadius = CPLAtof(szLineBuf);
1975 14 : break;
1976 :
1977 14 : case 50:
1978 : // This is apparently always degrees regardless of AUNITS
1979 14 : dfEndAngle = -1 * CPLAtof(szLineBuf);
1980 14 : break;
1981 :
1982 14 : case 51:
1983 : // This is apparently always degrees regardless of AUNITS
1984 14 : dfStartAngle = -1 * CPLAtof(szLineBuf);
1985 14 : break;
1986 :
1987 104 : default:
1988 104 : TranslateGenericProperty(poFeature.get(), nCode, szLineBuf);
1989 104 : break;
1990 : }
1991 : }
1992 14 : if (nCode < 0)
1993 : {
1994 0 : DXF_LAYER_READER_ERROR();
1995 0 : return nullptr;
1996 : }
1997 :
1998 14 : if (nCode == 0)
1999 14 : poDS->UnreadValue();
2000 :
2001 : /* -------------------------------------------------------------------- */
2002 : /* Create geometry */
2003 : /* -------------------------------------------------------------------- */
2004 14 : if (dfStartAngle > dfEndAngle)
2005 13 : dfEndAngle += 360.0;
2006 :
2007 14 : if (fabs(dfEndAngle - dfStartAngle) <= 361.0)
2008 : {
2009 : auto poArc = std::unique_ptr<OGRGeometry>(
2010 : OGRGeometryFactory::approximateArcAngles(
2011 : dfX1, dfY1, dfZ1, dfRadius, dfRadius, 0.0, dfStartAngle,
2012 28 : dfEndAngle, 0.0, poDS->InlineBlocks()));
2013 14 : if (!bHaveZ)
2014 0 : poArc->flattenTo2D();
2015 :
2016 14 : poFeature->ApplyOCSTransformer(poArc.get());
2017 14 : poFeature->SetGeometryDirectly(poArc.release());
2018 : }
2019 : else
2020 : {
2021 : // TODO: emit error ?
2022 : }
2023 :
2024 14 : PrepareLineStyle(poFeature.get());
2025 :
2026 14 : return poFeature.release();
2027 : }
2028 :
2029 : /************************************************************************/
2030 : /* TranslateSPLINE() */
2031 : /************************************************************************/
2032 :
2033 : void rbspline2(int npts, int k, int p1, double b[], double h[],
2034 : bool bCalculateKnots, double knots[], double p[]);
2035 :
2036 25 : OGRDXFFeature *OGRDXFLayer::TranslateSPLINE()
2037 :
2038 : {
2039 : char szLineBuf[257];
2040 : int nCode;
2041 50 : auto poFeature = std::make_unique<OGRDXFFeature>(poFeatureDefn);
2042 :
2043 50 : std::vector<double> adfControlPoints(FORTRAN_INDEXING, 0.0);
2044 50 : std::vector<double> adfKnots(FORTRAN_INDEXING, 0.0);
2045 50 : std::vector<double> adfWeights(FORTRAN_INDEXING, 0.0);
2046 25 : int nDegree = -1;
2047 25 : int nControlPoints = -1;
2048 25 : int nKnots = -1;
2049 25 : bool bInsertNullZ = false;
2050 25 : bool bHasZ = false;
2051 :
2052 : /* -------------------------------------------------------------------- */
2053 : /* Process values. */
2054 : /* -------------------------------------------------------------------- */
2055 1363 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
2056 : {
2057 1339 : bool bStop = false;
2058 1339 : switch (nCode)
2059 : {
2060 138 : case 10:
2061 138 : if (bInsertNullZ)
2062 : {
2063 0 : adfControlPoints.push_back(0.0);
2064 0 : bInsertNullZ = false;
2065 : }
2066 138 : adfControlPoints.push_back(CPLAtof(szLineBuf));
2067 138 : break;
2068 :
2069 138 : case 20:
2070 138 : adfControlPoints.push_back(CPLAtof(szLineBuf));
2071 138 : bInsertNullZ = true;
2072 138 : break;
2073 :
2074 138 : case 30:
2075 138 : adfControlPoints.push_back(CPLAtof(szLineBuf));
2076 138 : bHasZ = true;
2077 138 : bInsertNullZ = false;
2078 138 : break;
2079 :
2080 227 : case 40:
2081 : {
2082 227 : double dfVal = CPLAtof(szLineBuf);
2083 : // Ad-hoc fix for https://github.com/OSGeo/gdal/issues/1969
2084 : // where the first knot is at a very very close to zero negative
2085 : // value and following knots are at 0.
2086 227 : if (dfVal < 0 && dfVal > -1.0e-10)
2087 1 : dfVal = 0;
2088 227 : adfKnots.push_back(dfVal);
2089 227 : break;
2090 : }
2091 :
2092 10 : case 41:
2093 10 : adfWeights.push_back(CPLAtof(szLineBuf));
2094 10 : break;
2095 :
2096 25 : case 70:
2097 25 : break;
2098 :
2099 25 : case 71:
2100 25 : nDegree = atoi(szLineBuf);
2101 : // Arbitrary threshold
2102 25 : if (nDegree < 0 || nDegree > 100)
2103 : {
2104 0 : DXF_LAYER_READER_ERROR();
2105 0 : return nullptr;
2106 : }
2107 25 : break;
2108 :
2109 25 : case 72:
2110 25 : nKnots = atoi(szLineBuf);
2111 : // Arbitrary threshold
2112 25 : if (nKnots < 0 || nKnots > 10000000)
2113 : {
2114 0 : DXF_LAYER_READER_ERROR();
2115 0 : return nullptr;
2116 : }
2117 25 : break;
2118 :
2119 25 : case 73:
2120 25 : nControlPoints = atoi(szLineBuf);
2121 : // Arbitrary threshold
2122 25 : if (nControlPoints < 0 || nControlPoints > 10000000)
2123 : {
2124 0 : DXF_LAYER_READER_ERROR();
2125 0 : return nullptr;
2126 : }
2127 25 : break;
2128 :
2129 51 : case 100:
2130 51 : if (EQUAL(szLineBuf, "AcDbHelix"))
2131 1 : bStop = true;
2132 51 : TranslateGenericProperty(poFeature.get(), nCode, szLineBuf);
2133 51 : break;
2134 :
2135 537 : default:
2136 537 : TranslateGenericProperty(poFeature.get(), nCode, szLineBuf);
2137 537 : break;
2138 : }
2139 :
2140 1339 : if (bStop)
2141 1 : break;
2142 : }
2143 25 : if (nCode < 0)
2144 : {
2145 0 : DXF_LAYER_READER_ERROR();
2146 0 : return nullptr;
2147 : }
2148 :
2149 25 : if (nCode == 0)
2150 24 : poDS->UnreadValue();
2151 :
2152 25 : if (bInsertNullZ)
2153 : {
2154 0 : adfControlPoints.push_back(0.0);
2155 : }
2156 :
2157 25 : if (static_cast<int>(adfControlPoints.size() % 3) != FORTRAN_INDEXING)
2158 : {
2159 0 : CPLError(CE_Failure, CPLE_AppDefined,
2160 : "Invalid number of values for spline control points");
2161 0 : DXF_LAYER_READER_ERROR();
2162 0 : return nullptr;
2163 : }
2164 :
2165 : /* -------------------------------------------------------------------- */
2166 : /* Use the helper function to check the input data and insert */
2167 : /* the spline. */
2168 : /* -------------------------------------------------------------------- */
2169 : auto poLS =
2170 : InsertSplineWithChecks(nDegree, adfControlPoints, bHasZ, nControlPoints,
2171 50 : adfKnots, nKnots, adfWeights);
2172 :
2173 25 : if (!poLS)
2174 : {
2175 0 : DXF_LAYER_READER_ERROR();
2176 0 : return nullptr;
2177 : }
2178 :
2179 25 : poFeature->SetGeometryDirectly(poLS.release());
2180 :
2181 25 : PrepareLineStyle(poFeature.get());
2182 :
2183 25 : return poFeature.release();
2184 : }
2185 :
2186 : /************************************************************************/
2187 : /* InsertSplineWithChecks() */
2188 : /* */
2189 : /* Inserts a spline based on unchecked DXF input. The arrays are */
2190 : /* one-based. */
2191 : /************************************************************************/
2192 :
2193 27 : std::unique_ptr<OGRLineString> OGRDXFLayer::InsertSplineWithChecks(
2194 : const int nDegree, std::vector<double> &adfControlPoints, bool bHasZ,
2195 : int nControlPoints, std::vector<double> &adfKnots, int nKnots,
2196 : std::vector<double> &adfWeights)
2197 : {
2198 : /* -------------------------------------------------------------------- */
2199 : /* Sanity checks */
2200 : /* -------------------------------------------------------------------- */
2201 27 : const int nOrder = nDegree + 1;
2202 :
2203 27 : bool bResult = (nOrder >= 2);
2204 27 : if (bResult == true)
2205 : {
2206 : // Check whether nctrlpts value matches number of vertices read
2207 : int nCheck =
2208 27 : (static_cast<int>(adfControlPoints.size()) - FORTRAN_INDEXING) / 3;
2209 :
2210 27 : if (nControlPoints == -1)
2211 0 : nControlPoints =
2212 0 : (static_cast<int>(adfControlPoints.size()) - FORTRAN_INDEXING) /
2213 : 3;
2214 :
2215 : // min( num(ctrlpts) ) = order
2216 27 : bResult = (nControlPoints >= nOrder && nControlPoints == nCheck);
2217 : }
2218 :
2219 27 : bool bCalculateKnots = false;
2220 27 : if (bResult == true)
2221 : {
2222 27 : int nCheck = static_cast<int>(adfKnots.size()) - FORTRAN_INDEXING;
2223 :
2224 : // Recalculate knots when:
2225 : // - no knots data present, nknots is -1 and ncheck is 0
2226 : // - nknots value present, no knot vertices
2227 : // nknots is (nctrlpts + order), ncheck is 0
2228 27 : if (nCheck == 0)
2229 : {
2230 1 : bCalculateKnots = true;
2231 12 : for (int i = 0; i < (nControlPoints + nOrder); i++)
2232 11 : adfKnots.push_back(0.0);
2233 :
2234 1 : nCheck = static_cast<int>(adfKnots.size()) - FORTRAN_INDEXING;
2235 : }
2236 : // Adjust nknots value when:
2237 : // - nknots value not present, knot vertices present
2238 : // nknots is -1, ncheck is (nctrlpts + order)
2239 27 : if (nKnots == -1)
2240 0 : nKnots = static_cast<int>(adfKnots.size()) - FORTRAN_INDEXING;
2241 :
2242 : // num(knots) = num(ctrlpts) + order
2243 27 : bResult = (nKnots == (nControlPoints + nOrder) && nKnots == nCheck);
2244 : }
2245 :
2246 27 : if (bResult == true)
2247 : {
2248 27 : int nWeights = static_cast<int>(adfWeights.size()) - FORTRAN_INDEXING;
2249 :
2250 27 : if (nWeights == 0)
2251 : {
2252 158 : for (int i = 0; i < nControlPoints; i++)
2253 134 : adfWeights.push_back(1.0);
2254 :
2255 24 : nWeights = static_cast<int>(adfWeights.size()) - FORTRAN_INDEXING;
2256 : }
2257 :
2258 : // num(weights) = num(ctrlpts)
2259 27 : bResult = (nWeights == nControlPoints);
2260 : }
2261 :
2262 27 : if (bResult == false)
2263 0 : return nullptr;
2264 :
2265 : /* -------------------------------------------------------------------- */
2266 : /* Interpolate spline */
2267 : /* -------------------------------------------------------------------- */
2268 27 : int p1 = nControlPoints * 8;
2269 54 : std::vector<double> p(3 * p1 + FORTRAN_INDEXING);
2270 :
2271 27 : rbspline2(nControlPoints, nOrder, p1, &(adfControlPoints[0]),
2272 27 : &(adfWeights[0]), bCalculateKnots, &(adfKnots[0]), &(p[0]));
2273 :
2274 : /* -------------------------------------------------------------------- */
2275 : /* Turn into OGR geometry. */
2276 : /* -------------------------------------------------------------------- */
2277 54 : auto poLS = std::make_unique<OGRLineString>();
2278 :
2279 27 : poLS->setNumPoints(p1);
2280 27 : if (bHasZ)
2281 : {
2282 1129 : for (int i = 0; i < p1; i++)
2283 2208 : poLS->setPoint(i, p[i * 3 + FORTRAN_INDEXING],
2284 1104 : p[i * 3 + FORTRAN_INDEXING + 1],
2285 1104 : p[i * 3 + FORTRAN_INDEXING + 2]);
2286 : }
2287 : else
2288 : {
2289 106 : for (int i = 0; i < p1; i++)
2290 208 : poLS->setPoint(i, p[i * 3 + FORTRAN_INDEXING],
2291 104 : p[i * 3 + FORTRAN_INDEXING + 1]);
2292 : }
2293 :
2294 27 : return poLS;
2295 : }
2296 :
2297 : /************************************************************************/
2298 : /* Translate3DFACE() */
2299 : /************************************************************************/
2300 :
2301 10 : OGRDXFFeature *OGRDXFLayer::Translate3DFACE()
2302 :
2303 : {
2304 : char szLineBuf[257];
2305 10 : int nCode = 0;
2306 20 : auto poFeature = std::make_unique<OGRDXFFeature>(poFeatureDefn);
2307 10 : double dfX1 = 0.0;
2308 10 : double dfY1 = 0.0;
2309 10 : double dfZ1 = 0.0;
2310 10 : double dfX2 = 0.0;
2311 10 : double dfY2 = 0.0;
2312 10 : double dfZ2 = 0.0;
2313 10 : double dfX3 = 0.0;
2314 10 : double dfY3 = 0.0;
2315 10 : double dfZ3 = 0.0;
2316 10 : double dfX4 = 0.0;
2317 10 : double dfY4 = 0.0;
2318 10 : double dfZ4 = 0.0;
2319 :
2320 : /* -------------------------------------------------------------------- */
2321 : /* Process values. */
2322 : /* -------------------------------------------------------------------- */
2323 172 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
2324 : {
2325 162 : switch (nCode)
2326 : {
2327 10 : case 10:
2328 10 : dfX1 = CPLAtof(szLineBuf);
2329 10 : break;
2330 :
2331 10 : case 11:
2332 10 : dfX2 = CPLAtof(szLineBuf);
2333 10 : break;
2334 :
2335 10 : case 12:
2336 10 : dfX3 = CPLAtof(szLineBuf);
2337 10 : break;
2338 :
2339 10 : case 13:
2340 10 : dfX4 = CPLAtof(szLineBuf);
2341 10 : break;
2342 :
2343 10 : case 20:
2344 10 : dfY1 = CPLAtof(szLineBuf);
2345 10 : break;
2346 :
2347 10 : case 21:
2348 10 : dfY2 = CPLAtof(szLineBuf);
2349 10 : break;
2350 :
2351 10 : case 22:
2352 10 : dfY3 = CPLAtof(szLineBuf);
2353 10 : break;
2354 :
2355 10 : case 23:
2356 10 : dfY4 = CPLAtof(szLineBuf);
2357 10 : break;
2358 :
2359 10 : case 30:
2360 10 : dfZ1 = CPLAtof(szLineBuf);
2361 10 : break;
2362 :
2363 10 : case 31:
2364 10 : dfZ2 = CPLAtof(szLineBuf);
2365 10 : break;
2366 :
2367 10 : case 32:
2368 10 : dfZ3 = CPLAtof(szLineBuf);
2369 10 : break;
2370 :
2371 10 : case 33:
2372 10 : dfZ4 = CPLAtof(szLineBuf);
2373 10 : break;
2374 :
2375 42 : default:
2376 42 : TranslateGenericProperty(poFeature.get(), nCode, szLineBuf);
2377 42 : break;
2378 : }
2379 : }
2380 10 : if (nCode < 0)
2381 : {
2382 0 : DXF_LAYER_READER_ERROR();
2383 0 : return nullptr;
2384 : }
2385 :
2386 10 : if (nCode == 0)
2387 10 : poDS->UnreadValue();
2388 :
2389 : /* -------------------------------------------------------------------- */
2390 : /* Create geometry */
2391 : /* -------------------------------------------------------------------- */
2392 20 : auto poPoly = std::make_unique<OGRPolygon>();
2393 10 : OGRLinearRing *poLR = new OGRLinearRing();
2394 10 : poLR->addPoint(dfX1, dfY1, dfZ1);
2395 10 : poLR->addPoint(dfX2, dfY2, dfZ2);
2396 10 : poLR->addPoint(dfX3, dfY3, dfZ3);
2397 10 : if (dfX4 != dfX3 || dfY4 != dfY3 || dfZ4 != dfZ3)
2398 9 : poLR->addPoint(dfX4, dfY4, dfZ4);
2399 10 : poPoly->addRingDirectly(poLR);
2400 10 : poPoly->closeRings();
2401 :
2402 10 : poFeature->ApplyOCSTransformer(poLR);
2403 10 : poFeature->SetGeometryDirectly(poPoly.release());
2404 :
2405 10 : PrepareLineStyle(poFeature.get());
2406 :
2407 10 : return poFeature.release();
2408 : }
2409 :
2410 : /* -------------------------------------------------------------------- */
2411 : /* PointXAxisComparer */
2412 : /* */
2413 : /* Returns true if oP1 is to the left of oP2, or they have the */
2414 : /* same x-coordinate and oP1 is below oP2. */
2415 : /* -------------------------------------------------------------------- */
2416 :
2417 150 : static bool PointXAxisComparer(const OGRPoint &oP1, const OGRPoint &oP2)
2418 : {
2419 256 : return oP1.getX() == oP2.getX() ? oP1.getY() < oP2.getY()
2420 256 : : oP1.getX() < oP2.getX();
2421 : }
2422 :
2423 : /* -------------------------------------------------------------------- */
2424 : /* PointXYZEqualityComparer */
2425 : /* */
2426 : /* Returns true if oP1 is equal to oP2 in the X, Y and Z axes. */
2427 : /* -------------------------------------------------------------------- */
2428 :
2429 75 : static bool PointXYZEqualityComparer(const OGRPoint &oP1, const OGRPoint &oP2)
2430 : {
2431 86 : return oP1.getX() == oP2.getX() && oP1.getY() == oP2.getY() &&
2432 86 : oP1.getZ() == oP2.getZ();
2433 : }
2434 :
2435 : /************************************************************************/
2436 : /* TranslateSOLID() */
2437 : /************************************************************************/
2438 :
2439 25 : OGRDXFFeature *OGRDXFLayer::TranslateSOLID()
2440 :
2441 : {
2442 : char szLineBuf[257];
2443 25 : int nCode = 0;
2444 50 : auto poFeature = std::make_unique<OGRDXFFeature>(poFeatureDefn);
2445 25 : double dfX1 = 0.0;
2446 25 : double dfY1 = 0.0;
2447 25 : double dfZ1 = 0.0;
2448 25 : double dfX2 = 0.0;
2449 25 : double dfY2 = 0.0;
2450 25 : double dfZ2 = 0.0;
2451 25 : double dfX3 = 0.0;
2452 25 : double dfY3 = 0.0;
2453 25 : double dfZ3 = 0.0;
2454 25 : double dfX4 = 0.0;
2455 25 : double dfY4 = 0.0;
2456 25 : double dfZ4 = 0.0;
2457 :
2458 472 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
2459 : {
2460 447 : switch (nCode)
2461 : {
2462 25 : case 10:
2463 25 : dfX1 = CPLAtof(szLineBuf);
2464 25 : break;
2465 :
2466 25 : case 20:
2467 25 : dfY1 = CPLAtof(szLineBuf);
2468 25 : break;
2469 :
2470 25 : case 30:
2471 25 : dfZ1 = CPLAtof(szLineBuf);
2472 25 : break;
2473 :
2474 25 : case 11:
2475 25 : dfX2 = CPLAtof(szLineBuf);
2476 25 : break;
2477 :
2478 25 : case 21:
2479 25 : dfY2 = CPLAtof(szLineBuf);
2480 25 : break;
2481 :
2482 25 : case 31:
2483 25 : dfZ2 = CPLAtof(szLineBuf);
2484 25 : break;
2485 :
2486 25 : case 12:
2487 25 : dfX3 = CPLAtof(szLineBuf);
2488 25 : break;
2489 :
2490 25 : case 22:
2491 25 : dfY3 = CPLAtof(szLineBuf);
2492 25 : break;
2493 :
2494 25 : case 32:
2495 25 : dfZ3 = CPLAtof(szLineBuf);
2496 25 : break;
2497 :
2498 25 : case 13:
2499 25 : dfX4 = CPLAtof(szLineBuf);
2500 25 : break;
2501 :
2502 25 : case 23:
2503 25 : dfY4 = CPLAtof(szLineBuf);
2504 25 : break;
2505 :
2506 25 : case 33:
2507 25 : dfZ4 = CPLAtof(szLineBuf);
2508 25 : break;
2509 :
2510 147 : default:
2511 147 : TranslateGenericProperty(poFeature.get(), nCode, szLineBuf);
2512 147 : break;
2513 : }
2514 : }
2515 25 : if (nCode < 0)
2516 : {
2517 0 : DXF_LAYER_READER_ERROR();
2518 0 : return nullptr;
2519 : }
2520 25 : if (nCode == 0)
2521 25 : poDS->UnreadValue();
2522 :
2523 : // do we want Z-coordinates?
2524 25 : const bool bWantZ =
2525 25 : dfZ1 != 0.0 || dfZ2 != 0.0 || dfZ3 != 0.0 || dfZ4 != 0.0;
2526 :
2527 : // check how many unique corners we have
2528 250 : OGRPoint oCorners[4];
2529 25 : oCorners[0].setX(dfX1);
2530 25 : oCorners[0].setY(dfY1);
2531 25 : if (bWantZ)
2532 3 : oCorners[0].setZ(dfZ1);
2533 25 : oCorners[1].setX(dfX2);
2534 25 : oCorners[1].setY(dfY2);
2535 25 : if (bWantZ)
2536 3 : oCorners[1].setZ(dfZ2);
2537 25 : oCorners[2].setX(dfX3);
2538 25 : oCorners[2].setY(dfY3);
2539 25 : if (bWantZ)
2540 3 : oCorners[2].setZ(dfZ3);
2541 25 : oCorners[3].setX(dfX4);
2542 25 : oCorners[3].setY(dfY4);
2543 25 : if (bWantZ)
2544 3 : oCorners[3].setZ(dfZ4);
2545 :
2546 25 : std::sort(&oCorners[0], &oCorners[4], PointXAxisComparer);
2547 : int nCornerCount = static_cast<int>(
2548 25 : std::unique(&oCorners[0], &oCorners[4], PointXYZEqualityComparer) -
2549 25 : &oCorners[0]);
2550 25 : if (nCornerCount < 1)
2551 : {
2552 0 : DXF_LAYER_READER_ERROR();
2553 0 : return nullptr;
2554 : }
2555 :
2556 25 : std::unique_ptr<OGRGeometry> poFinalGeom;
2557 :
2558 : // what kind of object do we need?
2559 25 : if (nCornerCount == 1)
2560 : {
2561 1 : poFinalGeom.reset(oCorners[0].clone());
2562 :
2563 1 : PrepareLineStyle(poFeature.get());
2564 : }
2565 24 : else if (nCornerCount == 2)
2566 : {
2567 2 : auto poLS = std::make_unique<OGRLineString>();
2568 1 : poLS->setPoint(0, &oCorners[0]);
2569 1 : poLS->setPoint(1, &oCorners[1]);
2570 1 : poFinalGeom.reset(poLS.release());
2571 :
2572 1 : PrepareLineStyle(poFeature.get());
2573 : }
2574 : else
2575 : {
2576 : // SOLID vertices seem to be joined in the order 1-2-4-3-1.
2577 : // See trac ticket #7089
2578 23 : OGRLinearRing *poLinearRing = new OGRLinearRing();
2579 23 : int iIndex = 0;
2580 23 : poLinearRing->setPoint(iIndex++, dfX1, dfY1, dfZ1);
2581 23 : if (dfX1 != dfX2 || dfY1 != dfY2 || dfZ1 != dfZ2)
2582 23 : poLinearRing->setPoint(iIndex++, dfX2, dfY2, dfZ2);
2583 23 : if (dfX2 != dfX4 || dfY2 != dfY4 || dfZ2 != dfZ4)
2584 23 : poLinearRing->setPoint(iIndex++, dfX4, dfY4, dfZ4);
2585 23 : if (dfX4 != dfX3 || dfY4 != dfY3 || dfZ4 != dfZ3)
2586 17 : poLinearRing->setPoint(iIndex++, dfX3, dfY3, dfZ3);
2587 23 : poLinearRing->closeRings();
2588 :
2589 23 : if (!bWantZ)
2590 20 : poLinearRing->flattenTo2D();
2591 :
2592 46 : auto poPoly = std::make_unique<OGRPolygon>();
2593 23 : poPoly->addRingDirectly(poLinearRing);
2594 23 : poFinalGeom.reset(poPoly.release());
2595 :
2596 23 : PrepareBrushStyle(poFeature.get());
2597 : }
2598 :
2599 25 : poFeature->ApplyOCSTransformer(poFinalGeom.get());
2600 25 : poFeature->SetGeometryDirectly(poFinalGeom.release());
2601 :
2602 25 : return poFeature.release();
2603 : }
2604 :
2605 : /************************************************************************/
2606 : /* TranslateASMEntity() */
2607 : /* */
2608 : /* Translate Autodesk ShapeManager entities (3DSOLID, REGION, */
2609 : /* SURFACE), also known as ACIS entities. */
2610 : /************************************************************************/
2611 :
2612 2 : OGRDXFFeature *OGRDXFLayer::TranslateASMEntity()
2613 :
2614 : {
2615 : char szLineBuf[257];
2616 2 : int nCode = 0;
2617 4 : auto poFeature = std::make_unique<OGRDXFFeature>(poFeatureDefn);
2618 :
2619 23 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
2620 : {
2621 21 : TranslateGenericProperty(poFeature.get(), nCode, szLineBuf);
2622 : }
2623 :
2624 2 : if (nCode < 0)
2625 : {
2626 0 : DXF_LAYER_READER_ERROR();
2627 0 : return nullptr;
2628 : }
2629 :
2630 2 : poDS->UnreadValue();
2631 :
2632 2 : const char *pszEntityHandle = poFeature->GetFieldAsString("EntityHandle");
2633 :
2634 : // The actual data is located at the end of the DXF file (sigh).
2635 : const GByte *pabyBinaryData;
2636 : size_t nDataLength =
2637 2 : poDS->GetEntryFromAcDsDataSection(pszEntityHandle, &pabyBinaryData);
2638 2 : if (!pabyBinaryData)
2639 : {
2640 0 : CPLError(CE_Warning, CPLE_AppDefined,
2641 : "ACDSRECORD data for entity %s was not found.",
2642 : pszEntityHandle);
2643 0 : return poFeature.release();
2644 : }
2645 :
2646 : // Return a feature with no geometry but with one very interesting field.
2647 2 : poFeature->SetField(poFeatureDefn->GetFieldIndex("ASMData"),
2648 : static_cast<int>(nDataLength), pabyBinaryData);
2649 :
2650 : // Set up an affine transformation matrix so the user will be able to
2651 : // transform the resulting 3D geometry
2652 2 : poFeature->poASMTransform = std::make_unique<OGRDXFAffineTransform>();
2653 :
2654 2 : poFeature->poASMTransform->SetField(poFeature.get(), "ASMTransform");
2655 :
2656 : #ifdef notdef
2657 : FILE *fp;
2658 : fopen_s(&fp,
2659 : CPLString().Printf("C:\\Projects\\output.sab", pszEntityHandle),
2660 : "wb");
2661 :
2662 : if (fp != nullptr)
2663 : {
2664 : fprintf(fp, "Entity handle: %s\r\n\r\n", pszEntityHandle);
2665 : fwrite(pabyBinaryData, sizeof(GByte), nDataLength, fp);
2666 : if (ferror(fp) != 0)
2667 : {
2668 : fputs("Error writing .sab file", stderr);
2669 : }
2670 : fclose(fp);
2671 : }
2672 : #endif
2673 :
2674 2 : PrepareBrushStyle(poFeature.get());
2675 :
2676 2 : return poFeature.release();
2677 : }
2678 :
2679 : /************************************************************************/
2680 : /* SimplifyBlockGeometry() */
2681 : /************************************************************************/
2682 :
2683 : OGRGeometry *
2684 75 : OGRDXFLayer::SimplifyBlockGeometry(OGRGeometryCollection *poCollection)
2685 : {
2686 : /* -------------------------------------------------------------------- */
2687 : /* If there is only one geometry in the collection, just return */
2688 : /* it. */
2689 : /* -------------------------------------------------------------------- */
2690 75 : if (poCollection->getNumGeometries() == 1)
2691 : {
2692 64 : OGRGeometry *poReturn = poCollection->getGeometryRef(0);
2693 64 : poCollection->removeGeometry(0, FALSE);
2694 64 : delete poCollection;
2695 64 : return poReturn;
2696 : }
2697 :
2698 : /* -------------------------------------------------------------------- */
2699 : /* Convert to polygon, multipolygon, multilinestring or multipoint */
2700 : /* -------------------------------------------------------------------- */
2701 :
2702 : OGRwkbGeometryType eType =
2703 11 : wkbFlatten(poCollection->getGeometryRef(0)->getGeometryType());
2704 : int i;
2705 37 : for (i = 1; i < poCollection->getNumGeometries(); i++)
2706 : {
2707 27 : if (wkbFlatten(poCollection->getGeometryRef(i)->getGeometryType()) !=
2708 : eType)
2709 : {
2710 1 : eType = wkbUnknown;
2711 1 : break;
2712 : }
2713 : }
2714 11 : if (eType == wkbPoint || eType == wkbLineString)
2715 : {
2716 : OGRGeometryCollection *poNewColl;
2717 9 : if (eType == wkbPoint)
2718 0 : poNewColl = new OGRMultiPoint();
2719 : else
2720 9 : poNewColl = new OGRMultiLineString();
2721 43 : while (poCollection->getNumGeometries() > 0)
2722 : {
2723 34 : OGRGeometry *poGeom = poCollection->getGeometryRef(0);
2724 34 : poCollection->removeGeometry(0, FALSE);
2725 34 : poNewColl->addGeometryDirectly(poGeom);
2726 : }
2727 9 : delete poCollection;
2728 9 : return poNewColl;
2729 : }
2730 2 : else if (eType == wkbPolygon)
2731 : {
2732 2 : std::vector<OGRGeometry *> aosPolygons;
2733 3 : while (poCollection->getNumGeometries() > 0)
2734 : {
2735 2 : OGRGeometry *poGeom = poCollection->getGeometryRef(0);
2736 2 : poCollection->removeGeometry(0, FALSE);
2737 2 : if (!aosPolygons.empty() && aosPolygons[0]->Equals(poGeom))
2738 : {
2739 : // Avoids a performance issue as in
2740 : // https://bugs.chromium.org/p/oss-fuzz/issues/detail?id=8067
2741 0 : delete poGeom;
2742 : }
2743 : else
2744 : {
2745 2 : aosPolygons.push_back(poGeom);
2746 : }
2747 : }
2748 1 : delete poCollection;
2749 : int bIsValidGeometry;
2750 1 : return OGRGeometryFactory::organizePolygons(&aosPolygons[0],
2751 1 : (int)aosPolygons.size(),
2752 1 : &bIsValidGeometry, nullptr);
2753 : }
2754 :
2755 1 : return poCollection;
2756 : }
2757 :
2758 : /************************************************************************/
2759 : /* InsertBlockReference() */
2760 : /* */
2761 : /* Returns a point geometry located at the block's insertion */
2762 : /* point. */
2763 : /************************************************************************/
2764 : OGRDXFFeature *
2765 801 : OGRDXFLayer::InsertBlockReference(const CPLString &osBlockName,
2766 : const OGRDXFInsertTransformer &oTransformer,
2767 : OGRDXFFeature *const poFeature)
2768 : {
2769 : // Store the block's properties in the special DXF-specific members
2770 : // on the feature object
2771 801 : poFeature->bIsBlockReference = true;
2772 801 : poFeature->osBlockName = osBlockName;
2773 801 : poFeature->dfBlockAngle = oTransformer.dfAngle * 180 / M_PI;
2774 1602 : poFeature->oBlockScale = DXFTriple(
2775 801 : oTransformer.dfXScale, oTransformer.dfYScale, oTransformer.dfZScale);
2776 1602 : poFeature->oOriginalCoords = DXFTriple(
2777 801 : oTransformer.dfXOffset, oTransformer.dfYOffset, oTransformer.dfZOffset);
2778 :
2779 : // Only if DXF_INLINE_BLOCKS is false should we ever need to expose these
2780 : // to the end user as fields.
2781 801 : if (poFeature->GetFieldIndex("BlockName") != -1)
2782 : {
2783 13 : poFeature->SetField("BlockName", poFeature->osBlockName);
2784 13 : poFeature->SetField("BlockAngle", poFeature->dfBlockAngle);
2785 13 : poFeature->SetField("BlockScale", 3, &(poFeature->oBlockScale.dfX));
2786 13 : poFeature->SetField("BlockOCSNormal", 3, &(poFeature->oOCS.dfX));
2787 13 : poFeature->SetField("BlockOCSCoords", 3,
2788 13 : &(poFeature->oOriginalCoords.dfX));
2789 : }
2790 :
2791 : // For convenience to the end user, the point geometry will be located
2792 : // at the WCS coordinates of the insertion point.
2793 : OGRPoint *poInsertionPoint = new OGRPoint(
2794 801 : oTransformer.dfXOffset, oTransformer.dfYOffset, oTransformer.dfZOffset);
2795 :
2796 801 : poFeature->ApplyOCSTransformer(poInsertionPoint);
2797 801 : poFeature->SetGeometryDirectly(poInsertionPoint);
2798 :
2799 801 : return poFeature;
2800 : }
2801 :
2802 : /************************************************************************/
2803 : /* InsertBlockInline() */
2804 : /* */
2805 : /* Inserts the given block at the location specified by the given */
2806 : /* transformer. Returns poFeature, or NULL if all features on */
2807 : /* the block have been pushed to the extra feature queue. */
2808 : /* If poFeature is not returned, it is deleted. */
2809 : /* Throws std::invalid_argument if the requested block */
2810 : /* doesn't exist. */
2811 : /* */
2812 : /* - poFeature: The feature to use as a template. This feature's */
2813 : /* OCS will be applied to the block. */
2814 : /* - bInlineRecursively: If true, INSERTs within this block */
2815 : /* will be recursively inserted. Otherwise, they will be */
2816 : /* represented as a point geometry using InsertBlockReference. */
2817 : /* - bMergeGeometry: If true, all features in the block, */
2818 : /* apart from text features, are merged into a */
2819 : /* GeometryCollection which is returned by the function. */
2820 : /************************************************************************/
2821 :
2822 1224 : OGRDXFFeature *OGRDXFLayer::InsertBlockInline(
2823 : GUInt32 nInitialErrorCounter, const CPLString &osBlockName,
2824 : OGRDXFInsertTransformer oTransformer, OGRDXFFeature *const poFeature,
2825 : OGRDXFFeatureQueue &apoExtraFeatures, const bool bInlineRecursively,
2826 : const bool bMergeGeometry)
2827 : {
2828 : /* -------------------------------------------------------------------- */
2829 : /* Set up protection against excessive recursion on this layer. */
2830 : /* -------------------------------------------------------------------- */
2831 1224 : if (!poDS->PushBlockInsertion(osBlockName))
2832 : {
2833 1003 : delete poFeature;
2834 1003 : return nullptr;
2835 : }
2836 :
2837 : /* -------------------------------------------------------------------- */
2838 : /* Transform the insertion point from OCS into */
2839 : /* world coordinates. */
2840 : /* -------------------------------------------------------------------- */
2841 : OGRPoint oInsertionPoint(oTransformer.dfXOffset, oTransformer.dfYOffset,
2842 442 : oTransformer.dfZOffset);
2843 :
2844 221 : poFeature->ApplyOCSTransformer(&oInsertionPoint);
2845 :
2846 221 : oTransformer.dfXOffset = oInsertionPoint.getX();
2847 221 : oTransformer.dfYOffset = oInsertionPoint.getY();
2848 221 : oTransformer.dfZOffset = oInsertionPoint.getZ();
2849 :
2850 : /* -------------------------------------------------------------------- */
2851 : /* Lookup the block. */
2852 : /* -------------------------------------------------------------------- */
2853 221 : DXFBlockDefinition *poBlock = poDS->LookupBlock(osBlockName);
2854 :
2855 221 : if (poBlock == nullptr)
2856 : {
2857 : // CPLDebug( "DXF", "Attempt to insert missing block %s", osBlockName );
2858 9 : poDS->PopBlockInsertion();
2859 9 : throw std::invalid_argument("osBlockName");
2860 : }
2861 :
2862 : /* -------------------------------------------------------------------- */
2863 : /* If we have complete features associated with the block, push */
2864 : /* them on the pending feature stack copying over key override */
2865 : /* information. */
2866 : /* */
2867 : /* If bMergeGeometry is true, we merge the features */
2868 : /* (except text) into a single GeometryCollection. */
2869 : /* -------------------------------------------------------------------- */
2870 212 : OGRGeometryCollection *poMergedGeometry = nullptr;
2871 212 : if (bMergeGeometry)
2872 89 : poMergedGeometry = new OGRGeometryCollection();
2873 :
2874 424 : OGRDXFFeatureQueue apoInnerExtraFeatures;
2875 :
2876 1823 : for (unsigned int iSubFeat = 0; iSubFeat < poBlock->apoFeatures.size();
2877 : iSubFeat++)
2878 : {
2879 : OGRDXFFeature *poSubFeature =
2880 1613 : poBlock->apoFeatures[iSubFeat]->CloneDXFFeature();
2881 :
2882 : // If the template feature is in PaperSpace, set this on the
2883 : // subfeature too
2884 1613 : if (poFeature->GetFieldAsInteger("PaperSpace"))
2885 48 : poSubFeature->SetField("PaperSpace", 1);
2886 :
2887 : // Does this feature represent a block reference? If so,
2888 : // insert that block
2889 1613 : if (bInlineRecursively && poSubFeature->IsBlockReference())
2890 : {
2891 : // Unpack the transformation data stored in fields of this
2892 : // feature
2893 0 : OGRDXFInsertTransformer oInnerTransformer;
2894 1073 : oInnerTransformer.dfXOffset = poSubFeature->oOriginalCoords.dfX;
2895 1073 : oInnerTransformer.dfYOffset = poSubFeature->oOriginalCoords.dfY;
2896 1073 : oInnerTransformer.dfZOffset = poSubFeature->oOriginalCoords.dfZ;
2897 1073 : oInnerTransformer.dfAngle = poSubFeature->dfBlockAngle * M_PI / 180;
2898 1073 : oInnerTransformer.dfXScale = poSubFeature->oBlockScale.dfX;
2899 1073 : oInnerTransformer.dfYScale = poSubFeature->oBlockScale.dfY;
2900 1073 : oInnerTransformer.dfZScale = poSubFeature->oBlockScale.dfZ;
2901 :
2902 1073 : poSubFeature->bIsBlockReference = false;
2903 :
2904 : // Keep a reference to the attributes that need to be inserted
2905 : std::vector<std::unique_ptr<OGRDXFFeature>> apoInnerAttribFeatures =
2906 1073 : std::move(poSubFeature->apoAttribFeatures);
2907 :
2908 : // Insert this block recursively
2909 : try
2910 : {
2911 2146 : poSubFeature = InsertBlockInline(
2912 1073 : nInitialErrorCounter, poSubFeature->osBlockName,
2913 1073 : std::move(oInnerTransformer), poSubFeature,
2914 : apoInnerExtraFeatures, true, bMergeGeometry);
2915 : }
2916 0 : catch (const std::invalid_argument &)
2917 : {
2918 : // Block doesn't exist. Skip it and keep going
2919 0 : delete poSubFeature;
2920 0 : if (CPLGetErrorCounter() > nInitialErrorCounter + 1000)
2921 : {
2922 0 : break;
2923 : }
2924 0 : continue;
2925 : }
2926 :
2927 1073 : if (!poSubFeature)
2928 : {
2929 1067 : if (CPLGetErrorCounter() > nInitialErrorCounter + 1000)
2930 : {
2931 2 : break;
2932 : }
2933 :
2934 : // Append the attribute features to the pending feature stack
2935 1067 : for (auto &poAttribFeature : apoInnerAttribFeatures)
2936 : {
2937 : // Clear the attribute tag so the feature doesn't get mistaken
2938 : // for an ATTDEF and skipped
2939 2 : poAttribFeature->osAttributeTag = "";
2940 :
2941 2 : apoInnerExtraFeatures.push(poAttribFeature.release());
2942 : }
2943 :
2944 1065 : if (apoInnerExtraFeatures.empty())
2945 : {
2946 : // Block is empty and has no attributes. Skip it and keep going
2947 1002 : continue;
2948 : }
2949 : else
2950 : {
2951 : // Load up the first extra feature ready for
2952 : // transformation
2953 63 : poSubFeature = apoInnerExtraFeatures.front();
2954 63 : apoInnerExtraFeatures.pop();
2955 : }
2956 : }
2957 : }
2958 :
2959 : // Go through the current feature and any extra features generated
2960 : // by the recursive insert, and apply transformations
2961 : while (true)
2962 : {
2963 696 : OGRGeometry *poSubFeatGeom = poSubFeature->GetGeometryRef();
2964 696 : if (poSubFeatGeom != nullptr)
2965 : {
2966 : // Rotation and scaling first
2967 : OGRDXFInsertTransformer oInnerTrans =
2968 1386 : oTransformer.GetRotateScaleTransformer();
2969 693 : poSubFeatGeom->transform(&oInnerTrans);
2970 :
2971 : // Then the OCS to WCS transformation
2972 693 : poFeature->ApplyOCSTransformer(poSubFeatGeom);
2973 :
2974 : // Offset translation last
2975 693 : oInnerTrans = oTransformer.GetOffsetTransformer();
2976 693 : poSubFeatGeom->transform(&oInnerTrans);
2977 : }
2978 : // Transform the specially-stored data for ASM entities
2979 3 : else if (poSubFeature->poASMTransform)
2980 : {
2981 : // Rotation and scaling first
2982 : OGRDXFInsertTransformer oInnerTrans =
2983 6 : oTransformer.GetRotateScaleTransformer();
2984 3 : poSubFeature->poASMTransform->ComposeWith(oInnerTrans);
2985 :
2986 : // Then the OCS to WCS transformation
2987 3 : poFeature->ApplyOCSTransformer(
2988 : poSubFeature->poASMTransform.get());
2989 :
2990 : // Offset translation last
2991 3 : oInnerTrans = oTransformer.GetOffsetTransformer();
2992 3 : poSubFeature->poASMTransform->ComposeWith(oInnerTrans);
2993 :
2994 3 : poSubFeature->poASMTransform->SetField(poSubFeature,
2995 : "ASMTransform");
2996 : }
2997 :
2998 : // If we are merging features, and this is not text or a block
2999 : // reference, merge it into the GeometryCollection
3000 876 : if (bMergeGeometry &&
3001 180 : (poSubFeature->GetStyleString() == nullptr ||
3002 163 : strstr(poSubFeature->GetStyleString(), "LABEL") == nullptr) &&
3003 981 : !poSubFeature->IsBlockReference() &&
3004 105 : poSubFeature->GetGeometryRef())
3005 : {
3006 102 : poMergedGeometry->addGeometryDirectly(
3007 102 : poSubFeature->StealGeometry());
3008 102 : delete poSubFeature;
3009 : }
3010 : // Import all other features, except ATTDEFs when inlining
3011 : // recursively
3012 594 : else if (!bInlineRecursively || poSubFeature->osAttributeTag == "")
3013 : {
3014 : // If the subfeature is on layer 0, this is a special case: the
3015 : // subfeature should take on the style properties of the layer
3016 : // the block is being inserted onto.
3017 : // But don't do this if we are inserting onto a Blocks layer
3018 : // (that is, the owning feature has no layer).
3019 773 : if (EQUAL(poSubFeature->GetFieldAsString("Layer"), "0") &&
3020 211 : !EQUAL(poFeature->GetFieldAsString("Layer"), ""))
3021 : {
3022 180 : poSubFeature->SetField(
3023 : "Layer", poFeature->GetFieldAsString("Layer"));
3024 : }
3025 :
3026 : // Update the style string to replace ByBlock and ByLayer
3027 : // values.
3028 562 : PrepareFeatureStyle(poSubFeature, poFeature);
3029 :
3030 562 : ACAdjustText(oTransformer.dfAngle * 180 / M_PI,
3031 : oTransformer.dfXScale, oTransformer.dfYScale,
3032 : poSubFeature);
3033 :
3034 562 : if (!EQUAL(poFeature->GetFieldAsString("EntityHandle"), ""))
3035 : {
3036 525 : poSubFeature->SetField(
3037 : "EntityHandle",
3038 : poFeature->GetFieldAsString("EntityHandle"));
3039 : }
3040 :
3041 562 : apoExtraFeatures.push(poSubFeature);
3042 : }
3043 : else
3044 : {
3045 32 : delete poSubFeature;
3046 : }
3047 :
3048 696 : if (apoInnerExtraFeatures.empty())
3049 : {
3050 609 : break;
3051 : }
3052 : else
3053 : {
3054 87 : poSubFeature = apoInnerExtraFeatures.front();
3055 87 : apoInnerExtraFeatures.pop();
3056 : }
3057 87 : }
3058 : }
3059 :
3060 214 : while (!apoInnerExtraFeatures.empty())
3061 : {
3062 2 : auto poFeatureToDelete = apoInnerExtraFeatures.front();
3063 2 : apoInnerExtraFeatures.pop();
3064 2 : delete poFeatureToDelete;
3065 : }
3066 :
3067 212 : poDS->PopBlockInsertion();
3068 :
3069 : /* -------------------------------------------------------------------- */
3070 : /* Return the merged geometry if applicable. Otherwise */
3071 : /* return NULL and let the machinery find the rest of the */
3072 : /* features in the pending feature stack. */
3073 : /* -------------------------------------------------------------------- */
3074 212 : if (bMergeGeometry)
3075 : {
3076 89 : if (poMergedGeometry->getNumGeometries() == 0)
3077 : {
3078 14 : delete poMergedGeometry;
3079 : }
3080 : else
3081 : {
3082 75 : poFeature->SetGeometryDirectly(
3083 : SimplifyBlockGeometry(poMergedGeometry));
3084 :
3085 75 : PrepareLineStyle(poFeature);
3086 75 : return poFeature;
3087 : }
3088 : }
3089 :
3090 137 : delete poFeature;
3091 137 : return nullptr;
3092 : }
3093 :
3094 : /************************************************************************/
3095 : /* TranslateINSERT() */
3096 : /************************************************************************/
3097 :
3098 130 : bool OGRDXFLayer::TranslateINSERT()
3099 :
3100 : {
3101 : char szLineBuf[257];
3102 130 : int nCode = 0;
3103 :
3104 130 : m_oInsertState.m_poTemplateFeature.reset(new OGRDXFFeature(poFeatureDefn));
3105 130 : m_oInsertState.m_oTransformer = OGRDXFInsertTransformer();
3106 130 : m_oInsertState.m_osBlockName.clear();
3107 130 : m_oInsertState.m_nColumnCount = 1;
3108 130 : m_oInsertState.m_nRowCount = 1;
3109 130 : m_oInsertState.m_iCurCol = 0;
3110 130 : m_oInsertState.m_iCurRow = 0;
3111 130 : m_oInsertState.m_dfColumnSpacing = 0.0;
3112 130 : m_oInsertState.m_dfRowSpacing = 0.0;
3113 :
3114 130 : bool bHasAttribs = false;
3115 130 : m_oInsertState.m_apoAttribs.clear();
3116 130 : m_oInsertState.m_aosAttribs.Clear();
3117 :
3118 : /* -------------------------------------------------------------------- */
3119 : /* Process values. */
3120 : /* -------------------------------------------------------------------- */
3121 1368 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
3122 : {
3123 1238 : switch (nCode)
3124 : {
3125 128 : case 10:
3126 128 : m_oInsertState.m_oTransformer.dfXOffset = CPLAtof(szLineBuf);
3127 128 : break;
3128 :
3129 128 : case 20:
3130 128 : m_oInsertState.m_oTransformer.dfYOffset = CPLAtof(szLineBuf);
3131 128 : break;
3132 :
3133 121 : case 30:
3134 121 : m_oInsertState.m_oTransformer.dfZOffset = CPLAtof(szLineBuf);
3135 121 : break;
3136 :
3137 28 : case 41:
3138 28 : m_oInsertState.m_oTransformer.dfXScale = CPLAtof(szLineBuf);
3139 28 : break;
3140 :
3141 31 : case 42:
3142 31 : m_oInsertState.m_oTransformer.dfYScale = CPLAtof(szLineBuf);
3143 31 : break;
3144 :
3145 16 : case 43:
3146 16 : m_oInsertState.m_oTransformer.dfZScale = CPLAtof(szLineBuf);
3147 16 : break;
3148 :
3149 5 : case 44:
3150 5 : m_oInsertState.m_dfColumnSpacing = CPLAtof(szLineBuf);
3151 5 : break;
3152 :
3153 5 : case 45:
3154 5 : m_oInsertState.m_dfRowSpacing = CPLAtof(szLineBuf);
3155 5 : break;
3156 :
3157 23 : case 50:
3158 : // We want to transform this to radians.
3159 : // It is apparently always in degrees regardless of $AUNITS
3160 23 : m_oInsertState.m_oTransformer.dfAngle =
3161 23 : CPLAtof(szLineBuf) * M_PI / 180.0;
3162 23 : break;
3163 :
3164 13 : case 66:
3165 13 : bHasAttribs = atoi(szLineBuf) == 1;
3166 13 : break;
3167 :
3168 1 : case 70:
3169 1 : m_oInsertState.m_nColumnCount = atoi(szLineBuf);
3170 1 : if (m_oInsertState.m_nColumnCount <= 0)
3171 : {
3172 0 : DXF_LAYER_READER_ERROR();
3173 0 : m_oInsertState.m_nRowCount = 0;
3174 0 : m_oInsertState.m_nColumnCount = 0;
3175 0 : return false;
3176 : }
3177 1 : break;
3178 :
3179 3 : case 71:
3180 3 : m_oInsertState.m_nRowCount = atoi(szLineBuf);
3181 3 : if (m_oInsertState.m_nRowCount <= 0)
3182 : {
3183 0 : DXF_LAYER_READER_ERROR();
3184 0 : m_oInsertState.m_nRowCount = 0;
3185 0 : m_oInsertState.m_nColumnCount = 0;
3186 0 : return false;
3187 : }
3188 3 : break;
3189 :
3190 128 : case 2:
3191 128 : m_oInsertState.m_osBlockName = szLineBuf;
3192 128 : break;
3193 :
3194 608 : default:
3195 608 : TranslateGenericProperty(
3196 : m_oInsertState.m_poTemplateFeature.get(), nCode, szLineBuf);
3197 608 : break;
3198 : }
3199 : }
3200 130 : if (nCode < 0)
3201 : {
3202 0 : DXF_LAYER_READER_ERROR();
3203 0 : m_oInsertState.m_nRowCount = 0;
3204 0 : m_oInsertState.m_nColumnCount = 0;
3205 0 : return false;
3206 : }
3207 :
3208 : /* -------------------------------------------------------------------- */
3209 : /* Process any attribute entities. */
3210 : /* -------------------------------------------------------------------- */
3211 :
3212 130 : if (bHasAttribs)
3213 : {
3214 40 : while (nCode == 0 && !EQUAL(szLineBuf, "SEQEND"))
3215 : {
3216 27 : if (!EQUAL(szLineBuf, "ATTRIB"))
3217 : {
3218 0 : DXF_LAYER_READER_ERROR();
3219 0 : m_oInsertState.m_nRowCount = 0;
3220 0 : m_oInsertState.m_nColumnCount = 0;
3221 0 : return false;
3222 : }
3223 :
3224 : auto poAttribFeature =
3225 27 : std::unique_ptr<OGRDXFFeature>(TranslateTEXT(true));
3226 :
3227 27 : if (poAttribFeature && poAttribFeature->osAttributeTag != "")
3228 : {
3229 : m_oInsertState.m_apoAttribs.emplace_back(
3230 27 : std::move(poAttribFeature));
3231 : }
3232 :
3233 27 : nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf));
3234 : }
3235 : }
3236 117 : else if (nCode == 0)
3237 : {
3238 117 : poDS->UnreadValue();
3239 : }
3240 :
3241 : /* -------------------------------------------------------------------- */
3242 : /* Prepare a string list of the attributes and their text values */
3243 : /* as space-separated entries, to be stored in the */
3244 : /* BlockAttributes field if we are not inlining blocks. */
3245 : /* -------------------------------------------------------------------- */
3246 :
3247 134 : if (!poDS->InlineBlocks() && bHasAttribs &&
3248 4 : poFeatureDefn->GetFieldIndex("BlockAttributes") != -1)
3249 : {
3250 10 : for (const auto &poAttr : m_oInsertState.m_apoAttribs)
3251 : {
3252 14 : CPLString osAttribString = poAttr->osAttributeTag;
3253 7 : osAttribString += " ";
3254 7 : osAttribString += poAttr->GetFieldAsString("Text");
3255 :
3256 7 : m_oInsertState.m_aosAttribs.AddString(osAttribString);
3257 : }
3258 : }
3259 :
3260 130 : return true;
3261 : }
3262 :
3263 : /************************************************************************/
3264 : /* GenerateINSERTFeatures() */
3265 : /************************************************************************/
3266 :
3267 897 : bool OGRDXFLayer::GenerateINSERTFeatures()
3268 : {
3269 : OGRDXFFeature *poFeature =
3270 897 : m_oInsertState.m_poTemplateFeature->CloneDXFFeature();
3271 :
3272 897 : const double dfExtraXOffset =
3273 897 : m_oInsertState.m_iCurCol * m_oInsertState.m_dfColumnSpacing *
3274 897 : cos(m_oInsertState.m_oTransformer.dfAngle) +
3275 897 : m_oInsertState.m_iCurRow * m_oInsertState.m_dfRowSpacing *
3276 897 : -sin(m_oInsertState.m_oTransformer.dfAngle);
3277 897 : const double dfExtraYOffset =
3278 897 : m_oInsertState.m_iCurCol * m_oInsertState.m_dfColumnSpacing *
3279 897 : sin(m_oInsertState.m_oTransformer.dfAngle) +
3280 897 : m_oInsertState.m_iCurRow * m_oInsertState.m_dfRowSpacing *
3281 897 : cos(m_oInsertState.m_oTransformer.dfAngle);
3282 :
3283 1794 : OGRDXFInsertTransformer oTransformer(m_oInsertState.m_oTransformer);
3284 897 : oTransformer.dfXOffset += dfExtraXOffset;
3285 897 : oTransformer.dfYOffset += dfExtraYOffset;
3286 :
3287 : // If we are not inlining blocks, just insert a point that refers
3288 : // to this block
3289 897 : if (!poDS->InlineBlocks())
3290 : {
3291 801 : poFeature = InsertBlockReference(m_oInsertState.m_osBlockName,
3292 : oTransformer, poFeature);
3293 :
3294 801 : auto papszAttribs = m_oInsertState.m_aosAttribs.List();
3295 801 : if (papszAttribs)
3296 3 : poFeature->SetField("BlockAttributes", papszAttribs);
3297 :
3298 801 : poFeature->apoAttribFeatures = std::move(m_oInsertState.m_apoAttribs);
3299 :
3300 801 : apoPendingFeatures.push(poFeature);
3301 : }
3302 : // Otherwise, try inlining the contents of this block
3303 : else
3304 : {
3305 96 : OGRDXFFeatureQueue apoExtraFeatures;
3306 : try
3307 : {
3308 192 : poFeature = InsertBlockInline(
3309 96 : CPLGetErrorCounter(), m_oInsertState.m_osBlockName,
3310 96 : std::move(oTransformer), poFeature, apoExtraFeatures, true,
3311 96 : poDS->ShouldMergeBlockGeometries());
3312 : }
3313 0 : catch (const std::invalid_argument &)
3314 : {
3315 : // Block doesn't exist
3316 0 : CPLError(CE_Warning, CPLE_AppDefined, "Block %s does not exist",
3317 : m_oInsertState.m_osBlockName.c_str());
3318 0 : delete poFeature;
3319 0 : return false;
3320 : }
3321 :
3322 96 : if (poFeature)
3323 52 : apoPendingFeatures.push(poFeature);
3324 :
3325 303 : while (!apoExtraFeatures.empty())
3326 : {
3327 207 : apoPendingFeatures.push(apoExtraFeatures.front());
3328 207 : apoExtraFeatures.pop();
3329 : }
3330 :
3331 : // Append the attribute features to the pending feature stack
3332 96 : if (!m_oInsertState.m_apoAttribs.empty())
3333 : {
3334 16 : OGRDXFInsertTransformer oAttribTransformer;
3335 16 : oAttribTransformer.dfXOffset = dfExtraXOffset;
3336 16 : oAttribTransformer.dfYOffset = dfExtraYOffset;
3337 :
3338 42 : for (const auto &poAttr : m_oInsertState.m_apoAttribs)
3339 : {
3340 26 : OGRDXFFeature *poAttribFeature = poAttr->CloneDXFFeature();
3341 :
3342 26 : if (poAttribFeature->GetGeometryRef())
3343 : {
3344 26 : poAttribFeature->GetGeometryRef()->transform(
3345 26 : &oAttribTransformer);
3346 : }
3347 :
3348 26 : apoPendingFeatures.push(poAttribFeature);
3349 : }
3350 : }
3351 : }
3352 897 : return true;
3353 : }
3354 :
3355 : /************************************************************************/
3356 : /* GetNextUnfilteredFeature() */
3357 : /************************************************************************/
3358 :
3359 1905 : OGRDXFFeature *OGRDXFLayer::GetNextUnfilteredFeature()
3360 :
3361 : {
3362 1905 : OGRDXFFeature *poFeature = nullptr;
3363 3653 : while (poFeature == nullptr)
3364 : {
3365 : /* --------------------------------------------------------------------
3366 : */
3367 : /* If we have pending features, return one of them. */
3368 : /* --------------------------------------------------------------------
3369 : */
3370 3121 : if (!apoPendingFeatures.empty())
3371 : {
3372 1283 : poFeature = apoPendingFeatures.front();
3373 1283 : apoPendingFeatures.pop();
3374 :
3375 1283 : poFeature->SetFID(iNextFID++);
3376 1373 : return poFeature;
3377 : }
3378 :
3379 : /* --------------------------------------------------------------------
3380 : */
3381 : /* Emit INSERT features. */
3382 : /* --------------------------------------------------------------------
3383 : */
3384 1838 : if (m_oInsertState.m_iCurRow < m_oInsertState.m_nRowCount)
3385 : {
3386 998 : if (m_oInsertState.m_iCurCol == m_oInsertState.m_nColumnCount)
3387 : {
3388 862 : m_oInsertState.m_iCurRow++;
3389 862 : m_oInsertState.m_iCurCol = 0;
3390 862 : if (m_oInsertState.m_iCurRow == m_oInsertState.m_nRowCount)
3391 : {
3392 101 : m_oInsertState.m_nRowCount = 0;
3393 101 : m_oInsertState.m_nColumnCount = 0;
3394 998 : continue;
3395 : }
3396 : }
3397 897 : if (GenerateINSERTFeatures())
3398 : {
3399 897 : m_oInsertState.m_iCurCol++;
3400 : }
3401 : else
3402 : {
3403 0 : m_oInsertState.m_nRowCount = 0;
3404 0 : m_oInsertState.m_nColumnCount = 0;
3405 : }
3406 897 : continue;
3407 : }
3408 :
3409 : // read ahead to an entity.
3410 : char szLineBuf[257];
3411 840 : int nCode = 0;
3412 960 : while ((nCode = poDS->ReadValue(szLineBuf, sizeof(szLineBuf))) > 0)
3413 : {
3414 : }
3415 840 : if (nCode < 0)
3416 : {
3417 5 : DXF_LAYER_READER_ERROR();
3418 5 : return nullptr;
3419 : }
3420 :
3421 835 : if (EQUAL(szLineBuf, "ENDSEC"))
3422 : {
3423 : // CPLDebug( "DXF", "Clean end of features at ENDSEC." );
3424 13 : poDS->UnreadValue();
3425 13 : return nullptr;
3426 : }
3427 :
3428 822 : if (EQUAL(szLineBuf, "ENDBLK"))
3429 : {
3430 : // CPLDebug( "DXF", "Clean end of block at ENDBLK." );
3431 72 : poDS->UnreadValue();
3432 72 : return nullptr;
3433 : }
3434 :
3435 : /* --------------------------------------------------------------------
3436 : */
3437 : /* Handle the entity. */
3438 : /* --------------------------------------------------------------------
3439 : */
3440 750 : if (EQUAL(szLineBuf, "POINT"))
3441 : {
3442 33 : poFeature = TranslatePOINT();
3443 : }
3444 717 : else if (EQUAL(szLineBuf, "MTEXT"))
3445 : {
3446 40 : poFeature = TranslateMTEXT();
3447 : }
3448 677 : else if (EQUAL(szLineBuf, "TEXT"))
3449 : {
3450 26 : poFeature = TranslateTEXT(false);
3451 : }
3452 651 : else if (EQUAL(szLineBuf, "ATTDEF"))
3453 : {
3454 10 : poFeature = TranslateTEXT(true);
3455 : }
3456 641 : else if (EQUAL(szLineBuf, "LINE"))
3457 : {
3458 163 : poFeature = TranslateLINE();
3459 : }
3460 478 : else if (EQUAL(szLineBuf, "POLYLINE"))
3461 : {
3462 18 : poFeature = TranslatePOLYLINE();
3463 : }
3464 460 : else if (EQUAL(szLineBuf, "LWPOLYLINE"))
3465 : {
3466 37 : poFeature = TranslateLWPOLYLINE();
3467 : }
3468 423 : else if (EQUAL(szLineBuf, "MLINE"))
3469 : {
3470 3 : poFeature = TranslateMLINE();
3471 : }
3472 420 : else if (EQUAL(szLineBuf, "CIRCLE"))
3473 : {
3474 23 : poFeature = TranslateCIRCLE();
3475 : }
3476 397 : else if (EQUAL(szLineBuf, "ELLIPSE"))
3477 : {
3478 29 : poFeature = TranslateELLIPSE();
3479 : }
3480 368 : else if (EQUAL(szLineBuf, "ARC"))
3481 : {
3482 14 : poFeature = TranslateARC();
3483 : }
3484 354 : else if (EQUAL(szLineBuf, "SPLINE") || EQUAL(szLineBuf, "HELIX"))
3485 : {
3486 25 : poFeature = TranslateSPLINE();
3487 : }
3488 329 : else if (EQUAL(szLineBuf, "3DFACE"))
3489 : {
3490 10 : poFeature = Translate3DFACE();
3491 : }
3492 319 : else if (EQUAL(szLineBuf, "INSERT"))
3493 : {
3494 130 : if (!TranslateINSERT())
3495 0 : return nullptr;
3496 : }
3497 189 : else if (EQUAL(szLineBuf, "DIMENSION"))
3498 : {
3499 30 : poFeature = TranslateDIMENSION();
3500 : }
3501 159 : else if (EQUAL(szLineBuf, "HATCH"))
3502 : {
3503 28 : poFeature = TranslateHATCH();
3504 : }
3505 131 : else if (EQUAL(szLineBuf, "SOLID") || EQUAL(szLineBuf, "TRACE"))
3506 : {
3507 25 : poFeature = TranslateSOLID();
3508 : }
3509 106 : else if (EQUAL(szLineBuf, "LEADER"))
3510 : {
3511 14 : poFeature = TranslateLEADER();
3512 : }
3513 92 : else if (EQUAL(szLineBuf, "MLEADER") || EQUAL(szLineBuf, "MULTILEADER"))
3514 : {
3515 18 : poFeature = TranslateMLEADER();
3516 : }
3517 74 : else if (EQUAL(szLineBuf, "3DSOLID") || EQUAL(szLineBuf, "BODY") ||
3518 72 : EQUAL(szLineBuf, "REGION") || EQUAL(szLineBuf, "SURFACE"))
3519 : {
3520 2 : if (poDS->In3DExtensibleMode())
3521 : {
3522 2 : poFeature = TranslateASMEntity();
3523 : }
3524 0 : else if (oIgnoredEntities.count(szLineBuf) == 0)
3525 : {
3526 0 : oIgnoredEntities.insert(szLineBuf);
3527 0 : CPLDebug("DXF", "3D mode is off; ignoring all '%s' entities.",
3528 : szLineBuf);
3529 : }
3530 : }
3531 : else
3532 : {
3533 72 : if (oIgnoredEntities.count(szLineBuf) == 0)
3534 : {
3535 13 : oIgnoredEntities.insert(szLineBuf);
3536 13 : CPLDebug("DXF", "Ignoring one or more of entity '%s'.",
3537 : szLineBuf);
3538 : }
3539 : }
3540 : }
3541 :
3542 : /* -------------------------------------------------------------------- */
3543 : /* Set FID. */
3544 : /* -------------------------------------------------------------------- */
3545 532 : poFeature->SetFID(iNextFID++);
3546 532 : m_nFeaturesRead++;
3547 :
3548 532 : return poFeature;
3549 : }
3550 :
3551 : /************************************************************************/
3552 : /* GetNextFeature() */
3553 : /************************************************************************/
3554 :
3555 824 : OGRFeature *OGRDXFLayer::GetNextFeature()
3556 :
3557 : {
3558 : while (true)
3559 : {
3560 824 : OGRFeature *poFeature = GetNextUnfilteredFeature();
3561 :
3562 824 : if (poFeature == nullptr)
3563 15 : return nullptr;
3564 :
3565 1618 : if ((m_poFilterGeom == nullptr ||
3566 1618 : FilterGeometry(poFeature->GetGeometryRef())) &&
3567 809 : (m_poAttrQuery == nullptr || m_poAttrQuery->Evaluate(poFeature)))
3568 : {
3569 809 : return poFeature;
3570 : }
3571 :
3572 0 : delete poFeature;
3573 0 : }
3574 : }
3575 :
3576 : /************************************************************************/
3577 : /* TestCapability() */
3578 : /************************************************************************/
3579 :
3580 0 : int OGRDXFLayer::TestCapability(const char *pszCap)
3581 :
3582 : {
3583 0 : if (EQUAL(pszCap, OLCStringsAsUTF8))
3584 0 : return true;
3585 0 : else if (EQUAL(pszCap, OLCZGeometries))
3586 0 : return true;
3587 0 : return false;
3588 : }
3589 :
3590 : /************************************************************************/
3591 : /* GetDataset() */
3592 : /************************************************************************/
3593 :
3594 1 : GDALDataset *OGRDXFLayer::GetDataset()
3595 : {
3596 1 : return poDS;
3597 : }
|