File indexing completed on 2025-04-03 08:25:54
0001
0002
0003
0004
0005
0006
0007
0008
0009
0010
0011
0012
0013
0014 #ifndef BOOST_GEOMETRY_ALGORITHMS_DETAIL_OVERLAY_TRAVERSAL_SWITCH_DETECTOR_HPP
0015 #define BOOST_GEOMETRY_ALGORITHMS_DETAIL_OVERLAY_TRAVERSAL_SWITCH_DETECTOR_HPP
0016
0017 #include <boost/range/value_type.hpp>
0018
0019 #include <boost/geometry/algorithms/detail/ring_identifier.hpp>
0020 #include <boost/geometry/algorithms/detail/overlay/copy_segments.hpp>
0021 #include <boost/geometry/algorithms/detail/overlay/cluster_info.hpp>
0022 #include <boost/geometry/algorithms/detail/overlay/is_self_turn.hpp>
0023 #include <boost/geometry/algorithms/detail/overlay/turn_info.hpp>
0024
0025 #if defined(BOOST_GEOMETRY_DEBUG_TRAVERSAL_SWITCH_DETECTOR)
0026 #include <boost/geometry/core/access.hpp>
0027 #endif
0028
0029 #include <boost/geometry/util/constexpr.hpp>
0030
0031 #include <cstddef>
0032 #include <map>
0033
0034 namespace boost { namespace geometry
0035 {
0036
0037 #ifndef DOXYGEN_NO_DETAIL
0038 namespace detail { namespace overlay
0039 {
0040
0041
0042
0043
0044
0045
0046
0047
0048
0049
0050
0051
0052
0053
0054
0055
0056
0057
0058
0059
0060
0061
0062
0063
0064
0065
0066
0067
0068
0069
0070
0071
0072
0073
0074
0075
0076
0077 template
0078 <
0079 bool Reverse1,
0080 bool Reverse2,
0081 overlay_type OverlayType,
0082 typename Geometry1,
0083 typename Geometry2,
0084 typename Turns,
0085 typename Clusters,
0086 typename RobustPolicy,
0087 typename Visitor
0088 >
0089 struct traversal_switch_detector
0090 {
0091 static const operation_type target_operation
0092 = operation_from_overlay<OverlayType>::value;
0093
0094 enum isolation_type
0095 {
0096 isolation_no = 0,
0097 isolation_yes = 1,
0098 isolation_multiple = 2
0099 };
0100
0101 using turn_type = typename boost::range_value<Turns>::type;
0102 using set_type= std::set<signed_size_type>;
0103
0104
0105
0106 struct merged_ring_properties
0107 {
0108 signed_size_type region_id = -1;
0109 set_type turn_indices;
0110 };
0111
0112 struct connection_properties
0113 {
0114 std::size_t count = 0;
0115
0116 set_type unique_turn_ids;
0117 };
0118
0119
0120 using connection_map = std::map<signed_size_type, connection_properties>;
0121
0122
0123
0124 struct region_properties
0125 {
0126 signed_size_type region_id = -1;
0127 isolation_type isolated = isolation_no;
0128 set_type unique_turn_ids;
0129 connection_map connected_region_counts;
0130 };
0131
0132
0133 using merge_map = std::map<ring_identifier, merged_ring_properties>;
0134
0135
0136 using region_connection_map = std::map<signed_size_type, region_properties>;
0137
0138 inline traversal_switch_detector(Geometry1 const& geometry1,
0139 Geometry2 const& geometry2,
0140 Turns& turns, Clusters const& clusters,
0141 RobustPolicy const& robust_policy, Visitor& visitor)
0142 : m_geometry1(geometry1)
0143 , m_geometry2(geometry2)
0144 , m_turns(turns)
0145 , m_clusters(clusters)
0146 , m_robust_policy(robust_policy)
0147 , m_visitor(visitor)
0148 {
0149 }
0150
0151 bool one_connection_to_another_region(region_properties const& region) const
0152 {
0153
0154
0155
0156
0157
0158
0159
0160
0161
0162 if (region.connected_region_counts.size() == 1)
0163 {
0164 auto const& cprop = region.connected_region_counts.begin()->second;
0165 return cprop.count <= 1;
0166 }
0167 return region.connected_region_counts.empty();
0168 }
0169
0170
0171 bool multiple_connections_to_one_region(region_properties const& region) const
0172 {
0173
0174
0175
0176
0177
0178
0179
0180
0181
0182
0183
0184
0185 if (region.connected_region_counts.size() == 1)
0186 {
0187 auto const& cprop = region.connected_region_counts.begin()->second;
0188 return cprop.count > 1;
0189 }
0190 return false;
0191 }
0192
0193 bool one_connection_to_multiple_regions(region_properties const& region) const
0194 {
0195
0196
0197
0198
0199
0200
0201
0202
0203
0204 bool first = true;
0205 signed_size_type first_turn_id = 0;
0206 for (auto const& key_val : region.connected_region_counts)
0207 {
0208 auto const& cprop = key_val.second;
0209
0210 if (cprop.count != 1)
0211 {
0212 return false;
0213 }
0214 auto const unique_turn_id = *cprop.unique_turn_ids.begin();
0215 if (first)
0216 {
0217 first_turn_id = unique_turn_id;
0218 first = false;
0219 }
0220 else if (first_turn_id != unique_turn_id)
0221 {
0222 return false;
0223 }
0224 }
0225 return true;
0226 }
0227
0228 bool ii_turn_connects_two_regions(region_properties const& region,
0229 region_properties const& connected_region,
0230 signed_size_type turn_index) const
0231 {
0232 turn_type const& turn = m_turns[turn_index];
0233 if (! turn.both(operation_intersection))
0234 {
0235 return false;
0236 }
0237
0238 signed_size_type const id0 = turn.operations[0].enriched.region_id;
0239 signed_size_type const id1 = turn.operations[1].enriched.region_id;
0240
0241 return (id0 == region.region_id && id1 == connected_region.region_id)
0242 || (id1 == region.region_id && id0 == connected_region.region_id);
0243 }
0244
0245
0246 bool isolated_multiple_connection(region_properties const& region,
0247 region_properties const& connected_region) const
0248 {
0249 if (connected_region.isolated != isolation_multiple)
0250 {
0251 return false;
0252 }
0253
0254
0255 set_type turn_ids = region.unique_turn_ids;
0256 for (auto turn_id : connected_region.unique_turn_ids)
0257 {
0258 turn_ids.erase(turn_id);
0259 }
0260
0261
0262 if (turn_ids.size() != 1)
0263 {
0264 return false;
0265 }
0266
0267 for (auto id_or_index : connected_region.unique_turn_ids)
0268 {
0269 if (id_or_index >= 0)
0270 {
0271 if (! ii_turn_connects_two_regions(region, connected_region, id_or_index))
0272 {
0273 return false;
0274 }
0275 }
0276 else
0277 {
0278 signed_size_type const cluster_id = -id_or_index;
0279 auto it = m_clusters.find(cluster_id);
0280 if (it != m_clusters.end())
0281 {
0282 cluster_info const& cinfo = it->second;
0283 for (auto turn_index : cinfo.turn_indices)
0284 {
0285 if (! ii_turn_connects_two_regions(region, connected_region, turn_index))
0286 {
0287 return false;
0288 }
0289 }
0290 }
0291 }
0292 }
0293
0294 return true;
0295 }
0296
0297 bool has_only_isolated_children(region_properties const& region) const
0298 {
0299 bool first_with_turn = true;
0300 signed_size_type first_turn_id = 0;
0301
0302 for (auto const& key_val : region.connected_region_counts)
0303 {
0304 signed_size_type const region_id = key_val.first;
0305 connection_properties const& cprop = key_val.second;
0306
0307 auto mit = m_connected_regions.find(region_id);
0308 if (mit == m_connected_regions.end())
0309 {
0310
0311 return false;
0312 }
0313
0314 region_properties const& connected_region = mit->second;
0315
0316 if (cprop.count != 1)
0317 {
0318
0319 if (! isolated_multiple_connection(region, connected_region))
0320 {
0321 return false;
0322 }
0323 }
0324
0325 if (connected_region.isolated != isolation_yes
0326 && connected_region.isolated != isolation_multiple)
0327 {
0328 signed_size_type const unique_turn_id = *cprop.unique_turn_ids.begin();
0329 if (first_with_turn)
0330 {
0331 first_turn_id = unique_turn_id;
0332 first_with_turn = false;
0333 }
0334 else if (first_turn_id != unique_turn_id)
0335 {
0336 return false;
0337 }
0338 }
0339 }
0340
0341
0342
0343 return true;
0344 }
0345
0346 void get_isolated_regions()
0347 {
0348
0349
0350
0351
0352 for (auto& key_val : m_connected_regions)
0353 {
0354 region_properties& properties = key_val.second;
0355 if (one_connection_to_another_region(properties))
0356 {
0357 properties.isolated = isolation_yes;
0358 }
0359 else if (multiple_connections_to_one_region(properties))
0360 {
0361 properties.isolated = isolation_multiple;
0362 }
0363 else if (one_connection_to_multiple_regions(properties))
0364 {
0365 properties.isolated = isolation_yes;
0366 }
0367 }
0368
0369
0370
0371 std::size_t defensive_check = 0;
0372 bool changed = true;
0373 while (changed && defensive_check++ < m_connected_regions.size())
0374 {
0375 changed = false;
0376 for (auto& key_val : m_connected_regions)
0377 {
0378 region_properties& properties = key_val.second;
0379
0380 if (properties.isolated == isolation_no
0381 && has_only_isolated_children(properties))
0382 {
0383 properties.isolated = isolation_yes;
0384 changed = true;
0385 }
0386 }
0387 }
0388 }
0389
0390 void assign_isolation_to_enriched()
0391 {
0392 for (turn_type& turn : m_turns)
0393 {
0394 constexpr auto order1 = geometry::point_order<Geometry1>::value;
0395 constexpr bool reverse1 = (order1 == boost::geometry::counterclockwise)
0396 ? ! Reverse1 : Reverse1;
0397
0398 constexpr auto order2 = geometry::point_order<Geometry2>::value;
0399 constexpr bool reverse2 = (order2 == boost::geometry::counterclockwise)
0400 ? ! Reverse2 : Reverse2;
0401
0402
0403
0404
0405 bool const reverseMeaningInTurn
0406 = (reverse1 || reverse2)
0407 && ! turn.is_self()
0408 && ! turn.is_clustered()
0409 && uu_or_ii(turn)
0410 && turn.operations[0].enriched.region_id
0411 != turn.operations[1].enriched.region_id;
0412
0413 for (auto& op : turn.operations)
0414 {
0415 auto mit = m_connected_regions.find(op.enriched.region_id);
0416 if (mit != m_connected_regions.end())
0417 {
0418 bool const reverseMeaningInOp
0419 = reverseMeaningInTurn
0420 && ((op.seg_id.source_index == 0 && reverse1)
0421 || (op.seg_id.source_index == 1 && reverse2));
0422
0423
0424
0425
0426
0427 region_properties const& prop = mit->second;
0428 op.enriched.isolated
0429 = reverseMeaningInOp
0430 ? false
0431 : prop.isolated == isolation_yes;
0432 }
0433 }
0434 }
0435 }
0436
0437 void assign_region_ids_to_enriched()
0438 {
0439 for (auto const& key_val : m_turns_per_ring)
0440 {
0441 ring_identifier const& ring_id = key_val.first;
0442 merged_ring_properties const& properties = key_val.second;
0443
0444 for (auto turn_index : properties.turn_indices)
0445 {
0446 turn_type& turn = m_turns[turn_index];
0447
0448 if (! acceptable(turn))
0449 {
0450
0451 continue;
0452 }
0453
0454 for (auto& op : turn.operations)
0455 {
0456 if (ring_id_by_seg_id(op.seg_id) == ring_id)
0457 {
0458 op.enriched.region_id = properties.region_id;
0459 }
0460 }
0461 }
0462 }
0463 }
0464
0465 void assign_connected_regions()
0466 {
0467 for (std::size_t turn_index = 0; turn_index < m_turns.size(); ++turn_index)
0468 {
0469 turn_type const& turn = m_turns[turn_index];
0470
0471 signed_size_type const unique_turn_id
0472 = turn.is_clustered() ? -turn.cluster_id : turn_index;
0473
0474 signed_size_type const& id0 = turn.operations[0].enriched.region_id;
0475 signed_size_type const& id1 = turn.operations[1].enriched.region_id;
0476
0477
0478 if (id0 != -1)
0479 {
0480 m_connected_regions[id0].region_id = id0;
0481 m_connected_regions[id0].unique_turn_ids.insert(unique_turn_id);
0482 }
0483 if (id1 != -1 && id0 != id1)
0484 {
0485 m_connected_regions[id1].region_id = id1;
0486 m_connected_regions[id1].unique_turn_ids.insert(unique_turn_id);
0487 }
0488
0489 if (id0 != id1 && id0 != -1 && id1 != -1)
0490 {
0491
0492 connection_properties& prop0 = m_connected_regions[id0].connected_region_counts[id1];
0493 connection_properties& prop1 = m_connected_regions[id1].connected_region_counts[id0];
0494
0495
0496 if (prop0.unique_turn_ids.count(unique_turn_id) == 0)
0497 {
0498 prop0.count++;
0499 prop0.unique_turn_ids.insert(unique_turn_id);
0500 }
0501 if (prop1.unique_turn_ids.count(unique_turn_id) == 0)
0502 {
0503 prop1.count++;
0504 prop1.unique_turn_ids.insert(unique_turn_id);
0505 }
0506 }
0507 }
0508 }
0509
0510 inline bool acceptable(turn_type const& turn) const
0511 {
0512
0513
0514
0515 return ! turn.discarded && ! turn.both(operation_blocked);
0516 }
0517
0518 inline bool uu_or_ii(turn_type const& turn) const
0519 {
0520 return turn.both(operation_union) || turn.both(operation_intersection);
0521 }
0522
0523 inline bool connects_same_region(turn_type const& turn) const
0524 {
0525 if (! acceptable(turn))
0526 {
0527 return false;
0528 }
0529
0530 if (! turn.is_clustered())
0531 {
0532
0533 return ! uu_or_ii(turn);
0534 }
0535
0536 if BOOST_GEOMETRY_CONSTEXPR (target_operation == operation_union)
0537 {
0538
0539
0540 return turn.operations[0].enriched.zone
0541 == turn.operations[1].enriched.zone;
0542 }
0543 else
0544 {
0545
0546
0547
0548 return ! (turn.both(operation_intersection)
0549 || turn.combination(operation_intersection, operation_union));
0550 }
0551 }
0552
0553 void create_region(signed_size_type& new_region_id, ring_identifier const& ring_id,
0554 merged_ring_properties& properties, signed_size_type region_id = -1)
0555 {
0556 if (properties.region_id > 0)
0557 {
0558
0559 return;
0560 }
0561
0562
0563 if (region_id == -1)
0564 {
0565 region_id = new_region_id++;
0566 }
0567
0568
0569 properties.region_id = region_id;
0570
0571 #if defined(BOOST_GEOMETRY_DEBUG_TRAVERSAL_SWITCH_DETECTOR)
0572 std::cout << " ADD " << ring_id << " TO REGION " << region_id << std::endl;
0573 #endif
0574
0575
0576 for (auto turn_index : properties.turn_indices)
0577 {
0578 turn_type const& turn = m_turns[turn_index];
0579 if (! connects_same_region(turn))
0580 {
0581
0582 continue;
0583 }
0584
0585
0586
0587 for (auto const& op : turn.operations)
0588 {
0589 ring_identifier connected_ring_id = ring_id_by_seg_id(op.seg_id);
0590 if (connected_ring_id != ring_id)
0591 {
0592 propagate_region(new_region_id, connected_ring_id, region_id);
0593 }
0594 }
0595 }
0596 }
0597
0598 void propagate_region(signed_size_type& new_region_id,
0599 ring_identifier const& ring_id, signed_size_type region_id)
0600 {
0601 auto it = m_turns_per_ring.find(ring_id);
0602 if (it != m_turns_per_ring.end())
0603 {
0604 create_region(new_region_id, ring_id, it->second, region_id);
0605 }
0606 }
0607
0608 #if defined(BOOST_GEOMETRY_DEBUG_TRAVERSAL_SWITCH_DETECTOR)
0609 void debug_show_results()
0610 {
0611 auto isolation_to_string = [](isolation_type const& iso) -> std::string
0612 {
0613 switch(iso)
0614 {
0615 case isolation_no : return "no";
0616 case isolation_yes : return "yes";
0617 case isolation_multiple : return "multiple";
0618 }
0619 return "error";
0620 };
0621 auto set_to_string = [](auto const& s) -> std::string
0622 {
0623 std::ostringstream result;
0624 for (auto item : s) { result << " " << item; }
0625 return result.str();
0626 };
0627
0628 for (auto const& kv : m_connected_regions)
0629 {
0630 auto const& prop = kv.second;
0631
0632 std::ostringstream sub;
0633 sub << "[turns" << set_to_string(prop.unique_turn_ids)
0634 << "] regions";
0635 for (auto const& kvs : prop.connected_region_counts)
0636 {
0637 sub << " { " << kvs.first
0638 << " : via [" << set_to_string(kvs.second.unique_turn_ids)
0639 << " ] }";
0640 }
0641
0642 std::cout << "REGION " << prop.region_id
0643 << " " << isolation_to_string(prop.isolated)
0644 << " " << sub.str()
0645 << std::endl;
0646 }
0647
0648 for (std::size_t turn_index = 0; turn_index < m_turns.size(); ++turn_index)
0649 {
0650 turn_type const& turn = m_turns[turn_index];
0651
0652 if (uu_or_ii(turn) && ! turn.is_clustered())
0653 {
0654 std::cout << (turn.both(operation_union) ? "UU" : "II")
0655 << " " << turn_index
0656 << " (" << geometry::get<0>(turn.point)
0657 << ", " << geometry::get<1>(turn.point) << ")"
0658 << " -> " << std::boolalpha
0659 << " [" << turn.operations[0].seg_id.source_index
0660 << "/" << turn.operations[1].seg_id.source_index << "] "
0661 << "(" << turn.operations[0].enriched.region_id
0662 << " " << turn.operations[0].enriched.isolated
0663 << ") / (" << turn.operations[1].enriched.region_id
0664 << " " << turn.operations[1].enriched.isolated << ")"
0665 << std::endl;
0666 }
0667 }
0668
0669 for (auto const& key_val : m_clusters)
0670 {
0671 cluster_info const& cinfo = key_val.second;
0672 std::cout << "CL RESULT " << key_val.first
0673 << " -> " << cinfo.open_count << std::endl;
0674 }
0675 }
0676 #endif
0677
0678 void iterate()
0679 {
0680 #if defined(BOOST_GEOMETRY_DEBUG_TRAVERSAL_SWITCH_DETECTOR)
0681 std::cout << "BEGIN SWITCH DETECTOR (region_ids and isolation)"
0682 << (Reverse1 ? " REVERSE_1" : "")
0683 << (Reverse2 ? " REVERSE_2" : "")
0684 << std::endl;
0685 #endif
0686
0687
0688 m_turns_per_ring.clear();
0689 m_connected_regions.clear();
0690
0691 for (std::size_t turn_index = 0; turn_index < m_turns.size(); ++turn_index)
0692 {
0693 turn_type const& turn = m_turns[turn_index];
0694
0695 if BOOST_GEOMETRY_CONSTEXPR (target_operation == operation_intersection)
0696 {
0697 if (turn.discarded)
0698 {
0699
0700 continue;
0701 }
0702 }
0703
0704 for (auto const& op : turn.operations)
0705 {
0706 m_turns_per_ring[ring_id_by_seg_id(op.seg_id)].turn_indices.insert(turn_index);
0707 }
0708 }
0709
0710
0711 {
0712 signed_size_type new_region_id = 1;
0713 for (auto& key_val : m_turns_per_ring)
0714 {
0715 create_region(new_region_id, key_val.first, key_val.second);
0716 }
0717
0718 assign_region_ids_to_enriched();
0719 assign_connected_regions();
0720 get_isolated_regions();
0721 assign_isolation_to_enriched();
0722 }
0723
0724 #if defined(BOOST_GEOMETRY_DEBUG_TRAVERSAL_SWITCH_DETECTOR)
0725 std::cout << "END SWITCH DETECTOR" << std::endl;
0726 debug_show_results();
0727 #endif
0728
0729 }
0730
0731 private:
0732
0733 Geometry1 const& m_geometry1;
0734 Geometry2 const& m_geometry2;
0735 Turns& m_turns;
0736 Clusters const& m_clusters;
0737 merge_map m_turns_per_ring;
0738 region_connection_map m_connected_regions;
0739 RobustPolicy const& m_robust_policy;
0740 Visitor& m_visitor;
0741 };
0742
0743 }}
0744 #endif
0745
0746 }}
0747
0748 #endif