#pragma once #include #include #include #include #include #include #include #include #include #ifdef DEBUG #include #include #include #include // // void* callstack[128]; // int i, frames = backtrace(callstack, 128); // char** strs = backtrace_symbols(callstack, frames); // for (i = 0; i < frames; ++i) { // printf("%s\n", strs[i]); // } // free(strs); #endif namespace mapbox { namespace geometry { namespace wagyu { // NOTE: ring and ring_ptr are forward declared in wagyu/point.hpp template using ring_vector = std::vector>; template using ring_list = std::list>; template struct ring { std::size_t ring_index; // To support unset 0 is undefined and indexes offset by 1 std::size_t size; double area; ring_ptr parent; ring_list children; point_ptr points; point_ptr bottom_point; ring(ring const&) = delete; ring& operator=(ring const&) = delete; ring() : ring_index(0), size(0), area(std::numeric_limits::quiet_NaN()), parent(nullptr), children(), points(nullptr), bottom_point(nullptr) { } }; template using hot_pixel_vector = std::vector>; template using hot_pixel_itr = typename hot_pixel_vector::iterator; template using hot_pixel_rev_itr = typename hot_pixel_vector::reverse_iterator; template struct ring_manager { ring_list children; std::vector> all_points; hot_pixel_vector hot_pixels; hot_pixel_itr current_hp_itr; std::deque> points; std::deque> rings; std::vector> storage; std::size_t index; ring_manager(ring_manager const&) = delete; ring_manager& operator=(ring_manager const&) = delete; ring_manager() : children(), all_points(), hot_pixels(), current_hp_itr(hot_pixels.end()), points(), rings(), storage(), index(0) { } }; template void preallocate_point_memory(ring_manager& rings, std::size_t size) { rings.storage.reserve(size); rings.all_points.reserve(size); } template ring_ptr create_new_ring(ring_manager& rings) { rings.rings.emplace_back(); ring_ptr result = &rings.rings.back(); result->ring_index = rings.index++; return result; } template point_ptr create_new_point(ring_ptr r, mapbox::geometry::point const& pt, ring_manager& rings) { point_ptr point; if (rings.storage.size() < rings.storage.capacity()) { rings.storage.emplace_back(r, pt); point = &rings.storage.back(); } else { rings.points.emplace_back(r, pt); point = &rings.points.back(); } rings.all_points.push_back(point); return point; } template point_ptr create_new_point(ring_ptr r, mapbox::geometry::point const& pt, point_ptr before_this_point, ring_manager& rings) { point_ptr point; if (rings.storage.size() < rings.storage.capacity()) { rings.storage.emplace_back(r, pt, before_this_point); point = &rings.storage.back(); } else { rings.points.emplace_back(r, pt, before_this_point); point = &rings.points.back(); } rings.all_points.push_back(point); return point; } template void ring1_child_of_ring2(ring_ptr ring1, ring_ptr ring2, ring_manager& manager) { assert(ring1 != ring2); if (ring1->parent == ring2) { return; } if (ring1->parent == nullptr) { manager.children.remove(ring1); } else { ring1->parent->children.remove(ring1); } if (ring2 == nullptr) { ring1->parent = nullptr; manager.children.push_back(ring1); } else { ring1->parent = ring2; ring2->children.push_back(ring1); } } template void ring1_sibling_of_ring2(ring_ptr ring1, ring_ptr ring2, ring_manager& manager) { assert(ring1 != ring2); if (ring1->parent == ring2->parent) { return; } if (ring1->parent == nullptr) { manager.children.remove(ring1); } else { ring1->parent->children.remove(ring1); } if (ring2->parent == nullptr) { manager.children.push_back(ring1); } else { ring2->parent->children.push_back(ring1); } ring1->parent = ring2->parent; } template void ring1_replaces_ring2(ring_ptr ring1, ring_ptr ring2, ring_manager& manager) { assert(ring1 != ring2); if (ring2->parent == nullptr) { manager.children.remove(ring2); } else { ring2->parent->children.remove(ring2); } for (auto& c : ring2->children) { c->parent = ring1; } if (ring1 == nullptr) { manager.children.splice(manager.children.end(), ring2->children); } else { ring1->children.splice(ring1->children.end(), ring2->children); } ring2->parent = nullptr; } template void remove_ring(ring_ptr r, ring_manager& manager) { if (r->parent == nullptr) { manager.children.remove(r); for (auto& c : r->children) { c->parent = nullptr; } manager.children.splice(manager.children.end(), r->children); } else { r->parent->children.remove(r); for (auto& c : r->children) { c->parent = r->parent; } r->parent->children.splice(r->parent->children.end(), r->children); r->parent = nullptr; } } template inline std::size_t ring_depth(ring_ptr r) { std::size_t depth = 0; if (!r) { return depth; } while (r->parent) { depth++; r = r->parent; } return depth; } template inline bool ring_is_hole(ring_ptr r) { return ring_depth(r) & 1; } template void set_next(const_point_ptr& node, const const_point_ptr& next_node) { node->next = next_node; } template point_ptr get_next(const_point_ptr& node) { return node->next; } template point_ptr get_prev(const_point_ptr& node) { return node->prev; } template void set_prev(const_point_ptr& node, const const_point_ptr& prev_node) { node->prev = prev_node; } template void init(const_point_ptr& node) { set_next(node, node); set_prev(node, node); } template std::size_t point_count(const const_point_ptr& orig_node) { std::size_t size = 0; point_ptr n = orig_node; do { n = get_next(n); ++size; } while (n != orig_node); return size; } template void link_before(point_ptr& node, point_ptr& new_node) { point_ptr prev_node = get_prev(node); set_prev(new_node, prev_node); set_next(new_node, node); set_prev(node, new_node); set_next(prev_node, new_node); } template void link_after(point_ptr& node, point_ptr& new_node) { point_ptr next_node = get_next(node); set_prev(new_node, node); set_next(new_node, next_node); set_next(node, new_node); set_prev(next_node, new_node); } template void transfer_point(point_ptr& p, point_ptr& b, point_ptr& e) { if (b != e) { point_ptr prev_p = get_prev(p); point_ptr prev_b = get_prev(b); point_ptr prev_e = get_prev(e); set_next(prev_e, p); set_prev(p, prev_e); set_next(prev_b, e); set_prev(e, prev_b); set_next(prev_p, b); set_prev(b, prev_p); } else { link_before(p, b); } } template void reverse_ring(point_ptr pp) { if (!pp) { return; } point_ptr pp1; point_ptr pp2; pp1 = pp; do { pp2 = pp1->next; pp1->next = pp1->prev; pp1->prev = pp2; pp1 = pp2; } while (pp1 != pp); } template double area_from_point(point_ptr op, std::size_t& size) { point_ptr startOp = op; size = 1; double a = 0.0; do { ++size; a += static_cast(op->prev->x + op->x) * static_cast(op->prev->y - op->y); op = op->next; } while (op != startOp); return a * 0.5; } template double area(ring_ptr r) { assert(r != nullptr); if (std::isnan(r->area)) { r->area = area_from_point(r->points, r->size); } return r->area; } #ifdef DEBUG template inline std::basic_ostream& operator<<(std::basic_ostream& out, const ring& r) { out << " ring_index: " << r.ring_index << std::endl; if (!r.parent) { // out << " parent_ring ptr: nullptr" << std::endl; out << " parent_index: -----" << std::endl; } else { // out << " parent_ring ptr: " << r.parent << std::endl; out << " parent_ring idx: " << r.parent->ring_index << std::endl; } ring_ptr n = const_cast>(&r); if (ring_is_hole(n)) { out << " is_hole: true " << std::endl; } else { out << " is_hole: false " << std::endl; } auto pt_itr = r.points; if (pt_itr) { out << " area: " << r.area << std::endl; out << " points:" << std::endl; out << " [[[" << pt_itr->x << "," << pt_itr->y << "],"; pt_itr = pt_itr->next; while (pt_itr != r.points) { out << "[" << pt_itr->x << "," << pt_itr->y << "],"; pt_itr = pt_itr->next; } out << "[" << pt_itr->x << "," << pt_itr->y << "]]]" << std::endl; } else { out << " area: NONE" << std::endl; out << " points: NONE" << std::endl; } return out; } template std::string output_as_polygon(ring_ptr r) { std::ostringstream out; auto pt_itr = r->points; if (pt_itr) { out << "["; out << "[[" << pt_itr->x << "," << pt_itr->y << "],"; pt_itr = pt_itr->next; while (pt_itr != r->points) { out << "[" << pt_itr->x << "," << pt_itr->y << "],"; pt_itr = pt_itr->next; } out << "[" << pt_itr->x << "," << pt_itr->y << "]]"; for (auto const& c : r->children) { pt_itr = c->points; if (pt_itr) { out << ",[[" << pt_itr->x << "," << pt_itr->y << "],"; pt_itr = pt_itr->next; while (pt_itr != c->points) { out << "[" << pt_itr->x << "," << pt_itr->y << "],"; pt_itr = pt_itr->next; } out << "[" << pt_itr->x << "," << pt_itr->y << "]]"; } } out << "]" << std::endl; } else { out << "[]" << std::endl; } return out.str(); } template inline std::basic_ostream& operator<<(std::basic_ostream& out, const ring_list& rings) { out << "START RING LIST" << std::endl; for (auto& r : rings) { out << " ring: " << r->ring_index << " - " << r << std::endl; out << *r; } out << "END RING LIST" << std::endl; return out; } template inline std::basic_ostream& operator<<(std::basic_ostream& out, const ring_vector& rings) { out << "START RING VECTOR" << std::endl; for (auto& r : rings) { if (!r->points) { continue; } out << " ring: " << r->ring_index << " - " << r << std::endl; out << *r; } out << "END RING VECTOR" << std::endl; return out; } template inline std::basic_ostream& operator<<(std::basic_ostream& out, const std::deque>& rings) { out << "START RING VECTOR" << std::endl; for (auto& r : rings) { if (!r.points) { continue; } out << " ring: " << r.ring_index << std::endl; out << r; } out << "END RING VECTOR" << std::endl; return out; } template inline std::basic_ostream& operator<<(std::basic_ostream& out, const hot_pixel_vector& hp_vec) { out << "Hot Pixels: " << std::endl; for (auto& hp : hp_vec) { out << hp << std::endl; } return out; } #endif } } }