#include "wire_planner.h" namespace qucs_s { namespace wire { inline bool is_horizontal_or_vertical(const QPoint from, const QPoint to) { return from.x() == to.x() || from.y() == to.y(); } // TODO: migrate to C++20 and yonger and use std::midpoint instead int midpoint(int a, int b) { return a + (b - a) / 2; } std::vector straight(const QPoint from, const QPoint to) { return {from, to}; } std::vector two_step_xy(const QPoint from, const QPoint to) { if (is_horizontal_or_vertical(from, to)) { return {from, to}; }; /* From o---+ | +---o To */ return {from, {to.x(), from.y()}, to}; } std::vector two_step_yx(const QPoint from, const QPoint to) { if (is_horizontal_or_vertical(from, to)) { return {from, to}; }; /* From o | +---o To */ return {from, {from.x(), to.y()}, to}; } std::vector three_step_xy(const QPoint from, const QPoint to) { if (is_horizontal_or_vertical(from, to)) { return {from, to}; }; /* From o---+ | +---o To */ int mid_x = midpoint(from.x(), to.x()); return {from, {mid_x, from.y()}, {mid_x, to.y()}, to}; } std::vector three_step_yx(const QPoint from, const QPoint to) { if (is_horizontal_or_vertical(from, to)) { return {from, to}; }; /* o From | +---+ | o To */ int mid_y = midpoint(from.y(), to.y()); return {from, {from.x(), mid_y}, {to.x(), mid_y}, to}; } static const std::map routers = { {Planner::PlanType::TwoStepXY, two_step_xy}, {Planner::PlanType::TwoStepYX, two_step_yx}, {Planner::PlanType::ThreeStepXY,three_step_xy}, {Planner::PlanType::ThreeStepYX, three_step_yx} }; Planner::Planner() : current{routers.begin()} {} std::vector Planner::plan(PlanType type, const QPoint from, const QPoint to) { return routers.at(type)(from, to); } std::vector Planner::plan(const QPoint from, const QPoint to) const { return current->second(from, to); } void Planner::next() { ++current; if (current != routers.cend()) { return; } current = routers.cbegin(); } } // namespace wire } // namespace qucs_s