1
1
/*
2
2
* Copyright 2012, 2013 Thomas Schöps
3
- * Copyright 2014, 2015 Kai Pastor
3
+ * Copyright 2014-2020 Kai Pastor
4
4
*
5
5
* This file is part of OpenOrienteering.
6
6
*
27
27
#include < memory>
28
28
#include < stdexcept>
29
29
#include < type_traits>
30
+ #include < utility>
30
31
31
32
#include < QtGlobal>
32
33
#include < QDebug>
34
+ #include < QFlags>
35
+ #include < QHash>
36
+ #include < QHashFunctions>
33
37
#include < QScopedPointer>
34
38
39
+ #include < clipper.hpp>
40
+
35
41
#include " core/map.h"
36
42
#include " core/map_coord.h"
37
43
#include " core/map_part.h"
@@ -72,6 +78,128 @@ uint qHash(const IntPoint& point, uint seed)
72
78
73
79
namespace OpenOrienteering {
74
80
81
+ namespace {
82
+
83
+ using PathObjects = BooleanTool::PathObjects;
84
+
85
+ using PathCoordInfo = std::pair<const PathPart*, const PathCoord*>;
86
+ using PolyMap = QHash<ClipperLib::IntPoint, PathCoordInfo>;
87
+
88
+ /* *
89
+ * Converts a ClipperLib::PolyTree to PathObjects.
90
+ *
91
+ * @see BooleanTool::outerPolyNodeToPathObjects()
92
+ */
93
+ static void polyTreeToPathObjects (
94
+ const ClipperLib::PolyTree& tree,
95
+ PathObjects& out_objects,
96
+ const PathObject* proto,
97
+ const PolyMap& polymap );
98
+
99
+ /* *
100
+ * Converts a ClipperLib::PolyNode to PathObjects.
101
+ *
102
+ * The given ClipperLib::PolyNode must represent an outer polygon, not a hole.
103
+ *
104
+ * This method operates recursively on all outer children.
105
+ */
106
+ static void outerPolyNodeToPathObjects (
107
+ const ClipperLib::PolyNode& node,
108
+ PathObjects& out_objects,
109
+ const PathObject* proto,
110
+ const PolyMap& polymap );
111
+
112
+ /* *
113
+ * Constructs ClipperLib::Paths from a PathObject.
114
+ */
115
+ static void pathObjectToPolygons (
116
+ const PathObject* object,
117
+ ClipperLib::Paths& polygons,
118
+ PolyMap& polymap );
119
+
120
+ /* *
121
+ * Reconstructs a PathObject from a polygon given as ClipperLib::Path.
122
+ *
123
+ * Curves are reconstructed with the help of the polymap, mapping locations
124
+ * to path coords of the original objects.
125
+ */
126
+ static void polygonToPathPart (
127
+ const ClipperLib::Path& polygon,
128
+ const PolyMap& polymap,
129
+ PathObject* object );
130
+
131
+ /* *
132
+ * Tries to reconstruct a straight or curved segment with given start and
133
+ * end indices from the polygon.
134
+ * The first coordinate of the segment is assumed to be already added.
135
+ */
136
+ static void rebuildSegment (
137
+ ClipperLib::Path::size_type start_index,
138
+ ClipperLib::Path::size_type end_index,
139
+ bool sequence_increasing,
140
+ const ClipperLib::Path& polygon,
141
+ const PolyMap& polymap,
142
+ PathObject* object );
143
+
144
+ /* *
145
+ * Approximates a curved segment from the result polygon alone.
146
+ */
147
+ static void rebuildSegmentFromPathOnly (
148
+ const ClipperLib::IntPoint& start_point,
149
+ const ClipperLib::IntPoint& second_point,
150
+ const ClipperLib::IntPoint& second_last_point,
151
+ const ClipperLib::IntPoint& end_point,
152
+ PathObject* object );
153
+
154
+ /* *
155
+ * Special case of rebuildSegment() for straight or very short lines.
156
+ */
157
+ static void rebuildTwoIndexSegment (
158
+ ClipperLib::Path::size_type start_index,
159
+ ClipperLib::Path::size_type end_index,
160
+ bool sequence_increasing,
161
+ const ClipperLib::Path& polygon,
162
+ const PolyMap& polymap,
163
+ PathObject* object );
164
+
165
+ /* *
166
+ * Reconstructs one polygon coordinate and adds it to the object.
167
+ *
168
+ * Uses the polymap to check whether the coordinate should be a dash point.
169
+ */
170
+ static void rebuildCoordinate (
171
+ ClipperLib::Path::size_type index,
172
+ const ClipperLib::Path& polygon,
173
+ const PolyMap& polymap,
174
+ PathObject* object,
175
+ bool start_new_part = false );
176
+
177
+ /* *
178
+ * Compares a PathObject segment to a ClipperLib::Path polygon segment.
179
+ *
180
+ * Returns true if the segments match. In this case, the out_... parameters are set.
181
+ *
182
+ * @param original The original PathObject.
183
+ * @param coord_index The index of the segment start at the original.
184
+ * @param polygon The ClipperLib::Path polygon.
185
+ * @param start_index The start of the segment at the polygon.
186
+ * @param end_index The end of the segment at the polygon.
187
+ * @param out_coords_increasing If the segments match, will be set to
188
+ * either true if a matching segment's point at coord_index corresponds to the point at start_index,
189
+ * or false otherwise.
190
+ * @param out_is_curve If the segments match, will be set to
191
+ * either true if the original segment is a curve,
192
+ * or false otherwise.
193
+ */
194
+ static bool checkSegmentMatch (
195
+ const PathObject* original,
196
+ int coord_index,
197
+ const ClipperLib::Path& polygon,
198
+ ClipperLib::Path::size_type start_index,
199
+ ClipperLib::Path::size_type end_index,
200
+ bool & out_coords_increasing,
201
+ bool & out_is_curve );
202
+
75
203
/* *
76
204
* Removes flags from the coordinate to be able to use it in the reconstruction.
77
205
*/
@@ -97,6 +225,8 @@ bool operator==(const ClipperLib::IntPoint& lhs, const MapCoord& rhs)
97
225
return rhs == lhs;
98
226
}
99
227
228
+ } // namespace
229
+
100
230
101
231
102
232
// ### BooleanTool ###
@@ -111,7 +241,7 @@ BooleanTool::BooleanTool(Operation op, Map* map)
111
241
bool BooleanTool::execute ()
112
242
{
113
243
// Check basic prerequisite
114
- Object* const primary_object = map->getFirstSelectedObject ();
244
+ const Object* const primary_object = map->getFirstSelectedObject ();
115
245
if (primary_object->getType () != Object::Path)
116
246
{
117
247
qWarning (" The first selected object must be a path." );
@@ -211,7 +341,7 @@ bool BooleanTool::executePerSymbol()
211
341
return have_changes;
212
342
}
213
343
214
- bool BooleanTool::executeForObjects (PathObject* subject, PathObjects& in_objects, PathObjects& out_objects, CombinedUndoStep& undo_step)
344
+ bool BooleanTool::executeForObjects (const PathObject* subject, const PathObjects& in_objects, PathObjects& out_objects, CombinedUndoStep& undo_step)
215
345
{
216
346
if (!executeForObjects (subject, in_objects, out_objects))
217
347
{
@@ -258,7 +388,7 @@ bool BooleanTool::executeForObjects(PathObject* subject, PathObjects& in_objects
258
388
return true ;
259
389
}
260
390
261
- bool BooleanTool::executeForObjects (PathObject* subject, PathObjects& in_objects, PathObjects& out_objects)
391
+ bool BooleanTool::executeForObjects (const PathObject* subject, const PathObjects& in_objects, PathObjects& out_objects) const
262
392
{
263
393
// Convert the objects to Clipper polygons and
264
394
// create a hash map, mapping point positions to the PathCoords.
@@ -312,40 +442,7 @@ bool BooleanTool::executeForObjects(PathObject* subject, PathObjects& in_objects
312
442
return success;
313
443
}
314
444
315
- void BooleanTool::polyTreeToPathObjects (const ClipperLib::PolyTree& tree, PathObjects& out_objects, const PathObject* proto, const PolyMap& polymap)
316
- {
317
- for (int i = 0 , count = tree.ChildCount (); i < count; ++i)
318
- outerPolyNodeToPathObjects (*tree.Childs [i], out_objects, proto, polymap);
319
- }
320
-
321
- void BooleanTool::outerPolyNodeToPathObjects (const ClipperLib::PolyNode& node, PathObjects& out_objects, const PathObject* proto, const PolyMap& polymap)
322
- {
323
- auto object = std::unique_ptr<PathObject>{ proto->duplicate () };
324
- object->clearCoordinates ();
325
-
326
- try
327
- {
328
- polygonToPathPart (node.Contour , polymap, object.get ());
329
- for (int i = 0 , i_count = node.ChildCount (); i < i_count; ++i)
330
- {
331
- polygonToPathPart (node.Childs [i]->Contour , polymap, object.get ());
332
-
333
- // Add outer polygons contained by (nested within) holes ...
334
- for (int j = 0 , j_count = node.Childs [i]->ChildCount (); j < j_count; ++j)
335
- outerPolyNodeToPathObjects (*node.Childs [i]->Childs [j], out_objects, proto, polymap);
336
- }
337
-
338
- out_objects.push_back (object.release ());
339
- }
340
- catch (std::range_error&)
341
- {
342
- // Do nothing
343
- }
344
- }
345
-
346
-
347
-
348
- void BooleanTool::executeForLine (const PathObject* area, const PathObject* line, BooleanTool::PathObjects& out_objects)
445
+ void BooleanTool::executeForLine (const PathObject* area, const PathObject* line, BooleanTool::PathObjects& out_objects) const
349
446
{
350
447
if (op != BooleanTool::Intersection && op != BooleanTool::Difference)
351
448
{
@@ -428,7 +525,42 @@ void BooleanTool::executeForLine(const PathObject* area, const PathObject* line,
428
525
}
429
526
}
430
527
431
- void BooleanTool::pathObjectToPolygons (
528
+
529
+
530
+ namespace {
531
+
532
+ void polyTreeToPathObjects (const ClipperLib::PolyTree& tree, PathObjects& out_objects, const PathObject* proto, const PolyMap& polymap)
533
+ {
534
+ for (int i = 0 , count = tree.ChildCount (); i < count; ++i)
535
+ outerPolyNodeToPathObjects (*tree.Childs [i], out_objects, proto, polymap);
536
+ }
537
+
538
+ void outerPolyNodeToPathObjects (const ClipperLib::PolyNode& node, PathObjects& out_objects, const PathObject* proto, const PolyMap& polymap)
539
+ {
540
+ auto object = std::unique_ptr<PathObject>{ proto->duplicate () };
541
+ object->clearCoordinates ();
542
+
543
+ try
544
+ {
545
+ polygonToPathPart (node.Contour , polymap, object.get ());
546
+ for (int i = 0 , i_count = node.ChildCount (); i < i_count; ++i)
547
+ {
548
+ polygonToPathPart (node.Childs [i]->Contour , polymap, object.get ());
549
+
550
+ // Add outer polygons contained by (nested within) holes ...
551
+ for (int j = 0 , j_count = node.Childs [i]->ChildCount (); j < j_count; ++j)
552
+ outerPolyNodeToPathObjects (*node.Childs [i]->Childs [j], out_objects, proto, polymap);
553
+ }
554
+
555
+ out_objects.push_back (object.release ());
556
+ }
557
+ catch (std::range_error&)
558
+ {
559
+ // Do nothing
560
+ }
561
+ }
562
+
563
+ void pathObjectToPolygons (
432
564
const PathObject* object,
433
565
ClipperLib::Paths& polygons,
434
566
PolyMap& polymap)
@@ -475,7 +607,7 @@ void BooleanTool::pathObjectToPolygons(
475
607
}
476
608
}
477
609
478
- void BooleanTool:: polygonToPathPart (const ClipperLib::Path& polygon, const PolyMap& polymap, PathObject* object)
610
+ void polygonToPathPart (const ClipperLib::Path& polygon, const PolyMap& polymap, PathObject* object)
479
611
{
480
612
auto num_points = polygon.size ();
481
613
if (num_points < 3 )
@@ -538,7 +670,7 @@ void BooleanTool::polygonToPathPart(const ClipperLib::Path& polygon, const PolyM
538
670
if (cur_info.first && cur_info.first == new_info.first )
539
671
{
540
672
// Same original part
541
- auto cur_coord_index = cur_info.second ->index ;
673
+ auto cur_coord_index = cur_info.second ->index ; // NOLINT
542
674
const auto cur_coord = cur_info.first ->path ->getCoordinate (cur_coord_index);
543
675
544
676
auto new_coord_index = new_info.second ->index ;
@@ -644,7 +776,7 @@ void BooleanTool::polygonToPathPart(const ClipperLib::Path& polygon, const PolyM
644
776
object->parts ().back ().connectEnds ();
645
777
}
646
778
647
- void BooleanTool:: rebuildSegment (
779
+ void rebuildSegment (
648
780
ClipperLib::Path::size_type start_index,
649
781
ClipperLib::Path::size_type end_index,
650
782
bool sequence_increasing,
@@ -938,7 +1070,7 @@ void BooleanTool::rebuildSegment(
938
1070
}
939
1071
}
940
1072
941
- void BooleanTool:: rebuildSegmentFromPathOnly (
1073
+ void rebuildSegmentFromPathOnly (
942
1074
const ClipperLib::IntPoint& start_point,
943
1075
const ClipperLib::IntPoint& second_point,
944
1076
const ClipperLib::IntPoint& second_last_point,
@@ -961,7 +1093,7 @@ void BooleanTool::rebuildSegmentFromPathOnly(
961
1093
object->addCoordinate (end_point_c);
962
1094
}
963
1095
964
- void BooleanTool:: rebuildTwoIndexSegment (
1096
+ void rebuildTwoIndexSegment (
965
1097
ClipperLib::Path::size_type start_index,
966
1098
ClipperLib::Path::size_type end_index,
967
1099
bool sequence_increasing,
@@ -1030,7 +1162,7 @@ void BooleanTool::rebuildTwoIndexSegment(
1030
1162
}
1031
1163
}
1032
1164
1033
- void BooleanTool:: rebuildCoordinate (
1165
+ void rebuildCoordinate (
1034
1166
ClipperLib::Path::size_type index,
1035
1167
const ClipperLib::Path& polygon,
1036
1168
const PolyMap& polymap,
@@ -1049,7 +1181,7 @@ void BooleanTool::rebuildCoordinate(
1049
1181
object->addCoordinate (coord, start_new_part);
1050
1182
}
1051
1183
1052
- bool BooleanTool:: checkSegmentMatch (
1184
+ bool checkSegmentMatch (
1053
1185
const PathObject* original,
1054
1186
int coord_index,
1055
1187
const ClipperLib::Path& polygon,
@@ -1082,5 +1214,7 @@ bool BooleanTool::checkSegmentMatch(
1082
1214
return found;
1083
1215
}
1084
1216
1217
+ } // namespace
1218
+
1085
1219
1086
1220
} // namespace OpenOrienteering
0 commit comments